莫队-小Z的袜子
----普通莫队
首先
清楚概率怎么求
假设我们要求从区间l到r中拿出一对袜子的概率
sum[i]为第i种袜子在l到r中的数量
$$\frac{\sum_{i=l}^{r} {[sum[i] \times (sum[i]-1)]}}{ (r-l+1) \times {(r-l)}}\qquad$$
转化一下可以得到
$$\frac{\sum_{i=l}^{r} {sum[i]^{2}}-(r-l+1)}{ (r-l+1)\times {(r-l)}}\qquad$$
普通莫队是一种离线算法 并且充分利用上一个得到的答案来求得当前询问的答案
怎么由上一个答案来得到当前的答案呢?
设
$$ans=\sum_{i=l}^{r} {sum[i]^{2}}$$
即分母的一部分(减去(r-l+1)即可得到分母)
现在要求[l+1,r]这个区间的答案
若第l只袜子的编号为x 则只有sum[x]减少了1
更新ans的操作如下:
ans-=sum[x]*sum[x];
sum[x]-=1;
ans+=sum[x]*sum[x]
即:减去从前对答案的贡献 加上现在的贡献
求[l-1,r],[l,r-1],[l,r+1]的做法类比可得
整理一下可以得到change函数
//x为新增加或减少的点
//若为新增加的点 如l指针左移和r指针右移 则w=1
//反之 w=-1
/*例子:求[l-1,r] change(l,-1)
求[l,r+1] change(r+1,1) ……*/
change(int x,int w)
{
ans-=sum[x]*sum[x];sum[x]+=w;ans+=sum[x]*sum[x];
}
为了使l和r指针尽可能少的移动(优化时间)
我们需要给所有的问题的l和r排序
构造cmp函数
要用分块
将整个长度为n的序列分为sqrt(n)块
cmp为:若l与r在同一块中 则按照l从小到大排序
否则 按照r从小到大排序
这样l指针每次最多跳2*(n/sqrt(n))次
最后求gcd 即可
还要记得l==r的特判
一种优秀的gcd求法
//普通gcd
int gcd(int x,int y)
{
return y==0?x:gcd(y,x%y);
}
其实可以看成是把x赋值为x%y 再调换x,y的位置 求gcd(x,y)
假设我们要调换a,b的值
//普通做法:
int tmp;tmp=a;a=b;b=tmp; //其实可以利用位运算 异或
x^=y;y^=x;x^=y; /* 第一步:x=x^y
第二步:y=y^(x^y) 由于y^y=0 所以 y=x
第三步:x=(x^y)^x 同理可得 x=y
*/
可以得到gcd函数
ll gcd(ll a,ll b)
{
while(b^=a^=b^=a%=b);return a;
}
//从右至左运算
分析over
code
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cmath>
#define ll long long
#define yes(i,a,b) for(register int i=a;i<=b;i++)
#define M 50010
using namespace std;
int n,m,tot,len;
ll ans,f_ans[M][2],sum[M];
int c[M],be[M];
struct node {int l,r,id;} q[M];
bool cmp(node x,node y)
{
if(be[x.l]==be[y.l]) return x.r<y.r;
return x.l<y.l;
}
void change(int x,int w)
{
ans-=(ll)(sum[c[x]]*sum[c[x]]);sum[c[x]]+=w;ans+=(ll)(sum[c[x]]*sum[c[x]]);
}
ll gcd(ll a,ll b)
{
while(b^=a^=b^=a%=b);return a;
}
int main()
{
//freopen("1.in","r",stdin);
//freopen("1.out","w",stdout);
scanf("%d%d",&n,&m);
len=sqrt(n);
yes(i,1,n) scanf("%d",&c[i]),be[i]=i/len+1;
yes(i,1,m) scanf("%d%d",&q[i].l,&q[i].r),q[i].id=i;
sort(q+1,q+m+1,cmp);
int l=1,r=0;
yes(i,1,m)
{
while(l<q[i].l) change(l,-1),l++;
while(l>q[i].l) change(l-1,1),l--;
while(r<q[i].r) change(r+1,1),r++;
while(r>q[i].r) change(r,-1),r--;
ll ans1,ans2;
ans1=ans-(ll)(q[i].r-q[i].l+1);ans2=(ll)(q[i].r-q[i].l+1)*(q[i].r-q[i].l);
if(ans1==0) ans2=1;
else
{
ll g=gcd(ans2,ans1);
ans1/=g;ans2/=g;
}
f_ans[q[i].id][0]=ans1;f_ans[q[i].id][1]=ans2;
}
yes(i,1,m) printf("%lld/%lld\n",f_ans[i][0],f_ans[i][1]);
return 0;
}
莫队-小Z的袜子的更多相关文章
- 初识莫队——小Z的袜子
以前一直觉得莫队是多么高大上的一种算法,然而仔细看了下发现其实并不复杂,实质上就是技巧性的暴力美学. 在我看来莫队是一种分块排序后降低复杂度的算法,当答案可以通过左右端点一个一个移动维护出来的时候就可 ...
- BZOJ 2038: [2009国家集训队]小Z的袜子(hose) [莫队算法]【学习笔记】
2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 Sec Memory Limit: 259 MBSubmit: 7687 Solved: 3516[Subm ...
- 莫队算法 2038: [2009国家集训队]小Z的袜子(hose)
链接:http://www.lydsy.com/JudgeOnline/problem.php?id=2038 2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 ...
- BZOJ-2038 小Z的袜子(hose) 莫队算法
2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 Sec Memory Limit: 259 MB Submit: 5573 Solved: 2568 [Subm ...
- BZOJ 2038 [2009国家集训队]小Z的袜子 莫队
2038: [2009国家集训队]小Z的袜子(hose) 题目连接: http://www.lydsy.com/JudgeOnline/problem.php?id=2038 Descriptionw ...
- Bzoj 2038: [2009国家集训队]小Z的袜子(hose) 莫队,分块,暴力
2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 Sec Memory Limit: 259 MBSubmit: 5763 Solved: 2660[Subm ...
- BZOJ2038: [2009国家集训队]小Z的袜子(hose) -- 莫队算法 ,,分块
2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 Sec Memory Limit: 259 MBSubmit: 3577 Solved: 1652[Subm ...
- BZOJ 2038: [2009国家集训队]小Z的袜子(hose) ( 莫队 )
莫队..先按sqrt(n)分块, 然后按块的顺序对询问排序, 同块就按右端点排序. 然后就按排序后的顺序暴力求解即可. 时间复杂度O(n1.5) --------------------------- ...
- BZOJ 2038: [2009国家集训队]小Z的袜子(hose)【莫队算法裸题&&学习笔记】
2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 Sec Memory Limit: 259 MBSubmit: 9894 Solved: 4561[Subm ...
随机推荐
- 用Html5与Asp.net MVC上传多个文件
html: <form action="/home/upload" method="post" enctype="multipart/form- ...
- 2018 焦作icpc现场赛总结
Day 0 没有直达焦作的飞机,所以选择了先到新郑机场,再转乘城际列车.城际列车猜是专门给学生开通的吧,每天只有来和回一共两趟(所以机票选择的余地也不多).买的时候只有无座票了,本来以为会一直站着,但 ...
- MT【147】又见最大最小
(2018浙江省赛12题)设$a\in R$,且对任意的实数$b$均有$\max\limits_{x\in[0,1]}|x^2+ax+b|\ge1$求$a$的范围_____解答:由题意$\min\li ...
- NOI前总结
最近也就是天天考试,总结一下. 7.1 开场T1T2都是不可做的概率期望,只有T3看起来可做,于是怒干4h+,将题解里面的所有结论都推出来了,大模拟写的一点毛病都没有,可还是因为2-SAT掌握不熟结果 ...
- 【ARC068F】Solitaire
Description 你有一个双端队列和 \(N\) 个数字,先按 \(1\) 到 \(N\) 的顺序每次从任意一端插入当前数字,再进行 \(N\) 次操作每次可以从两端弹出,求有多少种弹出序列 ...
- keepalived使用nc命令检测udp端口
keepalived支持的健康检测方式有:HTTP_GET|SSL_GET.TCP_CHECK.SMTP_CHECK.MISC_CHECK. 由于keepalived自身并不支持udp检测,有TCP_ ...
- 遇到问题---java---安装新版本jdk后Failed reading value of registry key
情况 情况是原本安装有jdk1.7,能正常运行,现在要升级到1.8. 直接在oracle的网站下载1.8安装后修改配置为1.8后: 能用javac编译成功,但java命令运行时报错: Failed r ...
- 用Gradle命令行编译Android工程
在Android sdk 目录下的samples/android-21/ 文件夹下,任找一个工程,如果在命令行直接编译 可能会报这种错误:gradle buile.gradle FAILURE: Bu ...
- bzoj2755【SCOI2012】喵星人的入侵
输入格式 第一行为三个整数n,m,K,分别表示地图的长和宽,以及最多能放置的炮塔数量. 接下来的n行,每行包含m个字符,‘#’表示地图上原有的障碍,‘.’表示该处为空地,数据保证在原地图上存在S到T的 ...
- bzoj2300【HAOI2011】防线修建
题目描述 近来A国和B国的矛盾激化,为了预防不测,A国准备修建一条长长的防线,当然修建防线的话,肯定要把需要保护的城市修在防线内部了.可是A国上层现在还犹豫不决,到底该把哪些城市作为保护对象呢?又由于 ...