康托(Cantor)展开
直接进入正题。
康托展开
现在有"ABCDEFGHIJ”10个字符,将其所有的排列中按字典序排列,给出任意一种排列,说出这个排列在所有的排列中是第几小的?
Input
第一行有一个整数n(0<n<=100000);
随后有n行,每行是一个排列;
Output
输出一个整数m,占一行,m表示排列是第几位;
ABCDEFGHIJ
HGEBFACDJI
GDEDHJBXIA
2803322
1911924
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取模的值。
输入输出样例
3
2 1 3
3
4
1 2 4 3
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)展开的更多相关文章
- poj 1077 Eight (八数码问题——A*+cantor展开+奇偶剪枝)
题目来源: http://poj.org/problem?id=1077 题目大意: 给你一个由1到8和x组成的3*3矩阵,x每次可以上下左右四个方向交换.求一条路径,得到12345678x这样的矩阵 ...
- HDU 1027 Ignatius and the Princess II(康托逆展开)
Ignatius and the Princess II Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K ( ...
- 康托展开&&康托逆展开
康托展开 简介:对于给定的一个排列,求它是第几个,比如54321是n=5时的第120个.(对于不是1~n的排列可以离散化理解) 做法: ans=a[n]*(n-1)!+a[n-1]*(n-2)!+~~ ...
- 康托展开&康托逆展开 的写法
康托展开 康托展开解决的是当前序列在全排序的名次的问题. 例如有五个数字组成的数列:1,2,3,4,5 那么1,2,3,4,5就是全排列的第0个[注意从0开始计数] 1,2,3,5,4就是第1个 1, ...
- CDOJ 485 UESTC 485 Game (八数码变形,映射,逆cantor展开)
题意:八数码,但是转移的方式是转动,一共十二种,有多组询问,初态唯一,终态不唯一. 题解:初态唯一,那么可以预处理出012345678的所有转移情况,然后将初态对012345678做一个映射,再枚举一 ...
- [知识点]Cantor展开
// 此博文为迁移而来,写于2015年3月14日,不代表本人现在的观点与看法.原始地址:http://blog.sina.com.cn/s/blog_6022c4720102vtyo.html 1.含 ...
- 分形之康托(Cantor)三分集
1883年,德国数学家康托(G.Cantor)提出了如今广为人知的三分康托集,或称康托尔集.三分康托集是很容易构造的,然而,它却显示出许多最典型的分形特征.它是从单位区间出发,再由这个区间不断地去掉部 ...
- 【数学】康托展开 && 康托逆展开
(7.15)康托展开,就是把全排列转化为唯一对应自然数的算法.它可以建立1 - n的全排列与[1, n!]之间的自然数的双向映射. 1.康托展开: 尽管我并不清楚康托展开的原理何在,这个算法的过程还是 ...
- lightoj1060【康托逆展开】
可以先看些资料:http://blog.csdn.net/keyboarderqq/article/details/53388936 参考谷巨巨:http://blog.csdn.net/azx736 ...
- 康托展开+逆展开(Cantor expension)详解+优化
康托展开 引入 康托展开(Cantor expansion)用于将排列转换为字典序的索引(逆展开则相反) 百度百科 维基百科 方法 假设我们要求排列 5 2 4 1 3 的字典序索引 逐位处理: 第一 ...
随机推荐
- Java底层技术系列文章-线程池框架
一.线程池结构图 二.示例 定义线程接口 public class MyThread extends Thread { @Override publicvoid run() { System.o ...
- CAD2014学习笔记-文字编辑与尺寸标注
基于 虎课网huke88.com CAD教程 文字与表格 输入文字:TEXT.MTEXT 插入表格:table 新建表格样式 尺寸标注 测量工具:Di.DLI 开启标注:打开工具-工具栏-标注 对齐/ ...
- css基础4
今天是2019年6月21日,周五了.在这里写上一篇随笔,主要内容是css基础中的一些细节部分,话不多说,直接上! 一.背景渐变: background-image 线性渐变:linear-gradie ...
- Golang 高效实践之defer、panic、recover实践
前言 我们知道Golang处理异常是用error返回的方式,然后调用方根据error的值走不同的处理逻辑.但是,如果程序触发其他的严重异常,比如说数组越界,程序就要直接崩溃.Golang有没有一种异常 ...
- 数据结构-哈夫曼树(python实现)
好,前面我们介绍了一般二叉树.完全二叉树.满二叉树,这篇文章呢,我们要介绍的是哈夫曼树. 哈夫曼树也叫最优二叉树,与哈夫曼树相关的概念还有哈夫曼编码,这两者其实是相同的.哈夫曼编码是哈夫曼在1952年 ...
- Visual Studio 2015 WinForm应用程序打包教程
最近开发了一个小软件.由于需要打包.网上找了一些资料.然后整合了起来.希望对大家有所帮助.不全面请见谅. 本人开发工具用的是Visual Studio 2015 打包控件 InstallShield ...
- 《VR入门系列教程》之6---VR硬件介绍及DK1
第二章 VR硬件介绍 本章主要介绍当前比较流行的消费版VR设备,包括VR头显以及应用运行的PC和手机平台. 即使是在这工业高速发展的时代,一些大厂(比如Facebook的Oculus ...
- 基础篇-1.2Java世界的规章制度(下)
1 Java运算符 Java世界中的运算其实就是数学运算,而运算符就是其中的媒介. 算术运算符 操作符 描述 + 加法,对符号两边的数值相加 - 减法,符号左边的数减去右边的数 * 乘法,符号两边的数 ...
- Servlet的介绍
Servlet由来 做过BS项目的人都知道,浏览器能够根据HTML静态标记语言来显示各式各样的网页.但是如果我们需要在网页上完成一些业务逻辑:比如登陆验证.或者说网页显示的内容在服务器的数据库中.如果 ...
- TestNG中DataProvider的用法一
目录 为什么要使用DataProvider DataProvider的常规用法 带Method参数的DataProvider 带ITestContext的DataProvider DataProvid ...