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 离线 + 线段树的更多相关文章

  1. hdu 4288 离线线段树+间隔求和

    Coder Time Limit: 20000/10000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Su ...

  2. HDU 5700 区间交 离线线段树

    区间交 题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=5700 Description 小A有一个含有n个非负整数的数列与m个区间.每个区间可以表示为 ...

  3. BZOJ 3626 [LNOI2014]LCA 树剖+(离线+线段树 // 在线+主席树)

    BZOJ 4012 [HNOI2015]开店 的弱化版,离线了,而且没有边权(长度). 两种做法 1 树剖+离线+线段树 这道题求的是一个点zzz与[l,r][l,r][l,r]内所有点的lcalca ...

  4. BZOJ2333 [SCOI2011]棘手的操作 【离线 + 线段树】

    题目 有N个节点,标号从1到N,这N个节点一开始相互不连通.第i个节点的初始权值为a[i],接下来有如下一些操作: U x y: 加一条边,连接第x个节点和第y个节点 A1 x v: 将第x个节点的权 ...

  5. 【BZOJ 3443】 3443: 装备合成 (离线+线段树)

    3443: 装备合成 Time Limit: 15 Sec  Memory Limit: 128 MBSubmit: 63  Solved: 31 Description [背景]     lll69 ...

  6. 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 ...

  7. 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 ...

  8. 【BZOJ 2333 】[SCOI2011]棘手的操作(离线+线段树)

    2333: [SCOI2011]棘手的操作 Description 有N个节点,标号从1到N,这N个节点一开始相互不连通.第i个节点的初始权值为a[i],接下来有如下一些操作: U x y: 加一条边 ...

  9. HDU 4605 Magic Ball Game (在线主席树|| 离线 线段树)

    转载请注明出处,谢谢http://blog.csdn.net/ACM_cxlove?viewmode=contents    by---cxlove 题意:给出一棵二叉树,每个结点孩子数目为0或者2. ...

随机推荐

  1. Modification Default Identity Table

    Step 1. Open IdentityModel.cs to add following code. protected override void OnModelCreating(DbModel ...

  2. hdu-4763(kmp+拓展kmp)

    题意:给你一个串,问你满足最大字串既是前后缀,也在字符串除去前后缀的位置中出现过: 思路:我用的是拓展kmp求的前后缀,只用kmp也能解,在字符串2/3的位置后开始遍历,如果用一个maxx保存前2/3 ...

  3. 洛谷 P1102 A−B数对

    题目描述 出题是一件痛苦的事情! 题目看多了也有审美疲劳,于是我舍弃了大家所熟悉的 A+BA+BA+B ProblemProblemProblem ,改用 A−BA-BA−B 了哈哈! 好吧,题目是这 ...

  4. Centos虚拟环境工具virtualenvwrapper

    下载安装virtualenvwrapper pip3 install virtualenvwrapper !!!!注意安装virtualenvwrapper必须是在本地环境下!!! 设置Linux的用 ...

  5. 常用的redis服务命令。

    卸载服务:redis-server --service-uninstall 开启服务:redis-server --service-start 停止服务:redis-server --service- ...

  6. nginx压测工具--wrk

    基本使用 命令行敲下wrk,可以看到使用帮助 Usage: wrk <options> <url> Options: -c, --connections <N> C ...

  7. eclipse中无法查看引用的jar包源码

    本文来源:http://blog.csdn.NET/zljjava/article/details/7545270(这篇博客也是转载的,向最原始的作者致敬) 1.下载JAD jad官方地址的官方下载地 ...

  8. Tarjan求强连通分量,缩点,割点

    Tarjan算法是由美国著名计算机专家发明的,其主要特点就是可以求强连通分量和缩点·割点. 而强联通分量便是在一个图中如果有一个子图,且这个子图中所有的点都可以相互到达,这个子图便是一个强连通分量,并 ...

  9. python打印log重复问题

    本博客转载于:http://www.cnblogs.com/huang-yc/p/9209096.html,写得真不错 浅析python日志重复输出问题 目录 问题起源: 问题解析 解决办法 1.改名 ...

  10. 【XSY2786】Mythological VI 数学

    题目描述 有\(1\sim n\)一共\(n\)个数.保证\(n\)为偶数. 你要把这\(2n\)个数两两配对,一共配成\(n\)对.每一对的权值是他们两个数的和. 你想要知道这\(n\)对里最大的权 ...