直接进入正题。

康托展开

Description

现在有"ABCDEFGHIJ”10个字符,将其所有的排列中按字典序排列,给出任意一种排列,说出这个排列在所有的排列中是第几小的?

Input

第一行有一个整数n(0<n<=100000);

随后有n行,每行是一个排列;

Output

输出一个整数m,占一行,m表示排列是第几位;

Sample Input
3
ABCDEFGHIJ
HGEBFACDJI
GDEDHJBXIA
Sample Output
1
2803322
1911924
 
其实思路可能不用说也能想出来
设总长度为len,枚举每一个字母,找它后面比它小的字母个数,记为k。那么每一位的答案就是k*(len-i)!。
下面给出草率证明:
先举个栗子,比如说一个字符串BACED。(样例太长懒得解释)
第一位B找到了A一个比它小的字母,说明这个字符串排在所有以A开头的字符串之后(显然共有4!个),所以B对最终答案的贡献是4!。

第二位A没找到比它小的字母,说明这个字符是所有以B开头的字符串中第二位最小的,所以B对最终答案没有贡献。

第三位C没找到比它小的字母,说明这个字符是所有以BA开头的字符串中第二位最小的,所以C对最终答案没有贡献。

第四位E找到了E一个比它小的字母,说明这个字符串排在所有以BACD开头的字符串之后(显然共有1!个),所以E对最终答案的贡献是1!。

第五位D没找到比它小的字母,说明这个字符是所有以BACE开头的字符串中第二位最小的,所以D对最终答案没有贡献。
又因为第一个字符串排名为1,所以所有字符串排名都应加1。
所以答案是1+4!+1!=26;
 
放代码,算是很短的。
 1 #include<bits/stdc++.h>
2 using namespace std;
3
4 int n,f[19];
5 int main(){
6 scanf("%d",&n);
7 f[0]=1;//注意
8 for(int i=1;i<=10;i++) f[i]=f[i-1]*i;//预处理阶乘
9 while(n--)//数据组数
10 {
11 int ans=1;//每个字符串排名都加1,上面有提到
12 string s;
13 cin>>s;
14 int len=s.length();//取字符串长度
15 for(int i=0;i<len;i++)
16 {
17 int k=0;
18 for(int j=i+1;j<len;j++)
19 if(s[i]>s[j]) k++;//记录在当前字符之后有几个比当前字符小的
20 ans+=k*f[len-i-1];//注意i从0开始,所以是len-i-1
21 }
22 printf("%d\n",ans);
23 }
24 return 0;
25 }

但是时间复杂度达到了len2(当然这里len只有11,除了穷举都能过)

但如果是这道题呢?(参考洛谷P5637)

P5367 【模板】康托展开

题目描述

求1∼N的一个给定全排列在所有1∼N全排列中的排名。结果对998244353取模。

输入格式

第一行一个正整数N。

第二行N个正整数,表示1∼N的一种全排列。

输出格式

一行一个非负整数,表示答案对9982443539取模的值。

输入输出样例

输入 #1 

3
2 1 3
输出 #1 

3
输入 #2 

4
1 2 4 3
输出 #2 

2

说明/提示

对于10%数据,1≤N≤10。

对于50%数据,1≤N≤5000。

对于100%数据,1≤N≤1000000

如果用传统的康托展开做,时间复杂度O(n2),只能拿50分。

我们考虑用数据结构维护每个数后面的它小的数的个数,自然而然想到了权值线段树(其实是我不会树状数组)。

线段树的叶子节点维护的是该叶子结点所对应的编号出现的次数(我也说不清楚,具体看代码)。首先插入所有数,随后第i次找1到a[i]-1区间内的和,做完之后删除这个数,防止后面的数重复记录产生错误答案。时间复杂度O(logn)。

上代码:

 #include<bits/stdc++.h>//随时记得取模,尽可能降低WA的概率
#define P 998244353
#define N 1000009
using namespace std; int n,ans=,a[N],sum[N<<];
long long f[N];//这道题内存限制只有31.25MB,第一次全开了long long发现MLE,第二次全开了int爆WA,最后部分开了long long才压内存过
void Update(int rt,int l,int r,int x,int c,int fg)//fg是flag,标记
{
if(l==r && l==x)
{
if(fg==) sum[rt]+=c;//如果是插入,sum就加c
else sum[rt]-=c;//如果是删除,sum就减c
return;
}
int mid=(l+r)>>;
if(x<=mid) Update(rt<<,l,mid,x,c,fg);
else Update(rt<<|,mid+,r,x,c,fg);
sum[rt]=sum[rt<<]+sum[rt<<|];
} int Query(int rt,int l,int r,int x,int y)
{
if(l==x && r==y) return sum[rt];
int mid=(l+r)>>;
if(y<=mid) return Query(rt<<,l,mid,x,y);
else if(x>mid) return Query(rt<<|,mid+,r,x,y);
else return Query(rt<<,l,mid,x,mid)+Query(rt<<|,mid+,r,mid+,y);
}
int main(){
memset(sum,,sizeof(sum));
scanf("%d",&n);
f[]=;//注意
for(int i=;i<=;i++) f[i]=f[i-]%P*i%P;//计算阶乘
for(int i=;i<=n;i++)
{
scanf("%d",&a[i]);
Update(,,n,a[i],,);//将每一个数插入到权值线段树中
}
for(int i=;i<=n;i++)
{
if(a[i]==)//如果a[i]==1,那么该数一定对答案没有贡献,直接删除
{
Update(,,n,a[i],,);
continue;
}
ans=(ans+Query(,,n,,a[i]-)%P*f[n-i]%P)%P;//更新答案
Update(,,n,a[i],,);//删除这个数
}
printf("%d\n",(ans+P)%P);
return ;
}

