[Codeforces 1053C] Putting Boxes Together
Link:
Solution:
先推出一个结论:
最后必有一个点不动且其为权值上最中间的一个点
证明用反证证出如果不在中间的点必有一段能用代价少的替代多的
这样问题转换为求出区间第一个大于权值和一半的点,并求结果
如果只考虑半边的结果为$\sum_{i=1}^{mid} (pos[mid]-pos[i]-(mid-i))*w[i]$
将$pos[i]$修改为$pos[i]-i$之后维护$\sum pos[i]*w[i]$的和即可
以上操作可以用两颗线段树来维护
注意:
1、维护$w[i]$时不能取模!
2、对于求权值中间点可以二分+树状数组也可以直接在线段树上二分
线段树上二分时注意如果已经在区间内查询值要不断更新
Code:
#include <bits/stdc++.h> using namespace std;
#define X first
#define Y second
#define pb push_back
#define mid ((l+r)>>1)
#define lc k<<1,l,mid
#define rc k<<1|1,mid+1,r
typedef double db;
typedef long long ll;
typedef pair<int,int> P;
const int MAXN=5e5+,INF=<<,MOD=1e9+; struct SGT
{
ll seg[MAXN<<];
//分清何时要取模
void Update(int a,ll x,int f,int k,int l,int r)
{
if(l==r)
{seg[k]+=x;if(f) seg[k]%=MOD;return;}
if(a<=mid) Update(a,x,f,lc);
else Update(a,x,f,rc);
seg[k]=seg[k<<]+seg[k<<|];
if(f) seg[k]%=MOD;
}
ll Query(int a,int b,int f,int k,int l,int r)
{
if(a<=l&&r<=b) return seg[k];
ll ret=;
if(a<=mid) ret+=Query(a,b,f,lc);
if(b>mid) ret+=Query(a,b,f,rc);
return f?ret%MOD:ret;
}
//注意线段树二分的处理及此处的引用
int Find(int a,int b,ll &x,int k,int l,int r)
{
if(a<=l&&r<=b)
{
if(seg[k]<x){x-=seg[k];return INF;}
if(l==r) return l;
if(x<=seg[k<<]) return Find(a,b,x,lc);
else{x-=seg[k<<];return Find(a,b,x,rc);}
}
int ret=INF;
if(a<=mid)
ret=min(ret,Find(a,b,x,lc));
if(b>mid&&ret==INF)
ret=min(ret,Find(a,b,x,rc));
return ret;
}
}sw,sm;
int n,q,x,y,pos[MAXN],w[MAXN]; int main()
{
scanf("%d%d",&n,&q);
for(int i=;i<=n;i++)
scanf("%d",&pos[i]),pos[i]-=i;
for(int i=;i<=n;i++)
{
scanf("%d",&w[i]);
sw.Update(i,w[i],,,,n);
sm.Update(i,1ll*w[i]*pos[i],,,,n);
} while(q--)
{
scanf("%d%d",&x,&y);
if(x<)
{
x=-x;
sw.Update(x,-w[x],,,,n);
sm.Update(x,-1ll*w[x]*pos[x],,,,n);
w[x]=y;
sw.Update(x,w[x],,,,n);
sm.Update(x,1ll*w[x]*pos[x],,,,n);
}
else
{
//此时不能取模
ll sum=(sw.Query(x,y,,,,n)+)/;
int p=sw.Find(x,y,sum,,,n);
ll t1=(sw.Query(x,p,,,,n)*pos[p]%MOD-sm.Query(x,p,,,,n)+MOD)%MOD;
ll t2=(-sw.Query(p+,y,,,,n)*pos[p]%MOD+sm.Query(p+,y,,,,n)+MOD)%MOD;
printf("%lld\n",(t1+t2)%MOD);
}
}
return ;
}
[Codeforces 1053C] Putting Boxes Together的更多相关文章
- Codeforces 1053C Putting Boxes Together 树状数组
原文链接https://www.cnblogs.com/zhouzhendong/p/CF1053C.html 题目传送门 - CF1053C 题意 有 $n$ 个物品,第 $i$ 个物品在位置 $a ...
- 2018.09.24 codeforces 1053C. Putting Boxes Together(线段树)
传送门 就是让你维护动态的区间带权中位数. 然而昨晚比赛时并没有调出来. 想找到带权中位数的中点可以二分(也可以直接在线段树上找). 也就是二分出第一个断点,使得断点左边的和恰好大于或等于断点右边的和 ...
- Codeforces 1053 C - Putting Boxes Together
C - Putting Boxes Together 思路: 求带权中位数 用树状数组维护修改 代码: #pragma GCC optimize(2) #pragma GCC optimize(3) ...
- CodeForces 1058 F Putting Boxes Together 树状数组,带权中位数
Putting Boxes Together 题意: 现在有n个物品,第i个物品他的位置在a[i],他的重量为w[i].每一个物品移动一步的代价为他的w[i].目前有2种操作: 1. x y 将第x的 ...
- Putting Boxes Together CodeForces - 1030F (带权中位数)
#include <iostream> #include <algorithm> #include <cstdio> #include <math.h> ...
- Codeforces 488B - Candy Boxes
B. Candy Boxes 题目链接:http://codeforces.com/problemset/problem/488/B time limit per test 1 second memo ...
- Educational Codeforces Round 31- D. Boxes And Balls
D. Boxes And Balls time limit per test2 seconds memory limit per test256 megabytes 题目链接:http://codef ...
- [CF1053C]Putting Boxes Together(线段树)
http://codeforces.com/blog/entry/62013 两个结论: 1.一定有一个箱子不用动. 2.不动的箱子一定是加权前缀和为S/2的那个. 1显然,2由1易得. 于是问题变为 ...
- CF1030F Putting Boxes Together
昨晚的比赛题.(像我这种蒟蒻只能打打div2) 题意 给你$n$个物品,每一个物品$i$,有一个权值$w_i$和一个位置$a_i$,定义移动一个物品$i$到位置$t$的代价为$w_i * \left ...
随机推荐
- [IOS]VMware上虚拟机MAC安装XCode
1:VMware上虚拟机MAC安装前 VMware上安装Xcode之后 2:安装Xcode过程:把Xcode复制到虚拟机桌面上 3:复制完成之后,双击Xcode_6.4.dmg 文件 4:把Xcode ...
- js 合并多个对象 Object.assign
Object.assign() 方法用于将所有可枚举属性的值从一个或多个源对象复制到目标对象.它将返回目标对象. var o1 = { a: 1 };var o2 = { b: 2 };var o3 ...
- 利用certutil.exe 传文件
certutill.exe 在Windows 7 及其之后的所有Windows Server和Workstation版本均预装 1. Encode file: certutil -encode kk. ...
- 12-6 NSArray
原文:http://rypress.com/tutorials/objective-c/data-types/nsarray NSArray NSArray 是 Objective-C中最常用的数组类 ...
- FineReport——弹出新窗体选值并回调
主要实现的功能: 在主页面,通过单击按钮,弹出窗体,在窗体中通过下拉框选择值并查询,如果是多值,可以通过复选框选择,点击确定,将选中的行的字段值传递给主页面的下拉复选框,定义其编辑后事件进行查询.将想 ...
- mysql远程访问cannot connect(10038) 问题解决的过程
今天用Navicat访问虚拟机上的mysql,无法访问报cannot connect(10038). 首先看是否可以telnet,本机cmd,telnet 192.168.209.128 3306,结 ...
- POJ 2387 Til the Cows Come Home(dijkstra裸题)
题目链接:http://poj.org/problem?id=2387 题目大意:给你t条边(无向图),n个顶点,让你求点1到点n的最短距离. 解题思路:裸的dijsktra,注意判重边. 代码: # ...
- POJ 1177 Picture(线段树:扫描线求轮廓周长)
题目链接:http://poj.org/problem?id=1177 题目大意:若干个矩形,求这些矩形重叠形成的图形的轮廓周长. 解题思路:这里引用一下大牛的思路:kuangbin 总体思路: 1. ...
- ZOJ-3430
Detect the Virus Time Limit: 2 Seconds Memory Limit: 65536 KB One day, Nobita found that his c ...
- MySQL索引基础知识点
什么是索引 索引类似于书本目录,是数据库存储引擎维护的用于快速查找到记录的一种数据结构,它是对查询性能优化的最有效手段. MySQL索引是在存储引擎层而不是服务器层实现的,不同存储引擎的索引工作方式也 ...