HDU 6578 Blank
Time limit 1000 ms
Memory limit 262144 kB
OS Windows
Source 2019 Multi-University Training Contest 1
感想
火车上AK这场比赛的dls,在两天后直播讲题,讲到这题的时候,说得好轻松啊,一个简单的dp就完事了,然后说了一下dp的参数,然后就到下一题了……
解题思路
扔几个链接,融会贯通一下
- https://blog.csdn.net/m0_43448982/article/details/96988882
- https://blog.csdn.net/Ratina/article/details/97237438
- https://www.cnblogs.com/zhuyou/p/11273721.html
大概就是设\(dp[x][y][z][i]\)代表如下的意思:从前向后扫,扫到目前第\(i\)个格子时,4种颜色最后出现的位置分别为\(x、y、z、i\)时的方案数,其中\(x\leqslant y\leqslant z\leqslant i\),在\(x、y、z、i\)为0时取等号,表示目前有一些颜色还没有出现,即最后一次出现的位置为0。
颜色种类不重要,因为计算过程已经把不同颜色的组合方式计算进去了,重要的是颜色的。然后染这一格(\(i\))的时候的方案数是从染上一个格子(\(i-1\))的方案数转移过来的。
扩展当前这一个格子的时候,我们可以取之前出现过的4种颜色中的一种。举个例子,假设我们取的是之前最后一次出现在\(x\)的那个颜色,那么会发生这样的转移——\(dp[y][z][i-1][i]+=dp[x][y][z][i-1]\)(想想dp数组四个参数的含义和大小关系,现在这个状态的数量增加了上一个状态的数量那么多种),取的是最后一次出现在\(y\)的那个颜色的话,发生的转移就是——\(dp[x][z][i-1][i]+=dp[x][y][z][i-1]\),同样的道理还有\(dp[x][y][i-1][i]+=dp[x][y][z][i-1]\)和\(dp[x][y][i-1][i]+=dp[x][y][z][i]\)。对于判断条件,当我们跑完第\(i\)个格子的时候,就检查以\(i\)为右端点的所有条件,对于一个条件——\([l,r]\)之间有\(x\)种颜色,我们可以看那些状态是否满足条件,看每个颜色最后一次出现的位置在区间内还是区间外,然后根据这个统计区间内颜色数量,不满足条件就把它们清零,防止转移到下一个\(i\)。
统计的时候就统计所有\(i=n\)的格子,把数量加起来就好。但是这种方法会爆空间,注意到\(i\)那一维可以滚动,所以把\(i\)那一维滚动起来。另外注意取模 993244853 998244353。
源代码
#include<vector>
#include<cstdio>
#include<cstring>
#include<algorithm>
const int mod=998244353;
int T;
int n,m;
long long dp[105][105][105][2];
//x<y<z<i
struct Condition{
int l,x;
};
std::vector<Condition> c[105];
int main()
{
//freopen("test.in","r",stdin);
scanf("%d",&T);
while(T--)
{
scanf("%d%d",&n,&m);
for(int i=0;i<105;i++) c[i].clear();
for(int i=1,l,r,x;i<=m;i++)
{
scanf("%d%d%d",&l,&r,&x);
c[r].push_back({l,x});
}
dp[0][0][0][0]=1;
int cur=1;//i&1
int last=0;//!(i&1)
for(int i=1;i<=n;i++)
{
for(int z=0;z<=i;z++)
{
for(int y=0;y<=z;y++)
{
for(int x=0;x<=y;x++)
{
dp[x][y][z][cur]=0;//清零
}
}
}
for(int z=0;z<i;z++)
{
for(int y=0;y<=z;y++)
{
for(int x=0;x<=y;x++)//转移
{
long long temp=dp[x][y][z][last];
dp[x][y][z][cur]+=temp;
dp[y][z][i-1][cur]+=temp;
dp[x][z][i-1][cur]+=temp;
dp[x][y][i-1][cur]+=temp;
dp[x][y][z][cur]%=mod;
dp[y][z][i-1][cur]%=mod;
dp[x][z][i-1][cur]%=mod;
dp[x][y][i-1][cur]%=mod;
}
}
}
for(unsigned int cc=0;cc<c[i].size();cc++)//判断
{
for(int z=0;z<i;z++)
for(int y=0;y<=z;y++)
for(int x=0;x<=y;x++)
{
int temp=1+(x>=c[i][cc].l?1:0)+(y>=c[i][cc].l?1:0)+(z>=c[i][cc].l?1:0);//统计区间内颜色数量
if(temp!=c[i][cc].x) dp[x][y][z][cur]=0;
}
}
std::swap(cur,last);
}
long long ans=0;
for(int z=0;z<n;z++)//统计答案
{
for(int y=0;y<=z;y++)
{
for(int x=0;x<=y;x++)
{
ans+=dp[x][y][z][last];
ans%=mod;
}
}
}
printf("%lld\n",ans);
}
return 0;
}
HDU 6578 Blank的更多相关文章
- HDU - 6578 Blank DP + 滚动数组
HDU - 6578 Blank 题意 给你\(\{0,1,2, 3\}\)四个数,分别填入长度为\(n\)的数列中,有\(m\)个限制条件,\(l_{i}, r_{i}, x_{i}\)表示在\([ ...
- [2019HDU多校第一场][HDU 6578][A. Blank]
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6578 题目大意:长度为\(n\)的数组要求分别填入\(\{0,1,2,3\}\)四个数中的任意一个,有 ...
- HDU 3572 Task Schedule(拆点+最大流dinic)
Task Schedule Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) To ...
- hdu 1003 MAX SUM 简单的dp,测试样例之间输出空行
测试样例之间输出空行,if(t>0) cout<<endl; 这样出最后一组测试样例之外,其它么每组测试样例之后都会输出一个空行. dp[i]表示以a[i]结尾的最大值,则:dp[i ...
- HDU 2274 Magic WisKey
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2274 Magic WisKey Time Limit: 2000/1000 MS (Java/Othe ...
- HDU 1505 City Game (hdu1506 dp二维加强版)
F - City Game Time Limit:1000MS Memory Limit:32768KB 64bit IO Format:%I64d & %I64u Submi ...
- HDU 1231 最大连续子序列 &&HDU 1003Max Sum (区间dp问题)
C - 最大连续子序列 Time Limit:1000MS Memory Limit:32768KB 64bit IO Format:%I64d & %I64u Submit ...
- HDU 1707 简单模拟 Spring-outing Decision
Spring-outing Decision Time Limit: 3000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/O ...
- HDU 1708 简单dp问题 Fibonacci String
Fibonacci String Time Limit: 3000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) ...
随机推荐
- 启用hdfs的高可用
cm-HDFS: 选择另外一个节点的做NN, 生产选node3 选择三个节点作journalNode, node2,3,4 填入journalNode的目录/dfs/jn 经过一系列步骤,如果没报错 ...
- oracle 数据库备份与恢复
oracle 数据库备份与恢复 包含四个部分: 1.数据泵备份与恢复 2.rman备份与恢复 3.CSV增量备份恢复 4.截库操作 1.数据泵备份与恢复 expdp/ / impdp 时的 CONTE ...
- oracle sid_name service_name
在工作中也遇到了这种情况,使用oracle这么长时间一直使用的都是SID的概念.也是给自己扩充了一下知识,所以后面在配置oracle数据库连接的时候需要确认拿到的是service还是sid再进行相应的 ...
- HDU-5155 Harry And Magic Box
题目描述 在\(n*m\)的矩阵内每一行每一列都有钻石,问钻石分布的种类? 答案有可能很大,所以输出答案对\(1000000007\)取模. Input 对于每个测试用例,有两个整数\(n\)和\(m ...
- ckfinder的使用
引入<script type="text/javascript" src="${ctxStatic}/ckfinder/ckfinder.js">& ...
- springboot在集成mybatis的时候老是报错 The server time zone value '�й���ʱ��' is unrecognized
我已经解决了,感谢万能网友. 解决办法参见:https://blog.csdn.net/yunfeng482/article/details/86698133
- vue——echarts更换主题
链接:https://blog.csdn.net/Sunshine0508/article/details/90067437 //等配置安装好了以后 在main.js里引入echarts主题的js,一 ...
- kali下纯文本与窗口环境切换
切到纯文本环境,想返回 试了半天ctrl+alt+f7不行, 最后我想试试ctrl+alt+f8竟然成了: 而且那是之前以root账户登录图像界面时切换回去是f8,普通用户是f9 ,为何如此,我还不 ...
- PAT Basic 1086 就不告诉你 (15 分)
做作业的时候,邻座的小盆友问你:“五乘以七等于多少?”你应该不失礼貌地围笑着告诉他:“五十三.”本题就要求你,对任何一对给定的正整数,倒着输出它们的乘积. 输入格式: 输入在第一行给出两个不超过 10 ...
- GetShortPathName函数
Declare Function GetShortPathName Lib "kernel32" Alias "GetShortPathName" (ByVal ...