P6973 [NEERC2016]List of Primes
题意简述:将质数集合的所有子集按照子集和为第一关键字,字典序为第二关键字从小到大排序,求最终形成的字符串的第 \(l\sim r\) 个字符。
又是一道妙妙题。
首先考虑当 \(r\leq 10^5\) 时直接搜索,首先枚举子集和 \(i\),状态是 \(sum,len,lim\) 表示剩余子集和为 \(sum\),大小为 \(len\),接下来只能使用第 \(lim\) 个及以后的质数。边界是 \(sum=0\),表示找到一个符合题意的子集。
直接暴力跑大概可以过 \(r\leq 10^5\),不过有一个显然的剪枝:预处理 \(f_{i,j}\) 表示只用第 \(j\) 个及以后的质数能否拼出 \(sum=i\),爆搜下一个分支时可以快速判断是否合法,从而避免进入不合法的分支。这样一来复杂度就变成了线性 \(\mathcal{O}(r)\)。
\(r\leq 10^{18}\) 时,我们继续选择搜索。观察到 \(r-l\leq 10^5\),所以如果已经遍历过的子集长度总和 与 接下来的分支所形成的的子集长度总和 之和仍小于 \(l\),那么根本没有必要遍历该分支。对此,我们预处理 \(g_{i,j}\) 表示只用第 \(j\) 个及以后的质数拼出的所有子集的长度之和(不考虑两边的中括号和逗号 \(\texttt{[],}\)),同时更改 \(f_{i,j}\) 的定义:只用第 \(j\) 个及以后的质数拼出来的子集个数。这是一个显然的 DP:从大到小枚举 \(j\),那么有 \(f_{i,j}\gets f_{i,j+1}+f_{i+pr_j,j+1}\),\(g_{i,j}\gets g_{i,j+1}+(g_{i+pr_j,j+1}+(bit(pr_j)+2)\times f_{i+pr_j,j+1})\),其中 \(pr_j\) 表示第 \(j\) 个质数,\(bit(x)\) 表示 \(x\) 在十进制下的位数,有 \(+2\) 是因为两个数之间有长度为 \(2\) 的逗号 + 空格。同时你可以根据 DP 很容易地求出来 \(i\) 的上界大约在 \(2.1\times 10^3\) 左右,那么 \(j\) 只需要开到 \(350\) 即可。
爆搜时,我们记录一个全局变量 \(acc(umulation)\) 表示已经遍历过的子集长度,此外还要在状态中加入 \(slen\) 表示当前分支已经选择的质数的长度之和,即 \(\sum_{p\in \mathrm{chosen}}bit(p)\)(原因接下来会讲)。
首先判断搜索边界:\(sum\) 是否 \(=0\)。是 \(0\) 就表示遍历到了一个子集,将该子集计入答案并返回,否则根据已有的信息算出该分支接下来所能产生的所有子集的长度之和,与 \(acc\) 求和后看是否小于 \(l\),若小于,则直接更新 \(acc\) 并返回即可。否则继续搜索即可。
接着考虑到了搜索边界该干什么:如果遍历到了一个子集,那么一个字符一个字符地考虑:插入一个字符时,首先将 \(acc\) 自增 \(1\),如果 \(acc>r\) 那么退出程序;否则,如果 \(acc\geq l\) 那么输出该字符;否则啥也不干。
怎么根据已有的信息算出该分支的子集长度总和呢?因为能产生 \(f_{sum,lim}\) 个子集,因此 \(acc\) 长度加上:
- \(4f_{sum,lim}\) 表示边界的四个字符:\(\texttt{'[', ']', ',', ' '}\)。
- \((len-1)\times 2f_{sum,lim}\) 表示已经选择的 \(len\) 个数所产生的 \(len-1\) 个长度为 \(2\) 的间隔 \(\texttt{‘,’, ‘ ’}\)(代码中 \(len\) 的初始值为 \(1\),所以是 \(2len\))。
- \(g_{sum,lim}\)(这个就不用解释了吧 = . =)。
- \(slen\times f_{sum,lim}\) 表示已经选择的质数长度在所有 \(f_{sum,lim}\) 个子集中所贡献的长度之和,这也是我们要记录 \(slen\) 的原因。
时间复杂度 \(\mathcal{O}(r-l)\),也可以说是 \(\mathcal{O}(n\pi(n))\),其中 \(n\approx 2100\)。
此外,今天是 2021 年的七夕节(8.14),祝大家七夕快乐,早日脱单(大雾)!
const int N=2100+5;
const int P=350;
int cnt,vis[N],pr[N],bt[N];
ll f[N][P],g[N][P];
void init(){
for(int i=2;i<N;i++){
if(vis[i])continue;
pr[++cnt]=i,f[i][cnt]=1,g[i][cnt]=bt[i]=log10(i)+1;
for(int j=i+i;j<N;j+=i)vis[j]=1;
}
for(int i=cnt;i;i--){
int len=bt[pr[i]]+2;
for(int j=pr[i];j<N;j++)
f[j][i]+=f[j-pr[i]][i+1]+f[j][i+1],
g[j][i]+=g[j-pr[i]][i+1]+g[j][i+1]+f[j-pr[i]][i+1]*len;
}
}
ll l,r,acc,p[N];
string to_str(int x){
string s;
while(x)s+=x%10+'0',x/=10;
reverse(s.begin(),s.end());
return s;
}
void add(char s){
if(++acc>r)exit(0);
if(acc>=l)cout<<s;
}
void dfs(int sum,int len,int lim,ll slen){
if(!sum){
add('[');
for(int i=1;i<=len;i++){
string s=to_str(p[i]);
for(int j=0;j<s.size();j++)add(s[j]);
if(i<len)add(','),add(' ');
} add(']'),add(','),add(' ');
return;
}
ll nw=acc+(len*2+4+slen)*f[sum][lim]+g[sum][lim];
if(nw<l)return acc=nw,void();
len++;
for(int j=lim;j<=cnt;j++){
int res=sum-pr[j]; p[len]=pr[j];
if(res<0)break;
if(res==0||f[res][j])dfs(res,len,j+1,slen+bt[pr[j]]);
}
}
int main(){
init(),cin>>l>>r;
for(int i=2;i<N;i++)dfs(i,0,1,0);
return 0;
}
P6973 [NEERC2016]List of Primes的更多相关文章
- [LeetCode] Count Primes 质数的个数
Description: Count the number of prime numbers less than a non-negative number, n click to show more ...
- projecteuler Summation of primes
The sum of the primes below 10 is 2 + 3 + 5 + 7 = 17. Find the sum of all the primes below two milli ...
- leetcode-Count Primes 以及python的小特性
题目大家都非常熟悉,求小于n的所有素数的个数. 自己写的python 代码老是通不过时间门槛,无奈去看了看大家写的code.下面是我看到的投票最高的code: class Solution: # @p ...
- Count Primes - LeetCode
examination questions Description: Count the number of prime numbers less than a non-negative number ...
- [leetcode] Count Primes
Count Primes Description: Count the number of prime numbers less than a non-negative number, n click ...
- Count Primes
Count the number of prime numbers less than a non-negative number, n public int countPrimes(int n) { ...
- hduoj 4715 Difference Between Primes 2013 ACM/ICPC Asia Regional Online —— Warmup
http://acm.hdu.edu.cn/showproblem.php?pid=4715 Difference Between Primes Time Limit: 2000/1000 MS (J ...
- HDU 4715:Difference Between Primes
Difference Between Primes Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Jav ...
- hdu 4715 Difference Between Primes
题目连接 http://acm.hdu.edu.cn/showproblem.php?pid=4715 Difference Between Primes Description All you kn ...
随机推荐
- 【c++ Prime 学习笔记】第16章 模板与泛型编程
面向对象编程(OOP)和泛型编程(GP)都能处理在编写程序时类型未知的情况 OOP能处理运行时获取类型的情况 GP能处理编译期可获取类型的情况 标准库的容器.迭代器.算法都是泛型编程 编写泛型程序时独 ...
- BUAA_2020_软件工程_提问回顾与总结
项目 内容 这个作业属于哪个课程 2020春季计算机学院软件工程(罗杰 任建) 这个作业的要求在哪里 提问回顾与总结作业要求 我在这个课程的目标 了解软件工程的技术,掌握工程化开发的能力 这个作业在哪 ...
- 关于下载pyton第三方库的细节
1.下载Python第三方库有时候国外的网站网速很不好,需要选择国内的镜像网站去下载 阿里云 http://mirrors.aliyun.com/pypi/simple 中国科技大学 https: ...
- Swift-方法调度-类的普通方法底层探究
1. 类的普通方法调度 写一个结构体和一个类,对比看看方法调用的方式: // 结构体 struct PersonStruct { func changClassName() {} } let s = ...
- 『学了就忘』Linux基础命令 — 24、文件基本权限的相关命令
目录 1.chmod命令 2.权限模式 (1)用户身份. (2)赋予方式. (3)权限. 3.数字权限 4.文件常用权限 5.chown命令 6.chgrp命令 7.总结 常用基本权限操作命令: ch ...
- SimpleNVR流媒体服务在多分屏直播实时阅览时所遇到问题的解决
视频有一个流的概念,称为流媒体.当大量的客户端或WEB访问监控摄像机的时候,大多数的录像机无法承受那么大的网络压力,这时候SimpleNVR流媒体服务器的优势就显示出来了.其能将客户端的访问压力转到服 ...
- [JavaScript闭包]Javascript闭包的判别,作用和示例
闭包是JavaScript最重要的特性之一,也是全栈/前端/JS面试的考点. 那闭包究竟该如何理解呢? 如果不爱看文字,喜欢看视频.那本文配套讲解视频已发送到B站上供大家参考学习. 如果觉得有所收获, ...
- js深拷贝你还不会吗
js深拷贝 在讲正题之前我们要先了解数据存储的方式 数据存储方式 在讲之前我们要先知道值类型和引用类型的存储方式. 在JavaScript数据类型中有两种数据类型. 值类型:字符串(String).数 ...
- PHP、TP6框架及JavaScript的单步调试
目录 一.PHP程序的调试 1. 单个PHP程序的调试 2. PHP框架代码的调试 二.JavaScript程序的调试 三.总结 参考资料:https://www.bilibili.com/video ...
- Django 小实例S1 简易学生选课管理系统 7 修改个人信息
Django 小实例S1 简易学生选课管理系统 第7节--修改个人信息 点击查看教程总目录 作者自我介绍:b站小UP主,时常直播编程+红警三,python1对1辅导老师. 用户模块除了注册登录之外,还 ...