题目描述

我们说一个可重集 AA 比可重集 BB 小,当且仅当对于两个可重集中出现次数不同的最小元素 xx ,元素 xx 在 AA 中出现次数更多。

例如,可重集 {1,2,3}1,2,3 就比可重集 {1,3,3,5}1,3,3,5 小,类似的,{1,1,4,4}1,1,4,4 小于 {1,1,4}1,1,4 。

小 CC 给你了一个长度为 nn 的正整数序列 SS 。考虑 SS 的所有连续子序列,可以把它们分别看做一个可重集(也就是说,恰好存在着 \frac{n(n+1)}{2}2n(n+1)​ 个可重集)。

小 CC 想知道第 kk 小可重集,想请你帮她找到答案。

输入格式

输入共两行 。

第一行包含两个整数 n,kn,k 。

接下来的一行包含序列 S_iSi​ 。

输出格式

输出共一行。

输出你找到的可重集,请输出可重集排序后的结果。

可以用类似整体二分的缩小答案集合的思想,对于每个数确定在答案中应该出现几次。使用set来存答案的集合。

  1 #include <bits/stdc++.h>
2 #define N 100005
3 //#define loveGsy
4 using namespace std;
5 int n, k, s[N], hv[N], ans[N];
6 vector<int> pos[N];//存一个数出现在哪些位置
7 struct node {
8 int ll, lr, rl, rr;
9 node() {}
10 node(int a, int b, int c, int d) {ll = a, lr = b, rl = c, rr = d;}
11 friend bool operator < (node a, node b) {
12 return a.ll < b.ll;
13 }
14 };
15 set<node> st, hc;
16
17 int query(int p, int num) {
18 int l = 1, r = l + num - 1, js = 0, ll, lr, rl, rr, LL, LR, RL, RR;
19 set<node>::iterator zz;
20 while (r + 1 < pos[p].size()) {
21 LL = pos[p][l - 1] + 1;
22 LR = pos[p][l];
23 RL = pos[p][r];
24 RR = pos[p][r + 1] - 1;
25 zz = st.upper_bound(node(LR, 0, 0, 0));
26 while (zz != st.begin()) {
27 --zz;
28 if ((*zz).lr < LL) break;
29 ll = max((*zz).ll, LL);
30 lr = min((*zz).lr, LR);
31 rl = max((*zz).rl, RL);
32 rr = min((*zz).rr, RR);
33 if (lr >= ll && rr >= rl)
34 js += (lr - ll + 1) * (rr - rl + 1);
35 }
36 ++l, ++r;
37 }
38 return js;
39 }
40
41 void getnum(int p) {
42 ans[p] = hv[p];
43 while (ans[p]) {
44 int hc = query(p, ans[p]);//查询st集合内有多少区间满足有ans[p]个p
45 if (hc >= k) return ;
46 --ans[p];
47 k -= hc;//类似整体二分板题的答案分拆,缩小答案集合
48 }
49 }
50
51 void merge(int p, int num) {
52 int l = 1, r = l + num - 1, js = 0, ll, lr, rl, rr, LL, LR, RL, RR;
53 set<node>::iterator zz;
54 while (r + 1 < pos[p].size()) {
55 LL = pos[p][l-1] + 1;
56 LR = pos[p][l];
57 RL = pos[p][r];
58 RR = pos[p][r + 1] - 1;
59 zz = st.upper_bound(node(LR, 0, 0, 0));
60 while (zz != st.begin()) {
61 --zz;
62 if((*zz).lr < LL) break;
63 ll = max((*zz).ll, LL);
64 lr = min((*zz).lr, LR);
65 rl = max((*zz).rl, RL);
66 rr = min((*zz).rr, RR);//取交集
67 if(lr >= ll && rr >= rl)
68 hc.insert(node(ll,lr,rl,rr));
69 }
70 ++l,++r;
71 }
72 st.clear();//清空原来的集合
73 zz = hc.begin();
74 while (zz != hc.end()) {
75 st.insert((*zz));
76 ++zz;
77 }
78 hc.clear();
79 }
80
81 int main() {
82 #ifdef loveGsy
83 freopen("tree.in", "r", stdin);
84 freopen("tree.out", "w", stdout);
85 #endif
86 scanf("%d %d", &n, &k);
87 for (int i = 1; i <= n; i++) pos[i].push_back(0);//存边界条件
88 for (int i = 1; i <= n; i++) {
89 scanf("%d", &s[i]);
90 ++hv[s[i]];
91 pos[s[i]].push_back(i);
92 }
93 for (int i = 1; i <= n; i++) pos[i].push_back(n + 1);
94 st.insert(node(1, n, 1, n));//插入初始答案集合
95 for (int i = 1; i <= n; i++) {//确定答案中每个数字应该出现几次
96 if (hv[i]) {
97 getnum(i);//确定答案中应该有ans[i]个i
98 merge(i, ans[i]);//从st集合中筛选出含有ans[i]个i的集合
99 }
100 }
101 for (int i = 1; i <= n; i++) {
102 for (int j = 1; j <= ans[i]; j++)
103 printf("%d ", i);
104 }
105 return 0;
106 }

