【GMOJ6377】幽曲[埋骨于弘川]
2024-09-05 18:43:51
Description
- \(n\in[1,500],k\in[2,10]\)。
Solution
- 这是一道有点很有难度的题。
- 先考虑判断一个数是否在数列\(a\)中。由于每次加的数是在\([0,k)\)的范围内,所以个位不定,但除个以外的位可以任意取值。
- 考虑DP。记个位为第\(1\)位,设\(g_{i,p,x,a}\)表示我们构造的数第\(2\sim i\)位为0,第\(i\sim\infty\)位中最大的位值为\(p\),个位为\(a\),此时我们要将第\(i\)位刚好填到\(x\),个位变成了多少。
- 初值的话,可以暴力算出\(i\in[1,2]\)时的\(g\)。
- 不进位的转移显然。进位时,比如我们想算\(g_{i+1,p,1,a}\)的值,它可以由\(g_{i,k-1,1,a'}\)转移而来(\(a'\)表示我们在第\(i\)位填\(k-1\)次\(1\)后\(a\)会变成的数)。
- 然后做树形DP。如果我们沿着子树节点序列转移,那么实际上就是沿着dfs序转移,可以转移到\(dfn[i]\)的范围是\([dfn[fa[i]],dfn[i])\)。
- 设\(dp_{dfn[x],j,p,a}\)表示我们做到点\(x\),考虑到第\(j\)位(第\(j\)位放\(d[x]\)),前面的位的最大值为\(p\),个位为\(a\)的合法方案数。初值显然是\(dp_{1,j,d[1],g_{j,0,d[1],1}}=1\)。
- 转移的话,一定是从\(dp_{i',j,p,x}\)转移到\(dp_{i,j-1,max(p,d),g_{j-1,p,d,x}}\),其中\(i'<i\)。这个可用前缀和优化。
- 时空复杂度:\(O(nk^2(n+k))\)。
Code
#include <cstdio>
#include <cstring>
#define max(x,y) ((x)>(y)?(x):(y))
#define fo(i,a,b) for(int i=a;i<=b;i++)
using namespace std;
const int N=505,K=11,mo=998244353;
int n,k,g[N][K][K][K],u,d[N],x,y,ti,dfn[N],dp[N][N][K][K],ans;
bool e[N][N];
void P(int&x,int y) {x=(x+y)%mo;}
void dfs(int u,int fa)
{
if(!ti++)
fo(j,1,n)
{
int x=g[j][0][d[u]][1];
dp[ti][j][d[u]][x]=1;
}
else
fo(j,2,n) fo(p,0,k-1) fo(x,0,k-1)
{
int p1=max(p,d[u]), x1=g[j-1][p][d[u]][x];
if(~x1) P(dp[ti][j-1][p1][x1],(mo+dp[ti-1][j][p][x]-dp[dfn[fa]-1][j][p][x])%mo);
P(dp[ti][j-1][p][x],dp[ti-1][j-1][p][x]);
}
fo(x,d[u],k-1) P(ans,(mo+dp[ti][1][x][d[u]]-dp[ti-1][1][x][d[u]])%mo);
dfn[u]=ti;
fo(v,1,n) if(v^fa&&e[u][v]) dfs(v,u);
}
int main()
{
freopen("buried.in","r",stdin);
freopen("buried.out","w",stdout);
scanf("%d%d",&n,&k);
memset(g,-1,sizeof g);
fo(p,0,k-1)
{
g[1][p][0][0]=0;
fo(x,0,k-1)
if(p|x)
{
int y=x;
while(y<k)
{
g[1][p][y][x]=y;
y+=max(p,y);
}
g[2][p][1][x]=y-k;
}
}
fo(i,2,n)
fo(p,0,k-1)
fo(a,0,k-1)
{
g[i][p][0][a]=a;
if(!~(u=g[i][p][1][a])) continue;
fo(x,2,k-1) if(!~(u=g[i][p][x][a]=g[i][max(p,x-1)][1][u])) break;
if(~u) g[i+1][p][1][a]=g[i][k-1][1][u];
}
fo(i,1,n) scanf("%d",&d[i]);
fo(i,1,n-1) scanf("%d%d",&x,&y),e[x][y]=e[y][x]=1;
dfs(1,0);
printf("%d",ans);
}
最新文章
- 对Castle Windsor的Resolve方法的解析时new对象的探讨
- seajs学习一天后的总结归纳
- Spring结合Quartz实现多任务定时调用(转载)
- SharePoint 使用代码为页面添加WebPart
- 枚举Enumerations
- IOS开发之带格式的文本
- 三级联动数据表db_nove.sql
- xcode UILabel创建和隐藏
- Oracle SQL Developer 设置自动提示(完成设置)
- 类设计的SOLID原则
- Nginx与前端开发
- python学习:缩进
- python -- 函数进阶
- 右键菜单添加打开CMD选项
- linux操作系统中oracle数据库的密码过期问题解决
- php 根据日期获取星座
- linux基础11-bash编程(字符串测试 和 for循环)
- 【iCore4 双核心板_FPGA】例程十六:基于双口RAM的ARM+FPGA数据存取实验
- Universal-Image-Loader源码分析(一)——ImageLoaderConfiguration分析
- JS中的history对象
热门文章
- 织梦自定义表单导出为excel功能
- 十六、简单配置jenkins执行本地的robotframework项目
- source ~/.bash_profile是什么意思
- Spring Boot系列(四) Spring Cloud 之 Config Client
- jmeter _Random函数生成随机数
- .Net core 3.0 SignalR+Vue 实现简单的即时通讯/聊天IM (无jq依赖)
- [Git] 005 初识 Git 与 GitHub 之分支
- SocketChannel 读取ByteBuf 的过程
- Golang环境配置
- Elasticsearch学习,请先看这一篇!