康托(Cantor)展开的更多相关文章

  1. poj 1077 Eight (八数码问题——A*+cantor展开+奇偶剪枝)

    题目来源: http://poj.org/problem?id=1077 题目大意: 给你一个由1到8和x组成的3*3矩阵,x每次可以上下左右四个方向交换.求一条路径,得到12345678x这样的矩阵 ...

  2. HDU 1027 Ignatius and the Princess II(康托逆展开)

    Ignatius and the Princess II Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K ( ...

  3. 康托展开&&康托逆展开

    康托展开 简介:对于给定的一个排列,求它是第几个,比如54321是n=5时的第120个.(对于不是1~n的排列可以离散化理解) 做法: ans=a[n]*(n-1)!+a[n-1]*(n-2)!+~~ ...

  4. 康托展开&康托逆展开 的写法

    康托展开 康托展开解决的是当前序列在全排序的名次的问题. 例如有五个数字组成的数列:1,2,3,4,5 那么1,2,3,4,5就是全排列的第0个[注意从0开始计数] 1,2,3,5,4就是第1个 1, ...

  5. CDOJ 485 UESTC 485 Game (八数码变形,映射,逆cantor展开)

    题意:八数码,但是转移的方式是转动,一共十二种,有多组询问,初态唯一,终态不唯一. 题解:初态唯一,那么可以预处理出012345678的所有转移情况,然后将初态对012345678做一个映射,再枚举一 ...

  6. [知识点]Cantor展开

    // 此博文为迁移而来,写于2015年3月14日,不代表本人现在的观点与看法.原始地址:http://blog.sina.com.cn/s/blog_6022c4720102vtyo.html 1.含 ...

  7. 分形之康托(Cantor)三分集

    1883年,德国数学家康托(G.Cantor)提出了如今广为人知的三分康托集,或称康托尔集.三分康托集是很容易构造的,然而,它却显示出许多最典型的分形特征.它是从单位区间出发,再由这个区间不断地去掉部 ...

  8. 【数学】康托展开 && 康托逆展开

    (7.15)康托展开,就是把全排列转化为唯一对应自然数的算法.它可以建立1 - n的全排列与[1, n!]之间的自然数的双向映射. 1.康托展开: 尽管我并不清楚康托展开的原理何在,这个算法的过程还是 ...

  9. lightoj1060【康托逆展开】

    可以先看些资料:http://blog.csdn.net/keyboarderqq/article/details/53388936 参考谷巨巨:http://blog.csdn.net/azx736 ...

  10. 康托展开+逆展开(Cantor expension)详解+优化

    康托展开 引入 康托展开(Cantor expansion)用于将排列转换为字典序的索引(逆展开则相反) 百度百科 维基百科 方法 假设我们要求排列 5 2 4 1 3 的字典序索引 逐位处理: 第一 ...

随机推荐

  1. py+selenium 老是定位不到文本内容【已解决】

    问题:定位不到文本内容,路径也正确,该加frame也有加,等待时间也够长 测试: 上图看不出差异,但是测试1就定位得到,测试2就定位不到,为什么? 看下图就知道了 区别就在于,测试2后面多了个空格!! ...

  2. NOIP2018&2013提高组T1暨洛谷P5019 铺设道路

    题目链接:https://www.luogu.org/problemnew/show/P5019 花絮:普及蒟蒻终于A了一道提高的题目?emm,写一篇题解纪念一下吧.求过! 分析: 这道题我们可以采用 ...

  3. Hive调优策略

    Hive调优策略 Fetch抓取 Fetch抓取是指,Hive中对某些情况的查询可以不必使用MapReduce计算. 例如:select * from employee:在这种情况下,Hive可以简单 ...

  4. TensorFlow笔记-模型的保存,恢复,实现线性回归

    模型的保存 tf.train.Saver(var_list=None,max_to_keep=5) •var_list:指定将要保存和还原的变量.它可以作为一个 dict或一个列表传递. •max_t ...

  5. C#3.0新增功能06 对象和集合初始值设定项

    连载目录    [已更新最新开发文章,点击查看详细] 使用 C# 可以在单条语句中实例化对象或集合并执行成员分配. 对象初始值设定项 使用对象初始值设定项,你可以在创建对象时向对象的任何可访问字段或属 ...

  6. [leetcode] 406. Queue Reconstruction by Height (medium)

    原题 思路: 一开始完全没有思路..看了别人的思路才解出来. 先按照他们的高度从高到低(因为我后面用的从前往后遍历插入,当然也可以从低到高)排序,如果高度一样,那么按照k值从小到大排序. 排完序后我们 ...

  7. [leetcode] 110. Balanced Binary Tree (easy)

    原题链接 水题 深度搜索每一节点的左右深度,左右深度差大于1就返回false. class Solution { public: bool isBalanced(TreeNode *root) { b ...

  8. Java简单公式计算器

    最近给公司开发业务代码时,碰到一个场景,简单描述是这样的: 客户要向咱们公司定制一件产品,这个产品呢,有很多属性,那公司得根据这些属性报价呀,怎么报价呢?公司针对某种类型的产品有一个基准价,在同类产品 ...

  9. centos7更新yum库为aliyun库

    1. 备份原来的yum源$sudo cp /etc/yum.repos.d/CentOS-Base.repo /etc/yum.repos.d/CentOS-Base.repo.bak 2.设置ali ...

  10. 【iOS】创建真机调试证书

    今天第一次完整的在一个开发者账号里添加证书,刚接触,还真有些不熟悉,还好找到了一篇不错的文章:iOS开发:创建真机调试证书 ,做了很详细的介绍. 分享一下!!