做题记录 // 230221
听说是两道思维题!可是我并没有思维!
A. 1-11F. JM的赃物被盗
http://222.180.160.110:1024/contest/3341/problem/1
题目一大堆 P 话,总而言之就是一个长方体,里面的每一个单位体积有自己的权值,接下来询问里面的子长方体内单位体积的权值之和。
所以这跟思维有什么关系呢?我不知道,但是我确实没写过三维前缀和。
设 \((x,y,z)\) 为左上角,\((a,b,c)\) 为右下角。
不妨画个图,令 \(x \gets x - 1, y\gets y - 1, z\gets z - 1\),然后得到下面这个式子(\(s_{a,b,c}\) 表示前缀和,\(f\) 表示待求):
那么问题来辽,\(s\) 本身要怎么求呢?再画个图就可以了:
注意到我们的式子是轮换的,所以不管你在思考和画图的时候三个坐标分别代表什么,只要你按着题目输入顺序来就可以了,不然可能会加空!!
namespace XSC062 {
using namespace fastIO;
const int maxn = 135;
int s[maxn][maxn][maxn];
int n, m, w, x, y, z, a, b, c, q;
inline int f(int x, int y, int z,
int a, int b, int c) {
--x, --y, --z;
return s[a][b][c] - s[x][b][c] - s[a][y][c] +
s[x][y][c] - s[a][b][z] + s[x][b][z] +
s[a][y][z] - s[x][y][z];
}
int main() {
read(n), read(m), read(w);
for (int i = 1; i <= n; ++i) {
for (int j = 1; j <= m; ++j) {
for (int k = 1; k <= w; ++k) {
read(s[i][j][k]);
s[i][j][k] += s[i - 1][j - 1][k - 1]
+ f(1, j, 1, i, j, k - 1)
+ f(i, 1, 1, i, j - 1, k)
+ f(1, 1, k, i - 1, j, k);
}
}
}
read(q);
while (q--) {
read(x), read(y), read(z);
read(a), read(b), read(c);
print(f(x, y, z, a, b, c), '\n');
}
return 0;
}
} // namespace XSC062
#undef int
B. 1-11H. cyy的养殖基地
http://222.180.160.110:1024/contest/3341/problem/2
重要提醒:我一开始一直陷在寻找一条路线最终以哪个蚂蚁结尾的误区!但题目所求其实是蚂蚁最后会在哪个位置!注意不能混淆!
🌿!(拍桌跃起)Cube 写的题解 我看不懂!果然是不配了!(可是他真的写的好草率!)
总觉得 mj 确乎是讲过这么一道题的…… 错误的,mj 讲的是 神 jzm 出的题,应该是要简单很多的(因为不用求具体蚂蚁编号),不过思维转化点可以参考。
总之一个点就是,两个蚂蚁碰头了,你以为他们一起掉头,实际上他们并没有掉头,只是相当于他们继续沿原路线走,但是编号互换了。
那么我们用脚算都能把蚂蚁的最终坐标算出来,只是我们不知道这个位置上的蚂蚁是谁啊!
模拟是万万不能模拟的,因为但凡长度为 \(2\) 的木棍上两只蚂蚁相向爬 \(10^9\) 秒就会起飞。
不难想到,其实影响蚂蚁编号的只有碰头这一个操作,但神 Cube 的题解比我想得更深入:其实是碰头的次数。
那么怎么去计算蚂蚁碰头的次数呢?先从简单的想起,我们把问题放到直线上。
对于直线最左边的蚂蚁,他如果往右边走(因为往左走就什么都碰不到了),对于从最左边一直走到右的这条路线(而不是 1 号蚂蚁本身),我们找到这条路线最后到达的 坐标(注意不是蚂蚁编号,其实就是从起点开始往右走 \(T\) 步)。
按照 Cube 说的,我们需要找到这条路线一共撞上了几条其他路线。对于直线上的情况,我们只需要找到往左边的路线的条数就可以了,非常之简单,输入时记录之即可。
接下来的内容就是思维关键点了,我们已知这条路线共经历了 \(cnt\) 次交换,又知道这条路线最终到达的位置,我们怎么知道 1 号蚂蚁会走向哪里?
这个已知和待求就显得很奇怪。照理来说 1 号蚂蚁已经走到其他路线上去了,他最开始所在的路线就不会对它产生影响了呀?
—— · EOF · ——
真的什么也不剩啦 😖