CF643G Choosing Ads

\(n\) 和 \(m\) 和 \(p\) 和序列 \(a_i(1\le i\le n)\)。\(m\) 种如下操作:

  1. 1 l r id 令 \(i\in[l,r]:a_i=id\)。
  2. 2 l r 输出至多 \(\lfloor\frac{100}{p}\rfloor\) 个数,其中包括所有 \([l,r]\) 区间内出现 \(\ge\lceil\frac{p(r-l+1)}{100}\rceil\) 次的 \(a_i\)。

数据范围:\(1\le n,m,id,a_i\le 150000\),\(20\le p\le 100\),\(1\le l\le r\le n\)


一眼想法:搞个线段树,节点存该区间出现频率 \(\ge\frac{p}{100}\) 的 \(a_i\) 值和出现次数存下来,合并。

这个做法的依据: 如果区间 \([l,r]\) 中 \(a_i\) 出现频率 \(\ge\frac{p}{100}\),\([l,mid]\) 或 \([mid+1,r]\) 必中有一个也满足。

这个做法的 \(\tt bug\): 若 \(a_i\) 存在于 \([l,mid]\) 的节点中而不存在于 \([mid+1,r]\) 的节点中(或反过来),节点的信息难以合并。

于是,我这个思维愚钝的大蒟蒻就没在考场上做出来。


这题的正解类似以前的一道经典题:一个序列求众数(出现频率 \(\ge \frac 12\))空间复杂度 只能 \(\Theta(1)\)。

做法是记录 \(now\) 和 \(cnt\)。新加入数 \(a\) 的时候,如果 \([now=a]\),\(cnt++\);如果 \([now\not=a]\),\(cnt--\),如果 \([cnt<0]\),\(cnt=1\) 并且 \(now=a\)。


这题也类似。出现频率 \(\ge\frac{p}{100}\) 的数至多 \(\lfloor\frac{100}{p}\rfloor\) 个,所以可以记录 \(\lfloor\frac{100}{p}\rfloor\) 个 \(now_i\) 和 \(cnt_i\)。

每次加入 \(a\),如果 \([now_i=a]\),\(cnt_i++\);如果 \([now_i\not=a]\),\(cnt_i--\),如果 \([cnt_i<0]\),\(cnt_i=1\) 并且 \(now_i=a\)。

这样的话虽然只存了 \(\lfloor\frac{100}{p}\rfloor\) 种数,但已经完全反映了区间内数的数量对比,所以节点信息可以直接合并。

最后每个节点或许会存下不满足出现频率 \(\ge\frac{p}{100}\) 的值,但是输出只要包含答案就行了(题中说的)。


  • 代码

稍微长了点,但是没有坑人的细节,直接写就好了。

#include <bits/stdc++.h>
using namespace std; //Start
typedef long long ll;
typedef double db;
#define mp(a,b) make_pair(a,b)
#define x first
#define y second
#define b(a) a.begin()
#define e(a) a.end()
#define sz(a) int((a).size())
#define pb(a) push_back(a)
const int inf=0x3f3f3f3f;
const ll INF=0x3f3f3f3f3f3f3f3f; //Data
const int N=150000;
int n,m,p,pl,a[N+7]; //Segmenttree
typedef vector<pair<int,int>> vpii;
#define mid ((l+r)>>1)
#define ls k<<1,l,mid
#define rs k<<1|1,mid+1,r
int mk[N<<2]; vpii ca[N<<2]; //pair.x 表示 now_i,pair.y 表示 cnt_i
void fill(vpii&v,int id,int c){v.clear(),v.pb(mp(id,c));}
vpii operator+(vpii p,vpii q){ //将 p 中的数一一加入 q
vpii r;
for(auto&u:p){
int ok=0; for(auto&v:q)if(u.x==v.x){v.y+=u.y,ok=1;break;}
if(ok) continue; q.pb(u); if(sz(q)<=pl) continue;
int mn=n; for(auto&v:q) mn=min(mn,v.y);
r.clear(); for(auto&v:q)if(v.y-mn) r.pb(mp(v.x,v.y-mn)); q=r;
}
return q;
}
void down(int k,int l,int r){
if(!mk[k]) return;
fill(ca[k<<1],mk[k],mid-l+1),fill(ca[k<<1|1],mk[k],r-mid);
mk[k<<1]=mk[k<<1|1]=mk[k],mk[k]=0;
}
void build(int k=1,int l=1,int r=n){
if(l==r) return void(fill(ca[k],a[l],1));
build(ls),build(rs),ca[k]=ca[k<<1]+ca[k<<1|1];
}
void fix(int x,int y,int z,int k=1,int l=1,int r=n){
if(x<=l&&r<=y) return void((fill(ca[k],z,r-l+1),mk[k]=z)); down(k,l,r);
if(mid>=x) fix(x,y,z,ls); if(mid<y) fix(x,y,z,rs);
ca[k]=ca[k<<1]+ca[k<<1|1];
}
vpii query(int x,int y,int k=1,int l=1,int r=n){
if(x<=l&&r<=y) return ca[k]; down(k,l,r);
vpii res; if(mid>=x) res=res+query(x,y,ls); if(mid<y) res=res+query(x,y,rs);
return res;
}
void Print(int k=1,int l=1,int r=n){ //调试用的,我这个蒟蒻,总是代码写挂
printf("[%d,%d,%d]:mk=%d\n",k,l,r,mk[k]);
for(auto&u:ca[k]) printf("(%d,%d)",u.x,u.y);puts("");
if(l==r) return; down(k,l,r);
Print(ls),Print(rs);
} //Main
int main(){
scanf("%d%d%d",&n,&m,&p),pl=100/p;
for(int i=1;i<=n;i++) scanf("%d",&a[i]); build();
for(int i=1;i<=m;i++){
int o; scanf("%d",&o);
if(o==1){int l,r,id; scanf("%d%d%d",&l,&r,&id); fix(l,r,id);}
else {
int l,r; scanf("%d%d",&l,&r);
vpii res=query(l,r); printf("%d",sz(res));
for(auto&u:res) printf(" %d",u.x); puts("");
}
// puts("+++++");
// Print();
// puts("-----");
}
return 0;
}

