不论是哪种全排列生成算法,都遵循着“原排列”→“原中介数”→“新中介数”→“新排列”的过程。其中中介数依据算法的不同会的到递增进位制数和递减进位制数。关于排列和中介数的一一对应性的证明我们不做讨论,这里仅仅给出了排列和中介数的详细映射方法。相信熟练掌握了方法就可以顺利通过这部分的考察。
所谓递增进位制和递减进位制数字是指数字的进制随着数字位置的不同递增或递减。通常我们见到的都是固定进制数字,如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,即 (1*8 + 3)*9 + 1 = 100。将一个序号转换成其递减进位制数,需要对序号用9取余数,就可以得到递减进位制的最末位(这点和递增进位制先算出最高位相反)。用余下的数的整数除结果重复此过程(当然,依次对8、7、6……取余),直到余数为0。
关于递增进位制和递减进位制需要注意的重点:一是其加减法的进位需要小心;二是序号和数字的转换。除了100之外,常见的转换有:999的递增数是121211,递减数是1670;99的递增数是4011,递减数是130。大家可以以此为参考测试自己是否真正理解了计算的方法。下文将省略递增进位制或递减进位制的详细计算过程。
从现在开始我们将详细介绍六种排列生成算法。具体的理论介绍将被忽略,下文所注重的就是如何将排列映射为中介数以及如何将中介数还原为排列。我全部以求839647521的下100个排列为例。
映射方法:将原排列数字从左到右(最末尾不用理会),依次查看数字右侧比其小的数字有几个,个数就是中介数的一位。例如,对于排列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。
新中介数当前位 |
辅助数字 |
新排列数 |
初始 |
1 2 3 4 5 6 7 8 9 |
|
7 |
1 2 3 4 5 6 7 8 9 |
8 |
2 |
1 2 3 4 5 6 7 8 9 |
8 3 |
6 |
1 2 3 4 5 6 7 8 9 |
8 3 9 |
5 |
1 2 3 4 5 6 7 8 9 |
8 3 9 7 |
2 |
1 2 3 4 5 6 7 8 9 |
8 3 9 7 4 |
0 |
1 2 3 4 5 6 7 8 9 |
8 3 9 7 4 1 |
1 |
1 2 3 4 5 6 7 8 9 |
8 3 9 7 4 1 5 |
1 |
1 2 3 4 5 6 7 8 9 |
8 3 9 7 4 1 5 6 |
最后补位 |
|
8 3 9 7 4 1 5 6 2 |
映射方法:将原排列按照从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。
新中介数当前位 |
新排列数 |
备注 |
初始 |
|
|
6 |
9 |
从右向左数6个空格,第7个空格里填“9” |
7 |
8 9 |
从右向左数7个空格,第8个空格里填“8” |
3 |
8 9 7 |
从右向左数3个空格,第4个空格里填“7” |
5 |
8 6 9 7 |
从右向左数5个空格,第6个空格里填“6” |
1 |
8 6 9 7 5 |
从右向左数1个空格,第2个空格里填“5” |
3 |
8 6 9 4 7 5 |
从右向左数3个空格,第4个空格里填“4” |
1 |
8 6 9 4 7 3 5 |
从右向左数1个空格,第2个空格里填“3” |
1 |
8 6 9 4 2 7 3 5 |
从右向左数1个空格,第2个空格里填“2” |
最后补位 |
8 6 9 4 2 7 3 5 1 |
余下的空格填“1” |
映射方法:递减进位制的映射方法刚好和递增进位制相反,即按照从9到2的顺序,依次查看其右侧比其小数字的个数。但是,生成中介数的顺序不再是从左向右,而是从右向左。生成的递减进制中介数刚好是递增进位排列生成算法得到中介数的镜像。例如839647521的递减进制中介数就是12224376。(此中介数加上100的递减进制数131后得到新中介数12224527)
还原方法:递减进位制中介数的还原方法也刚好和递增进位制中介数相反。递增进位制还原方法是按照从中介数最高位(左侧)到最低位(右侧)的顺序来填数。而递减仅位置还原方法则从中介数的最低位向最高位进行依次计数填空。例如对于新中介数12224527,我给出了详细的还原步骤。红色代表当前正在填充的空格。最终得到新排列397645821。
新中介数当前位 |
新排列数 |
备注 |
初始 |
|
|
7 |
9 |
从右向左数7个空格,第8个空格里填“9” |
2 |
9 8 |
从右向左数2个空格,第3个空格里填“8” |
5 |
9 7 8 |
从右向左数5个空格,第6个空格里填“7” |
4 |
9 7 6 8 |
从右向左数4个空格,第5个空格里填“6” |
2 |
9 7 6 5 8 |
从右向左数2个空格,第3个空格里填“5” |
2 |
9 7 6 4 5 8 |
从右向左数2个空格,第3个空格里填“4” |
2 |
3 9 7 6 4 5 8 |
从右向左数2个空格,第3个空格里填“3” |
1 |
3 9 7 6 4 5 8 2 |
从右向左数1个空格,第2个空格里填“2” |
最后补位 |
3 9 7 6 4 5 8 2 1 |
余下的空格填“1” |
映射方法:循环左移排列生成算法与递减进位排列生成算法非常相似,同样是在原始排列中找到比当前数字小的数字个数。只不过循环左移使用的是左循环搜索法,而不是递减进位法使用的右侧搜索。所谓左循环搜索法是指从起始数字开始向左搜索,如果到了左边界还没有发现终止数字,则从右边界开始继续寻找,直到终止数字被发现。图中展示了839647521种以开始数字为6,结束数字为5和开始数字为7,结束数字为6的左循环搜索区间,注意开始和结束数字是不包括在区间内的。
具体到循环左移的排列生成法得映射方法,就是要为每一个数字确定这样一个区间。方法是以从9到2的顺序依次产生中介数中的数字,中介数从右向左产生(即原排列的9产生的数字放到中介数右数第一位,8产生的数字放到中介数右数第二位,以此类推。这样才能得到递减进位制数)。对于9,产生的中介数字依然是9的右边比9小的数字的个数。但是从8开始一直到2中的每一个数i,都是要做一个以i+1为开始数字,i为结束数字的左循环区间,并看这个左循环区间中比i小的数字的个数。例如对于排列839647521,9的右侧比9小的数字有6个;9到8的左循环区间比8小的数字有1个;8到7的左循环区间比7小的数字有3个;7到6的左循环区间比6小的数字有1个(如上面图下半部所示);6到5的左循环区间比5小的右3个数字(如上图上半部分所示);……;3到2的左循环区间里比2小的数字有1个。所以得到递减中介数10031316(此中结束加上100的递减进制数131得新中介数10031447)
还原方法:循环左移的还原方法很自然。首先画9个空格。当拿到新的中介数后,从中介数的最右边向左边开始计数逐个计数。计数的方法是,对于中介数最右边的数x9,从空格的最右端起数x9个空格,在第x9+1个空格上填上9。然后对于中介数的每一位xi,沿着左循环的方向数xi个未占用空格,在第xi+1个空格里填上i,即可完成还原。我给出了将中介数10031447还原的步骤,其中底纹代表为了定位下一个数字而数过的空格。红色代表被填充的空格。最终得到结果592138647。
新中介数当前位 |
新排列数 |
备注 |
初始 |
|
|
7 |
9 |
从右向左数7个空格,第8个空格提填“9” |
4 |
9 8 |
从9开始左循环数4个空格,第5个空格提填“8” |
4 |
9 8 7 |
从8开始左循环数4个空格,第5个空格提填“7” |
1 |
9 8 6 7 |
从7开始左循环数1个空格,第2个空格提填“6” |
3 |
5 9 8 6 7 |
从6开始左循环数3个空格,第4个空格提填“5” |
0 |
5 9 8 6 4 7 |
从5开始左循环数0个空格,第1个空格提填“4” |
0 |
5 9 3 8 6 4 7 |
从4开始左循环数0个空格,第1个空格提填“3” |
1 |
5 9 2 3 8 6 4 7 |
从3开始左循环数1个空格,第2个空格提填“2” |
最后补位 |
5 9 2 1 3 8 6 4 7 |
余下的空格填“1” |
邻位对换法对连续生成排列作了优化,即通过保存数字的“方向性”来快速得到下一个排列。然而当任意给定一个排列数,想恢复每个数字的方向性相对比较麻烦。邻位对换法的关键就在于方向性。
映射方法:我们需要确定每一个数字的方向性,从2开始。同时,设定b2b3b4b5b6b7b8b9为我们要求的中介数。2的方向一定是向左(关于向左原因的讨论这里省略)。b2就是从2开始,背向2的方向直到排列的边界比2小的数字。知道了b2的值之后就可以依次推导出b3b4……直到b9的值,规则如下:对于每一个大于2的数字i,如果i为奇数,其方向性决定于bi-1的奇偶性,奇向右、偶向左。如果i为偶数,其方向性决定于bi-1+bi-2的奇偶性,同样是奇向右、偶向左。当得到方向性后,bi的值就是背向i的方向直到排列边界这个区间里比i小的数字的个数。如此就能得到邻位对换法的中结束。例如对于排列839647521:
2的方向向左,背向2的方向中比2小的数字有1个,b2=1
3的方向由b2为奇可以断定向右,背向3的方向中比3小的数字有0个,b3=0
4的方向由b2+b3为奇可以断定向右,背向4的方向中比4小的数字有1个,b4=1
5的方向由b4为奇可以断定向右,背向5的方向中比5小的数字有2个,b5=2
6的方向由b4+b5为奇可以断定向右,背向6的方向中比6小的数字有1个,b6=1
7的方向由b6为奇可以断定向右,背向7的方向中比7小的数字有3个,b7=3
8的方向由b6+b7为偶可以断定向左,背向8的方向中比8小的数字有7个,b8=7
9的方向由b8为奇可以断定向右,背向9的方向中比9小的数字有2个,b9=2
最终得到中介数10121372(此中介数加上100的递减数131后得到新的中介数10121523)
还原方法:还原方法完全为映射方法的逆过程。当我们知道了b2b3b4b5b6b7b8b9,自然而然就可以得到每个数字的方向性,具体规则同上。我们先画9个空格,以从9到2的填充顺序依次计算他们的位置。每个数字数格的方向就是那个数字的方向。例如如果一个数字i的方向性是向右,则从空格的左侧起向右数bi个未占用的空格,在第bi+1个未占用空格里填上i。例如对于中介数10121523,还原步骤如下:
9的方向由b8为偶知道向左,从空格右起向左数3个空格,在第4个空格填上9。 _ _ _ _ _ 9 _ _ _ ←
8的方向由b6+b7为偶知道向左,从空格左起数2个空格,在第3个空格上填上8。 _ _ _ _ _ 9 8 _ _ ←
7的方向由b6为奇知道向右,从空格左起向右数5个空格,在第6个空格填上7。→ _ _ _ _ _ 9 8 7 _
6的方向由b4+b5为奇知道向右,从空格左起数1个空格,在第2个空格上填上6。→_ 6 _ _ _ 9 8 7 _
5的方向由b4为奇知道向右,从空格左起向右数2个空格,在第3个空格填上5。→ _ 6 _ 5 _ 9 8 7 _
4的方向由b2+b3为奇知道向右,从空格左起数1个空格,在第2个空格上填上4。→ _ 6 4 5 _ 9 8 7 _
3的方向由b2为奇知道向右,从空格左起向右数0个空格,在第1个空格填上3。→ 3 6 4 5 _ 9 8 7 _
2的方向必为向左,从空格右起向左数1个空格,再第2个空格上填上2。3 6 4 5 2 9 8 7 _ ←
最后补上1,得到最终结果364529871。
循环左右移法结合了循环左移的循环区间思想和邻位对换的数字方向性思想。在循环左右移全排列生成算法当中,也是要首先确定数字的方向性。数字的方向性决定了搜索区间到底是左循环区间还是右循环区间。
映射方法:我们需要确定每一个数字的方向性,从2开始。同时,设定b2b3b4b5b6b7b8b9为我们要求的中介数。对于每一个小于9的数i,如果i为奇数,其方向性决定于bi-1的奇偶性,奇向右、偶向左。如果i为偶数,其方向性决定于i-1+bi-2的奇偶性,同样是奇向右、偶向左(当i=2时,方向一定向左)。确定方向性后,就要构造一个以i开始,以i+1为结束的循环区间,此循环区间的方向是背向i的方向。在此区间里比i小的数字个数就是bi的值。但到计算b9时是没法构造循环空间的,b9的值就是背向9的方向到排列数字边界之间比9小的数字的个数(在此退化为邻位对换法)。例如对于排列839647521:
2的方向向左,背向2的方向到3的循环区间(1 8)里向中比2小的数字有1个,b2=1
3的方向由b2为奇可以断定向右,背向3的方向到4的循环区间(8 1 2 5 7)中比3小的数字有2个,b3=2
4的方向由b2+b3为奇可以断定向右,背向4的方向到5的循环区间(6 9 3 8 1 2)中比4小的数字有3个,b4=3
5的方向由b4为奇可以断定向右,背向5的方向到6的循环区间(7 4)中比5小的数字有1个,b5=1
6的方向由b4+b5为偶可以断定向左,背向6的方向到7的循环区间(4)中比6小的数字有1个,b6=1
7的方向由b6为奇可以断定向右,背向7的方向到8的循环区间(4 6 9 3)中比7小的数字有3个,b7=3
8的方向由b6+b7为偶可以断定向左,背向8的方向到9的循环区间(3)中比8小的数字有1个,b8=1
9的方向由b8为奇可以断定向右,背向9的方向到排列边界(3 8)中比9小的数字有2个,b9=2
最后得到中介数12311312(此中介数加上100的递减数131得到新的中介数12311443)。
还原方法:当我们知道了b2b3b4b5b6b7b8b9,自然而然就可以得到每个数字的方向性,具体规则同上。我们先画9个空格,以从9到2的填充顺序依次计算他们的位置。数格的方向除了9是从排列边界沿着9的方向之外,其它的数字都是以上一个数字开始,沿着此数字的方向对未占用空格的格数进行循环计数。例如对于新中介数12311443,还原步骤如下:
9的方向由b8为偶知道向左,从空格右起向左数3个空格,在第4个空格填上9。 _ _ _ _ _ 9 _ _ _ ←
8的方向由b6+b7为奇知道向右,从9所在位置向右循环数4个空格,在第5个空格上填上8。→ _ 8 _ _ _ 9 _ _ _
7的方向由b6为奇知道向右,从8所在位置向右循环数4个空格,在第5个空格填上7。→ _ 8 _ _ _ 9 _ 7 _
6的方向由b4+b5为偶知道向左,从7所在位置向左循环数1个空格,在第2个空格上填上6。_ 8 _ _ 6 9 _ 7 _ ←
5的方向由b4为奇知道向右,从6所在位置向右循环数1个空格,在第2个空格填上5。→ _ 8 _ _ 6 9 _ 7 5
4的方向由b2+b3为奇知道向右,从5所在位置向右循环数3个空格,在第4个空格上填上4。→ _ 8 _ _ 6 9 4 7 5
3的方向由b2为奇知道向右,从4所在位置向右循环数2个空格,在第3个空格填上3。→ _ 8 _ 3 6 9 4 7 5
2的方向必为向左,从3所在位置向左循环数1个空格,再第2个空格上填上2。2 8 _ 3 6 9 4 7 5 ←
最后补上1,得到最终结果281369475。
我们从这六种全排列生成算法可以看出这些方法实际上都是将排列数映射到递增/递减进位制的中介数,只是映射的策略略有不同而已。同时,不难发现很多映射和还原方法的细则都是硬性规定的(例如,求每一个数右侧比这个数小的数字个数,而不是左侧;循环左移,循环左右移采用递减进位制数,而不是递增进位制数,作为中介数等)实际上,这些规定仅仅与我们的考试相符而已。不同的组合可以衍生出许许多多不同的算法。在任何一个关于排列生成算法的综述上都会有超过50种的算法。此外,可以看到像邻位对换和循环左右移这样的算法特别适合连续生成排列数,但在生成指定序号的排列上比较复杂;但递增/递减进位法则无法快速的连续生成排列数。这样,算法的不同优势造就了其不同的适用环境。最后希望大家能够通过了解这6种算法来提高自己结合现有思想,创造新方法的能力和经验。