包含不小于$\sqrt n$列的只有不大于$\sqrt n$行,修改时这些行打标记,否则暴力更新,操作一列的时候暴力更新这些行。合并没啥影响直接搞就是了。更新需要访问位置,感觉必须用哈希表,并不是特别靠谱。平均复杂度$O(n\sqrt n)$。

#include<bits/stdc++.h>
#include<ext/pb_ds/assoc_container.hpp>
#include<ext/pb_ds/hash_policy.hpp>
using namespace std;
using namespace __gnu_pbds;
int n2;
inline void upd2(int&a,int b){
a<b?a=b:0;
}
struct vec{
int s,d;
gp_hash_table<int,int>w;
vec(){s=d=0;}
void ins(int i,int e){
auto j=w.find(i);
if(w.end()==j)w[i]=e-d;
else
upd2(j->second,e-d);
upd2(s,e);
}
};
gp_hash_table<int,vec>f[2];
gp_hash_table<int,int>g[2];
gp_hash_table<int,null_type>h[2];
void dev(int k,int x){
auto&a=f[k][x];
for(int y:h[k^1]){
auto&b=f[k^1][y];
auto i=b.w.find(x);
if(b.w.end()!=i)
a.ins(y,i->second+b.d);
}
}
int sol5(int k,int x){
if(h[k].end()==h[k].find(x))
dev(k,x);
return f[k][x].s;
}
int ask(int k,int x){
return g[k].end()!=g[k].find(x)?sol5(k,g[k][x]):0;
}
void inc(int k,int x,int d){
auto&a=f[k][x];
if(h[k].end()==h[k].find(x)){
dev(k,x);
for(auto&e:a.w){
e.second+=d;
f[k^1][e.first].ins(x,e.second);
}
}else{
a.d+=d;
for(int y:h[k^1]){
auto&b=f[k^1][y];
auto i=b.w.find(x);
if(b.w.end()!=i){
i->second+=d;
upd2(b.s,i->second+b.d);
}
}
}
a.s+=d;
}
void sol4(int k,int x,int nx){
g[k][nx]=g[k][x];
g[k].erase(x);
}
void sol3(int k,int x,int nx){
auto&a=f[k][x],&b=f[k][nx];
if(h[k].end()==h[k].find(x))
dev(k,x);
for(auto e:a.w){
auto&c=f[k^1][e.first];
upd2(c.w[nx],e.second+a.d-c.d);
c.w.erase(x);
b.ins(e.first,e.second+a.d);
}
if(h[k].erase(x)||b.w.size()>n2){
h[k].insert(nx);
dev(k,nx);
}
}
void sol2(int k,int x,int nx){
sol3(k,g[k][x],g[k][nx]);
g[k].erase(x);
}
void sol1(int k,int x,int nx){
if(f[k][g[k][x]].w.size()<=f[k][g[k][nx]].w.size())
sol2(k,x,nx);
else{
sol2(k,nx,x);
sol4(k,x,nx);
}
}
void mov(int k,int x,int nx){
if(x!=nx)
(g[k].end()!=g[k].find(nx)?sol1:sol4)(k,x,nx);
}
int main(){
int n,m,x,y,d;
char o[8];
scanf("%d",&n);
n2=sqrt(n+.5);
while(n--){
scanf("%d%d%d",&x,&y,&d);
f[0][x].ins(y,d);
f[1][y].ins(x,d);
g[0][x]=x;
g[1][y]=y;
}
for(int k=0;k<2;++k)
for(auto&e:f[k])
if(e.second.w.size()>n2)
h[k].insert(e.first);
scanf("%d",&m);
while(m--){
scanf("%s%d",o,&x);
int k=*o=='Y';
if(o[1]=='Q')
printf("%d\n",ask(k,x));
else{
scanf("%d",&d);
if(g[k].end()!=g[k].find(x))
o[1]=='M'?mov(k,x,x+d):inc(k,g[k][x],d);
}
}
}

最新文章

  1. String、StringBuffer和StringBuilder的深入解析
  2. Scalaz(26)- Lens: 函数式不可变对象数据操作方式
  3. 解析window.open链接的参数
  4. 深入分析ConcurrentHashMap
  5. 全选,不选,反选js
  6. 【云计算】Docker云平台—Docker基础
  7. Memcached使用入门
  8. 网站性能扩展案例:每天30-50亿请求,300K QPS是如何炼成的
  9. 写一个EF的CodeFirst的Demo
  10. VM中ubuntu已经正确配置了静态IP仍无法上网
  11. &lt;未测&gt;源码升级安装glibc和rpm升级glibc
  12. 【python问题系列--2】脚本运行出现语法错误:IndentationError: unindent does not match any outer indentation level
  13. HTTPS连接前的几毫秒发生了什么?
  14. [Swift]LeetCode979. 在二叉树中分配硬币 | Distribute Coins in Binary Tree
  15. JS根据屏幕分辨率改变背景宽高
  16. Linux之 Ngnix
  17. cas 资源
  18. AI模型训练/算法评估 测试员
  19. Huffman树与编码
  20. python 字符串格式化,使用f前缀

热门文章

  1. 非GUI模式下运行JMeter和远程启动JMeter
  2. C# Ftp方式下载文件(无用户认证方式,支持断点续传)
  3. iOS经常使用设计模式——单例模式
  4. 以面试官的角度看strcpy函数
  5. 5.4 heapq--堆队列算法
  6. dos alias/cname address
  7. java多个文件压缩下载
  8. GB28181对接视频流
  9. php自定义错误
  10. 【每日Scrum】第二天(4.23) TD学生助手Sprint2站立会议