祝大家学习愉快!

题解-CF643G Choosing Ads的更多相关文章

  1. [cf643G]Choosing Ads

    首先对于$p>50$,有经典的做法,即不断删去区间中不同的两数,最终剩下的即为出现次数超过一半的数(或没有),用线段树维护即可 那么对于$p\le 50$,类似的,即删去区间中不同的$\lflo ...

  2. 「CF643G」 Choosing Ads

    「CF643G」 Choosing Ads 传送门 如果你知道摩尔投票法可以扩展事实上是个一眼题,又好写又好调. 首先摩尔投票法是用来求众数定义为超过所有数个数一半的数的一个算法. 大致算法流程: 将 ...

  3. @codeforces - 674G@ Choosing Ads

    目录 @description@ @solution@ @accepted code@ @details@ @description@ 给定长度为 n 的序列,m 次询问以及参数 p.共有两类询问: ...

  4. Codeforces 643G - Choosing Ads(线段树)

    Codeforces 题目传送门 & 洛谷题目传送门 首先考虑 \(p>50\) 的时候怎么处理,也就是求一个区间的绝对众数.我们知道众数这个东西是不能用线段树直接维护的,因为对于区间 ...

  5. 多校联训 DS 专题

    CF1039D You Are Given a Tree 容易发现,当 \(k\) 不断增大时,答案不断减小,且 \(k\) 的答案不超过 \(\lfloor\frac {n}{k}\rfloor\) ...

  6. Codeforces Choosing Laptop 题解

    这题实在是太水了,具体看注释 蒟蒻的方法是一边找过时的电脑一边比大小 蒟蒻不才,只会C++ 其实还会free basic,但它已经过时了 附: 本题洛谷网址 Codeforces网址 希望蒟蒻的题解能 ...

  7. 【题解】codeforces 219D Choosing Capital for Treeland 树型dp

    题目描述 Treeland国有n个城市,这n个城市连成了一颗树,有n-1条道路连接了所有城市.每条道路只能单向通行.现在政府需要决定选择哪个城市为首都.假如城市i成为了首都,那么为了使首都能到达任意一 ...

  8. 【codeforce 219D】 Choosing Capital for Treeland (树形DP)

    Choosing Capital for Treeland Description The country Treeland consists of n cities, some pairs of t ...

  9. usaco training 4.2.4 Cowcycles 题解

    Cowcycles题解 Originally by Don Gillies [International readers should note that some words are puns on ...

随机推荐

  1. Win10系统下安装VC6.0教程

    学习一门语言最重要的一步是搭建环境,许多人搭建在搭建环境上撞墙了,就有些放弃的心理了:俗话说,工欲善其事,必先利其器:所以接下来我们进行学习C的第一步下载编程所用的工具;当然也有其它的软件,只不过初学 ...

  2. LeetCode 中等题解(4)

    40 组合总和 II Question 给定一个数组 candidates 和一个目标数 target ,找出 candidates 中所有可以使数字和为 target 的组合. candidates ...

  3. Leetcode 1329. 将矩阵按对角线排序 题解

    首先遍历对角线元素,顺序为: 先从第一列的最后一行到第一行 然后从第一行的第一列到最后一列 遍历的同时记录坐标和数值,对数值进行排序,然后坐标顺序放回. class Solution: def dia ...

  4. php批量转换时间戳

    //批量转换时间戳 array_map(array($this, 'myfunction'),'时间戳数组'); //如 array('time1'=>11,'time2'=>2222); ...

  5. 2个快速制作完成一幅思维导图的iMindMap思维导图用法

    随着思维导图的流行,与其相关的思维导图制作软件如雨后春笋,纷纷进入我们的视野中,更让人难以选择.那想要入门的萌新该如何开始这个新的旅途呢? 各式各样的思维导图制作软件当中,有一个软件得到了大家一致的好 ...

  6. Boom 3D的保真度是什么,如何应用

    Boom 3D是一款非常优秀的3D音频软件,拥有3D音效.环境模式.空间模式.夜间模式.保真度等多种音效模式,可以为用户提供多种音效体验感. 第一.什么是保真度 或许第一次接触音频软件的朋友就会问到什 ...

  7. Django rest framework 基础

    01: Django rest framework 基础 ​ ​ 1.1 什么是RESTful 1. REST与技术无关,代表的是一种软件架构风格(REST是Representational Stat ...

  8. Acwing 245.你能回答这些问题吗

    题目描述 给定长度为N的数列A,以及M条指令,每条指令可能是以下两种之一: 1."1 x y",查询区间 [x,y] 中的最大连续子段和,即 maxx≤l≤r≤y{∑ri=lA[i ...

  9. iOS如何实现语音播报及后台播放

    最近项目刚刚交付,偶然间用到了语音播报和语音搜索的功能.语音搜索我用的是讯飞的demo,感觉效果还不错,感兴趣的话可以去官网上面下载demo,里面讲的特别的详细,不过稍显麻烦一些.语音播报讯飞也有de ...

  10. Java基础教程——模拟浏览器发送请求

    JAVA访问网页 分别测试使用get和post方法访问网页,可以收到服务器的请求,并写入到html文件中. import java.io.*; import java.net.*; import ja ...