P1494 小Z的袜子

终于了解了莫队算法(更专业的名称Square Root Decomposition of Queries)

莫队算法:

  • 一般来说解决静态(实际上也有修改的但复杂度更高)的离线(离线意味着需要知道所有查询)区间查询
  • 我理解的特点是从区间[L,R]到区间[L,R-1],[L+1,R],[L-1,R],[L,R+1]复杂度低
  • 相较与线段树,线段树要求[l,mid],[mid,r]-->[l,r]复杂度较低
  • 莫队时间复杂度为O(nlogn)
  • 采用的是分块处理查询
  • 实现需要注意的点:
    • 为了保持原有查询顺序需要保存原序号,根据原序号存储答案
    • 区间的移动传入的参数是增加的点和删除的点,不要无脑写

关于该题

  • 稍微需要一些数学的变换,记下x,y,z出现的次数分别为a,b,c在区间[l,r],那么概率为C(a,2)+C(b,2)+...C(,2)/C(len,2)
  • 变换 a(a-1)+b(b-1).../len(len-1)=\(\frac{(\sum{a^2}-len)}{ len(len-1)}\)
  • 所以需要维护频度的平方和

代码

#include<bits/stdc++.h>
using namespace std;
#define ll long long
const int maxn=50005;
inline ll sqr(const ll &x){
return x*x;
}
inline ll gcd(const ll &a,const ll &b){
if(!b) return a;
else return gcd(b,a%b);
}
int belong[maxn];//每个点的分块预处理
ll ans1[maxn],ans2[maxn];//存储i次询问的结果
struct Cmd{
int l,r,id;
friend bool operator <(const Cmd &a,const Cmd &b){
if(belong[a.l]==belong[b.l]) return a.r<b.r;
else return belong[a.l]<belong[b.l];
}
}cmds[maxn];
//存储所有询问,因为要将所有询问存储(离线算法)
//所以有时需要非常毒瘤的读入优化
int n,m,c[maxn],sum[maxn];
//c[]存储n个元素
inline void upd(ll &now,int p,int v){
//更新平方和
now-=sqr(sum[c[p]]);
sum[c[p]]+=v;//c[p]的计数
now+=sqr(sum[c[p]]);
//now应该加上改变后的^2-之前^2
//巧妙的避免了讨论
}
inline void solve(){
int L=1,R=0;//[L,R]为当前维护好的区间
ll now=0;//now为当前区间答案
for(int i=1;i<=m;i++){//莫队主要部分
while(L<cmds[i].l){
upd(now,L,-1);L++;//表示L右移动
}
while(L>cmds[i].l){//加上点L-1
upd(now,L-1,1);L--;
}
while(R<cmds[i].r){//加上点R+1
upd(now,R+1,1);R++;
}
while(R>cmds[i].r){
upd(now,R,-1);R--;
}
if(cmds[i].l==cmds[i].r){
ans1[cmds[i].id]=0;
ans2[cmds[i].id]=1;
continue;
}
ans1[cmds[i].id]=now-(cmds[i].r-cmds[i].l+1);
ans2[cmds[i].id]=(ll)(cmds[i].r-cmds[i].l)*(cmds[i].r-cmds[i].l+1);
//printf("db %d %d %lld %lld %lld\n",cmds[i].l,cmds[i].r,now,ans1[cmds[i].id],ans2[cmds[i].id]);
ll g=gcd(ans1[cmds[i].id],ans2[cmds[i].id]);
ans1[cmds[i].id]/=g;
ans2[cmds[i].id]/=g; } }
int main(){
scanf("%d %d",&n,&m);//n个数,m次询问
int s=sqrt(n);//准备分块
for(int i=1;i<=n;i++){
scanf("%d",&c[i]);
belong[i]=((i-1)/s)+1;//每个点分块预处理
}
for(int i=1;i<=m;i++){
scanf("%d %d",&cmds[i].l,&cmds[i].r);
cmds[i].id=i;//需要标号记录原来顺序,因为后面要排序
}
sort(cmds+1,cmds+m+1);//对区间重新排序
solve();
for(int i=1;i<=m;i++){
printf("%lld/%lld\n",ans1[i],ans2[i]);
}
return 0;
}

