Rank
炸了,触底反弹
A. 五彩斑斓(colorful)
签,又没签上。
考虑如何一步步优化暴力。最暴力的思想 \(\mathcal{O(n^4)}\) 枚举每个矩形,判断四个顶点颜色。稍微优化些,两次 \(\mathcal{O(n^2)}\) 跑出对于行/列每个点下一个与之颜色相同的坐标,利用容斥全部减去不合法的方案数,然后再枚举每个点,随机数据下跑得很快,颜色数量越少效率越低。
考虑正解,依旧容斥,我们可以直接枚举两行,再枚举列,记录每一个相同颜色组成的点对出现的数量,简单组合计数可知对于 \(n\) 组点对能组成的方案数为 \(\sum_{i=1}^n\ i\),增加出现数量时直接在总 \(ans\) 里减一下就行了,每次统计后清空计数数组,这样复杂度为 \(\mathcal{O(n^3)}\),可以轻松过。
据说有 \(\mathcal{O(n^4)}\) 算法各种大力优化碾过去了。
点击查看代码
#include<bits/stdc++.h>
#define fo(x, y, z) for(register int (x) = (y); (x) <= (z); (x)++)
#define fu(x, y, z) for(register int (x) = (y); (x) >= (z); (x)--)
using namespace std;
typedef long long ll;
#define lx ll
inline lx qr()
{char ch = getchar();lx x = 0 , f = 1;for(;ch<'0'||ch>'9';ch = getchar()) if(ch == '-') f = -1;for(;ch>= '0' && ch<= '9';ch = getchar()) x = (x<<3) + (x<<1) + (ch^48);return x*f;
}
#undef lx
#define qr qr()
#define pii pair<int , int>
#define fi first
#define se second
const int Ratio = 0;
const int N = 400 + 5;
const int mod = 998244353;
int n, m;
int c[N][N], tim[N * N];
ll ans;
namespace Wisadel
{short main(){freopen("colorful.in", "r", stdin) , freopen("colorful.out", "w", stdout);n = qr, m = qr;fo(i, 1, n) fo(j, 1, m) c[i][j] = qr;ans = 1ll * n * (n + 1) * m * (m + 1) / 4;fo(i, 1, n) fo(j, i, n){fo(k, 1, m) if(c[i][k] == c[j][k]) tim[c[i][k]]++, ans -= tim[c[i][k]];fo(k, 1, m) if(c[i][k] == c[j][k]) tim[c[i][k]] = 0;}printf("%lld\n", ans);return Ratio;}
}
signed main(){return Wisadel::main();}
B. 错峰旅行(travel)
签,双没签上。
看到第一感觉是动态开点线段树,不过赛时怎么压都过不了最后一个大样例,卡到极限 70pts,自己画蛇添足唐错挂 10pts。
其实是可以用线段树做的。首先很显然的是,对于一段时间 \([s,t]\),如果不拥挤城市数都是 \(n\),那么总方案数为 \(\mathcal{n^{t-s+1}}\)。初始每一天合法城市数都是 \(n\),由于题目说明了同一城市没有交集,所以每个限制条件相当于做了一次区间减的操作。我们用标记永久化的思想,每次正常区间减,不下放标记,最后做一次最坏 \(\mathcal{O(n)}\) 的查询就做完了。
点击查看代码
#include<bits/stdc++.h>
#define fo(x, y, z) for(register int (x) = (y); (x) <= (z); (x)++)
#define fu(x, y, z) for(register int (x) = (y); (x) >= (z); (x)--)
using namespace std;
typedef long long ll;
#define lx ll
inline lx qr()
{char ch = getchar();lx x = 0 , f = 1;for(;ch<'0'||ch>'9';ch = getchar()) if(ch == '-') f = -1;for(;ch>= '0' && ch<= '9';ch = getchar()) x = (x<<3) + (x<<1) + (ch^48);return x*f;
}
#undef lx
#define qr qr()
#define pii pair<int , int>
#define fi first
#define se second
const int Ratio = 0;
const int N = 1e6 + 5;
const int mod = 1e9 + 7;
int n, m, s, t, tot, ans = 1;
int v[N * 20], son[N * 20][2];
namespace Wisadel
{int Wqp(ll x, int y){int res = 1;while(y){if(y & 1) res = 1ll * res * x % mod; x = x * x % mod; y >>= 1;}return res;}#define ls (son[rt][0])#define rs (son[rt][1])#define mid ((l + r) >> 1)void Wupd(int rt, int l, int r, int x, int y){if(x <= l && r <= y){v[rt]--;return;}if(x <= mid){if(!ls) ls = ++tot;Wupd(ls, l, mid, x, y);}if(y > mid){if(!rs) rs = ++tot;Wupd(rs, mid + 1, r, x, y);}}void Wq(int rt, int l, int r, int now){if(!rt){ans = 1ll * ans * Wqp(now, r - l + 1) % mod;return;}if(ls || rs) Wq(ls, l, mid, now + v[rt]), Wq(rs, mid + 1, r, now + v[rt]);else ans = 1ll * ans * Wqp(now + v[rt], r - l + 1) % mod;}short main(){freopen("travel.in", "r", stdin) , freopen("travel.out", "w", stdout);n = qr, m = qr, s = qr, t = qr;t -= s - 1;tot = 1, v[1] = n;fo(i, 1, m){int x = qr, l = qr - s + 1, r = qr - s + 1;Wupd(1, 1, t, l, r);}Wq(1, 1, t, 0);printf("%d\n", ans);return Ratio;}
}
signed main(){return Wisadel::main();}
正解给出的是差分的思想,也很简单,有用的信息最多 \(2\times m\) 个,我们记录每个限制信息的左右端点,到左端点处合法城市数减一,到右端点 +1 处合法城市数加一。实现也很简单,pair 记录后 sort 一遍,加入首末时间 \(\mathcal{O(m)}\) 跑一遍就行,效率比上面做法高很多。
点击查看代码
#include<bits/stdc++.h>
#define fo(x, y, z) for(register int (x) = (y); (x) <= (z); (x)++)
#define fu(x, y, z) for(register int (x) = (y); (x) >= (z); (x)--)
using namespace std;
typedef long long ll;
#define lx ll
inline lx qr()
{char ch = getchar();lx x = 0 , f = 1;for(;ch<'0'||ch>'9';ch = getchar()) if(ch == '-') f = -1;for(;ch>= '0' && ch<= '9';ch = getchar()) x = (x<<3) + (x<<1) + (ch^48);return x*f;
}
#undef lx
#define qr qr()
#define pii pair<int , int>
#define fi first
#define se second
const int Ratio = 0;
const int N = 1e6 + 5;
const int mod = 1e9 + 7;
int n, m, s, t, tot, ans = 1;
pii p[N << 1];
namespace Wisadel
{int Wqp(ll x, int y){int res = 1;while(y){if(y & 1) res = 1ll * res * x % mod; x = x * x % mod; y >>= 1;}return res;}short main(){freopen("travel.in", "r", stdin) , freopen("travel.out", "w", stdout);n = qr, m = qr, s = qr, t = qr;fo(i, 1, m){int x = qr, l = qr, r = qr;p[++tot] = {l, -1}, p[++tot] = {r + 1, 1};}sort(p + 1, p + 1 + tot);p[0] = {s, 0};p[++tot] = {t + 1, 0};fo(i, 0, tot - 1){n += p[i].se;if(p[i].fi != p[i + 1].fi)ans = 1ll * ans * Wqp(n, p[i + 1].fi - p[i].fi) % mod;}printf("%d\n", ans);return Ratio;}
}
signed main(){return Wisadel::main();}
C. 线段树(segment)
签(?),叒没签上。
还以为真是线段树,赛时脑子被禁锢了转不了,暴力都没写出来。
据 wang54321 说这是他们暑假第二场模拟赛原题。
正解区间 dp。考虑维护包含 \([i,j]\) 的询问数量 \(s_{i,j}\),设 \(f_{i,j}\) 表示所有询问在区间 \([i,j]\) 子树内的答案,我们枚举所有区间,然后枚举其中可行的分割点,则有状态转移方程:
边界处理上,初始化 \(f_{i,i} = s_{i,i}\),其余赋为极大值即可。复杂度 \(\mathcal{O(n^3)}\)。
点击查看代码
#include<bits/stdc++.h>
#define fo(x, y, z) for(register int (x) = (y); (x) <= (z); (x)++)
#define fu(x, y, z) for(register int (x) = (y); (x) >= (z); (x)--)
using namespace std;
typedef long long ll;
#define lx ll
inline lx qr()
{char ch = getchar();lx x = 0 , f = 1;for(;ch<'0'||ch>'9';ch = getchar()) if(ch == '-') f = -1;for(;ch>= '0' && ch<= '9';ch = getchar()) x = (x<<3) + (x<<1) + (ch^48);return x*f;
}
#undef lx
#define qr qr()
#define pii pair<int , int>
#define fi first
#define se second
const int Ratio = 0;
const int N = 500 + 5;
const int mod = 1e9 + 7;
int n, m;
int sum[N][N], f[N][N];
namespace Wisadel
{short main(){// freopen(".in", "r", stdin) , freopen(".out", "w", stdout);freopen("segment.in", "r", stdin) , freopen("segment.out", "w", stdout);n = qr, m = qr;memset(f, 0x3f, sizeof f);fo(i, 1, m){int a = qr, b = qr;sum[a][b]++;}fo(i, 1, n) fu(j, n, i) sum[i][j] += sum[i][j + 1];fo(j, 1, n) fo(i, 1, j) sum[i][j] += sum[i - 1][j];fo(i, 1, n) f[i][i] = sum[i][i];fo(len, 2, n) fo(l, 1, n - len + 1){int r = l + len - 1;fo(k, l, r - 1)f[l][r] = min(f[l][r], f[l][k] + f[k + 1][r] - sum[l][r]);}printf("%d\n", f[1][n]);return Ratio;}
}
signed main(){return Wisadel::main();}
D. 量子隧穿问题(experiment)
叕没做出来(?)
一个 trick 是计数转概率,答案 \(=\) 总方案数 \(\times\) 合法概率。
这样一来,盒子的状态可以得出有猫的概率,状态转移方程如下:
然后就可以做 dp 了。吗?
显然这么简单就不至于放 T4 了。发现每个隧道有可能连成环,整个图是一个基环树。当 \(n\) 存在与一个环内时转移时就会因为二者概率不独立而出错。因此我们需要断一条边,枚举两点的概率 1/0,然后再根据上面的转移方程就能无后效性地 dp 了。
注意由于 巴甫洛夫的狗 是从 1 到 \(n\) 运动的,我们断的必须是第一个与 \(n\) 在一个环内的盒子的边,可以用并查集辅助解决。
点击查看代码
#include<bits/stdc++.h>
#define fo(x, y, z) for(register int (x) = (y); (x) <= (z); (x)++)
#define fu(x, y, z) for(register int (x) = (y); (x) >= (z); (x)--)
using namespace std;
typedef long long ll;
#define lx ll
inline lx qr()
{char ch = getchar();lx x = 0 , f = 1;for(;ch<'0'||ch>'9';ch = getchar()) if(ch == '-') f = -1;for(;ch>= '0' && ch<= '9';ch = getchar()) x = (x<<3) + (x<<1) + (ch^48);return x*f;
}
#undef lx
#define qr qr()
#define pii pair<int , int>
#define fi first
#define se second
const int Ratio = 0;
const int N = 1e5 + 5;
const int mod = 1e9 + 7;
int n, m;
int to[N], p[N], fx[N];
bool in[N];
string s;
namespace Wisadel
{int Wfind(int x){if(x == fx[x]) return x;return fx[x] = Wfind(fx[x]);}short main(){freopen("experiment.in", "r", stdin) , freopen("experiment.out", "w", stdout);int T = qr;while(T--){n = qr; cin >> s;memset(in, 0, sizeof in);fo(i, 1, n) to[i] = qr, fx[i] = i;int st = 0, ans = 0;fo(i, 1, n) fx[Wfind(i)] = Wfind(to[i]);fo(i, 1, n) in[i] = (Wfind(i) == Wfind(n));fo(i, 1, n) fx[i] = i;fu(i, n, 1){int _ = Wfind(i), __ = Wfind(to[i]);if(in[i] && _ != __) fx[_] = __;else if(in[i]) st = i;}fo(x, 0, 1) fo(y, 0, 1){fo(i, 0, n - 1)if(s[i] == 'C') p[i + 1] = 1;else if(s[i] == '.') p[i + 1] = 0;else p[i + 1] = (1 + mod) / 2;int res = 1;fo(i, 1, n){if(i == st){res = 1ll * (x ? p[i] : (1 - p[i])) * (y ? p[to[i]] : (1 - p[to[i]])) % mod;p[i] = x;p[to[i]] = y;}int xx = p[i], yy = p[to[i]];p[i] = 1ll * xx * yy % mod;p[to[i]] = (1ll * yy + 1ll * xx * (1 - yy) % mod) % mod;}ans = (ans + 1ll * res * p[n] % mod) % mod;}fo(i, 0, n - 1) if(s[i] == '?') ans = ans * 2 % mod;printf("%d\n", (ans + mod) % mod);}return Ratio;}
}
signed main(){return Wisadel::main();}
末
怀疑这个出题人看过 某人 和 某人 的唐博,开幕巴甫洛夫的狗雷击整个机房。
挺烂的,作为 17 岁 的开局有点糟糕,希望低开高走吧。
一整天状态不太好,好在下午及时调整过来了。
一些消极的话,有反转
今天的超链接好像放的过于多了
完结撒花~
最后再祝自己生日快乐!