想查看其他题的真题及题解的同学可以前往查看:CCF-CSP真题附题解大全
试题编号: | 202309-5 | ||||||||||||||||||||||||||||||||||||||||||||||||||||
试题名称: | 阻击 | ||||||||||||||||||||||||||||||||||||||||||||||||||||
时间限制: | 2.0s | ||||||||||||||||||||||||||||||||||||||||||||||||||||
内存限制: | 512.0MB | ||||||||||||||||||||||||||||||||||||||||||||||||||||
问题描述: | 问题描述上回提到,西西艾弗岛下方有一个庞大的遗迹群,栖息着一种名为“阴阳龙”的神兽。然而隔壁的狄迪吉岛盯上了西西艾弗岛,决定发动一场战争,试图从遗迹群中掠夺有价值的宝物。由此,西西艾弗岛不得不陷入一场漫长的阻击战中,史称“阴阳龙阻击战”。 狄迪吉岛拥有胜过西西艾弗岛的科技实力和武装水平,西西艾弗岛很快发现形势不妙:全歼敌军似乎是不可能的,唯一的策略是凭借主场作战的优势和人海战术,尽可能给敌军带来损失,当敌军发现发动进攻的损失明显超过收益时,就会无趣而归。 具体而言,西西艾弗岛共有 n 座城市,有 n−1 条道路连接这些城市,使得所有城市之间均可以通过道路互相到达。容易发现,任意两座城市之间都有唯一一条不经过重复城市的路径。 由于缺乏城市巷战的实力,西西艾弗岛决定将防御重心放在道路上。在每条道路上均派遣了一定的军队防守,当敌军经过时对其发动阻击。虽然由于实力的差距,并不能阻止敌军通过道路,但仍然可以对敌军造成一定的损失。 然而,敌军具有更强的科技,可以趁机对道路附近的遗迹进行探索,并掠夺其中的宝物——这也正是敌军发动战争的意义所在。如此,敌军通过一条道路时,“发掘宝物的收益”w 和“受到阻击的损失”b 两个值是独立的。 西西艾弗岛事先在狄迪吉岛中安插了一系列间谍,得到的情报消息如下:敌军将选择西西艾弗岛的两座城市作为进攻的“起点”和“终点”,并派遣军队在进攻起点城市登陆,沿两座城市间唯一的路径进攻至终点城市。同时,间谍还背负着另外一个重要的使命:影响敌军对于起点和终点城市的决策,使得敌军受到的总损失尽可能大,其中“总损失”定义为敌军经过的每条道路上的“受到阻击的损失”减去“发掘宝物的收益”之和,即 总损失是路径上的每条边总损失=∑e是路径上的每条边(be−we)。 此外,遗迹中宝物的价值与所处的环境属性密切相关,而阴阳龙的“现身”会使得环境的阴阳属性发生变化,这会使得敌军通过现身位置处的某一条道路时“发掘宝物的收益”w 发生变化。 这样的“阴阳龙现身”事件共会发生 m 次,你的任务就是帮助间谍计算出在所有事件前及每次事件后,敌军对于起点和终点城市的决策应当怎样改变,以最大化敌军的总损失。 输入格式从标准输入读入数据。 第 1 行,两个非负整数 n,m,分别表示西西艾弗岛的城市数和“阴阳龙现身”事件数。 接下来 n−1 行,每行 4 个非负整数 ui,vi,wi,bi,表示第 i 条道路连接城市 ui 和 vi,敌军在这条道路上“发掘宝物的收益”为 wi,“受到阻击的损失”为 bi。 接下来 m 行,每行 2 个非负整数 xi,yi,表示一次“阴阳龙现身”事件,使得第 xi 条道路的“发掘宝物的收益”变为 yi。 输出格式输出到标准输出中。 输出 m+1 行,每行一个非负整数,分别表示在所有事件前及每次事件后,对敌军造成的最大总损失。 样例输入
样例输出
样例说明在最初,由于敌人攻打每一条道路都会有正收益,因此间谍最好的策略就是将进攻起点和终点选为同一座城市,这样敌军的总损失为 0。 第 1 次事件后,间谍可以将进攻起点和终点分别选在城市 3 和 4,这样敌军的总损失为 3−2=1。 第 2 次事件后,间谍可以将进攻起点和终点分别选在城市 4 和 5,这样敌军的总损失为 (3−2)+(5−3)=3。 第 3 次事件后,间谍可以将进攻起点和终点分别选在城市 1 和 5,这样敌军的总损失为 (4−1)+(1−2)+(5−3)=4。 评测用例规模与约定对于所有测试数据保证:2≤n≤105,0≤m≤105,1≤ui,vi≤n,1≤xi≤n−1,0≤wi,bi,yi≤109。
特殊性质 A:ui=i,vi=i+1。 特殊性质 B:0≤wi,yi≤10^8≤bi。 特殊性质 C:保证任意两座城市均可在经过不超过 100 条道路的前提下互相到达。 |
真题来源:阻击
感兴趣的同学可以如此编码进去进行练习提交
c++满分题解:
#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
const int N = 1e5 + 8;
class segment{
#define lson root << 1
#define rson root << 1 | 1
LL ans[N << 2];
LL lsum[N << 2];
LL rsum[N << 2];
LL sum[N << 2];
public:
void popup(int root){
ans[root] = max({ans[lson], ans[rson], rsum[lson] + lsum[rson]});
lsum[root] = max(lsum[lson], sum[lson] + lsum[rson]);
rsum[root] = max(rsum[rson], sum[rson] + rsum[lson]);
sum[root] = sum[lson] + sum[rson];
}
void build(int root, int l, int r, vector<int>& a){
if (l == r){
ans[root] = (a[l] >= 0 ? a[l] : 0);
lsum[root] = (a[l] >= 0 ? a[l] : 0);
rsum[root] = (a[l] >= 0 ? a[l] : 0);
sum[root] = a[l];
return;
}
int mid = (l + r) >> 1;
build(lson, l, mid, a);
build(rson, mid + 1, r, a);
popup(root);
}
void update(int root, int l, int r, int pos, int val){
if (l == r){
ans[root] = (val >= 0 ? val : 0);
lsum[root] = (val >= 0 ? val : 0);
rsum[root] = (val >= 0 ? val : 0);
sum[root] = val;
return;
}
int mid = (l + r) >> 1;
if (pos <= mid)
update(lson, l, mid, pos, val);
else
update(rson, mid + 1, r, pos, val);
popup(root);
}
LL query(int root){
return ans[root];
}
}seg;
int main(){
ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
int n, m;
cin >> n >> m;
vector<vector<int>> G(n);
vector<array<int, 4>> edge;
for(int i = 1; i < n; ++ i){
int u, v, w, b;
cin >> u >> v >> w >> b;
-- u, -- v;
G[u].push_back(edge.size());
G[v].push_back(edge.size());
edge.push_back({u, v, w, b});
}
vector<LL> maxd(n, 0);
LL ans = 0;
function<void(int, int)> dfs = [&](int u, int fa){
for(auto &i : G[u]){
int v = edge[i][0] ^ edge[i][1] ^ u;
if (v == fa)
continue;
int d = edge[i][3] - edge[i][2];
dfs(v, u);
ans = max(ans, maxd[u] + maxd[v] + d);
maxd[u] = max(maxd[u], maxd[v] + d);
}
};
dfs(1, 1);
cout << ans << '\n';
if (m < 100000){
for(int i = 1; i <= m; ++ i){
int x, y;
cin >> x >> y;
-- x;
edge[x][2] = y;
ans = 0;
fill(maxd.begin(), maxd.end(), 0);
dfs(1, 1);
cout << ans << '\n';
}
}else{
int ok1 = true;
for(int i = 0; i < n - 1; ++ i){
ok1 &= (edge[i][0] == i && edge[i][1] == i + 1);
}
if (ok1){
// A
vector<int> a(n);
for(int i = 1; i < n; ++ i){
a[i] = edge[i - 1][3] - edge[i - 1][2];
}
seg.build(1, 1, n - 1, a);
for(int i = 1; i <= m; ++ i){
int x, y;
cin >> x >> y;
-- x;
edge[x][2] = y;
seg.update(1, 1, n - 1, x + 1, edge[x][3] - edge[x][2]);
cout << seg.query(1) << '\n';
}
}else{
// C
vector<set<array<LL, 2>>> anss(n), maxs(n);
vector<unordered_map<int, LL>> anss_id(n), maxs_id(n);
vector<int> deep(n), f(n);
function<void(int, int)> dfs2 = [&](int u, int fa){
f[u] = fa;
int leave = true;
for(auto &i : G[u]){
int v = edge[i][0] ^ edge[i][1] ^ u;
if (v == fa)
continue;
leave = false;
deep[v] = deep[u] + 1;
dfs2(v, u);
int d = edge[i][3] - edge[i][2];
LL ans_val = (*anss[v].rbegin())[0];
anss[u].insert({ans_val, v});
anss_id[u][v] = ans_val;
LL maxs_val = (*maxs[v].rbegin())[0] + d;
maxs[u].insert({maxs_val, v});
maxs_id[u][v] = maxs_val;
}
anss[u].insert({0, -1});
maxs[u].insert({0, -1});
if (maxs[u].size() > 1){
auto c1 = maxs[u].rbegin();
auto c2 = next(c1);
anss[u].insert({(*c1)[0] + (*c2)[0], -2});
anss_id[u][-2] = (*c1)[0] + (*c2)[0];
}
};
dfs2(0, -1);
for(int i = 0; i < m; ++ i){
int x, y;
cin >> x >> y;
-- x;
edge[x][2] = y;
int d = edge[x][3] - edge[x][2];
ans = 0;
int cur = (deep[edge[x][0]] < deep[edge[x][1]] ? edge[x][0] : edge[x][1]);
int son = cur ^ edge[x][0] ^ edge[x][1];
while(cur != -1){
maxs[cur].erase({maxs_id[cur][son], son});
maxs_id[cur][son] = (*maxs[son].rbegin())[0] + d;
maxs[cur].insert({maxs_id[cur][son], son});
anss[cur].erase({anss_id[cur][son], son});
anss_id[cur][son] = (*anss[son].rbegin())[0];
anss[cur].insert({anss_id[cur][son], son});
anss[cur].erase({anss_id[cur][-2], -2});
if (maxs[cur].size() > 1){
auto c1 = maxs[cur].rbegin();
auto c2 = next(c1);
anss_id[cur][-2] = (*c1)[0] + (*c2)[0];
anss[cur].insert({anss_id[cur][-2], -2});
}
son = cur;
cur = f[cur];
}
ans = max(0ll, (*anss[0].rbegin())[0]);
cout << ans << '\n';
}
}
}
return 0;
}
运行结果: