bzoj2333 离线 + 线段树
https://www.lydsy.com/JudgeOnline/problem.php?id=2333
有N个节点,标号从1到N,这N个节点一开始相互不连通。第i个节点的初始权值为a[i],接下来有如下一些操作: U x y: 加一条边,连接第x个节点和第y个节点 A1 x v: 将第x个节点的权值增加v A2 x v: 将第x个节点所在的连通块的所有节点的权值都增加v A3 v: 将所有节点的权值都增加v F1 x: 输出第x个节点当前的权值 F2 x: 输出第x个节点所在的连通块中,权值最大的节点的权值 F3: 输出所有节点中,权值最大的节点的权值
题意
看起来像一道可并堆的硬核数据结构题。
实际上确实是一道可并堆的数据结构题。
但我不会
那我有什么办法,只能用离线 + 线段树做了。
离线处理交换位置,保证所有连边的点都相邻,然后直接上区间修改+区间查询。
思路不难,就是写起来有点烦
#include <map>
#include <set>
#include <ctime>
#include <cmath>
#include <queue>
#include <stack>
#include <vector>
#include <string>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <sstream>
#include <iostream>
#include <algorithm>
#include <functional>
using namespace std;
inline int read(){int now=;register char c=getchar();for(;!isdigit(c);c=getchar());
for(;isdigit(c);now=now*+c-'',c=getchar());return now;}
#define For(i, x, y) for(int i=x;i<=y;i++)
#define _For(i, x, y) for(int i=x;i>=y;i--)
#define Mem(f, x) memset(f,x,sizeof(f))
#define Sca(x) scanf("%d", &x)
#define Sca2(x,y) scanf("%d%d",&x,&y)
#define Sca3(x,y,z) scanf("%d%d%d",&x,&y,&z)
#define Scl(x) scanf("%lld",&x);
#define Pri(x) printf("%d\n", x)
#define Prl(x) printf("%lld\n",x);
#define CLR(u) for(int i=0;i<=N;i++)u[i].clear();
#define LL long long
#define ULL unsigned long long
#define mp make_pair
#define PII pair<int,int>
#define PIL pair<int,long long>
#define PLL pair<long long,long long>
#define pb push_back
#define fi first
#define se second
typedef vector<int> VI;
const double eps = 1e-;
const int maxn = 3e5 + ;
const int INF = 0x3f3f3f3f;
const int mod = 1e9 + ;
template <class T>
inline bool scan_d(T &ret){
char c; int sgn;
if(c = getchar(),c == EOF) return ;
while(c != '-' && (c < '' || c > '')) c = getchar();
sgn = (c == '-')?-:;
ret = (c == '-')?:(c - '');
while(c = getchar(),c >= '' && c <= '') ret = ret * + (c - '');
ret *= sgn;
return ;
}
int N,M,K;
int a[maxn];
struct Node{
char op[];
int x,y;
}node[maxn];
int nxt[maxn],ed[maxn];
int fa[maxn];
int find(int p){
if(p == fa[p]) return p;
return fa[p] = find(fa[p]);
}
void Union(int a,int b){
a = find(a); b = find(b);
if(a == b) return;
nxt[ed[a]] = b;
ed[a] = ed[b];
fa[b] = a;
}
void init(){
For(i,,N) ed[i] = fa[i] = i;
}
int pos[maxn],id[maxn];
struct Tree{
int l,r;
int lazy,Max;
}tree[maxn << ];
void Pushup(int t){
tree[t].Max = max(tree[t << ].Max,tree[t << | ].Max);
}
void Pushdown(int t){
if(tree[t].lazy){
tree[t << ].lazy += tree[t].lazy; tree[t << | ].lazy += tree[t].lazy;
tree[t << ].Max += tree[t].lazy; tree[t << | ].Max += tree[t].lazy;
tree[t].lazy = ;
}
}
void build(int t,int l,int r){
tree[t].l = l; tree[t].r = r;
tree[t].lazy = ;
if(l == r){
tree[t].Max = a[id[l]];
return ;
}
int m = (l + r) >> ;
build(t << ,l,m); build(t << | ,m + ,r);
Pushup(t);
}
void update(int t,int l,int r,int v){
if(l <= tree[t].l && tree[t].r <= r){
tree[t].Max += v;
tree[t].lazy += v;
return ;
}
Pushdown(t);
int m = (tree[t].l + tree[t].r) >> ;
if(r <= m) update(t << ,l,r,v);
else if(l > m) update(t << | ,l,r,v);
else{
update(t << ,l,m,v); update(t << | ,m + ,r,v);
}
Pushup(t);
}
int query(int t,int l,int r){
if(l <= tree[t].l && tree[t].r <= r){
return tree[t].Max;
}
Pushdown(t);
int m = (tree[t].l + tree[t].r) >> ;
if(r <= m) return query(t << ,l,r);
else if(l > m) return query(t << | ,l,r);
return max(query(t << ,l,m),query(t << | ,m + ,r));
}
int main()
{
Sca(N);
For(i,,N) scan_d(a[i]);
scan_d(M); init();
for(int i = ; i < M ; i ++ ){
scanf("%s",node[i].op);
if(node[i].op[] == 'F'){
if(node[i].op[] == '') continue;
scan_d(node[i].x);
}else if(node[i].op[] == 'A' && node[i].op[] == ''){
scan_d(node[i].x);
}else{
scan_d(node[i].x); scan_d(node[i].y);
}
if(node[i].op[] == 'U') Union(node[i].x,node[i].y);
}
int cnt = ;
for(int i = ; i <= N ; i ++){
if(fa[i] != i) continue;
for(int j = i; j;j = nxt[j]){
pos[j] = ++cnt;
id[cnt] = j;
}
}
build(,,N); init();
int sum = ;
for(int i = ; i < M ; i ++){
if(node[i].op[] == 'U'){
Union(node[i].x,node[i].y);
}
if(node[i].op[] == 'A'){
if(node[i].op[] == ''){
node[i].x = pos[node[i].x];
update(,node[i].x,node[i].x,node[i].y);
// cout << "update" << node[i].x << " " << node[i].x << " " << node[i].y << endl;
}else if(node[i].op[] == ''){
int t = find(node[i].x);
update(,pos[t],pos[ed[t]],node[i].y);
// cout << "update" << pos[t] << " " << pos[ed[t]] << " " << node[i].y << endl;
}else{
sum += node[i].x;
}
}else if(node[i].op[] == 'F'){
if(node[i].op[] == ''){
Pri(query(,pos[node[i].x],pos[node[i].x]) + sum);
// cout << "query" << pos[node[i].x] << " " << pos[node[i].x] << endl;
}else if(node[i].op[] == ''){
int t = find(node[i].x);
// cout << "query" << pos[t] << " " << pos[ed[t]] << endl;
Pri(query(,pos[t],pos[ed[t]]) + sum);
}else{
Pri(tree[].Max + sum);
}
}
}
#ifdef VSCode
system("pause");
#endif
return ;
}
bzoj2333 离线 + 线段树的更多相关文章
- hdu 4288 离线线段树+间隔求和
Coder Time Limit: 20000/10000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Su ...
- HDU 5700 区间交 离线线段树
区间交 题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=5700 Description 小A有一个含有n个非负整数的数列与m个区间.每个区间可以表示为 ...
- BZOJ 3626 [LNOI2014]LCA 树剖+(离线+线段树 // 在线+主席树)
BZOJ 4012 [HNOI2015]开店 的弱化版,离线了,而且没有边权(长度). 两种做法 1 树剖+离线+线段树 这道题求的是一个点zzz与[l,r][l,r][l,r]内所有点的lcalca ...
- BZOJ2333 [SCOI2011]棘手的操作 【离线 + 线段树】
题目 有N个节点,标号从1到N,这N个节点一开始相互不连通.第i个节点的初始权值为a[i],接下来有如下一些操作: U x y: 加一条边,连接第x个节点和第y个节点 A1 x v: 将第x个节点的权 ...
- 【BZOJ 3443】 3443: 装备合成 (离线+线段树)
3443: 装备合成 Time Limit: 15 Sec Memory Limit: 128 MBSubmit: 63 Solved: 31 Description [背景] lll69 ...
- HDU 4031 Attack(离线+线段树)(The 36th ACM/ICPC Asia Regional Chengdu Site —— Online Contest)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4031 Problem Description Today is the 10th Annual of ...
- Codeforces Round #365 (Div. 2) D. Mishka and Interesting sum 离线+线段树
题目链接: http://codeforces.com/contest/703/problem/D D. Mishka and Interesting sum time limit per test ...
- 【BZOJ 2333 】[SCOI2011]棘手的操作(离线+线段树)
2333: [SCOI2011]棘手的操作 Description 有N个节点,标号从1到N,这N个节点一开始相互不连通.第i个节点的初始权值为a[i],接下来有如下一些操作: U x y: 加一条边 ...
- HDU 4605 Magic Ball Game (在线主席树|| 离线 线段树)
转载请注明出处,谢谢http://blog.csdn.net/ACM_cxlove?viewmode=contents by---cxlove 题意:给出一棵二叉树,每个结点孩子数目为0或者2. ...
随机推荐
- Sharepoint 2016 - Deploy Office Online Server
Step 1: Install prerequisite software for Office Online Server To install Office Online Server Ope ...
- UVA 12171 Sculpture
https://vjudge.net/problem/UVA-12171 题目 某人设计雕塑,用的是很扯的方法:把一堆长方体拼起来.给出长方体的坐标和长宽高,求外表面积.因为要将这雕塑进行酸洗,需要知 ...
- Nginx 针对建立TCP连接优化
L:124 sysctl -a | grep file-max //通过命令查看系统最大句柄数 [root@3 ~]# sysctl -a | grep file-max fs.file-max = ...
- Nginx 安装与详解
nginx简介 nginx是一个开源的,支持高性能,高并发的www服务和代理服务软件.它是一个俄罗斯人lgor sysoev开发的,作者将源代码开源出来供全球使用.nginx比它大哥apache性能改 ...
- [CEOI2007] 树的匹配Treasury
类型:树形 DP 传送门:>Here< 题意:给一棵树,你可以匹配有边相连的两个点,问你这棵树的最大匹配是多少,并且计算出有多少种最大匹配. 解题思路 首先树形Dp是很明显的,$f[i][ ...
- Codeforces Round #540 (Div. 3) A,B,C,D2,E,F1
A. Water Buying 链接:http://codeforces.com/contest/1118/problem/A 实现代码: #include<bits/stdc++.h> ...
- Ionic生成的App安装在手机上后无法联网的解决方案
在Ionic中使用inappbrowser.themeablebrowser 组件打开网页,刚开始是好的,后来不知添加什么插件,导致了安装在手机上以后没有网络访问权限. 尝试了很多,最后才发现,此时, ...
- git 本地推送远程仓库报错: error: failed to push some refs to 'https://github.com/yangtuothink/mxonline.git'
报错现象 添加远程仓库后 推送代码的时候报错 报错分析 远程代码和本地代码不匹配问题 远程初始仓库的创建有些默认 的 README什么的本地是没有的 需要先同步后再上传 报错解决 git push - ...
- 每天一个Linux命令(03):du命令
du命令 今天找开发定位问题,看到他使用了这个命令,查看文件,之前知道df,所以今天的每天系列把这命令 du命令也是查看使用空间的,但是与df命令不同的是Linux du命令是对文件和目录磁盘使用的空 ...
- Ubuntu相关配置
1.Ubuntu配置root密码 2.开机ssh连接--nat 编辑网络设置端口转发