bzoj4889
http://www.lydsy.com/JudgeOnline/problem.php?id=4889
人傻常数大 bzoj上跑不过 洛谷上能过两到三个点
我写的是树套树啊 怎么跑的比分块还慢
每次可以发现交换两个点 只对他们中间的点有影响 所以我们只用计算比x小的数的和 比x大的数的和 比y小的数的和 比y大的数的和 然后计算一下就可以了 很明显可以用各种数据结构维护
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = ;
const ll mod = ;
int n, m, cnt;
ll ans;
int q[N];
inline int read()
{
int x = , f = ; char c = getchar();
while(c < '' || c > '')
{
if(c == '-') f = -;
c = getchar();
}
while(c >= '' && c <= '')
{
x = x * + c - '';
c = getchar();
}
return x * f;
}
struct BIT {
ll tree[N], count[N];
int lowbit(int i)
{
return i & (-i);
}
void update(int pos, ll delta)
{
for(int i = pos; i <= ; i += lowbit(i))
{
tree[i] = (tree[i] + delta) % mod;
++count[i];
}
}
ll query(int pos, ll delta)
{
ll ret1 = , ret2 = ;
for(int i = pos; i; i -= lowbit(i))
{
ret1 = (ret1 + tree[i]) % mod;
ret2 += count[i];
}
ret1 = (ret1 + ret2 * delta) % mod;
return ret1;
}
} B;
struct data {
int a;
ll v;
} tree[N], a[N];
namespace splaytree
{
int fa[N], child[N][], root[N];
ll size[N], num[N], sum[N];
void update(int x)
{
size[x] = num[x] + size[child[x][]] + size[child[x][]];
sum[x] = (tree[x].v * num[x] + sum[child[x][]] + sum[child[x][]]) % mod;
}
void zig(int x)
{
int y = fa[x];
fa[x] = fa[y];
child[fa[x]][child[fa[x]][] == y] = x;
child[y][] = child[x][];
fa[child[x][]] = y;
fa[y] = x;
child[x][] = y;
update(y);
update(x);
}
void zag(int x)
{
int y = fa[x];
fa[x] = fa[y];
child[fa[x]][child[fa[x]][] == y] = x;
child[y][] = child[x][];
fa[child[x][]] = y;
fa[y] = x;
child[x][] = y;
update(y);
update(x);
}
void splay(int x, int t, int pos)
{
while(fa[x] != t)
{
int y = fa[x], z = fa[y];
if(z == t)
{
child[y][] == x ? zig(x) : zag(x);
break;
}
else if(y == child[z][] && x == child[y][]) { zig(y); zig(x); }
else if(y == child[z][] && x == child[y][]) { zag(y); zag(x); }
else if(y == child[z][] && x == child[y][]) { zag(x); zig(x); }
else if(y == child[z][] && x == child[y][]) { zig(x); zag(x); }
}
if(!t) root[pos] = x;
update(root[pos]);
}
void up(int x)
{
while(x)
{
update(x);
x = fa[x];
}
}
ll getbig(int pos, data &k)
{
int now = root[pos];
ll ret = ;
while(now)
{
if(tree[now].a > k.a)
{
ret = (ret + k.v + tree[now].v) % mod;
ret = (ret + k.v * size[child[now][]] +
sum[child[now][]]) % mod;
now = child[now][];
}
else now = child[now][];
}
return ret;
}
ll getsmall(int pos, data &k)
{
int now = root[pos];
ll ret = ;
while(now)
{
if(tree[now].a < k.a)
{
ret = (ret + k.v + tree[now].v) % mod;
ret = (ret + k.v * size[child[now][]] +
sum[child[now][]]) % mod;
now = child[now][];
}
else now = child[now][];
}
return ret;
}
int find(int x, data &k)
{
for(x = root[x]; x; x = child[x][k.a > tree[x].a])
if(tree[x].a == k.a) return x;
}
void del(int pos, int x)
{
splay(x, , pos);
if(num[x] > )
{
--num[x];
update(x);
return;
}
if(child[x][] * child[x][] == )
{
root[pos] = child[x][] + child[x][];
fa[root[pos]] = ;
child[x][] = child[x][] = fa[x] = ;
return;
}
int now = child[x][];
while(child[now][]) now = child[now][];
fa[child[x][]] = now;
child[now][] = child[x][];
root[pos] = child[x][];
fa[root[pos]] = ;
up(child[x][]);
child[x][] = child[x][] = num[x] = size[x]
= fa[x] = ;
splay(now, , pos);
}
void insert(int pos, data &k)
{
int now = root[pos];
if(!root[pos])
{
root[pos] = ++cnt;
tree[cnt] = k;
size[cnt] = num[cnt] = ;
sum[cnt] = k.v;
return;
}
while()
{
if(tree[now].a == k.a)
{
++num[now];
up(now);
break;
}
if(!child[now][k.a > tree[now].a])
{
child[now][k.a > tree[now].a] = ++cnt;
tree[cnt] = k;
size[cnt] = num[cnt] = ;
fa[cnt] = now;
sum[cnt] = k.v;
up(cnt);
splay(cnt, , pos);
break;
}
now = child[now][k.a > tree[now].a];
}
}
} using namespace splaytree;
namespace segmenttree
{
void update(int l, int r, int x, int pos, data &k)
{
if(l == r)
{
root[x] = ++cnt;
tree[cnt] = k;
num[cnt] = size[cnt] = ;
sum[cnt] = k.v;
return;
}
int mid = (l + r) >> ;
if(pos <= mid) update(l, mid, x << , pos, k);
else update(mid + , r, x << | , pos, k);
int t = find(x, a[pos]);
del(x, t); insert(x, k);
}
void build(int l, int r, int x)
{
if(l == r)
{
root[x] = ++cnt;
num[cnt] = size[cnt] = ;
tree[cnt] = a[l];
sum[cnt] = a[l].v;
return;
}
int mid = (l + r) >> ;
build(l, mid, x << );
build(mid + , r, x << | );
for(int i = l; i <= r; ++i) insert(x, a[i]);
}
ll querybig(int l, int r, int x, int a, int b, data &k)
{
if(l > b || r < a) return ;
if(l >= a && r <= b) return getbig(x, k) % mod;
int mid = (l + r) >> ;
return ((querybig(l, mid, x << , a, b, k) +
querybig(mid + , r, x << | , a, b, k)) % mod);
}
ll querysmall(int l, int r, int x, int a, int b, data &k)
{
if(l > b || r < a) return ;
if(l >= a && r <= b) return getsmall(x, k) % mod;
int mid = (l + r) >> ;
return ((querysmall(l, mid, x << , a, b, k) +
querysmall(mid + , r, x << | , a, b, k)) % mod);
}
} using namespace segmenttree;
int main()
{
n = read(); m = read();
for(int i = ; i <= n; ++i)
{
a[i].a = read(); a[i].v = read();
ans = ((ans + B.query(, a[i].v) - B.query(a[i].a, a[i].v)) % mod + mod) % mod;
B.update(a[i].a, a[i].v);
}
build(, n, );
while(m--)
{
int x, y; x = read(); y = read();
if(x > y) swap(x, y);
if(x == y)
{
printf("%lld\n", ans);
continue;
}
if(a[x].a < a[y].a) ans += a[x].v + a[y].v;
else ans -= a[x].v + a[y].v;
ll a1 = querybig(, n, , x + , y - , a[x]);
ll a2 = querysmall(, n, , x + , y - , a[x]);
ll a3 = querysmall(, n, , x + , y - , a[y]);
ll a4 = querybig(, n, , x + , y - , a[y]);
ans = ((ans + a1 - a2 + a3 - a4) % mod + mod) % mod;
update(, n, , x, a[y]);
update(, n, , y, a[x]);
swap(a[x], a[y]);
printf("%lld\n", ans);
}
return ;
}
bzoj4889的更多相关文章
- 【BZOJ4889】不勤劳的图书管理员(树套树)
[BZOJ4889]不勤劳的图书管理员(树套树) 题面 又是权限题,烦死了 洛谷真好 题解 分开考虑每一次交换产生的贡献. 假设交换\((x,y)\) 检查\(x\)与\(y\)对于区间\([x+1, ...
- 【bzoj4889】: [Tjoi2017]不勤劳的图书管理员 分块-BIT
[bzoj4889]: [Tjoi2017]不勤劳的图书管理员 题目大意:给定一个序列(n<=50000),每个数有一个编码ai(ai<=50000)和权值vi(vi<=100000 ...
- 【BZOJ4889】[Tjoi2017]不勤劳的图书管理员 分块+树状数组
[BZOJ4889][Tjoi2017]不勤劳的图书管理员 题目描述 加里敦大学有个帝国图书馆,小豆是图书馆阅览室的一个书籍管理员.他的任务是把书排成有序的,所以无序的书让他产生厌烦,两本乱序的书会让 ...
- BZOJ4889 & 洛谷3759:[TJOI2017]不勤劳的图书管理员——题解
https://www.lydsy.com/JudgeOnline/problem.php?id=4889 https://www.luogu.org/problemnew/show/P3759 加里 ...
- [bzoj4889] [Tjoi2017]不勤劳的图书管理员
Description 加里敦大学有个帝国图书馆,小豆是图书馆阅览室的一个书籍管理员.他的任务是把书排成有序的,所以无序的书让他产生厌烦,两本乱序的书会让小豆产生这两本书页数的和的厌烦度.现在有n本被 ...
- [TJOI2017][bzoj4889] 不勤劳的图书管理员 [线段树套线段树]
题面 传送门 思路 考虑两本书的位置交换对答案的贡献: (为了方便描述,用"左边那本"和"右边那本"称呼两本我们要交换的书,"中间那本"是我 ...
- BZOJ4889:[TJOI2017]不勤劳的图书管理员
浅谈树状数组与线段树:https://www.cnblogs.com/AKMer/p/9946944.html 题目传送门:https://www.lydsy.com/JudgeOnline/prob ...
- bzoj4889: [Tjoi2017]不勤劳的图书管理员(树套树)
传送门 据说正解线段树套平衡树 然而网上参考(抄)了一个树状数组套动态开点线段树的 思路比较清楚,看代码应该就明白了 //minamoto #include<iostream> #incl ...
- 【bzoj4889】[Tjoi2017]不勤劳的图书管理员 树状数组+分块+二分
题目描述(转自洛谷) 加里敦大学有个帝国图书馆,小豆是图书馆阅览室的一个书籍管理员.他的任务是把书排成有序的,所以无序的书让他产生厌烦,两本乱序的书会让小豆产生这两本书页数的和的厌烦度.现在有n本被打 ...
随机推荐
- js数组的处理
//重写Array中的indexOf方法,获取数组中指定值的元素的索引 Array.prototype.indexOf = function (val) { for (var i = 0; i < ...
- 2019西安多校联训 Day3
试题链接:http://www.accoders.com/contest.php?cid=1895 考试密码请私信; 特别鸣谢:zkc奆佬帮助我优化本篇题解(语言表达方面) T1 显然二分求解的 ...
- case when里的like功能 ////// 截取(substr)
case when里的like功能 假如要用到case when又要用到like这样的功能,即如果字符串包含‘语文’就怎么怎么样,包含‘数学’就怎么怎么样,包含‘英语’就怎么怎么样,like是用于wh ...
- 透彻分析C/C++中memset函数
在C语言中,经常需要对内存进行操作,里面涉及很多函数,但是memset函数的使用有一点需要大家格外注意,这也是我在做项目时遇到过的一个问题,调试了很久才找出来错误. 函数原型是:void *memse ...
- 微信小程序 setData动态修改数据数组的值
1.问题说明 有一组数据,用来存储图片路径,动态修改图片的路径来上传图片,而小程序JS只能通过事件获取时机和setData方法修改数据来改变view. 而用这样写的方式明显是错误的 2.解决办法 字符 ...
- The turtle Module 一个画图的模块
感悟: 这样写的介绍才有用 import turtle bob = turtle.Turtle() print(bob) turtle.mainloop() Turtle()方法打开一个窗口,中间带有 ...
- day21 02 包的进阶
day21 02 包的进阶 1._init_.py文件的操作---导入包 根据day21 01 包的初识,建立的glance包,直接import glance后通过“包点包..点方法”是不能执行所要的 ...
- Vue.Draggable实现拖拽效果(快速使用)
1.下载包:npm install vuedraggable 配置:package.json "dependencies": { "element-ui": & ...
- HDU5266 LCA 树链剖分LCA 线段树
HDU5266 LCA Description 给一棵 n 个点的树,Q 个询问 [L,R] : 求点 L , 点 L+1 , 点 L+2 -- 点 R 的 LCA. Input 多组数据. The ...
- spark之scala快速入门
scala和java都是在jvm之上的语言,相对来讲,scala热度比较低,其实并不是一个特别好的语言选择. 原因倒不是因为scala本身的缺点,而是使用人群不够多,论坛和社区不够活跃.这就跟社交软件 ...