[LibreOJ 2058][TJOI2016]求和
求\(\sum_{i = 0}^n\sum_{j = 0}^i \begin{Bmatrix}i\\ j\end{Bmatrix}2^jj!\)。
\(1\leq n\leq 10^5\)。
[LibreOJ 2538][PKUWC2018]Slay the Spire
给定一个\(2n\)张卡的卡组,其中有\(n\)张强化卡(其权值为一大于\(1\)的整数),\(n\)张攻击卡(其权值为一正整数)。在一次游戏中,你需要有序的打出一些卡牌,如果说你打了一张权值为\(x\)的攻击卡,那么对对方造成\(x\)点伤害;如果说你打出了一张权值为\(x\)的强化卡,那么之后所有伤害乘上\(x\)。
现在,你需要随机从卡组中抽出\(m\)张牌,然后选出其中的\(k\)张照一定顺序打出,要求使得产生的伤害最大化。求伤害的期望,答案乘\(\binom{2n}{m}\)再膜\(998244353\)输出。
\(1\leq k\leq m\leq 2n\leq 3000, 1\leq x\leq 10^8\)(其中\(x\)为牌的权值)。
多组数据,满足\(\sum 2n\leq 30000\)。
[LibreOJ 2803][CCC2018]平衡树
假设每个结点都有正整数权值,那么我们定义完美平衡树:权值为\(1\)的单点树是完美平衡树;根的权值为\(w(w\geq 2)\)的话,那么它一定有\(k(2\leq k\leq w)\)个子树,每个子树要完全一致,并且每个子树的根权值都要是\(\lfloor\frac{w}{k}\rfloor\)。
给定\(N\),计算根权值为\(N\)的完美平衡树的数量。
\(1\leq N\leq 10^9\)。
[LibreOJ 6433][PKUSC2018]最大前缀和
给你一个长为\(n\)的整数序列\(a\),求出将序列随机打乱之后的最大前缀和(不能选空前缀!)的期望,答案乘上\(n!\)之后对\(998244353\)输出。
\(1\leq n\leq 20, \sum_{i = 1}^n |a_i|\leq 10^9\)。
[LibreOJ 2316][NOIP2017]逛公园
给你一张\(n\)个点\(m\)条边的有向图,设从\(1\)到\(n\)的最短路为\(d\),那么请问有多少\(1\)到\(n\)的路径(可以是非简单路径)满足其长度在\([d, d + K]\)中,如果答案为无限大的话输出\(-1\)。
\(1\leq n\leq 10^5, 1\leq m\leq 2\times 10^5, 0\leq K\leq 50\)。
[LibreOJ 6268]分拆数
定义分拆数\(f(x)\)表示将\(x\)拆为若干正整数的本质不同方案数。对于\(i = 1\ldots n\),输出\(f(i)\)。
\(1\leq n\leq 10^5\)。
[LibreOJ 2185][SDOI2015]约数个数和
首先我们有这样一个式子:
\[d(ij) = \sum_{a | i}\sum_{b | j} [\gcd(\frac{i}{a}, b) = 1]\]
怎样理解这一结论呢?我们知道\(ij\)的约数\(d\)一定可以被分解为\(ab\)使得\(a | i, b | j\),但这种划分可能会非常的多,我们要保证每个约数\(d\)只有一种划分被统计过。
那么考虑上述划分方法。对于\(i\)和\(j\)中一个有一个没有的质因子,是一定会划分到\(a\)和\(b\)中固定的一个的。那么考虑一个在\(i\)和\(j\)中都出现的质因子\(p\),假设\(p\)在\(d\)中出现的次数超过了\(i\)和\(j\)中任何一个的话,那么\(p\)在\(a\)中一定会尽可能多的出现(否则\(\frac{i}{a}\)中会出现\(p\),而这样无论如何一定有\(p | b\),因此会不满足\(\gcd(\frac{i}{a}, b) = 1\))。如果没超过呢?那么一定会全部划分到\(a\)中(其他情况下有\(p | b\)并且会出现\(p | \frac{i}{a}\))。
下面考虑应用这个结论(为了方便,使用等价的\(d(ij) = \sum_{a | i}\sum_{b | j} [\gcd(a, b) = 1]\),假设\(n\le m\)):
\[
\begin{aligned}
\quad&\sum_{i = 1}^n\sum_{j = 1}^m \sum_{a | i}\sum_{b | j}\sum_{d | i, d | j}\mu(d)\\
=&\sum_{d = 1}^n\mu(d)\sum_{a = 1}^{\lfloor\frac{n}{d}\rfloor}\lfloor\frac{n}{ad}\rfloor\sum_{b = 1}^{\lfloor\frac{m}{d}\rfloor}\lfloor\frac{m}{bd}\rfloor\\
=&\sum_{d = 1}^n\mu(d)\sum_{a = 1}^{\lfloor\frac{n}{d}\rfloor}d(a)\sum_{b = 1}^{\lfloor\frac{m}{d}\rfloor}d(b)
\end{aligned}
\]
后面两个和式可以随便\(O(n)\)欲处理一下(然后我当初写了个\(O(n\sqrt{n})\)的神必预处理……),然后这个题做完力,,,
代码:
#include <cstdio> #include <cstring> #include <cstdlib> #include <cctype> #include <algorithm> #include <queue> #include <utility> typedef long long int_t; const int maxn = 50005; int_t miu[maxn], miu_s[maxn], f[maxn]; void process() { static bool vis[maxn]; static int prm[maxn]; int cnt = 0; const int N = 50000; miu[1] = 1LL; for(int i = 2; i <= N; i ++) { if(!vis[i]) { miu[i] = -1LL; prm[cnt ++] = i; } for(int j = 0; j < cnt && prm[j] * i <= N; j ++) { int v = prm[j] * i; vis[v] = true; if(i % prm[j] == 0) { miu[v] = 0; break; } else { miu[v] = miu[i] * -1; } } } for(int i = 1; i <= N; i ++) { miu_s[i] = miu_s[i - 1] + miu[i]; } for(int i = 1; i <= N; i ++) { for(int j = 1; j <= i;) { int nx = i / (i / j); f[i] += (int_t(nx - j + 1)) * (int_t(i / j)); j = nx + 1; } } } int_t solve(int n, int m) { int_t ans = 0; for(int i = 1; i <= std::min(n, m);) { int nx = std::min(m / (m / i), n / (n / i)); ans += (miu_s[nx] - miu_s[i - 1]) * f[m / i] * f[n / i]; i = nx + 1; } return ans; } #ifdef LOCAL #define LO "%I64d" #else #define LO "%lld" #endif int main() { process(); int T; scanf("%d", &T); while(T --) { int n, m; scanf("%d%d", &n, &m); printf(LO, solve(n, m)); puts(""); } return 0; }
[LibreOJ 2100][TJOI2015]线性代数
颓一波式子可以发现答案就是\(ABA^T-CA^T\)。然后发现对于一个\(B\)中的元素\(B_{i, j}\)要对答案做出贡献要求\(A_i\)和\(A_j\)都为1,而\(A_i\)为1会导致答案减去一个\(C_i\)。
因此我们可以分别对\(B\)中元素和\(A\)中元素建点。\(B\)中元素会带来收益,但是要求依赖两个\(A\)中元素。而\(A\)中元素会带来一定的笋丝。这个就已经事很明显的最大权闭合子图力,,,
代码:
#include <cstdio> #include <cstring> #include <cstdlib> #include <algorithm> #include <functional> #include <utility> #include <vector> #include <queue> #include <stack> const int maxn = 505; const int maxno = 500 * 500 + 500 + 5; const int maxm = 2 * (500 * 500 * 3 + 500 + 5); int first[maxno]; int next[maxm], to[maxm], flow[maxm], cap[maxm]; int gcnt = 0; void add_edge(int u, int v, int c) { gcnt ++; next[gcnt] = first[u]; first[u] = gcnt; to[gcnt] = v; cap[gcnt] = c; flow[gcnt] = 0; } void ins_edge(int u, int v, int c) { add_edge(u, v, c); add_edge(v, u, 0); } int rev(int i) { if(1 & i) { return i + 1; } else { return i - 1; } } int d[maxno]; int s, t, num; bool bfs() { static bool vis[maxno]; std::fill(vis, vis + num + 1, false); std::fill(d, d + num + 1, 0); std::queue<int> Q; Q.push(s); d[s] = 1; vis[s] = true; while(!Q.empty()) { int u = Q.front(); Q.pop(); for(int i = first[u]; i; i = next[i]) { int v = to[i]; if(!vis[v] && cap[i] > flow[i]) { d[v] = d[u] + 1; Q.push(v); vis[v] = true; } } } return vis[t]; } int cur[maxno]; int dfs(int x, int a) { if(a == 0 || x == t) return a; int ret = 0; for(int &i = cur[x]; i; i = next[i]) { int v = to[i], f; if(d[v] == d[x] + 1 && (f = dfs(v, std::min(a, cap[i] - flow[i]))) > 0) { ret += f; a -= f; flow[i] += f; flow[rev(i)] -= f; if(a == 0) break; } } if(a > 0) d[x] = -1; return ret; } int dinic() { int ret = 0; while(bfs()) { for(int i = 0; i <= num; i ++) cur[i] = first[i]; ret += dfs(s, 0x7f7f7f7f); } return ret; } int n; int main() { int ans = 0; scanf("%d", &n); s = 0, t = n * n + n + 1; num = t; for(int i = 1; i <= n; i ++) { for(int j = 1; j <= n; j ++) { int u = (i - 1) * n + j; int val; scanf("%d", &val); ans += val; ins_edge(s, u, val); ins_edge(u, n * n + i, 0x7f7f7f7f); ins_edge(u, n * n + j, 0x7f7f7f7f); } } for(int i = 1; i <= n; i ++) { int val; scanf("%d", &val); ins_edge(n * n + i, t, val); } ans -= dinic(); printf("%d\n", ans); return 0; }
[LibreOJ 2472][九省联考2018]IIIDX
一直没有填的坑……今天乱写了写跑过了然后厚颜无耻凑一篇题解
还有样例过于草(出题人钦点淫梦运营)
考虑从小到大枚举序列的每一位置\(p\),然后我们发现每一位置的值的合法性事单调的(小的合法,大了就不合法),这样可以对每一个位置二分答案。
然后我们考虑二分完了答案\(x\)怎么判定。把值离散化一下然后用一个线段树来维护\(T_i\)(表示当前还没有被预定的大于等于\(i\)的数有多少个),然后判定答案就看到\(x\)的前缀区间是否都不小于\(p\)所在子树的大小。同时注意顺次枚举每个点的时候要去除他对他父亲的影响,然后每次一个点确定是什么值了也要修改对应的前缀区间。
代码:
#include <cstdio> #include <cstring> #include <cstdlib> #include <cmath> #include <algorithm> #include <functional> #include <utility> using R = long double; const int maxn = 500005; const int maxno = maxn << 2; int n; R k; int minv[maxno], addv[maxno]; void maintain(int o) { int lc = o << 1, rc = o << 1 | 1; minv[o] = std::min(minv[lc], minv[rc]); } void paint(int o, int v) { addv[o] += v; minv[o] += v; } void pushdown(int o) { if(addv[o] != 0) { int lc = o << 1, rc = o << 1 | 1; paint(lc, addv[o]); paint(rc, addv[o]); addv[o] = 0; } } int left[maxn]; void build_tree(int o, int L, int R) { // addv[o] = 0; if(L == R) { minv[o] = left[L]; } else { int M = (L + R) / 2; build_tree(o << 1, L, M); build_tree(o << 1 | 1, M + 1, R); maintain(o); } } void update(int o, int L, int R, int ql, int qr, int v) { if(ql <= L && R <= qr) { paint(o, v); } else { pushdown(o); int M = (L + R) / 2; if(ql <= M) update(o << 1, L, M, ql, qr, v); if(qr > M) update(o << 1 | 1, M + 1, R, ql, qr, v); maintain(o); } } const int INF = 0x7f7f7f7f; int query_min(int o, int L, int R, int ql, int qr) { if(ql <= L && R <= qr) { return minv[o]; } else { pushdown(o); int M = (L + R) / 2, ans = INF; if(ql <= M) ans = std::min(ans, query_min(o << 1, L, M, ql, qr)); if(qr > M) ans = std::min(ans, query_min(o << 1 | 1, M + 1, R, ql, qr)); return ans; } } int d[maxn], d2[maxn]; int lsiz; void process() { std::sort(d + 1, d + 1 + n); std::sort(d2 + 1, d2 + 1 + n); lsiz = std::unique(d2 + 1, d2 + 1 + n) - d2 - 1; for(int i = 1; i <= n; i ++) { d[i] = std::lower_bound(d2 + 1, d2 + 1 + lsiz, d[i]) - d2; left[d[i]] ++; } for(int i = lsiz - 1; i >= 1; i --) { left[i] += left[i + 1]; } build_tree(1, 1, lsiz); } int A[maxn], siz[maxn]; void solve() { for(int i = n; i >= 1; i --) { siz[i] ++; int f = (int)floor((R(i)) / k); siz[f] += siz[i]; } for(int i = 1; i <= n; i ++) { int f = (int)floor((R(i)) / k); if(f > 0) { update(1, 1, lsiz, 1, A[f], siz[i]); } int L = (f == 0) ? 1 : A[f], R = lsiz, ret; while(true) { if(R - L <= 3) { for(int j = R; j >= L; j --) { if(query_min(1, 1, lsiz, 1, j) >= siz[i]) { ret = j; break; } } break; } int M = L + (R - L) / 2; if(query_min(1, 1, lsiz, 1, M) >= siz[i]) { L = M; } else { R = M; } } A[i] = ret; update(1, 1, lsiz, 1, ret, -siz[i]); } } int main() { scanf("%d%Lf", &n, &k); for(int i = 1; i <= n; i ++) { scanf("%d", &d[i]); d2[i] = d[i]; } process(); solve(); for(int i = 1; i <= n; i ++) { if(i > 1) putchar(' '); printf("%d", d2[A[i]]); } puts(""); return 0; }
[LibreOJ 2383][HNOI2013]游走
本野蛮人竟然没做过高消期望DP,,,泪,流了下来,,,
根据期望线性性,答案就是所有边的期望被走的次数乘上边的编号的和。一条边期望经过的次数可以根据他两个端点期望经过的次数来算(但是\(n\)要特判一下),要求所有点期望走过的次数当然就可以列\(n\)个方程然后高消力。然后期望走的次数多的边编号应该小,反之亦然,所以求完每条边走的次数的期望之后就贪心一下就好力。
代码:
#include <cstdio> #include <cstring> #include <cstdlib> #include <cassert> #include <algorithm> #include <utility> #include <cmath> const int maxn = 505; const int maxm = maxn * maxn; using R = double; const R eps = 1e-9; int sign(R x) { if(fabs(x) < eps) { return 0; } else { return ((x < 0.00) ? -1 : 1); } } R D[maxn][maxn]; int n; void gauss() { #ifdef LOCAL for(int i = 1; i <= n; i ++) { for(int j = 1; j <= n + 1; j ++) { printf("%.3lf ", D[i][j]); } puts(""); } #endif for(int i = 1; i <= n; i ++) { int r = i; for(int j = i + 1; j <= n; j ++) { if(fabs(D[j][i]) > fabs(D[r][i])) { r = j; } } assert(sign(D[r][i]) != 0); if(r != i) { for(int j = 1; j <= n + 1; j ++) { std::swap(D[i][j], D[r][j]); } } for(int k = i + 1; k <= n; k ++) { R rate = D[k][i] / D[i][i]; for(int j = i; j <= n + 1; j ++) { D[k][j] -= D[i][j] * rate; } } } for(int i = n; i >= 1; i --) { for(int j = i + 1; j <= n; j ++) { D[i][n + 1] -= D[j][n + 1] * D[i][j]; D[i][j] = 0; } D[i][n + 1] /= D[i][i]; D[i][i] = 1; #ifdef LOCAL printf("E[%d] : %.3lf\n", i, D[i][n + 1]); #endif } } int E[maxm][2], deg[maxn]; R tms[maxm]; void add_edge(int u, int v) { if(u != n) D[v][u] += 1.00 / (R(deg[u])); } int main() { int m; scanf("%d%d", &n, &m); for(int i = 1; i <= n; i ++) { D[i][i] = -1; } D[1][n + 1] = -1; for(int i = 1; i <= m; i ++) { scanf("%d%d", &E[i][0], &E[i][1]); deg[E[i][0]] ++; deg[E[i][1]] ++; } for(int i = 1; i <= m; i ++) { int u = E[i][0], v = E[i][1]; add_edge(u, v); add_edge(v, u); } gauss(); for(int i = 1; i <= m; i ++) { tms[i] = 0; int u = E[i][0], v = E[i][1]; if(u != n) tms[i] += D[u][n + 1] / (R(deg[u])); if(v != n) tms[i] += D[v][n + 1] / (R(deg[v])); } std::sort(tms + 1, tms + 1 + m, [&](const R &i, const R &j) { return i > j; }); R ans = 0; for(int i = 1; i <= m; i ++) { ans += tms[i] * (R(i)); } printf("%.3lf\n", ans); return 0; }