A. 卡门
https://www.luogu.com.cn/problem/P6370
http://222.180.160.110:61235/contest/4434/problem/1
我们惊讶地发现全场没多少人会做绿题。
就是说呢,我们把它「滚动到相邻行然后落下」的过程直接变成「往左下 / 右下滚」。那么球掉下去的过程呢,就可以理解为它是一直一次往下掉一格的,然后每一次掉落呢,会根据实际情况往左 / 往右 / 直着掉。
所以我们可以直接用 \(m\) 个长度为 \(n\) 的序列,记录从每 \(1\sim m\) 列扔下去的情况。然后我们很容易可以知道,如果任意两个序列发生了交汇,那么从交汇点开始至序列结束呢,这两个序列的内容都是完全一样的。因为之前序列长什么样子,完全不会对之后产生影响。所以一旦有任意一个状态是一样的,后面都会是一样的。
所以呢,一旦我们跟着计划好的路线走,却发现走不动了,那就说明从走不动的位置开始,到原定路线结束,这些位置都会被封掉。所以这个时候我们直接重新计算路线就好了。
对于这 \(m\) 列,就算每一列预定路线上的每个位置都被占了一次,也只会被更改 \(n\times m\) 次;总体时间复杂度只有 \(O(nm^2)\)。然后又由于不可能跑满,其实是完全没有问题,甚至跑得出溜快的。
namespace XSC062 {
using namespace fastIO;
const int maxm = 35;
const int maxn = 3e4 + 5;
char a[maxn][maxm];
int n, m, q, u, x, y;
std::vector<int> p[maxm];
int main() {
#ifdef ONLINE_JUDGE
freopen("kamen.in", "r", stdin);
freopen("kamen.out", "w", stdout);
#endif
scanf("%d %d", &n, &m);
for (int i = 1; i <= n; ++i)
scanf("%s", a[i] + 1);
for (int i = 1; i <= m; ++i) {
x = 1, y = i;
for (;;) {
p[i].push_back(y);
if (a[x + 1][y] == '.') ++x;
else {
if (a[x + 1][y] == 'X') break;
if (a[x][y - 1] == '.' && a[x + 1][y - 1] == '.')
++x, --y;
else if (a[x][y + 1] == '.' && a[x + 1][y + 1] == '.')
++x, ++y;
else break;
}
}
}
scanf("%d", &q);
while (q--) {
scanf("%d", &u);
while (a[p[u].size()][p[u].back()] != '.')
p[u].pop_back();
x = p[u].size(), y = p[u].back();
for (;;) {
p[u].push_back(y);
if (a[x + 1][y] == '.') ++x;
else {
if (a[x + 1][y] == 'X') break;
if (a[x][y - 1] == '.' && a[x + 1][y - 1] == '.')
++x, --y;
else if (a[x][y + 1] == '.' && a[x + 1][y + 1] == '.')
++x, ++y;
else break;
}
}
a[x][y] = 'O';
}
for (int i = 1; i <= n; ++i) puts(a[i] + 1);
return 0;
}
} // namespace XSC062
B. 商人
http://222.180.160.110:61235/contest/4434/problem/2
首先忽略图中的环带来的问题,假设 \(f_u\) 表示从 \(u\) 点出发的答案,不难想到 DP 式 \(f_u = \min\{\max(f_v-p_{(u,v)}, r_{(u,v)})\}\)。
那么问题来了,图中是会有环的,不能简单地去更新 DP 值。我们首先考虑「根源」,每一个 \(f_u\) 一定都是由另一个 \(f_v\) 转移得到的,那么一定会存在一个已知的 \(f_u\),不由其他任何 \(f_v\) 得到。
考虑一个环中 \(r\) 最大的一条边 \((u,v)\)。假设图中只有这一个环,那么 \(f_u\) 的值一定为 \(r\),因为 \(p\) 只能为非负。这样「根源」就被我们找到了。接下来进一步考虑转移方法。
一个点 \(u\) 只能被其相连的点 \(v\) 更新,那么当且仅当所有 \(f_v\) 都是确定的,\(f_u\) 才能被确定,进而去更新 \(u\) 的前驱。这让我们想到了拓扑排序。如果我们将所有边反向,那么上面的过程几乎就是拓扑排序。
为了保证任意一个环上的「根源」都能最先被确定,我们按照按 \(r\) 从大到小遍历每一条边的方式处理问题。对于每一条边 \((u,v)\),我们都假设它是「根源」并用 \(r_{(u,v)}\) 更新 \(f_u\)。我们其实并不关心 \(u\) 是否是我们刚刚定义的形式上的「根源」,毕竟,只要 \(f_u\) 能被 \(r_{(u,v)}\) 更新,它就能算作广义上的,当前时刻的「根源」。
我们在一开始将出度为 \(0\)(反图上就是入度为 \(0\))的点全部加入队列,在每次加边的时候进行拓扑排序。每次一条边以任意形式被「使用」后,都要打标记删除并不能再访问,以保证转移的正确性和高效。如果一条边被「使用」,分两种情况:
更新「根源」:
这个时候这条边已经发挥了它的所有作用了,作为 当前时刻的「根源」,这条边的源点在未来至少不会再经由这一条边被更新。故可以删除。
拓扑排序中转移:
一个点被加入队列,说明它的 DP 值是确定的,那么就不用拿一个已经确定的值多次更新另一个 DP 值。故可以删除。
所以正确性是可以保证的。类拓扑排序的结构也保证了算法复杂度 \(O(n+m)\)。
#define int long long
namespace XSC062 {
using namespace fastIO;
const int inf = 1e18;
const int maxn = 2e5 + 5;
const int maxm = 2e5 + 5;
struct _ {
int v, r, p, i;
_() {}
_(int v1, int r1, int p1, int i1) {
v = v1, r = r1, p = p1, i = i1;
}
};
struct __ {
int x, y, r, p;
bool operator< (const __ &q) const {
return r > q.r;
}
};
__ a[maxm];
bool del[maxn];
std::queue<int> q;
int n, m, x, y, r, p;
std::vector<_> g[maxn];
int f[maxn], deg[maxn];
int max(int x, int y) { return x > y ? x : y; }
int min(int x, int y) { return x < y ? x : y; }
void add(int x, int y, int r, int p, int i) {
g[x].emplace_back(y, r, p, i);
return;
}
int main() {
#ifdef ONLINE_JUDGE
freopen("merchant.in", "r", stdin);
freopen("merchant.out", "w", stdout);
#endif
read(n), read(m);
std::fill(f + 1, f + n + 1, inf);
for (int i = 1; i <= m; ++i) {
read(a[i].x), read(a[i].y);
read(a[i].r), read(a[i].p);
++deg[a[i].x];
}
std::sort(a + 1, a + m + 1);
for (int i = 1; i <= n; ++i)
if (!deg[i]) q.push(i);
for (int i = 1; i <= m; ++i)
add(a[i].y, a[i].x, a[i].r, a[i].p, i);
for (int i = 1; i <= m; ++i) {
while (!q.empty()) {
int t = q.front(); q.pop();
for (auto j : g[t]) {
if (del[j.i]) continue;
del[j.i] = 1;
if (f[t] != inf)
f[j.v] = min(f[j.v], max(f[t] - j.p, j.r));
if (!--deg[j.v]) q.push(j.v);
}
}
if (!del[i]) {
del[i] = 1, f[a[i].x] = min(f[a[i].x], a[i].r);
if (!--deg[a[i].x]) q.push(a[i].x);
}
}
for (int i = 1; i <= n; ++i)
print(f[i] == inf ? -1 : f[i], ' ');
putchar('\n');
return 0;
}
} // namespace XSC062
#undef int
A - Gerald and Giant Chess
https://vjudge.net/contest/594162#problem/A
这个呢,一道简单的数数题,但是因为我忘了 DP 容斥怎么打了所以自行思考了很久很久很久,还差最后一点点想出来的时候放弃了去抄题解 /cf
就是,这种「限制通行」的 DP 容斥(名字是我现场起的)类型的数数题有一种通法,就是首先我们只计算非法方案然后减掉。
那么非法方案怎么算呢?令 \(f_i\) 表示一个非法点都不经过,到达 \(i\) 这个非法点的方案数,那么有 \(f_i=calc(s,i)-\sum f_j\times calc(j,i)\),其中 \(s\) 是起点,\(calc(j,i)\) 是从 \(j\to i\) 的方案数。这样就能不重不漏地枚举完所有情况了。
我们把 \((n,m)\) 也视作一个非法点然后代入 DP 即可得到答案。
#define int long long
namespace XSC062 {
using namespace fastIO;
const int lim = 3e5;
const int mod = 1e9 + 7;
const int maxk = 2e3 + 5;
const int maxn = 3e5 + 5;
struct _ {
int x, y;
bool operator< (const _ &q) const {
return x == q.x ? y < q.y : x < q.x;
}
};
_ a[maxk];
int n, m, k, res;
int f[maxk], fac[maxn];
int qkp(int x, int y) {
int res = 1;
while (y) {
if (y & 1) (res *= x) %= mod;
(x *= x) %= mod, y >>= 1;
}
return res;
}
int inv(int x) { return qkp(x, mod - 2); }
int A(int n, int m) {
return fac[n] * inv(fac[n - m]) % mod;
}
int C(int n, int m) {
return A(n, m) * inv(fac[m]) % mod;
}
int calc(int i, int j) {
int x = a[j].x - a[i].x + 1,
y = a[j].y - a[i].y + 1;
return C(x + y - 2, y - 1);
}
int main() {
fac[0] = 1;
for (int i = 1; i <= lim; ++i)
fac[i] = fac[i - 1] * i % mod;
bool flag = 0;
read(n), read(m), read(k);
for (int i = 1; i <= k; ++i) {
read(a[i].x), read(a[i].y);
flag |= (a[i].x == 1 && a[i].y == 1);
flag |= (a[i].x == n && a[i].y == m);
}
a[0].x = a[0].y = 1;
if (flag) { print(0, '\n'); return 0; }
++k, a[k].x = n, a[k].y = m;
std::sort(a + 1, a + k + 1);
for (int i = 1; i <= k; ++i) {
f[i] = calc(0, i);
for (int j = 1; j < i; ++j) {
if (a[j].x > a[i].x || a[j].y > a[i].y) continue;
(f[i] -= f[j] * calc(j, i) % mod) %= mod;
}
}
f[k] = (f[k] % mod + mod) % mod;
print(f[k], '\n');
return 0;
}
} // namespace XSC062
#undef int