zoukankan      html  css  js  c++  java
  • CF 1027 E.Inverse Coloring(dp)

    题目链接:https://codeforces.com/contest/1027/problem/E

    题目大意:

      给一个n*n的矩阵(n<=500),要求将其填满,每一个为黑或白,任意两行或两列,要么完全相同,要么完全相反,并且填充的方格内,任意相同颜色矩形的面积不大于k,求填充方案数。

    题目分析:

      很好的一道题目,不难发现,若确定第一行和第一列,整个图形也就定了,并且构成相同颜色的矩形区域,在第一行和第一列部分颜色必然是相同连续的,例如第一行为黑黑黑白白黑,可以将其划分为长度为3,2,1的三个部分,第一列为黑黑白白黑黑可划分为2,2,2三个部分,则整个图形最大面积为3*3=6。我们只需要计算长度为n时,最大连续区域颜色相同的方案,然后枚举长和宽即可。

      用dp[i][j]表示铺到第i个,最长相同的连续段为j的方案数。那么到i,j的方案就有两种,第一种为最后一块的长度为k,即从i-j块直接铺过来,另一种最后一块长度不到k,而之前的最长连续长度为k,则最后一段从i-j开始,到i-1位置。

      因此,dp转移方程为$dp[i][j]= sum_{k=1}^{j}dp[i-j][k]+sum_{k=i-j}^{i-1}dp[k][j]$。最后最长连续字段长度为i的方案数即为dp[n][i],因为黑白两种颜色,所以最后答案要乘以2即可。

      

    #include<bits/stdc++.h>
    
    using namespace std;
    typedef long long ll;
    typedef long double ld;
    typedef unsigned long long ull;
    typedef pair <int,int> pii;
    #define rep(i,x,y) for(int i=x;i<y;i++)
    #define rept(i,x,y) for(int i=x;i<=y;i++)
    #define per(i,x,y) for(int i=x;i>=y;i--)
    #define all(x) x.begin(),x.end()
    #define pb push_back
    #define fi first
    #define se second
    #define mes(a,b) memset(a,b,sizeof a);
    #define mp make_pair
    #define dd(x) cout<<#x<<"="<<x<<" "
    #define de(x) cout<<#x<<"="<<x<<"
    "
    #define debug() cout<<"I love Miyamizu Mitsuha forever.
    "
    const int inf=0x3f3f3f3f;
    const int maxn=505;
    const int mod=998244353;
    ll dp[maxn][maxn];
    ll len[maxn];
    int main()
    {
        mes(dp,0);
        ios::sync_with_stdio(false);
        cin.tie(0);
        int n,s;
        cin>>n>>s;
        dp[0][1]=1;
        rept(i,1,n)
        {
            rept(j,1,i)//from 1 to i, the max length of contious block is j
            {
                rep(k,1,j) dp[i][j]=(dp[i][j]+dp[i-j][k])%mod;
                rept(k,i-j,i-1) dp[i][j]=(dp[i][j]+dp[k][j])%mod;
            }
        }
        ll ans=0;
        rept(i,1,n) len[i]=dp[n][i];
        rept(i,1,n)
        {
            rept(j,1,n)
            {
                if(i*j<s) ans+=len[i]*len[j];
                ans%=mod;
            }
        }
        cout<<(ans*2)%mod<<"
    ";
        return 0;
    }
  • 相关阅读:
    把Orchard部署到Windows Azure Web Sites
    使用Windows Live Writer 发布博客园博客
    使用Microsoft Word 2013 发布Blog到博客园
    Java栈的简单实现
    Java中的运算符
    Java简单双向链表实现 @version 1.0
    Java中的面向对象II
    认识和分析日志文件
    两数之和问题
    括号序列算法
  • 原文地址:https://www.cnblogs.com/FZUzyz/p/12255690.html
Copyright © 2011-2022 走看看