首页 > 其他分享 >LOJ 3276 JOISC 2020 Day2 遗迹 题解 (计数DP)

LOJ 3276 JOISC 2020 Day2 遗迹 题解 (计数DP)

时间:2023-03-01 21:36:05浏览次数:62  
标签:LOJ 题解 石柱 删掉 Day2 高度 初始 dp MOD

LOJ链接

UOJ链接

观察一下n次地震的过程,发现最后会有n个石柱高度为0,\(1,2\cdots n\)高度的石柱各有一个。假设现在已经确定了一种初始高度状态,我们来看看最后哪些石柱高度会大于0:

  • 首先,初始高度为n的两个石柱中,较右的会留下来,最后高度为n的石柱就是它
  • 接下来,最后高度为n-1的石柱是初始高度为n的两个石柱中较左的那个,以及初始高度为n-1的两个石柱中最右的那个
  • 以此类推\(\cdots\)

稍微推一下可以总结成以下过程:从大到小枚举每种初始高度,枚举到i时,把初始高度为i的两个石柱加入一个堆中,然后从堆中弹出最右的一个石柱,这个石柱就是最终高度为i的。最后堆中会剩下n个石柱,这些就是最后高度为0的。

知道了这个似乎还是不太好做。我们来看看在确定了初始状态后,能不能方便地确定每个石柱最后的高度。我们维护一个"剩余可用高度集合"S,初始\(S=\{1,2\cdots n\}\)。令第i个石柱的初始高度为\(h_i\),我们从右到左遍历每个石柱i,并在S中找出\(\le h_i\)的最大的元素。如果这个元素不存在,那i号石柱最终的高度就是0;否则最终的高度就是这个元素的值,我们需要把这个元素从S中删除。根据上面总结的过程,这部分也很好理解。用最右边的石柱举个例子吧,由于它在所有的"取出最右"的过程中都能杀穿,所以它一定能抢到最终高度为\(h_{2n}\)的位置。

现在题目钦定了每个石柱最终高度是0(被删除)还是大于0(不被删除),我们要求出满足这些条件的初始状态数。我们用dp来从右到左决定石柱的高度。一个合法的初始状态中每个高度都恰有2个石柱,为了便于dp,我们假设这两个石柱是不同的(相当于假设高度为i的石柱有红蓝两种,分配的时候任选一种不得重复),最后把答案除以\(2^n\)即可。令\(dp_{i,j}\)表示处理完了最靠右的i个石柱,S中的前j个元素都已经被删掉,第j+1个元素没被删掉(如果存在的话),且只决定了最右边的i个石柱中 钦定被删掉的那些 和 钦定没被删掉的那些中最终高度\(\le j\)的那些 的初始高度,其它石柱初始高度待定,此时的方案数。这句话包含的条件有点多,建议多看几遍。令最右的i个石柱中有\(cdel\)个钦定被删除,\(ckeep\)个钦定不被删除。(主动)转移有以下几种情况:

  • 当前石柱钦定被删掉。则它的初始高度必须\(\le j\),否则它的最终高度就是j+1了。高度范围\([1,j]\)中一共有2j种石柱,钦定不被删的消耗了j种,钦定被删的消耗了\(cdel\)种,所以一共有\(2j-j-cdel\)种方案。因此,\(dp_{i+1,j}+=dp_{i,j}\cdot(j-cdel)\)。
  • 当前石柱钦定不被删掉,且选择初始高度后S被删除的连续前缀长度不变。那么我们就先不选它的初始高度,先待定,以后再选。\(dp_{i+1,j}+=dp_{i,j}\)。
  • 当前石柱钦定不被删掉,且选择初始高度后S被删除的连续前缀变长了。我们枚举k,表示S被删除的连续前缀长度从j变成了\(j+k\)。要从S中删除这多出来的k个元素,需要k个石柱,当前石柱占了一个(它刚好从S中删掉了j+1,选择它的初始高度的方案数是\(2k-(k-1)=k+1\)),我们还需要从\(ckeep-j\)个初始高度待定的中选出\(k-1\)个,由于这k-1个之间是有顺序的,所以方案数\(\binom{ckeep-j}{k-1}\cdot (k-1)!\)。还要乘上一个额外的方案数\(f_{k-1}\),其中\(f_i\)表示在\(1-i\)的高度值,\(2i\)种不同石柱中选出i个,使得删掉的S中的元素刚好为\(1-i\)的方案数(注意只是选出,选出元素之间没有顺序)。综上,\(dp_{i+1,j+k}+=dp_{i,j}\cdot (k+1)\cdot \binom{ckeep-j}{k-1}\cdot (k-1)!\cdot f_{k-1}\)。

上面这部分的复杂度是\(O(n^3)\)的。

然后来求一下f。令\(dp2_{i,j}\)表示处理了最大的i种高度,选了j个石柱的方案数。为了保证合法,需要满足\(j总是\ge i\)(即j<i时dp值为0)。转移就枚举接下来这种高度选0、1还是2个即可。\(f_i=dp2_{i,i}\)。这部分时间复杂度\(O(n^2)\)。

总时间复杂度\(O(n^3)\)。

点击查看代码
#include <bits/stdc++.h>

#define rep(i,n) for(int i=0;i<n;++i)
#define repn(i,n) for(int i=1;i<=n;++i)
#define LL long long
#define fi first
#define se second
#define pb push_back
#define mpr make_pair

void fileio()
{
  #ifdef LGS
  freopen("in.txt","r",stdin);
  freopen("out.txt","w",stdout);
  #endif
}
void termin()
{
  #ifdef LGS
  std::cout<<"\n\nEXECUTION TERMINATED";
  #endif
  exit(0);
}

