problem1 linknode
若是$k$是先手必勝那麼$f(k)=1$不然$f(k)=0$code
經過對前面小的數字的計算能夠發現:(1)$f(2k+1)=0$,(2)$f(2^{2k+1})=0$,(3)其餘狀況都是1blog
這個能夠用數學概括法證實字符串
problem2 linkget
假設字符串的總長度爲$n$數學
首先,設$x_{k}$爲位置$i$通過$k$ 次交換後仍然在$i$的機率,那麼在其餘位置$j(j\ne i)$的機率爲$\frac{x}{n-1}$.能夠獲得關於$x_{k}$的轉移方程$x_{0}=1, x_{k}=x_{k-1}*\frac{C_{n-1}^{2}}{C_{n}^{2}}+(1-x_{k-1})*\frac{1}{C_{n}^{2}}$string
計算出了$x_{k}$以後。對於$[1,n]$的某個位置$i$,其對答案的貢獻爲$G(i)=(x_{k}*P+\frac{x_{k}}{n-1}*(T-P))*S_{i}$it
其中$S_{i}$表示位置$i$的數字。$P$表示任意選一個區間包含$i$的機率,$P=F_{i}=\frac{2i(n-i+1)}{n(n+1)}$,而$T=\sum_{i=1}^{n}F_{i}$ast
problem3 linkclass
這個能夠用最大流來解決。
(1)將每一個黑色的格子拆分紅兩個點,一個表示進入該格子,一個表示離開該格子。這兩個點之間連邊,流量爲1。
(2)將全部白色的格子,分爲兩類,第一類是列數爲偶數的,第二類是列數爲奇數的。源點向第一類連邊,流量爲1;第二類向匯點連邊,流量爲1。另外,第一類向周圍的四個黑色格子拆成的第一個連邊,黑色格子拆成的第二個向第二類連邊。
最後求最大流便可。
code for problem1
#include <string> class TheNumberGameDivOne { public: std::string find(long long n) { if (IsFirstWin(n)) { return "John"; } return "Brus"; } private: bool IsFirstWin(long long n) { if (n == 1 || n % 2 == 1) { return false; } int c = 0; while (n % 2 == 0) { ++c; n /= 2; } if (n == 1 && c % 2 == 1) { return false; } return true; } };
code for problem2
#include <string> #include <vector> class TheSwapsDivOne { public: double find(const std::vector<std::string> &seq, int k) { int n = 0; int sum = 0; for (const auto &e : seq) { n += static_cast<int>(e.size()); for (char c : e) { sum += c - '0'; } } auto Get = [&](int t) { return 2.0 * t * (n - t + 1) / n / (n + 1); }; double sum_rate = 0.0; for (int i = 1; i <= n; ++i) { sum_rate += Get(i); } double p = 1.0 * (n - 2) / n; double q = 2.0 / n / (n - 1); double x = 1.0; for (int i = 1; i <= k; ++i) { x = p * x + q * (1 - x); } double result = 0; int idx = 0; for (const auto &e : seq) { for (size_t i = 0; i < e.size(); ++i) { ++idx; int d = e[i] - '0'; double r = Get(idx); result += (x * r + (1 - x) / (n - 1) * (sum_rate - r)) * d; } } return result; } };
code for problem3
#include <limits> #include <string> #include <unordered_map> #include <vector> template <typename FlowType> class MaxFlowSolver { static constexpr FlowType kMaxFlow = std::numeric_limits<FlowType>::max(); static constexpr FlowType kZeroFlow = static_cast<FlowType>(0); struct node { int v; int next; FlowType cap; }; public: int VertexNumber() const { return used_index_; } FlowType MaxFlow(int source, int sink) { source = GetIndex(source); sink = GetIndex(sink); int n = VertexNumber(); std::vector<int> pre(n); std::vector<int> cur(n); std::vector<int> num(n); std::vector<int> h(n); for (int i = 0; i < n; ++i) { cur[i] = head_[i]; num[i] = 0; h[i] = 0; } int u = source; FlowType result = 0; while (h[u] < n) { if (u == sink) { FlowType min_cap = kMaxFlow; int v = -1; for (int i = source; i != sink; i = edges_[cur[i]].v) { int k = cur[i]; if (edges_[k].cap < min_cap) { min_cap = edges_[k].cap; v = i; } } result += min_cap; u = v; for (int i = source; i != sink; i = edges_[cur[i]].v) { int k = cur[i]; edges_[k].cap -= min_cap; edges_[k ^ 1].cap += min_cap; } } int index = -1; for (int i = cur[u]; i != -1; i = edges_[i].next) { if (edges_[i].cap > 0 && h[u] == h[edges_[i].v] + 1) { index = i; break; } } if (index != -1) { cur[u] = index; pre[edges_[index].v] = u; u = edges_[index].v; } else { if (--num[h[u]] == 0) { break; } int k = n; cur[u] = head_[u]; for (int i = head_[u]; i != -1; i = edges_[i].next) { if (edges_[i].cap > 0 && h[edges_[i].v] < k) { k = h[edges_[i].v]; } } if (k + 1 < n) { num[k + 1] += 1; } h[u] = k + 1; if (u != source) { u = pre[u]; } } } return result; } MaxFlowSolver() = default; void Clear() { edges_.clear(); head_.clear(); vertex_indexer_.clear(); used_index_ = 0; } void InsertEdge(int from, int to, FlowType cap) { from = GetIndex(from); to = GetIndex(to); AddEdge(from, to, cap); AddEdge(to, from, kZeroFlow); } private: int GetIndex(int idx) { auto iter = vertex_indexer_.find(idx); if (iter != vertex_indexer_.end()) { return iter->second; } int map_idx = used_index_++; head_.push_back(-1); return vertex_indexer_[idx] = map_idx; } void AddEdge(int from, int to, FlowType cap) { node p; p.v = to; p.cap = cap; p.next = head_[from]; head_[from] = static_cast<int>(edges_.size()); edges_.emplace_back(p); } std::vector<node> edges_; std::vector<int> head_; std::unordered_map<int, int> vertex_indexer_; int used_index_ = 0; }; class TheTilesDivOne { public: int find(const std::vector<std::string> &board) { MaxFlowSolver<int> solver; int n = static_cast<int>(board.size()); int m = static_cast<int>(board[0].size()); auto GetWhite = [&](int x, int y) { return x * m + y; }; auto GetBlackIn = [&](int x, int y) { return GetWhite(x, y); }; auto GetBlackOut = [&](int x, int y) { return GetWhite(x, y) + n * m; }; auto Valid = [&](int x, int y) { return 0 <= x && x < n && 0 <= y && y < m && board[x][y] != 'X'; }; int source = -1; int sink = -2; const int kDirX[] = {-1, 0, 1, 0}; const int kDirY[] = {0, 1, 0, -1}; for (int i = 0; i < n; ++i) { for (int j = i & 1; j < m; j += 2) { if (Valid(i, j)) { int in = GetBlackIn(i, j); int out = GetBlackOut(i, j); solver.InsertEdge(in, out, 1); for (int k = 0; k < 4; ++k) { int kx = i + kDirX[k]; int ky = j + kDirY[k]; if (Valid(kx, ky)) { if (kx % 2 == 0) { solver.InsertEdge(GetWhite(kx, ky), in, 1); } else { solver.InsertEdge(out, GetWhite(kx, ky), 1); } } } } } } for (int i = 0; i < n; ++i) { for (int j = 0; j < m; ++j) { if ((i + j) % 2 == 1 && Valid(i, j)) { if (i % 2 == 0) { solver.InsertEdge(source, GetWhite(i, j), 1); } else { solver.InsertEdge(GetWhite(i, j), sink, 1); } } } } return solver.MaxFlow(source, sink); } };