复制代码 代码如下:
<P>不论是哪种全排列生成算法,都遵循着“原排列”→“原中介数”→“新中介数”→“新排列”的过程。</P><P>其中中介数依据算法的不同会的到递增进位制数和递减进位制数。</P><P>关于排列和中介数的一一对应性的证明我们不做讨论,这里仅仅给出了排列和中介数的详细映射方法。</P>
· 递增进位制和递减进位制数 所谓递增进位制和递减进位制数字是指数字的进制随着数字位置的不同递增或递减。通常我们见到的都是固定进制数字,如
2进制,10进制等。m位n进制数可以表示的数字是m*n个。而m位递增或递减进位制数则可以表示数字m!个。例如递增进位制数4121,它的进制从右向左依次是2、3、4、5。即其最高位(就是数字4那位)最大值可能是4;第三高位最大可能是3;第二高位最大可能是2;最末位最大可能是1。如果将4121加上1的话,会使最末位得到0,同时进位;第二位的2与进位相加,也会得到0,同时进位;第三位的1与进位相加得到2,不再进位。最终得到结果是4200。递减进位制的道理是一样的,只不过进制从右向左依次是9、8、7、6……,正好与递增进位制相反。很明显,递减进位制的一个最大的好处就是加法不易进位,因为它在进行加法最频繁的末几位里(最右边)进制比较大。
接下来要了解的是递增进位制、递减进位制数和其序号的关系。递增、递减进位制数可以被看作一个有序的数字集合。如果规定递增进位制和递减进位制数的0的序号是十进制0,递增进位制数的987654321和递减进位制数的123456789对应十进制序号362880(即9!),则可以整理一套对应法则。其中,递增进位制数(a1 a2 a3 a4 a5 a6 a7 a8 a9)为:
a1*9! + a2*8! + ….+ a8*2! + a9*1! = 序号
例如序号100的递增进位制数就是4020,即4*4!+ 0*3!+ 2*2!+ 0*1!=100。将一个序号转换成其递增进位制数首先需要找到一个比序号小的最大阶乘数(即1、2、6、24、120、720……),对其进行整数除得到递增进位制的第一位;将除法的余数反复应用这个方法(当然,之后选择的余数是小一级的阶乘数),直到余数为0。
递减进位制数(a1 a2 a3 a4 a5 a6 a7 a8 a9)为:
(((((((((a1 * 1 + a2) * 2 + a3) * 3 + …… + a7) * 8 + a8) * 9 + a9= 序号
例如序号100的递减进位制数就是131(a7 a8 a9, 即从后对齐),即 (1*8 + 3)*9 + 1 = 100。将一个序号转换成其递减进位制数,需要对序号用9取余数,就可以得到递减进位制的最末位(这点和递增进位制先算出最高位相反)。用余下的数的整数除结果重复此过程(当然,依次对8、7、6……取余),直到余数为0。
关于递增进位制和递减进位制需要注意的重点:一是其加减法的进位需要小心;二是序号和数字的转换。除了100之外,常见的转换有:999的递增数是121211,递减数是1670;99的递增数是4011,递减数是130。大家可以以此为参考测试自己是否真正理解了计算的方法。下文将省略递增进位制或递减进位制的详细计算过程。
从现在开始我们将详细介绍六种排列生成算法。具体的理论介绍将被忽略,下文所注重的就是如何将排列映射为中介数以及如何将中介数还原为排列。
我全部以求839647521的下100个排列为例。
· 递增进位排列生成算法 映射方法:
将原排列按照从9到2的顺序,依次查看其右侧比其小的数字的个数。这个个数就是中介数的一位。例如对于原排列839647521。9的右侧比9小的数字有6个,8的右侧比8小的数字有7个,7的右侧比7小的数字有3个,……2的右侧比2小的数字有1个。最后得到递增进制中介数67342221。(此中介数加上
100的递增进制数4020得到新的中介数67351311)
还原方法:我们设新中介数的位置号从左向右依次是9、8、7、6、5、4、3、2。在还原前,画9个空格。对于每一个在位置x的中介数y,从空格的右侧向左数y个未被占用的空格。在第y+1个未占用的空格中填上数字x。重复这个过程直到中介数中所有的位都被数完。最后在余下的最后一个空格里填上1,完成新排列的生成。以新中介数67351311为例,我给出了详细的恢复步骤。其中红色数字代表新填上的数字。最后得到新排列869427351。
复制代码 代码如下:
void next_Permutations_by_increDecimal(int dataArr[],int size){
int i;
int *resultArr = new int[size];
int index = 0;
map<int,int>::iterator iter;
//第一步 求出中介数
//由大到小,得到并记录当前排列中,数字i的右边比其小的数的个数
map<int,int> agentMap;
for(i=0; i<size; ++i){
agentMap.insert(valType(dataArr[i],count(dataArr,i,size,dataArr[i])));
}
qsort(dataArr,0,size-1);
//第二步 得到新的中介数,在旧的中介数的基础上,根据递增进位制数法加1
while (true){
++countNum;
next_inter_num(dataArr,agentMap);
//第三步 根据新的中介数得到新的排列
index = size -1;
//清空记录当前排列的数组,以存放新产生的排列
for(i=0; i<size; ++i){
resultArr[i] = 0;
}
while(true){
iter = agentMap.find(dataArr[index]);
valType value = *iter;
resultArr[getNextPosition(resultArr,size,value.second,0)] = dataArr[index];
--index;
if(index == 0) break;
}
//将最后一个空位置为最小数
i = 0;
while(true){
if(resultArr[i] != 0){
++i;
}else{
resultArr[i] = dataArr[index];
break;
}
}
print(resultArr,size);
bool flag = true;
for(i=1; i<size; ++i){
if(resultArr[i] > resultArr[i-1]){
flag = false;
break;
}
}
if(flag) break;
}
delete [] resultArr;
}
void next_inter_num(int dataArr[],map<int,int>& agentMap){
map<int,int>::iterator iter;
//temp当前位需要增加得值,tmpResult为temp与当前位的值之和,start为末位开始的进制
int start = 2,temp=1,tmpResult;
int index = 1; //数组中的第一个数位最小数
while(true){
iter = agentMap.find(dataArr[index]);
valType value = *iter;
tmpResult = value.second + temp;
if(tmpResult < start){
//已经不产生进位
agentMap.erase(dataArr[index]);
agentMap.insert(valType(dataArr[index],tmpResult));
break;
}else{
agentMap.erase(dataArr[index]);
agentMap.insert(valType(dataArr[index],tmpResult % start));
temp = tmpResult / start;
++start;
}
++index;
}
}
· 递减进位排列生成算法 映射方法:递减进位制的映射方法刚好和递增进位制相反,即按照从
9到2的顺序,依次查看其右侧比其小数字的个数。但是,生成中介数的顺序不再是从左向右,而是从右向左。生成的递减进制中介数刚好是递增进位排列生成算法得到中介数的镜像。例如
839647521的递减进制中介数就是12224376。(此中介数加上100的递减进制数131后得到新中介数12224527)
还原方法:递减进位制中介数的还原方法也刚好和递增进位制中介数相反。递增进位制还原方法是按照从中介数最高位(左侧)到最低位(右侧)的顺序来填数。而递减仅位置还原方法则从中介数的最低位向最高位进行依次计数填空。例如对于新中介数12224527,我给出了详细的还原步骤。红色代表当前正在填充的空格。最终得到新排列397645821。
C++实现代码:
复制代码 代码如下:
void next_Permutations_by_DecreDecimal(int dataArr[],int size){
//创建一个结果数组,用来记录下一个排列
int *resultArr = new int[size];
int i;
//第一步 求出中介数
map<int,int> agentMap;
for(i=0; i<size; ++i){
int nums = count(dataArr,i,size,dataArr[i]);
agentMap.insert(valType(dataArr[i],nums));
}
//第二步 求新的中介数 此处最低位进制最高,故不会频繁产生进位,性能应该优于递增进位
//最低位进制为9,向前依次递减
int start = size,temp = 1;
int tmpResult;
int index = size-1;//中介数末位数位数字序列中最大的数右边比其小的数
map<int,int>::iterator iter;
qsort(dataArr,0,size-1);
while (true){
++countNum; //全局变量 记录排列个数
next_inter_num(dataArr,agentMap,size);
index = size -1;
//第三步 根据产生的中介数得到新的排列
for(i=0; i<size; ++i){
resultArr[i] = 0;
}
while(true){
iter = agentMap.find(dataArr[index]);
valType value = *iter;
//找到下一个填充空位
resultArr[getNextPosition(resultArr,size,value.second,0)] = dataArr[index];
--index;
if(index == 0) break;
}
i = 0;
while(true){
if(resultArr[i] != 0){
++i;
}else{
resultArr[i] = dataArr[index];
break;
}
}
print(resultArr,size);
bool flag = true;
for(i=1; i<size; ++i){
if(resultArr[i] > resultArr[i-1]){
flag = false;
break;
}
}
if(flag) break;
}
delete [] resultArr;
}
void next_inter_num(int dataArr[],map<int,int> &agentMap,int size){
int start = size,temp = 1;
int tmpResult;
int index = size-1;//中介数末位数位数字序列中最大的数右边比其小的数
map<int,int>::iterator iter;
while(true){
iter = agentMap.find(dataArr[index]);
valType value = *iter;
tmpResult = value.second + temp;
if(tmpResult < start){
//没有产生进位,直接改写末位值
agentMap.erase(dataArr[index]);
agentMap.insert(valType(dataArr[index],tmpResult));
break;
}else{
//产生进位
agentMap.erase(dataArr[index]);
agentMap.insert(valType(dataArr[index],tmpResult % start));
tmpResult = tmpResult / start;
--start;
}
--index;
}
}
· 字典全排列生成法
映射方法:将原排列数字从左到右(最末尾不用理会),依次查看数字右侧比其小的数字有几个,个数就是中介数的一位。例如,对于排列839647521。最高位8右侧比8小的有7个数字,次高位3右侧比3小的数字有2个,再次位的9的右侧比9小的有6个数字,……,2的右侧比2小的数有1个。得到递增进制中介数72642321。(此中介数加上100的递增进进制数4020后得到新中介数72652011)
还原方法:还原方法为映射方法的逆过程。你可以先写出辅助数字1 2 3 4 5 6 7 8 9,以及9个空位用于填充新排列。然后从新中介数的最高位数起。例如新中介数最高位是x,你就可以从辅助数字的第一个没有被使用的数字开始数起,数到x。第x+1个数字就应当是空位的第一个数字。我们将此数字标为“已用”,然后用其填充左起第一个空位。然后,再看新中介数的次高位y,从辅助数字的第一个未用数字数起,数到一。第y+1个数字就是下一个空位的数字。我们将此数字标为“已用”,然后用其填充左起第二个空位。依此类推,直到最后一个中介数中的数字被数完为止。例如对于新中介数72652011,我们给出其辅助数字和空位的每一步的情况。其中红色的数字代表“正在标记为已用”,“已用”的数字不会再被算在之后的计数当中。当新中介数中所有的数字都被数完了,辅助数字中剩下的唯一数字将被填入最后的空位中。最终得到新排列839741562。
C++实现:
复制代码 代码如下:
void next_Permutations_by_DicOrder(int dataArr[],int size){
int key = 0;
int index,temp,end,left,right;
int i;
bool flag ;
while(true){
++countNum;
print(dataArr,size);
flag = true;
index = 0,temp = 0,end=8,left = right = 0;
//从当前排列末尾向前找到第一次出现下降的点
for(i = size-1; i > 0; i--){
if(dataArr[i] > dataArr[i-1]){
key = i-1; //K记录下降的点
flag = false;
break;
}
}
//如果已经是由高到低有序,则操作完成
if(flag)
break;
index = key + 1;
//找到后缀中比第一次下降点的数大的数中最小的数
while(dataArr[key] < dataArr[index] && index < size){
++index;
}
index --;
//将找到的数和第一次出现下降的点交换
temp = dataArr[key];
dataArr[key] = dataArr[index];
dataArr[index] = temp;
left = key+1;
right = size - 1;
//将下降点后面的数逆转
while(left < right){
temp = dataArr[left];
dataArr[left] = dataArr[right];
dataArr[right] = temp;
++left;
--right;
}
}
}
回溯法:
复制代码 代码如下:
void next_Permutations_by_backtrack(int dataArr[],int size){
//创建结果数组
int *resultArr = new int[size+1];
backTrack(1,size+1,resultArr,dataArr);
delete [] resultArr;
}
//剪枝函数
bool place(int k,int resultArr[])
{
for (int j = 1; j < k; j++) {
if (resultArr[j] == resultArr[k]) {
return false;
}
}
return true;
}
void backTrack(int t,int size,int resultArr[],int dataArr[])
{
if (t > size-1) {
++ countNum;
for(int i = 1; i < size; i++) {
cout << resultArr[i] << " ";
}
cout << endl;
} else {
for(int i = 1; i < size; i++) {
resultArr[t] = dataArr[i-1];
if (place(t,resultArr)) {
backTrack(t+1,size,resultArr,dataArr);
}
}
}
}
相关推荐:
文章AI思维导图自动生成助力创作的智慧之源
SEO查:如何通过精准优化让网站流量飞速增长,吉林推广营销怎么样
ChatGPT进不去怎么办?解决方案与技巧,轻松畅享智能对话,ai va
ChatGPT当前不可用?如何应对AI服务中断的挑战,ai文章免费写作app
SEO师:如何在数字时代为企业赢得流量与销量,十堰网站关键词优化教程
ChatGPT出现错误503?你需要知道的解决方案和应对策略,ai拉伸字效
什么是seo手段,seo的具体手段 ,中国ai 企业
AI缩短短文-提升创作效率,写作新体验,光速写作业ai写作app
专业关键词助力SEO优化,让你的内容脱颖而出,东营响应式网站优化
SEO希望:如何通过SEO优化实现网站突破,迈向成功之路,品牌网站推广软件
SEO要好,网站流量翻倍的关键秘诀,品牌网站建设关键词优化
SEO搜索关键词是什么意思?让你轻松网站流量的秘密!,ai颜色不对
AI写作免费一键生成下载,助您轻松创作!
好用的AI写作软件,让创作更高效
为什么要做seo si,为什么要做* ,ai里面怎么扣二维码
SEO数量-如何提升你的网站排名与流量?,射阳seo优化五星服务
URL泄露:如何防止信息泄露带来的严重后果,抖音推广营销服务多少钱
优化标题:如何让你的文章更具吸引力与点击力,整站网站优化解决方案
seo都有什么问题,seo都有什么问题和答案 ,ai凯旋公主下载
在线AI文章生成:内容创作新革命
打破创作边界,无限可能无限制生成文章的AI
ChatGPT页面不自动显示最新消息:如何解决这一困扰,提升使用体验?,斑马ai幼儿百度云网盘
SEO获取流量的必杀技:如何通过优化轻松提升网站排名,德州全网营销推广价格
什么是SEO可以自学吗,seo零基础可以自学吗 ,ai1紫
人工润色:让文字更具魅力的秘密武器,快排seo软件哪个最好
ChatGPT4网页空白:重新定义智能交互的未来,怎么登ai
BingAdapter设置数据后没有显示数据?解决方案在这里!,ai中如何把图变形
SEO自从上线后的演变与未来发展趋势,拼多多增加关键词排名
OpenAI注册问题解决方案:如何顺利通过手机号验证,反对ai头像
ChatGPT中文版下载,开启智能对话新体验,婚纱ai男
ChatGPT打不开了吗?如何快速解决常见问题,恢复顺畅体验!,ai13212511845
ChatGPT无法加载?检查您的网络设置并尝试重启Chat,ai写作怎么写作文的
SEO表格:优化网站排名的秘密武器,大数据推广营销费用多少
文章AI生成软件高效创作新纪元
seo网页优化什么意思,seo网站优化必知的10个问答,问吧,【解决】百度不知道 ,ai识别点读机
ChatGPT宕机两小时,OpenAI紧急修复,用户期待AI恢复正常服务,oppo小布ai
AI免费生成文字,打造创作新时代
SEO很多,如何在竞争激烈的市场中脱颖而出?,在SEO优化中
SEO优化:提升网站流量的终极指南,揭阳网站快速推广
SEO优化快:如何在短时间内实现网站流量爆发,Lacey0508ai
AI免费生成:释放创造力的秘密武器
SEO与SEM:谁才是提升网站流量的王者?,Ai中字体如何变形立体
ChatGPT回答问题,网页无法线下滚动?解决方案轻松get!,谷歌AI铃声
SEO分类中内部优化的有:让你的网站脱颖而出!,by ai
SEO开发:数字营销的核心驱动力,园区网站建设
ChatGPT+维护页面:您的智能助手之旅,安全、高效、无忧,奥特曼画图ai
亚马逊产品seo什么意思,亚马逊平台产品专业术语 ,小新同学ai
seo需要什么器械,seo需要什么器械才能做 ,长续航ai纯电汽车
什么是seo艺术,什么是seo seo有何价值 ,ai写作生成器 推荐
SEO关键词推广软件官网-助力企业实现高效精准的网络营销,圈圈ai