【CF1443E】Long Permutation 题解(排列生成模板)
题意简介
给定一个长度为 n 的排列 {1,2,3,...,n} 。现有两种操作:
- 对某个区间 [l,r] 求和
- 将排列往后推 x 次 (按字典序)
其中 \(n,q \leq 2\times10^5 , x\leq 10^5\) 。
思路分析
乍一看毫无思路。
因为排列变换是毫无疑问的暴力,变换后怎么维护区间和是一个非常玄妙的问题。好像没有什么特殊的维护技巧。
仔细观察数据范围:\(x\leq 10^5 , q\leq 2\times 10^5\)
这意味着 \(\sum x_i \leq 2\times 10^{10}\)
而经过简单的打表观察我们不难发现 \(13! <2\times 10^{10}<14!\)
换句话说,所有的操作过后,会改变的实际上只有最后的 14 个数。
于是问题就简单了,每次更新暴力更改后 14 个数,维护一下前缀和就行了。
那么让我们来考虑如何生成一个排名为 x 的排列。
其实类比如何计算某个排列的排名,反过来就行了。
我的做法是利用树状数组维护比某个数小的数里有几个被选过,然后用二分查找确定当前位置的数字。
详见代码。
代码库
1. 排列生成模板
#include <cstdio>
typedef long long ll;
#define REG register
#define rep(i,a,b) for(REG int i=a;i<=b;i++)
#define Rep(i,a,b) for(REG int i=a;i>=b;i--)
int n,tr[25],A[25]; ll d,fact[25];
inline int lowbit(int x){
return x&-x;
}
inline int sum(int x){
REG int ans=0;
while(x) ans+=tr[x],x-=lowbit(x);
return ans;
}
inline void add(int x){
while(x<=n) tr[x]++,x+=lowbit(x);
}
inline bool check(int x,ll r,int i){
// x-sum(x) 表示这个数往下的没用过的数的个数
// x 在 i 位置上的所有情况之和仍不足以达到 d
return r+(x-sum(x))*fact[n-i]<d;
}
int main(){
fact[0]=1; rep(i,1,20) fact[i]=fact[i-1]*i;
while(scanf("%d%lld",&n,&d)==2){
//tr 用于存储用过的数字
rep(i,1,n) tr[i]=0;
REG ll rest=0;
rep(i,1,n){
REG int l=1,r=n,mid,ans=0;
while(l<=r){
//找到最大的恰不能使序号比 d 大的数字
mid=(l+r)>>1;
if(check(mid,rest,i)) ans=mid,l=mid+1;
else r=mid-1;
}
// ans+1 必然没用过,假如用过了,必然会被记为 ans
A[i]=ans+1; rest+=(ans-sum(ans))*fact[n-i]; add(A[i]);
}
rep(i,1,n) printf("%d ",A[i]); putchar('\n');
}
return 0;
}
2. 本题题解
#include <cstdio>
#include <cstring>
typedef long long ll;
#define REG register
#define rep(i,a,b) for(REG int i=a;i<=b;i++)
#define Rep(i,a,b) for(REG int i=a;i<=b;i++)
inline char getc(){
static char buf[1<<14],*p1=buf,*p2=buf;
return p1==p2&&(p2=(p1=buf)+fread(buf,1,1<<14,stdin),p1==p2)?EOF:*p1++;
}
inline ll scan(){
REG ll x=0; REG char ch=0;
while(ch<48) ch=getc();
while(ch>=48) x=x*10+ch-48,ch=getc();
return x;
}
inline int max(const int&a,const int&b){
return a>b?a:b;
}
inline int min(const int&a,const int&b){
return a<b?a:b;
}
const int N=2e5+5;
//注意初始排列的排名为 1
int n,q,tr[17]; ll sum[N],D=1,fact[17];
inline int lowbit(int x){
return x&-x;
}
inline int query(int x){
REG int ans=0;
while(x) ans+=tr[x],x-=lowbit(x);
return ans;
}
inline void add(int x){
//注意这里不是 x<=n
while(x<=14) tr[x]++,x+=lowbit(x);
}
inline bool check(int x,int i,ll r){
return r+(x-query(x))*fact[n-i]<D;
}
inline void test(){
printf("Now:\n");
rep(i,1,n) printf("%d ",sum[i]-sum[i-1]);
putchar('\n');
}
int main(){
n=scan(),q=scan();
rep(i,1,n) sum[i]=sum[i-1]+i;
fact[0]=1;
rep(i,1,14) fact[i]=fact[i-1]*i;
while(q--){
REG int opt=scan();
if(opt==1){
REG int l=scan(),r=scan();
printf("%lld\n",sum[r]-sum[l-1]);
}else{
D+=scan(); REG ll rest=0;
memset(tr,0,sizeof(tr));
//只有最后的 14 个数会发生变化
for(REG int i=max(1,n-13);i<=n;i++){
REG int l=1,r=min(14,n),mid,ans=0;
while(l<=r){
//找到最大的恰不能使序号比 d 大的数字
mid=(l+r)>>1;
if(check(mid,i,rest)) l=mid+1,ans=mid;
else r=mid-1;
}
sum[i]=sum[i-1]+ans+max(1,n-13);
rest+=(ans-query(ans))*fact[n-i];
add(ans+1);
}
//test();
}
}
return 0;
}
END
【CF1443E】Long Permutation 题解(排列生成模板)的更多相关文章
- 【原创】开源.NET排列组合组件KwCombinatorics使用(二)——排列生成
本博客所有文章分类的总目录:本博客博文总目录-实时更新 本博客其他.NET开源项目文章目录:[目录]本博客其他.NET开源项目文章目录 KwCombinatorics组件文章目录: 1. ...
- [BILL WEI]stimulsoft reports DEMO自动生成模板
stimulsoft reports是一款强大的报表开发工具,能够开发各式各样的报表. 对于初学者而言,任何报表开发,刚开始都是去模仿,熟练掌握之后,自己才能独立开发,而在报表开发实际过程中, 我们所 ...
- C# T4 模板 数据库实体类生成模板(带注释,娱乐用)
说明:..,有些工具生成实体类没注释,不能和SqlServer的MS_Description属性一起使用,然后照着网上的资源,随便写了个生成模板,自娱自乐向,其实卵用都没有参考教程 1.htt ...
- HDU 1027 Ignatius and the Princess II 排列生成
解题报告:1-n这n个数,有n!中不同的排列,将这n!个数列按照字典序排序,输出第m个数列. 第一次TLE了,没注意到题目上的n和m的范围,n的范围是小于1000的,然后m的范围是小于10000的,很 ...
- springboot mail整合freemark实现动态生成模板
目标:1:springboot 整合 mail2: mail 使用freemark 实现模板动态生成(就是通过字符串生成模板,不需要在工程中写入固定模板)3: springboot 整合aop 实现日 ...
- uniapp - 更改项目生成模板、页面
每次生成项目目录都需要删除一些再添加太麻烦了,就想着能不能修改一下模板 - 当然自定义模板也能实现 好了,被我找到了. 修改以后源文件名称和格式覆盖回去即可,重新启动hbuilderx 这里可以修改e ...
- mybatis配eclise模板,mybatis快速生成模板
eclipse中mybatis得mapper文件不提示(mybatis-3-mapper.dtd,mybatis-3-config.dtd) 1.下载该文件到你的硬盘文件夹下 2.windows -- ...
- Android Studio 配置快速生成模板代码
前言 Android studio 有提供快速生成模板代码的功能,其实这个功能也可以自定义配置.此篇博客将讲解如何使用此功能 进入Settings 选择 Editor > Live Templa ...
- ACM-挑战题之排列生成
题目描述:挑战题之排列生成 一自然数N,设N为3,则关于N的字典序排列为123,132,213,231,312,321.对于一个自然数N(1<= N <= 9 ) , 你要做的便是生成它的 ...
随机推荐
- Spring Cloud系列(二):Eureka应用详解
一.注册中心 1.注册中心演变过程 2.注册中心必备功能 ① 服务的上线 ② 服务的下线 ③ 服务的剔除 ④ 服务的查询 ⑤ 注册中心HA ⑥ 注册中心节点数据同步 ⑦ 服务信息的存储,比如mysql ...
- 放弃"指针常量"这种不严谨的中文描述!深度理解数组名、指针常量
ques1: 数组名完全等价于指针常量吗? int array[10] = { 10,11,12,13,14,15 }; printf("sizeof(array)= %d \n" ...
- IDEA文本编辑区的护眼绿豆沙色配置
第一步:打开IDEA -> File -> settings -> Editor -> Color Scheme -> General 第二步:找到右方Text -> ...
- Java知识系统回顾整理01基础05控制流程08综合练习
一.练习--黄金分割点 题目: 寻找某两个数相除,其结果 离黄金分割点 0.618最近 分母和分子不能同时为偶数 分母和分子 取值范围在[1,20] (即1到20) 要求效果: public clas ...
- Code Forces 1030E
题目大意: 给你n个数,你可以交换一个数的任意二进制位,问你可以选出多少区间经过操作后异或和是0. 思路分析: 根据题目,很容易知道,对于每个数,我们可以无视它的1在那些位置,只要关注它有几个1即可, ...
- Hello World -- 第一篇博客 -- 活着的意义
今年注定是不寻常的一年,因为技术,接触了许多大牛.通过一篇篇博文,看到了大牛们勤奋好学.孜孜不倦的精神,于是决定也开个博客,向大牛学习. 博客开了,写点什么呢?奈何肚子里墨水不多,吐出来也多是白沫,不 ...
- 有感于“U盘型人才”
先转载一篇互联网上转载比较多的一篇文章,文章是一名职业规划师写的: 上一阶段欠的债,下一阶段总要还,剩男剩女的家里比较着急也是这个道理,该结婚的时候不结婚,生涯任务没完成,必将影响下一段 ...
- 原生JS实现下拉列表
1 <div class="list"> 2 <ul> 3 <li> 4 <a href="#">Web部< ...
- day35 Pyhton 网络编程02
一内容回顾 网络基础 网络应用开发架构 B/S架构 浏览器直接作为客户端的程序 C/S架构 B/S是特殊的C/S osi七层模型 应用层 python代码 http https ftp ...
- python程序整理(1)
''' 用户登录验证 要求: 1. 系统⾃动⽣成4位随机数. 作为登录验证码. 直接用就好. 这里不用纠结 提示. 生成随机数的办法. from random import randint num = ...