zoukankan      html  css  js  c++  java
  • [TopCoder]Seatfriends

    题目

      点这里看题目。

    分析

      可以想到用 DP 解决。
      由于把空位放到状态里面太麻烦了,因此我们单独将 " 组 " 提出来进行 DP 。
      (f(i,j)):前(i)个人组成(j)个组的方案数。
      此时这个组是有顺序有编号的,并且按照编号相邻(由于在环上,(j) 组和 (1) 组也算相邻)。
      考虑三种转移:
      1.我们新建一组,并在原来的一个组后面插入新组:(f(i+1,j+1)+=j*f(i,j))
      2.我们将新的人安排到新的组里面,可以放在组的两头:(f(i+1,j)+=2j*f(i,j))
      3.我们用一个人将两个组合在一起,有(j)个组相邻:(f(i+1,j-1)+=j*f(i,j))
      可以发现这样 DP 只需要保证中途状态不会超过(G)组即可。
      环上问题,我们可以先固定第一个人的位置,计算出方案数然后再乘上(n)。因此(f(1,1)=1)
      现在考虑怎么再组与组之间插入空位组成结果。设现在分配(g)个组,第(i)个组后有(x_i)个空位,则可以得到:

    [sum_{i=1}^g x_i=n-k |x_ige 1 ]

      这是一个可用插板法解决的问题,因此可以快速计算方案数。
      因此一般情况的答案为:

    [sum_{i=1}^G f(k,i) imes C_{n-k-1}^{i-1} ]

      但是如果(n=k),则组合数下标为(-1),不能算。这实际上是位置会被坐满,只有一个组的情况。当第(k)个人入座的时候,他只可能有一个位置可坐,因此方案数为(n imes f(k-1,1))

    代码

    #include <cstdio>
    
    const int mod = 1e9 + 7;
    const int MAXN = 2005;
    
    template<typename _T>
    void read( _T &x )
    {
    	x = 0;char s = getchar();int f = 1;
    	while( s > '9' || s < '0' ){if( s == '-' ) f = -1; s = getchar();}
    	while( s >= '0' && s <= '9' ){x = ( x << 3 ) + ( x << 1 ) + ( s - '0' ), s = getchar();}
    	x *= f;
    }
    
    template<typename _T>
    void write( _T x )
    {
    	if( x < 0 ){ putchar( '-' ); x = ( ~ x ) + 1; }
    	if( 9 < x ){ write( x / 10 ); }
    	putchar( x % 10 + '0' );
    }
    
    int f[MAXN][MAXN], C[MAXN][MAXN];
    int N, K, G;
    
    void upt( int &x, const int v ) { x = ( x + v ) % mod; }
    
    void init()
    {
    	C[0][0] = 1;
    	for( int i = 1 ; i <= N ; i ++ )
    	{
    		C[i][0] = C[i][i] = 1;
    		for( int j = 1 ; j < i ; j ++ ) C[i][j] = ( C[i - 1][j] + C[i - 1][j - 1] ) % mod;
    	}
    }
    
    class Seatfriends
    {
    	public:
    	int countseatnumb( const int n, const int k, const int g )
    	{
    		N = n, K = k, G = g;
    		init();
    		f[1][1] = 1;
    		for( int i = 1 ; i < K ; i ++ )
    			for( int j = 1 ; j <= G ; j ++ )
    			{
    				upt( f[i + 1][j], 2ll * j * f[i][j] % mod );
    				upt( f[i + 1][j + 1], 1ll * j * f[i][j] % mod );
    				if( j > 1 ) upt( f[i + 1][j - 1], 1ll * j * f[i][j] % mod );
    			}
    		int ans = 0;
    		if( N == K ) return 1ll * f[K - 1][1] * N % mod;
    		for( int i = 1 ; i <= G ; i ++ ) 
    			upt( ans, 1ll * f[K][i] * ( N == K ? 1 : C[N - K - 1][i - 1] ) % mod );
    		return 1ll * ans * N % mod;
    	}
    };
    
  • 相关阅读:
    实现用户注册验证码
    自带的打印预览
    分页存储过程
    文章标题、内容、摘要的处理函数
    ASP常用函数收藏
    生活中的经典感人语句
    如何在某一数据库的所有表的所有列上搜索一个字符串?
    如何访问隐藏的列表 workflow history list
    Windows Server 2008下如果什么操作没能正常完成, 请尝试run as administrator
    Visual Studio Build Marcos
  • 原文地址:https://www.cnblogs.com/crashed/p/12627033.html
Copyright © 2011-2022 走看看