luoguP1505 [国家集训队]旅游(真的毒瘤)
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
using namespace std;
#define ll long long
#define mem(Arr,x) memset(Arr,x,sizeof(Arr))
const int maxN=;
const int maxM=maxN*;
const int inf=;
class SegmentData
{
public:
int sum;
int mn,mx;
int mega;
};
int n;
int edgecnt=,Head[maxN],Next[maxM],V[maxM],W[maxM];
int Fa[maxN],Size[maxN],Hson[maxN],Top[maxN],Faedge[maxN],Depth[maxN];
int idcnt=,Id[maxN];
SegmentData S[maxN*];
void Add_Edge(int u,int v,int w);
void _Add(int u,int v,int w);
void dfs1(int u);
void dfs2(int u,int top);
int QC_max(int u,int v);//查询路径最大值
int QC_min(int u,int v);//查询路径最小值
int QC_sum(int u,int v);//查询路径和
void QC_nega(int u,int v);//把路径上所有边权取负
void PushDown(int now);//线段树下放标记
void Update(int now);//线段树更新
void Modify(int now,int l,int r,int pos,int key);//线段树单点修改
void Makenega(int now,int l,int r,int ql,int qr);//线段树区间取负
int Query_max(int now,int l,int r,int ql,int qr);//线段树区间最大
int Query_min(int now,int l,int r,int ql,int qr);//线段树区间最小
int Query_sum(int now,int l,int r,int ql,int qr);//线段树区间求和
int main(){
mem(Head,-);
scanf("%d",&n);
for (int i=;i<n;i++){
int u,v,w;scanf("%d%d%d",&u,&v,&w);u++;v++;
Add_Edge(u,v,w);
}
Depth[]=;
dfs1();
dfs2(,);
int Q;scanf("%d",&Q);
char opt[];
while (Q--){
scanf("%s",opt);
if (opt[]=='C'){
int pos,key;scanf("%d%d",&pos,&key);pos++;
pos=Id[pos];
Modify(,,n,pos,key);
}
if (opt[]=='N'){
int u,v;scanf("%d%d",&u,&v);u++;v++;
QC_nega(u,v);
}
if (opt[]=='S'){
int u,v;scanf("%d%d",&u,&v);u++;v++;
printf("%d\n",QC_sum(u,v));
}
if (opt[]=='A'){
int u,v;scanf("%d%d",&u,&v);u++;v++;
printf("%d\n",QC_max(u,v));
}
if (opt[]=='I'){
int u,v;scanf("%d%d",&u,&v);u++;v++;
printf("%d\n",QC_min(u,v));
}
}
return ;
}
void Add_Edge(int u,int v,int w)
{
_Add(u,v,w);_Add(v,u,w);return;
}
void _Add(int u,int v,int w){
edgecnt++;Next[edgecnt]=Head[u];Head[u]=edgecnt;V[edgecnt]=v;W[edgecnt]=w;
return;
}
void dfs1(int u){
Size[u]=;
for (int i=Head[u];i!=-;i=Next[i])
if (V[i]!=Fa[u]){
Fa[V[i]]=u;Faedge[V[i]]=W[i];
Depth[V[i]]=Depth[u]+;
dfs1(V[i]);
Size[u]+=Size[V[i]];
if (Size[V[i]]>Size[Hson[u]]) Hson[u]=V[i];
}
return;
} void dfs2(int u,int top){
Id[u]=++idcnt;
Modify(,,n,idcnt,Faedge[u]);
Top[u]=top;
if (Hson[u]==) return;
dfs2(Hson[u],top);
for (int i=Head[u];i!=-;i=Next[i])
if ((V[i]!=Fa[u])&&(V[i]!=Hson[u])) dfs2(V[i],V[i]);
return;
} int QC_max(int u,int v){
int ret=-inf;
while (Top[u]!=Top[v]){
if (Depth[Top[u]]<Depth[Top[v]]) swap(u,v);
ret=max(ret,Query_max(,,n,Id[Top[u]],Id[u]));
u=Fa[Top[u]];
}
if (Depth[u]>Depth[v]) swap(u,v);
if (u!=v) ret=max(ret,Query_max(,,n,Id[u]+,Id[v]));
return ret;
} int QC_min(int u,int v){
int ret=inf;
while (Top[u]!=Top[v]){
if (Depth[Top[u]]<Depth[Top[v]]) swap(u,v);
ret=min(ret,Query_min(,,n,Id[Top[u]],Id[u]));
u=Fa[Top[u]];
}
if (Depth[u]>Depth[v]) swap(u,v);
if (u!=v) ret=min(ret,Query_min(,,n,Id[u]+,Id[v]));
return ret;
} int QC_sum(int u,int v){
int ret=;
while (Top[u]!=Top[v]){
if (Depth[Top[u]]<Depth[Top[v]]) swap(u,v);
ret=ret+Query_sum(,,n,Id[Top[u]],Id[u]);
u=Fa[Top[u]];
}
if (Depth[u]>Depth[v]) swap(u,v);
if (u!=v) ret=ret+Query_sum(,,n,Id[u]+,Id[v]);
return ret;
} void QC_nega(int u,int v){
while (Top[u]!=Top[v])
{
if (Depth[Top[u]]<Depth[Top[v]]) swap(u,v);
Makenega(,,n,Id[Top[u]],Id[u]);
u=Fa[Top[u]];
}
if (Depth[u]>Depth[v]) swap(u,v);
if (u!=v) Makenega(,,n,Id[u]+,Id[v]);
return;
} void PushDown(int now){
if (S[now].mega)
{
S[now].mega=;
int lson=now*,rson=now*+;
swap(S[lson].mx,S[lson].mn);swap(S[rson].mx,S[rson].mn);
S[lson].mx=-S[lson].mx;S[lson].mn=-S[lson].mn;
S[rson].mx=-S[rson].mx;S[rson].mn=-S[rson].mn;
S[lson].sum=-S[lson].sum;S[rson].sum=-S[rson].sum;
S[lson].mega^=;S[rson].mega^=;
}
return;
} void Update(int now){
int lson=now*,rson=now*+;
S[now].mx=max(S[lson].mx,S[rson].mx);
S[now].mn=min(S[lson].mn,S[rson].mn);
S[now].sum=S[lson].sum+S[rson].sum;
return;
} void Modify(int now,int l,int r,int pos,int key){
PushDown(now);
if (l==r){
S[now].mx=S[now].mn=S[now].sum=key;
return;
}
int mid=(l+r)/;
if (pos<=mid) Modify(now*,l,mid,pos,key);
if (pos>=mid+) Modify(now*+,mid+,r,pos,key);
Update(now);return;
} void Makenega(int now,int l,int r,int ql,int qr){
PushDown(now);
if ((l==ql)&&(r==qr)){
S[now].mega^=;swap(S[now].mx,S[now].mn);
S[now].mx=-S[now].mx;S[now].mn=-S[now].mn;
S[now].sum=-S[now].sum;
return;
}
int mid=(l+r)/;
if (qr<=mid) Makenega(now*,l,mid,ql,qr);
else if (ql>=mid+) Makenega(now*+,mid+,r,ql,qr);
else{
Makenega(now*,l,mid,ql,mid);Makenega(now*+,mid+,r,mid+,qr);
}
Update(now);return;
} int Query_max(int now,int l,int r,int ql,int qr){
PushDown(now);
if ((l==ql)&&(r==qr)) return S[now].mx;
int mid=(l+r)/;
if (qr<=mid) return Query_max(now*,l,mid,ql,qr);
else if (ql>=mid+) return Query_max(now*+,mid+,r,ql,qr);
else return max(Query_max(now*,l,mid,ql,mid),Query_max(now*+,mid+,r,mid+,qr));
}
int Query_min(int now,int l,int r,int ql,int qr){
PushDown(now);
if ((l==ql)&&(r==qr)) return S[now].mn;
int mid=(l+r)/;
if (qr<=mid) return Query_min(now*,l,mid,ql,qr);
else if (ql>=mid+) return Query_min(now*+,mid+,r,ql,qr);
else return min(Query_min(now*,l,mid,ql,mid),Query_min(now*+,mid+,r,mid+,qr));
}
int Query_sum(int now,int l,int r,int ql,int qr){
PushDown(now);
if ((l==ql)&&(r==qr)) return S[now].sum;
int mid=(l+r)/;
if (qr<=mid) return Query_sum(now*,l,mid,ql,qr);
else if (ql>=mid+) return Query_sum(now*+,mid+,r,ql,qr);
else return Query_sum(now*,l,mid,ql,mid)+Query_sum(now*+,mid+,r,mid+,qr);
}
luoguP1505 [国家集训队]旅游(真的毒瘤)的更多相关文章
- 洛谷 P1505 [国家集训队]旅游 解题报告
P1505 [国家集训队]旅游 题目描述 \(\tt{Ray}\) 乐忠于旅游,这次他来到了\(T\)城.\(T\)城是一个水上城市,一共有 \(N\) 个景点,有些景点之间会用一座桥连接.为了方便游 ...
- 树链剖分【洛谷P1505】 [国家集训队]旅游
P1505 [国家集训队]旅游 题目描述 Ray 乐忠于旅游,这次他来到了T 城.T 城是一个水上城市,一共有 N 个景点,有些景点之间会用一座桥连接.为了方便游客到达每个景点但又为了节约成本,T 城 ...
- 洛谷 P1505 [国家集训队]旅游 树链剖分
目录 题面 题目链接 题目描述 输入输出格式 输入格式 输出格式 输入输出样例 输入样例: 输出样例: 说明 思路 AC代码 总结 题面 题目链接 P1505 [国家集训队]旅游 题目描述 Ray 乐 ...
- ⌈洛谷1505⌋⌈BZOJ2157⌋⌈国家集训队⌋旅游【树链剖分】
题目链接 [洛谷] [BZOJ] 题目描述 Ray 乐忠于旅游,这次他来到了T 城.T 城是一个水上城市,一共有 N 个景点,有些景点之间会用一座桥连接.为了方便游客到达每个景点但又为了节约成本,T ...
- 2018.06.29 洛谷P1505 [国家集训队]旅游(树链剖分)
旅游 题目描述 Ray 乐忠于旅游,这次他来到了T 城.T 城是一个水上城市,一共有 N 个景点,有些景点之间会用一座桥连接.为了方便游客到达每个景点但又为了节约成本,T 城的任意两个景点之间有且只有 ...
- P1505 [国家集训队]旅游
\(\color{#0066ff}{题 目 描 述}\) Ray 乐忠于旅游,这次他来到了T 城.T 城是一个水上城市,一共有 N 个景点,有些景点之间会用一座桥连接.为了方便游客到达每个景点但又为了 ...
- 树链剖分【p1505】[国家集训队]旅游
Description Ray 乐忠于旅游,这次他来到了T 城.T 城是一个水上城市,一共有 N 个景点,有些景点之间会用一座桥连接.为了方便游客到达每个景点但又为了节约成本,T 城的任意两个景点之间 ...
- 洛谷P1505 [国家集训队]旅游
题目描述 \(Ray\) 乐忠于旅游,这次他来到了\(T\) 城.\(T\) 城是一个水上城市,一共有 \(N\) 个景点,有些景点之间会用一座桥连接.为了方便游客到达每个景点但又为了节约成本,\(T ...
- P1505 [国家集训队]旅游[树剖]
题目描述 Ray 乐忠于旅游,这次他来到了T 城.T 城是一个水上城市,一共有 N 个景点,有些景点之间会用一座桥连接.为了方便游客到达每个景点但又为了节约成本,T 城的任意两个景点之间有且只有一条路 ...
随机推荐
- Robot Framework 源码阅读 day1 __main__.py
robot文件夹下的__main__.py函数 是使用module运行时的入口函数: import sys # Allows running as a script. __name__ check n ...
- JS删除对象中的某一属性(delete)
var obj= {} 1.JS对象添加新属性 obj.address="shenzhen" 2.JS删除对象中的某一属性(delete) var obj= { height: , ...
- RabbitMQ 全套
本博客代码运行环境 ErLang: ErLang_X64_22 version RabbitMQ: RabbitMQ_Server_3.7.15 version python : Python 3.7 ...
- ps:点阵格式图像
我们所看到的图像,究竟是如何构成的呢?这就需要涉及到图像类型的概念. 电脑中的图像类型分为两大类,一类称为点阵图,一类称为矢量图. 点阵图顾名思义就是由点构成的,如同用马赛克去拼贴图案一样,每个马赛克 ...
- python笔记(2)---不定长参数
python自定义函数中有两种不定长参数, 第一种是*name:加了星号 * 的参数会以元组(tuple)的形式导入 第二种是**name:加了星号 * *的参数会以字典(dict)的形式导入 *na ...
- 【vue】vue不足 待补强
83719279 9:56:03尤其是路由 声明周期 父子通信 组件通信 以及钩子函数83719279 9:56:17这些我都不敢用,只能用最原始的方法83719279 9:56:32还有es6 js ...
- nginx安装配置目录
1.nginx日志轮转,用于lograte服务的日志切割. /etc/logrotate.d/nginx 2.cgi配置相关 /etc/nginx/fastcgi_params /etc/nginx/ ...
- c++11 指针空值
1. 引入nullptr的必要性: 典型的指针初始化是将其指向一个空的位置.比如: int* my_ptr = 0; int* my_ptr = NULL; 一般情况下,NULL是一个宏定义. #un ...
- Docker Swarm学习教程【转载】
Swarm介绍 Swarm是Docker公司在2014年12月初发布的一套较为简单的工具,用来管理Docker集群,它将一群Docker宿主机变成一个单一的,虚拟的主机.Swarm使用标准的Docke ...
- Python_006(is和==,小数据池)
一.is和==的区别 1.id()方法,可以查看一个变量的内存地址 lst = ["周杰伦", "燃哥"] lst1 = ["周杰伦", & ...