板子题 Sol
Cyber_Tree 出了一道板子题。。。
这题乍看之下貌似还不戳,但如果您做过类似的题,那么这就是一道板子题。。。。
首先明确要求的是什么,如果我们只考虑权值最大而不考虑最小距离,那么要求的显然就是森林中每棵基环树的最长链,即直径。。。
如果您已经知道了怎么求,那么可以直接关闭该题解了。。。。。
考虑如何求一棵基环树的直径。
发现如果直径上包含环上的边会很难求,因此可以分类讨论,分别求得不包含环上的边的最长链和包含环上的最长链即可。。。
首先考虑不包含链上的点的情况,显然可以忽略所有环上的边,然后用一个简单的树形 \(dp\) 解决。与此同时还可以求得对于环上的每一个点 \(u\) ,仅经过非环边与其距离最远的点距离是多少,记为 \(f_{u}\)。
接下来考虑若包含链上的边该怎么求,显然对于环上任意两个点 \(u, v\) ,显然经过这两个的最长链长度为 \(f_{u}+f_{v}+dis_{u, v}\) ,如果直接枚举每个点对显然是 \(O(n^{2})\) 的,因此考虑优化。
环上的问题不难想到破环为链,破环为链后两个点的距离可以用前缀和表示,则原式可以被表示为 \(f_{u}+f_{v}+sum_{u}-sum_{v}\) ,这个式子显然可以用单调队列进行优化,每次将 \(f_{u}-sum_{u}\) 插入队列即可。
以上,为基环树的直径求法。。。
然而此题还包含的一个最小距离,该怎么处理呢?
直接将 \(f\) 数组改成一个二元组,重载运算符后进行相同操作即可。
如果没有做过基环树直径的题,可以拿来练手,题中有亿些细节需要注意。。。
code
#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define t first
#define w second
const int N=1e6+10;
inline int read(){
int f=1, x=0; char ch=getchar();
while(!isdigit(ch)) { if(ch=='-') f=-1; ch=getchar(); }
while(isdigit(ch)) { x=x*10+ch-48; ch=getchar(); }
return f*x;
}
int n, deg[N], to[N], q1[N], stk[N<<1], top;
struct node{
ll x, y;
friend node operator + (const node &a, const node &b) { return (node){a.x+b.x, a.y+b.y}; }
friend node operator - (const node &a, const node &b) { return (node){a.x-b.x, a.y-b.y}; }
friend bool operator < (const node &a, const node &b) { return a.x==b.x ? a.y > b.y : a.x < b.x; }
friend bool operator <= (const node &a, const node &b) { return a.x==b.x ? a.y >= b.y : a.x < b.x; }
friend bool operator == (const node &a, const node &b) { return a.x==b.x&&a.y==b.y; }
}wr[N];
node sum[N<<1];
node f[N], maxn, ans;
vector<pair<int, node> > l[N];
queue<int> q;
bool cir[N], vis[N], in[N];
inline void topsort(){
for(int i=1; i<=n; ++i) if(deg[i]==1) q.push(i), cir[i]=1;
while(!q.empty()){
int u=q.front(); q.pop();
for(pair<int, node> v : l[u])
if(!cir[v.t]&&(--deg[v.t])==1) q.push(v.t), cir[v.t]=1;
}
}
void pre_dfs(int u, int fa){
vis[u]=1;
for(pair<int, node> v : l[u]){
if(!cir[v.t]&&v.t!=fa&&!in[v.t]&&!cir[u]) stk[++top]=v.t, sum[top]=v.w, in[v.t]=1;
if(vis[v.t]) continue;
pre_dfs(v.t, u);
}
}
void dfs1(int u, int fa){
f[u]=(node){0, 0};
for(pair<int, node> v : l[u]){
if(v.t==fa||!cir[v.t]) continue;
dfs1(v.t, u);
maxn=max(maxn, f[u]+v.w+f[v.t]);
f[u]=max(f[u], f[v.t]+v.w);
}
maxn=max(maxn, f[u]);
}
signed main(void){
freopen("naive.in", "r", stdin);
freopen("naive.out", "w", stdout);
n=read();
for(int i=1; i<=n; ++i) to[i]=read(), wr[i]=(node){read(), read()};
for(int i=1; i<=n; ++i){
if(to[to[i]]==i&&(wr[i]<wr[to[i]]||(wr[i]==wr[to[i]]&&to[i]<i))) continue;
l[i].push_back(make_pair(to[i], wr[i]));
l[to[i]].push_back(make_pair(i, wr[i]));
++deg[to[i]]; ++deg[i];
}
topsort();
for(int i=1; i<=n; ++i){
if(vis[i]) continue;
top=0; pre_dfs(i, 0); node tmp=(node){0, 0}; maxn=(node){0, 0};
if(!top){
dfs1(i, 0); tmp=maxn; maxn=(node){0, 0};
ans=ans+tmp; continue;
}
for(int j=1; j<=top; ++j)
dfs1(stk[j], 0), tmp=max(maxn, tmp), maxn=(node){0, 0};
for(int j=1; j<=top; ++j) stk[j+top]=stk[j], sum[j+top]=sum[j];
for(int j=1; j<=top<<1; ++j) sum[j]=sum[j]+sum[j-1];
int l=1, r=0;
for(int j=1; j<=top<<1; ++j){
while(j-q1[l]>=top&&l<=r) ++l;
if(l<=r) tmp=max(tmp, f[stk[j]]+f[stk[q1[l]]]+sum[j]-sum[q1[l]]);
while(f[stk[q1[r]]]-sum[q1[r]]<=f[stk[j]]-sum[j]&&l<=r) --r;
q1[++r]=j;
}
ans=ans+tmp;
}
printf("%lld %lld\n", ans.x, ans.y);
return 0;
}
板子题 Sol的更多相关文章
- POJ 1321 棋盘问题(DFS板子题,简单搜索练习)
棋盘问题 Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 44012 Accepted: 21375 Descriptio ...
- POJ 3278 Catch That Cow(BFS,板子题)
Catch That Cow Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 88732 Accepted: 27795 ...
- Gym 100952J&&2015 HIAST Collegiate Programming Contest J. Polygons Intersection【计算几何求解两个凸多边形的相交面积板子题】
J. Polygons Intersection time limit per test:2 seconds memory limit per test:64 megabytes input:stan ...
- hihoCoder #1038 : 01背包(板子题)
#1038 : 01背包 时间限制:20000ms 单点时限:1000ms 内存限制:256MB 描述 且说上一周的故事里,小Hi和小Ho费劲心思终于拿到了茫茫多的奖券!而现在,终于到了小Ho领取奖励 ...
- [LOJ 6270]数据结构板子题
Description 有n个区间,第i个区间是[li,ri],它的长度是ri−li. 有q个询问,每个询问给定L,R,K,询问被[L,R]包含的且长度不小于K的区间数量. 你想,像这种板子题,你随手 ...
- 堆以及一些用法 QWQ这是写得最认真的板子题
最近一直在学图论,然后吧,由于学的东西实在是太多太杂了,加上蒟蒻本蒻又经常颓,所以落了好多好多板子题的整理没写啊嘤嘤嘤,不过把这些东西学的差不多了,再一块写个整理,其实感觉还不错?????也算是很神奇 ...
- bzoj-1787-洛谷-4281(LCA板子题)
传送门(bzoj) 传送门(洛谷) 可以说这道也是一个板子题 由于题中是三个人需经过的路径最短 就会有一点点不太一样 那么 就两两求LCA 这样之后就会出现两种状况 一.所得到的三个LCA是相等的 那 ...
- bzoj3944: Sum 杜教筛板子题
板子题(卡常) 也可能是用map太慢了 /************************************************************** Problem: 3944 Us ...
- AC自动机板子题/AC自动机学习笔记!
想知道484每个萌新oier在最初知道AC自动机的时候都会理解为自动AC稽什么的,,,反正我记得我当初刚知道这个东西的时候,我以为是什么神仙东西,,,(好趴虽然确实是个对菜菜灵巧比较难理解的神仙知识点 ...
随机推荐
- Spring Boot中使用时序数据库InfluxDB
除了最常用的关系数据库和缓存之外,之前我们已经介绍了在Spring Boot中如何配置和使用MongoDB.LDAP这些存储的案例.接下来,我们继续介绍另一种特殊的数据库:时序数据库InfluxDB在 ...
- 使用Python玩转阿里云盘
项目地址: https://github.com/foyoux/aligo 这个项目起源于我的一个简单需求, 我有25000个文件, 已经上传了9000个, 但是现在我把这些文件重新整理了, 最后我不 ...
- Vue学习笔记(三)条件渲染和循环渲染
目录 一.条件渲染 1. v-if 2. 与v-else配合使用 3. 与v-else-if配合使用 4. v-show的使用 5. 类型切换案例 二.列表渲染 1. 遍历数组 2. 遍历对象 获取v ...
- 洛谷P3067题解
题面 首先,对于每个数,有三种状态:选入集合A,选入集合B,或者不选入集合.暴力枚举的时间复杂度是 \(O(n\times3^n)\) ,显然跑不过去. 因此考虑 \(\text{Meet in Mi ...
- 6轮面试辛苦拿到阿里Android开发offer,却从22k降到15k,在逗我?
一小伙工作快3年了,拿到了阿里云Android开发岗位P6的offer,算HR面一起,加起来有6轮面试了,将近3个月的时间,1轮同级 + 1轮Android用人部门leader + 1轮Android ...
- 字节跳动上传了一份“面试官版Android面试小册”,不讲一句废话,全是精华
前言 金三银四马上就到了,很多粉丝朋友私信希望我出一篇面试专题或者分享面试相关的笔记来学习,这不今天就给大家安排上了?(都是干货,错过就是亏.) 下面的面试笔记都是精心整理好免费分享给大家的,希望新朋 ...
- 大学同学做Java开发比我多5K,八年老Android只会crud该转Java吗?
最近在网上看到这样一个帖子: 做了八年Android开发,感觉这块做着也挺没意思,日常工作就是做一些架构优化,质量数据监控,改一改构建脚本,最主要的是业务负责人没有一个是做客户端的,都是后端的人. 最 ...
- Java代码搭建Dubbo+ZooKeeper 的示例
.personSunflowerP { background: rgba(51, 153, 0, 0.66); border-bottom: 1px solid rgba(0, 102, 0, 1); ...
- HTML5(十一)——WebSocket 基础教程
一.为什么要学 WebSocket? websocket 是 HTML5 提供的一种长链接双向通讯协议,使得客户端和服务器之间的数据交换更简单,允许服务端主动向客户端推送数据,并且客户端与服务端只需连 ...
- Golang语言系列-12-网络编程
网络编程 互联网协议介绍 互联网的核心是一系列协议,总称为"互联网协议"(Internet Protocol Suite),正是这一些协议规定了电脑如何连接和组网.我们理解了这些协 ...