CCF-CSP真题《202309-5 阻击》思路+ c++满分题解

2023-12-18 04:52

本文主要是介绍CCF-CSP真题《202309-5 阻击》思路+ c++满分题解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

 想查看其他题的真题及题解的同学可以前往查看: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 行,每行一个非负整数,分别表示在所有事件前及每次事件后,对敌军造成的最大总损失。

样例输入

5 3
1 2 6 4
2 3 2 1
3 4 5 3
3 5 8 5
3 2
4 3
1 1

样例输出

0
1
3
4

样例说明

在最初,由于敌人攻打每一条道路都会有正收益,因此间谍最好的策略就是将进攻起点和终点选为同一座城市,这样敌军的总损失为 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。

测试点编号n≤m≤特殊性质
12020
2300300
3∼430003000A
5∼630003000B
7∼930003000
101050A
111050B
121050
13∼15105105A
16∼18105105B
19∼21105105C
22∼25105105

特殊性质 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 | 1LL 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){// Avector<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{// Cvector<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;
}

运行结果:

这篇关于CCF-CSP真题《202309-5 阻击》思路+ c++满分题解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/507106

相关文章

C++使用栈实现括号匹配的代码详解

《C++使用栈实现括号匹配的代码详解》在编程中,括号匹配是一个常见问题,尤其是在处理数学表达式、编译器解析等任务时,栈是一种非常适合处理此类问题的数据结构,能够精确地管理括号的匹配问题,本文将通过C+... 目录引言问题描述代码讲解代码解析栈的状态表示测试总结引言在编程中,括号匹配是一个常见问题,尤其是在

使用C++实现链表元素的反转

《使用C++实现链表元素的反转》反转链表是链表操作中一个经典的问题,也是面试中常见的考题,本文将从思路到实现一步步地讲解如何实现链表的反转,帮助初学者理解这一操作,我们将使用C++代码演示具体实现,同... 目录问题定义思路分析代码实现带头节点的链表代码讲解其他实现方式时间和空间复杂度分析总结问题定义给定

C++初始化数组的几种常见方法(简单易懂)

《C++初始化数组的几种常见方法(简单易懂)》本文介绍了C++中数组的初始化方法,包括一维数组和二维数组的初始化,以及用new动态初始化数组,在C++11及以上版本中,还提供了使用std::array... 目录1、初始化一维数组1.1、使用列表初始化(推荐方式)1.2、初始化部分列表1.3、使用std::

C++ Primer 多维数组的使用

《C++Primer多维数组的使用》本文主要介绍了多维数组在C++语言中的定义、初始化、下标引用以及使用范围for语句处理多维数组的方法,具有一定的参考价值,感兴趣的可以了解一下... 目录多维数组多维数组的初始化多维数组的下标引用使用范围for语句处理多维数组指针和多维数组多维数组严格来说,C++语言没

linux进程D状态的解决思路分享

《linux进程D状态的解决思路分享》在Linux系统中,进程在内核模式下等待I/O完成时会进入不间断睡眠状态(D状态),这种状态下,进程无法通过普通方式被杀死,本文通过实验模拟了这种状态,并分析了如... 目录1. 问题描述2. 问题分析3. 实验模拟3.1 使用losetup创建一个卷作为pv的磁盘3.

c++中std::placeholders的使用方法

《c++中std::placeholders的使用方法》std::placeholders是C++标准库中的一个工具,用于在函数对象绑定时创建占位符,本文就来详细的介绍一下,具有一定的参考价值,感兴... 目录1. 基本概念2. 使用场景3. 示例示例 1:部分参数绑定示例 2:参数重排序4. 注意事项5.

使用C++将处理后的信号保存为PNG和TIFF格式

《使用C++将处理后的信号保存为PNG和TIFF格式》在信号处理领域,我们常常需要将处理结果以图像的形式保存下来,方便后续分析和展示,C++提供了多种库来处理图像数据,本文将介绍如何使用stb_ima... 目录1. PNG格式保存使用stb_imagephp_write库1.1 安装和包含库1.2 代码解

C++实现封装的顺序表的操作与实践

《C++实现封装的顺序表的操作与实践》在程序设计中,顺序表是一种常见的线性数据结构,通常用于存储具有固定顺序的元素,与链表不同,顺序表中的元素是连续存储的,因此访问速度较快,但插入和删除操作的效率可能... 目录一、顺序表的基本概念二、顺序表类的设计1. 顺序表类的成员变量2. 构造函数和析构函数三、顺序表

使用C++实现单链表的操作与实践

《使用C++实现单链表的操作与实践》在程序设计中,链表是一种常见的数据结构,特别是在动态数据管理、频繁插入和删除元素的场景中,链表相比于数组,具有更高的灵活性和高效性,尤其是在需要频繁修改数据结构的应... 目录一、单链表的基本概念二、单链表类的设计1. 节点的定义2. 链表的类定义三、单链表的操作实现四、

使用C/C++调用libcurl调试消息的方式

《使用C/C++调用libcurl调试消息的方式》在使用C/C++调用libcurl进行HTTP请求时,有时我们需要查看请求的/应答消息的内容(包括请求头和请求体)以方便调试,libcurl提供了多种... 目录1. libcurl 调试工具简介2. 输出请求消息使用 CURLOPT_VERBOSE使用 C