using namespace std;

const LL MOD=1e9+7;

LL qpow(LL x,LL a)
{
	LL res=x,ret=1;
	while(a>0)
	{
		if(a&1) (ret*=res)%=MOD;
		a>>=1;
		(res*=res)%=MOD;
	}
	return ret;
}

LL n,del[1210],dp2[610][610],f[610],dp[1210][610],fac[1210],inv[1210];

int main()
{
  fileio();
  freopen("temple.in","r",stdin);
  freopen("temple.out","w",stdout);

  fac[0]=1;repn(i,1205) fac[i]=fac[i-1]*i%MOD;
  rep(i,1203) inv[i]=qpow(fac[i],MOD-2);

  cin>>n;
  rep(i,n+n+3) del[i]=1;
  LL x;
  rep(i,n)
  {
    scanf("%lld",&x);--x;
    del[x]=0;
  }

  dp2[0][0]=1;
  rep(i,n+1) for(int j=i;j<=min((LL)i+i,n);++j)
  {
    (dp2[i+1][j]+=dp2[i][j])%=MOD;
    (dp2[i+1][j+1]+=dp2[i][j]*2)%=MOD;
    (dp2[i+1][j+2]+=dp2[i][j])%=MOD;
  }
  rep(i,n+2) f[i]=dp2[i][i];

  reverse(del,del+n+n);
  dp[0][0]=1;
  LL cdel=0,ckeep=0;
  rep(i,n+n)
  {
    rep(j,ckeep+1) if(dp[i][j])
    {
      if(del[i])
      {
        if(j-cdel>=0)
          (dp[i+1][j]+=dp[i][j]*(j-cdel))%=MOD;
      }
      else
      {
        (dp[i+1][j]+=dp[i][j])%=MOD;
        repn(k,n-j)
        {
          LL lftkeep=ckeep-j;
          if(k-1>lftkeep) break;
          LL val=dp[i][j]*(k+1)%MOD;
          if(k-1>0) (val*=fac[lftkeep]*inv[lftkeep-(k-1)]%MOD*f[k-1]%MOD)%=MOD;
          (dp[i+1][j+k]+=val)%=MOD;
        }
      }
    }
    cdel+=del[i];ckeep+=(1-del[i]);
  }
  LL ans=dp[n+n][n];
  (ans*=qpow(qpow(2,n),MOD-2))%=MOD;
  cout<<ans<<endl;

  termin();
}

标签:LOJ,题解,石柱,删掉,Day2,高度,初始,dp,MOD
From: https://www.cnblogs.com/legendstane/p/loj-3276-uoj-506-joisc-2020-day-2-ruins-2-solution-c

相关文章

  • P8631 [蓝桥杯 2015 国 AC] 切开字符串 题解
    P8631[蓝桥杯2015国AC]切开字符串题解前言看到这题没有人写题解,就打算写一篇。这也是蒟蒻的第一篇题解。前置知识\(manacher\),\(SA\)。如果不会可以转至mana......
  • 代码随想录算法训练营Day28 回溯算法 | 491.递增子序列 46.全排列 47.全排列 II
    代码随想录算法训练营491.递增子序列题目链接:491.递增子序列给定一个整型数组,你的任务是找到所有该数组的递增子序列,递增子序列的长度至少是2。示例:输入:[4,6,......
  • ABC275D-Yet-Another-Recursive-Function题解
    题目传送门题意:定义一个\(\mathbb{N}\to\mathbb{N}\)的函数\(f(x)=\begin{cases}1&x=0\\f(\lfloor\frac{x}{2}\rfloor)+f(\lfloor\frac{x}{3}\rfloor)&\text{otherwis......
  • CF71D-Solitaire题解
    题目传送门题意:一副扑克牌由54张牌组成,包括52张基本牌和两张“王”。在本题中每张牌用两个字符表示:对于基本牌,第一个字符为点数,有'2''3''4''5''6''7''8''9......
  • CF118C-Fancy-Number题解
    题目传送门题意:有一个\(n\)位的数字串,每位为\(0-9\)。每次操作可以更改一位的数字,代价为新旧两位数字之差。问使字符串存在某一个数的出现次数超过\(k\)的最小代价......
  • CF15D-Map题解
    题目传送门题意:有一个\(n\timesm\)的矩阵,每个格子有一个权值。每次操作会选择一个\(x\timesy\)的矩形区域,花费为“每个位置的权值减去最小权值”之和,区域之间不能......
  • CF杂题题解
    129B.StudentsandShoelaces题意:一个\(n\)个点\(m\)条边的无向图,每一轮删去所有度数为\(1\)的点,问删几轮停止。暴力模拟每一轮即可,每次删点更新邻居度数。{%......
  • CSP-J2022-C-逻辑表达式题解
    题意:给你一个由0、1、&、|、(、)组成的字符串,保证是一个合法的逻辑表达式。其中括号优先级最高,与运算优先级高于或运算,同级之间从左到右算。定义一次短路为,或运算的左边......
  • Gym100078H-History-of-Football题解
    题目传送门题意:有\(n\)支球队,每两支球队之间都会进行一场较量,赢者积\(3\)分,输者积\(0\)分,如果平局则各积\(1\)分。给出每支球队的最终积分,求方案数。\(n\le8\)......
  • Gym100753D-Carpets题解
    题目传送门题意:有一个\(H\timesW\)的地板和\(n\)个矩形地毯,问是否能不重不漏地填满地板。\(H,W\le100,n\le7\)。考虑朴素的搜索,每次考虑最左上角的没填的位置,枚......