通过K反演查找排列总数

时间:2016-09-11 08:14:12

标签: algorithm dynamic-programming

我已获得号码N。我必须找到数字1..N的所有排列,以便数组中的反转总数为K

For example: (all the following permutations have exactly 2 inversions) 
N=4 , K=2
(1,4,2,3) , (1,3,4,2) , (2,1,4,3) , (3,1,2,4) and (2,3,1,4).

我的方法:
动态编程:让dp[i][j]保持多种方式,使i位置负责j次反转。

dp[n][0]=1
 for(int i=n-1;i>=1;i--){

      int x = Math.min(k,n-i); // Maximum Number of inversion i position can do

      for(int j=0;j<=x;j++){

          for(int v=0;v<=j;v++){
                dp[i][j]+=dp[i+1][v];
            }

      } 

但是我的方法给了我个人的位置倒置,我怎样才能得到总体方式?

1 个答案:

答案 0 :(得分:0)

由于对1,2...N-1的任何排列,我们可以在N位置插入j,添加总共N-j个倒置,{1,2...N的排列总数带有k反转的1}}可以通过

计算
T(N,k) = T(N-1,k) + T(N-1,k-1) + T(N-1,k-2) ... + T(N-1,0)

JavaScript示例:

&#13;
&#13;
function f(N,K){
  var M = [[1]];

  for (var n=1; n<=N; n++){
    M[n] = new Array((n-1)*n/2 + 1).fill(0);
    
    for (var k=0; k<=(n-1)*n/2; k++){
      for (var j=Math.min(k,(n-2)*(n-1)/2); j>= Math.max(0,k-n+1); j--){
        M[n][k] += M[n-1][j];
      }
    }
  }
  
  return M;
}

var res = f(5,10);

for (var i=0; i< res.length; i++){
  console.log(JSON.stringify(res[i]));
}
&#13;
&#13;
&#13;