博客园 首页 私信博主 显示目录 隐藏目录 管理 动画

题解 [BZOJ1925][SDOI2010] 地精部落

题面

解析

这个似乎并不好讲啊

f[i][j]表示有i座山,

最后一座山到达高度是i座中第j大的,

且最后一座山是山谷.

注意,i是代表有i座山,并不代表高度一定是1~i.

j也是一个类似于离散化的东西.

然后我们考虑设g[i][j],

除了最后一座山是山峰以外其它的定义和f[i][j]一样.

那么有式子f[i][j]=k=ji1g[i1][k].

在这里讲一下最大和最小值的问题,

最大值可以随便往大了取,

但考虑到前i座山的最后一座一定比前i1座的小(因为是山谷),

所以最后一座是第j大那它前面一座在前i1座山中的排名就不可能小于j.

(可能表达有点不太清楚,感性理解下吧.)

还有一件显然的事,

如果有一个合法的方案,

那我们把每座山的高度hi换成nhi+1,方案依然合法(就是把山峰换成了山谷).

因此g[i][j]=f[i][ij+1].

合并一下就变成了f[i][j]=k=1ijf[i1][k].

而这个相当于一个后缀和的东西.

因此DP式就是f[i][j]=f[i][j+1]+f[i1][ij].

用滚动数组优化下空间就行了.

#include <iostream>
#include <cstdio>
#include <cstring>
#define fre(x) freopen(x".in","r",stdin),freopen(x".out","w",stdout)
using namespace std;

inline int read(){
	int sum=0,f=1;char ch=getchar();
	while(ch>'9' || ch<'0'){if(ch=='-')f=-1;ch=getchar();}
	while(ch>='0' && ch<='9'){sum=sum*10+ch-'0';ch=getchar();}
	return f*sum;
}

const int N=4301;
int n,Mod,f[2][N];
int ans=0;

int main(){
	n=read();Mod=read();
	f[1][1]=1;int now=1;
	for(int i=2;i<=n;i++){
		now^=1;memset(f[now],0,sizeof(f[now]));
		for(int j=i-1;j;j--) f[now][j]=(f[now][j+1]+f[now^1][i-j])%Mod;
	}
	for(int i=1;i<=n;i++) ans=(ans+f[now][i])%Mod;
	printf("%d\n",(ans<<1)%Mod);
	return 0;
}

posted @   Hastin  阅读(123)  评论(0编辑  收藏  举报
编辑推荐:
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
阅读排行:
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 单线程的Redis速度为什么快?
· SQL Server 2025 AI相关能力初探
· 展开说说关于C#中ORM框架的用法!
· AI编程工具终极对决:字节Trae VS Cursor,谁才是开发者新宠?
点击右上角即可分享
微信分享提示