把数列\((x_1,x_2,\cdots,x_n)\)变换顺序为\((x_{p(1)},x_{p(2)},\cdots,x_{p(n)})\),其中\(p\)是\(A=\{1,2,3,\cdots,n\}\)的一个排列,要求只使用\(O(1)\)的额外空间。例如,当数列为\((10,20,30,40)\),\(p\)为\((3,1,2,4)\)时获得的数列是\((30,10,20,40)\).c++
对于映射\(p:A\rightarrow A\),它的含义是“排列后的数组每一个元素从哪里来”。即:变换后,数组下标\(k\)处的数从变换前的下标\(p(k)\)处来。变换后下标为\(p(k)\)处的数从变换前下标为\(p(p(k))\)处的数来……所以咱们能够把这条变换的链记为:算法
每个下标都在惟一的一个“圈”内(原文这里的用词为"cycle")。举个例子:数组
对于给定的一个排列:函数
咱们能够观察到这样的规律:spa
对于括号中的每一行。最后一个等式右侧的数在\(p\)映射下的像,等于第一个等式左侧\(p\)做用的原像。对于每个这样的圈,咱们都能用大小为\(O(1)\)的额外空间完成数字的交换。而这个交换咱们从每一个圈中的最小数开始作。code
for (int j = 1;j <= n;j++) { int k = p(j);//n while (k > j) {//n+a k = p (k);//a } if(k == j) { int y = x[j],l = p[k];//b while (l != j) {//b+c x[k] = x[l];//c k = l;//c l = p(k);//c } x[k] = y;//b } } /* 这里b是变换p中"圈"的个数,c+b=n a的含义是:对于每一个j,在j所在的圈中第一个不大于j的数k距离p(j)的距离之和 */
最坏的状况以下class
此为\(a\)的最坏状况和\(b\)的最好状况。引用
此为\(a\)的最好状况和\(b\)的最坏状况。im
对于上面引用的\(p\)的实例:di
把全部的圈按照下述规则排列
1.圈内最小的数排在第一
2.圈内最小的数较大的,排在前面
则以上的\(p\)将变为\((5,6,9)(3,7)(2)(1,8,4)\),设\(q=(5,6,9,3,7,2,1,8,4)\).则这样的\(p\)到\(q\)的变换构成了一个排列到排列的双射。
这样,咱们就能够把求\(b\)的值转化为求\(\{1,2,\cdots,n\}\)中知足\(q(j)=\min\{q(i)|i\le i \le j \}\)的\(j\)的个数。使得知足这样条件的\(j\)的个数为\(k\)的\(n\)元排列数共有\(\left[\begin{array}{c}n\\k\end{array}\right]\)种。(第一类Stirling数)
因此:
(当\(n\)充分大时,\(O(1)\)的值收敛到欧拉常数)
其中
咱们定义以下函数
则
而
其中,\(r=j-i+1\)
由上式:
由上面计算的\(b\)的均值\(\overline b=\ln n +O(1)=O(\log n)\)
因此算法的平均时间复杂度是\(O(n\log n)\)