220726 T2 Multisets (思维)的更多相关文章

  1. EZ 2018 03 09 NOIP2018 模拟赛(三)

    最近挺久没写比赛类的blog了 链接:http://211.140.156.254:2333/contest/59 这次的题目主要考验的是爆搜+打表的能力 其实如果你上来就把所有题目都看过一次就可以知 ...

  2. HZ游记

    HZ 游记 Day -1 收拾东西,准备出发. 话说这几天一直比较懒,也没什么心情和效率学习,颓废好几天了,希望到衡水以后能感觉好点. 不知道衡水有没有妹子 非常想看看衡水的样子,但是又害怕封闭式教学 ...

  3. D - Number of Multisets 题解(思维dp)

    题目链接 题目大意 给你一个数k和n,表示用n个\(1/2^i(i=0,1,2.....)\)组成k有多少种方案数 题目思路 这个dp实属巧妙 设\(dp[i][j]表示i个数构成j\) 这i个数可以 ...

  4. 1250 Super Fast Fourier Transform(湘潭邀请赛 暴力 思维)

    湘潭邀请赛的一题,名字叫"超级FFT"最终暴力就行,还是思维不够灵活,要吸取教训. 由于每组数据总量只有1e5这个级别,和不超过1e6,故先预处理再暴力即可. #include&l ...

  5. Day4:T1小技巧(类似于指针操作)T2搜索+小细节

    Day4:其中有很多小技巧get T1 一直没有听到过像这样的小技巧的略专业名词,有点类似于指针操作,之前有碰到过很多这样的题目 每次都是以不同的形式出现,但是感觉思想还是有点接近的吧(就比如某天有一 ...

  6. Day1:T1 模拟 T2 拓扑排序

    T1:模拟 自己第一天的简直跟白痴一样啊...模拟都会打错.. 当时貌似在更新最大值的时候打逗比了... if((sum[x]==max && x<maxh) || sum[x] ...

  7. 带你深入理解STL之空间配置器(思维导图+源码)

    前不久把STL细看了一遍,由于看得太"认真",忘了做笔记,归纳和总结这步漏掉了.于是为了加深印象,打算重看一遍,并记录下来里面的一些实现细节.方便以后能较好的复习它. 以前在项目中 ...

  8. PJ考试可能会用到的数学思维题选讲-自学教程-自学笔记

    PJ考试可能会用到的数学思维题选讲 by Pleiades_Antares 是学弟学妹的讲义--然后一部分题目是我弄的一部分来源于洛谷用户@ 普及组的一些数学思维题,所以可能有点菜咯别怪我 OI中的数 ...

  9. 思维定势--AtCoder Regular Contest 092 D - Two Sequences

    $n \leq 100000$的俩序列,数字范围$2^{28}$,问所有$a_i+b_j$的$n^2$个数字的异或和. 这种东西肯定是按位考虑嘛,从低位开始然后补上进位.比如说第一位俩串分别有$c$个 ...

随机推荐

  1. 整数分解和for循环

    整数的分解: 一个整数是由多位数字组成的,那么如何能分解出整数的各个位上的数字呢 对一个整数做%10的操作,就可以得到它的个位数 对一个整数做/10的操作,就去掉了他的个位数 然后再对2的结果做%10 ...

  2. python在管道中执行命令

    简介 在实际开发中,可能在执行命令过程中,需要在命令的管道中输入相应命令后继续执行,因此需要在执行命令后在命令的管道中输入相应指令 方法一 直接使用communicate向管道传入所需指令,注意如果是 ...

  3. 用js获取当前页面的url

    1.获取当前或者指定页面的文件名或路径 window.location.pathname 2.设置或获取整个URl字符串 window.location.href 3.设置或获取URL相关的端口号 w ...

  4. 使用Python3将word文档和pdf电子书进行格式互转(兼容Windows/Linux)

    原文转载自「刘悦的技术博客」https://v3u.cn/a_id_96 一些重要文档格式之间的互转在目前显得尤为重要,pdf作为通用格式在现在各个平台上兼容性是最好的,所以写python脚本将这些w ...

  5. Percona XtraBackup 8.0.26使用说明

    欢迎来到 GreatSQL社区分享的MySQL技术文章,如有疑问或想学习的内容,可以在下方评论区留言,看到后会进行解答 Percona XtraBackup特性说明 Percona Xtrabacku ...

  6. java学习第一天.day05

    jvm的内存 栈:类方法使用后自动销毁,销毁的好处是释放内存 java方法执行时,在栈区执行 堆: 线程共享的一块内存区域      所有的对象实例以及 数组 都要在堆上分配      每次使用new ...

  7. ceph 008 ceph多区域网关(ceph对象容灾) cephfs文件系统

    clienta作为集群的管理人员.一部分.他是需要秘钥与配置文件的 但真正服务端只需要通过curl就好 ceph 多区域网关 对象存储容灾解决方案 zone与zone会做数据同步. 把会做同步的rgw ...

  8. Express 项目,res.cookie() 设置 Cookie 无法被保存在浏览器的 Application 中

    res.cookie() 给客户端响应头封装的 Cookie 无法被保存在客户端浏览器的 Application 中,只能在 Set-Cookie 中看到有这个值: 在前后端分离项目中,存在跨域问题, ...

  9. CM311-1a(S905L3系列)玩转桌面

    那安装了ambian后,玩转桌面(安装GUI桌面环境)有没有可能呢?那肯定啊!那桌面有什么用?当然有用,多一种玩法,可以写写代码,上网冲浪,学习linux语法什么的.而且单主机只要40左右,想想看可以 ...

  10. 获取进程产生了多少次pagefault

    怎么获取某个进程产生了多少次pagefault? 这个在ps 命令中可以看到,比如查看java的pagefault情况. ps -o maj_flt -o min_flt -p `ps -e|grep ...