BZOJ2752: [HAOI2012]高速公路(road)

时间:2023-03-09 03:24:44
BZOJ2752: [HAOI2012]高速公路(road)

2752: [HAOI2012]高速公路(road)

Time Limit: 20 Sec  Memory Limit: 128 MB
Submit: 608  Solved: 199
[Submit][Status]

Description

Y901高速公路是一条重要的交通纽带,*部门建设初期的投入以及使用期间的养护费用都不低,因此*在这条高速公路上设立了许多收费站。
Y901高速公路是一条由N-1段路以及N个收费站组成的东西向的链,我们按照由西向东的顺序将收费站依次编号为1~N,从收费站i行驶到i+1(或从i+1行驶到i)需要收取Vi的费用。高速路刚建成时所有的路段都是免费的。
*部门根据实际情况,会不定期地对连续路段的收费标准进行调整,根据政策涨价或降价。
无聊的小A同学总喜欢研究一些稀奇古怪的问题,他开车在这条高速路上行驶时想到了这样一个问题:对于给定的l,r(l<r),在第l个到第r个收费站里等概率随机取出两个不同的收费站a和b,那么从a行驶到b将期望花费多少费用呢?

Input

第一行2个正整数N,M,表示有N个收费站,M次调整或询问
接下来M行,每行将出现以下两种形式中的一种
C l r v 表示将第l个收费站到第r个收费站之间的所有道路的通行费全部增加v
Q l r   表示对于给定的l,r,要求回答小A的问题
所有C与Q操作中保证1<=l<r<=N

Output

对于每次询问操作回答一行,输出一个既约分数
若答案为整数a,输出a/1

Sample Input

4 5
C 1 4 2
C 1 2 -1
Q 1 2
Q 2 4
Q 1 4

Sample Output

1/1
8/3
17/6

HINT

数据规模

所有C操作中的v的绝对值不超过10000

在任何时刻任意道路的费用均为不超过10000的非负整数

所有测试点的详细情况如下表所示

Test N M

1 =10 =10

2 =100 =100

3 =1000 =1000

4 =10000 =10000

5 =50000 =50000

6 =60000 =60000

7 =70000 =70000

8 =80000 =80000

9 =90000 =90000

10 =100000 =100000

Source

题解:

我居然不知道 i*x,i*i*x是可以用线段树维护的,我是有多sb。。。

其实看到题目就已经想到了肯定转化一下就变成了只需要维护一个和,平方和啊什么乱七八糟的东西,可是还是没想下去。。。

说下正解:

假设x到y公路是可经过的

那么 i 对答案的贡献是  v[i]*(i-x+1)*(y-i+1)=v[i]*[(y-x-x*y+1)+i*(x+y)-i*i]

所以我们需要线段树来维护 sigma v[i],sigma i*v[i],sigma i*i*v[i],

维护的时候其实很简单,我脑洞太大没去想。。。

inline void update(int k,ll z)
{
ll l=t[k].l,r=t[k].r;
t[k].tag+=z;
t[k].num[]+=z*(r-l+);
t[k].num[]+=z*(r+l)*(r-l+)/;
t[k].num[]+=z*(r*(r+)*(*r+)-(l-)*l*(*l-))/;
}

这样就可以了。。。

代码:

 #include<cstdio>

 #include<cstdlib>

 #include<cmath>

 #include<cstring>

 #include<algorithm>

 #include<iostream>

 #include<vector>

 #include<map>

 #include<set>

 #include<queue>

 #include<string>

 #define inf 1000000000

 #define maxn 100000+100

 #define maxm 500+100

 #define eps 1e-10

 #define ll long long

 #define pa pair<int,int>

 #define for0(i,n) for(int i=0;i<=(n);i++)

 #define for1(i,n) for(int i=1;i<=(n);i++)

 #define for2(i,x,y) for(int i=(x);i<=(y);i++)

 #define for3(i,x,y) for(int i=(x);i>=(y);i--)

 #define mod 1000000007

 using namespace std;

 inline int read()

 {

     int x=,f=;char ch=getchar();

     while(ch<''||ch>''){if(ch=='-')f=-;ch=getchar();}

     while(ch>=''&&ch<=''){x=*x+ch-'';ch=getchar();}

     return x*f;

 }
struct seg{int l,r;ll num[],tag;}t[*maxn];
int n,m;ll ans[];
void build(int k,int l,int r)
{
t[k].l=l;t[k].r=r;t[k].tag=;
if(l==r)return;
int mid=(l+r)>>;
build(k<<,l,mid);build(k<<|,mid+,r);
}
inline void update(int k,ll z)
{
ll l=t[k].l,r=t[k].r;
t[k].tag+=z;
t[k].num[]+=z*(r-l+);
t[k].num[]+=z*(r+l)*(r-l+)/;
t[k].num[]+=z*(r*(r+)*(*r+)-(l-)*l*(*l-))/;
}
inline void pushdown(int k)
{
if(!t[k].tag)return;
update(k<<,t[k].tag);
update(k<<|,t[k].tag);
t[k].tag=;
}
inline void pushup(int k)
{
for1(i,)t[k].num[i]=t[k<<].num[i]+t[k<<|].num[i];
}
void change(int k,int x,int y,ll z)
{
int l=t[k].l,r=t[k].r,mid=(l+r)>>;
if(l==x&&r==y){update(k,z);return;}
pushdown(k);
if(y<=mid)change(k<<,x,y,z);
else if(x>mid)change(k<<|,x,y,z);
else change(k<<,x,mid,z),change(k<<|,mid+,y,z);
pushup(k);
}
void query(int k,int x,int y,int z)
{
int l=t[k].l,r=t[k].r,mid=(l+r)>>;
if(l==x&&r==y){ans[z]+=t[k].num[z];return;}
pushdown(k);
if(y<=mid)query(k<<,x,y,z);
else if(x>mid)query(k<<|,x,y,z);
else query(k<<,x,mid,z),query(k<<|,mid+,y,z);
}
inline ll gcd(ll a,ll b){return b?gcd(b,a%b):a;} int main() { freopen("input.txt","r",stdin); freopen("output.txt","w",stdout); n=read()-;m=read();
build(,,n);
char ch[];
while(m--)
{
scanf("%s",ch);
if(ch[]=='C')
{
int x=read(),y=read()-;ll z=read();
change(,x,y,z);
}
else
{
int xx=read(),yy=read()-;ll x=xx,y=yy;
ll sum=(y-x+)*(y-x+)/;
for1(i,)ans[i]=,query(,xx,yy,i);
ans[]=(y-x-x*y+)*ans[]+(y+x)*ans[]-ans[];
ll tmp=gcd(ans[],sum);
printf("%lld/%lld\n",ans[]/tmp,sum/tmp);
}
} return ; }