「BZOJ 5161」最长上升子序列「状压DP」

题意

求一个(1sim n)的排列LIS的期望长度,(nleq 28)

题解

考虑朴素的LIS:(f[i] = min(f[j]) + 1)

(mx[i])(f)的前缀最大值,那么可以得到一个性质(mx[i + 1] in [mx[i], mx[i] + 1])

(mx)数组进行差分,则差分数组只有(01),可以状压

由于(mx[1] - mx[0]=1),从第二位开始状压

然后考虑从(1sim i)的排列推到(1sim i+1)的排列,(1sim i)的差分数组为(S)。把i插入第(j)(1sim i+1))个数前面:

(j=1):新状态为(S << 1),即(mx[2])(0),而其余不改

(j>1)(mx[1])(mx[j - 1])不变,(mx[j])(mx[i])整体右移一位,新的(mx[j])为1,但要把新的(mx[j])后面第一个(1)删去(因为(j)处答案变大了,所以该点和新答案一样大,差分数组对应(0)

注意实现的时候第(k)个位置对应(1<<(k-2))

打表程序:

	static int dp[2][134217728], ans;
	for(int n = 1; n <= 28; n ++) {
		  memset(dp, 0, sizeof dp); 
		  int r = 0; dp[0][0] = 1; ans = 0;
		  for(int i = 1; i < n; i ++, r ^= 1) {
		      fill(dp[r ^ 1], dp[r ^ 1] + (1 << i), 0);
		      for(int j = 0; j < (1 << (i - 1)); j ++) if(dp[r][j]) {
		          upd(dp[r ^ 1][j << 1], dp[r][j]);
		          int la = -1;
		          for(int k = i + 1; k >= 2; k --) {
		              int t = j;
		              if(j >> (k - 2) & 1) la = k - 2;
		              if(~ la) t ^= 1 << la;
		              t = t >> (k - 2) << (k - 1) | ( 1 << (k - 2) ) | ( j & (( 1 << (k - 2) ) - 1) );
		              upd(dp[r ^ 1][t], dp[r][j]);
		          }
		      }
		  }
		  for(int i = 0; i < (1 << (n - 1)); i ++)
		      upd(ans, 1ll * dp[r][i] * (__builtin_popcount(i) + 1) % mo);
		  int fac = 1;
		  for(int i = 2; i <= n; i ++) fac = 1ll * fac * i % mo;
		  printf("%d, ", 1ll * ans * qpow(fac, mo - 2) % mo);
	}

提交程序:

#include <cstdio>
const int qwq[] = {1, 499122178, 2, 915057326, 540715694, 946945688, 
422867403, 451091574, 317868537, 200489273, 976705134, 705376344, 662845575, 
331522185, 228644314, 262819964, 686801362, 495111839, 947040129, 414835038, 
696340671, 749077581, 301075008, 314644758, 102117126, 819818153, 273498600, 267588741};
int main() {
    int n; scanf("%d", &n); printf("%d
", qwq[n - 1]);
    return 0;
}
原文地址:https://www.cnblogs.com/hongzy/p/11331756.html