P1494 [国家集训队]小Z的袜子(luogu)的更多相关文章

  1. P1494 [国家集训队]小Z的袜子

    题目 P1494 [国家集训队]小Z的袜子 解析 在区间\([l,r]\)内, 任选两只袜子,有 \[r-l+1\choose2\] \[=\frac{(r-l+1)!}{2!(r-l-1)!}\] ...

  2. P1494 [国家集训队]小Z的袜子/莫队学习笔记(误

    P1494 [国家集训队]小Z的袜子 题目描述 作为一个生活散漫的人,小\(Z\)每天早上都要耗费很久从一堆五颜六色的袜子中找出一双来穿.终于有一天,小\(Z\)再也无法忍受这恼人的找袜子过程,于是他 ...

  3. 洛谷 P1494 [国家集训队] 小Z的袜子

    题目概述: 小Z把N只袜子从1到N编号,然后从编号L到R(L 尽管小Z并不在意两只袜子是不是完整的一双,甚至不在意两只袜子是否一左一右,他却很在意袜子的颜色,毕竟穿两只不同色的袜子会很尴尬. 你的任务 ...

  4. luogu P1494 [国家集训队]小Z的袜子 ( 普 通 )

    题目:    链接:https://www.luogu.org/problemnew/show/P1494 题意:一些袜子排成一排,每个袜子有固定的颜色.                        ...

  5. 【luogu P1494 [国家集训队]小Z的袜子】 题解

    题目链接:https://www.luogu.org/problemnew/show/P1494 #include <cstdio> #include <algorithm> ...

  6. Luogu P1494 [国家集训队]小Z的袜子

    比较简单的莫队题,主要是为了熟练板子. 先考虑固定区间时我们怎么计算,假设区间\([l,r]\)内颜色为\(i\)的袜子有\(cnt_i\)只,那么对于颜色\(i\)来说,凑齐一双的情况个数为: \( ...

  7. P1494 [国家集训队]小Z的袜子(莫队)

    题目链接:https://www.luogu.org/problemnew/show/P1494 题目大意:中文题目 具体思路:计算概率的时候,每一次是区间的移动,每一次移动,记得先将原来的记录的影响 ...

  8. 洛谷 P1494 [国家集训队]小Z的袜子(莫队)

    题目链接:https://www.luogu.com.cn/problem/P1494 一道很经典的莫队模板题,然而每道莫队题的大体轮廓都差不多. 首先莫队是一种基于分块的算法,它的显著特点就是: 能 ...

  9. P1494 [国家集训队]小Z的袜子(莫队算法)

    莫队板子 代码 #include <cstdio> #include <algorithm> #include <cstring> #include <cma ...

随机推荐

  1. priority_deque作为Timer时间队列底层容器的一些思考

    https://www.bbsmax.com/A/D854VkZxzE/ 设置底层容器可以分离出两个逻辑上独立的问题: >如何存储构成优先级队列(容器)的实际元素,以及>如何组织这些元素以 ...

  2. Linux分布式测试

    在使用Jmeter进行性能测试时,如果并发数比较大(比如最近项目需要支持1000并发),单台电脑的配置(CPU和内存)可能无法支持,这时可以使用Jmeter提供的分布式测试的功能. 执行机和调度机做好 ...

  3. 【Git教程】Git教程及使用命令

      Git是目前世界上最先进的分布式版本控制系统,可以自动记录和管理文件的改动,还可以团队写作编辑,也就是帮助我们对不同的版本进行控制.2008年,GitHub网站上线,为开源项目提供免费存储,迅速发 ...

  4. 【UOJ139】【UER #4】被删除的黑白树

    题意: 很久很久以前,有一棵树加入了 UOJ 群. 这天,在它讨论“一棵树应该怎么旋转”的时候一不小心被删除了,变成了被删除的树. 突然间,它突然发现它失去了颜色,变成了一棵纯白的树.这让它感觉很焦躁 ...

  5. 自动化测试之firebug、firepath、IDE的使用

    1firebug安装-firefox添加组件-firebug 如图 firepath依赖于firebug 展示路径用,安装和firebug一样

  6. 《代码敲不队》第八次团队作业:Alpha冲刺 第四天

    项目 内容 这个作业属于哪个课程 任课教师博客主页链接 这个作业的要求在哪里 作业链接地址 团队名称 代码敲不队 作业学习目标 掌握软件编码实现的工程要求. 团队项目github仓库地址链接 GitH ...

  7. Java基础学习总结(37)——Java23中设计模式(Design Patterns)详解

    设计模式(Design Patterns) --可复用面向对象软件的基础 设计模式(Design pattern)是一套被反复使用.多数人知晓的.经过分类编目的.代码设计经验的总结.使用设计模式是为了 ...

  8. 阻止a标签跳页,使用单击事件函数处理该请求

    阻止a标签跳页href="javascript:void(0)" 单击该标签时 页面不跳页   使用单击事件函数来处理该单击请求  返回上一层页面 ---------------- ...

  9. switch 的穿透, 以及穿透利用

    switch 穿透测试: outputs: 添加break 阻止switch穿透: outputs: 利用switch的穿透功能:

  10. codevs——T1048 石子归并

     http://codevs.cn/problem/1048/  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 黄金 Gold 题解       题目描述 Descriptio ...