【BZOJ3612】【HEOI2014】平衡 整数划分、

题意:

有一个杠杆,半长为n,在(2n+1)个整数坐标各有一个质量相同的砝码。

现在给出n,然后再给出要取走的砝码个数,问使得最终杠杆依然平衡的最终方案数是多少。

(文末会给出原题帽)

分析:

数据范围不大,我们可以写整数划分。

f[i][j]表示将i划分成j个互不相同的正整数,且最大不超过n 的划分方案数。

这里说一下这道题的整数划分。

我们不妨先来反向思考一下。---------------------------------------------------------------------

首先考虑f[i][j](下图每一列都代表一个数,高度就是数值)

它可以在底下添加一行,进行转移

分为两种情况:

Ⅰ. 转移过后最小数不为1

Ⅱ. 转移过后最小数为1

然后f[i][j]可以向这两个方向转移。

而我们还需要保证最大数不能大于n,那么如下图

在最后加入一层,使得当前所有整数都+1时,发现出现了一个101,而不妨当作n是100

那么显然我们可以很方便地清掉这个数带来的情况。

即当前是f[i][j],那么最后一列是大于n的情况显然只能是有一个整数n+1

不看这个n+1,情况数是f[i-(n+1)][j-1] ,我们把这个情况集删掉就好了。

正向考虑:-----------------------------------------------------------------------------------

首先不妨把刚才的图片按顺序记作图1、2、3、4。

我们把f[i][j](图1)这么多方案分成两种情况:

Ⅰ. 最小的数不为1:

好说。 直接由f[i-j][j]在底下加一行得到。就是图2。

此时原来划分出来的整数不同,新的这些整数显然依然不同。

Ⅱ. 最小的数为1:

那么显然它可以由f[i-j][j-1]转移得到,

即在f[i-j][j-1]代表图形下面整体+1,最后加上一个整数1,即图3。

注意此时f[i-j][j-1]代表的所有图形整数都不同(性质/定义),那么新加1后所有整数依然不同,且均>=2

这个时候再来个整数1,依然满足所有整数不同。

而这两种情况显然互补,即这两种情况的转移包含了f[i][j]的所有情况(两个命题“最小数是1”,“最小数不是1”,显然包含全部情况),也就是说转移完成。

但是我们注意到还需要让最大数不能超过n,

所以有了图4。

也就是我们要减去最大数超过n的情况,方法前文图下有说明。

这道题难点解决了。

现在说一下其它细节:

f[i][j]算出来后直接暴力枚举两边的权值,及用点个数(不要忘了中心支点)

然后check。end。

代码:

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#define N 101000
#define M 15
using namespace std;
int n,m,p,w;
int f[N][M]; // f[i][j] 表示
//将i划分成j个互不相同的正整数,
// 且最大不超过n 的划分方案数
int main()
{
	int i,j,k,g;
	f[0][0]=1;
	for(scanf("%d",&g);g--;)
	{
		scanf("%d%d%d",&n,&m,&p);
		if(m==1)
		{
			puts("1");
			continue;
		}
		w=n*(m-1);
		for(i=1;i<=w;i++)
			for(j=1;j<m;j++)
			{
				f[i][j]=(i>=j?(f[i-j][j]+f[i-j][j-1]):0);
				f[i][j]=(i>=n+1)?(f[i][j]-f[i-n-1][j-1]):f[i][j];
				f[i][j]=(f[i][j]%p+p)%p;
			}
		long long ans=0;
		for(i=1;i<=w;i++)
			for(j=1;j<m;j++)
				ans+=f[i][j]*f[i][m-j],ans%=p;
		for(i=0;i<=w;i++)
			for(j=1;j<m-1;j++)
				ans+=f[i][j]*f[i][m-1-j],ans%=p;
		printf("%lld\n",ans);
	}
}

原题面:

Ⅰ. 最小的数不为1:

好说。 直接由f[i-j][j]在底下加一行得到。就是图2。

时间: 2024-10-14 06:19:55

【BZOJ3612】【HEOI2014】平衡 整数划分、的相关文章

bzoj 3612: [Heoi2014]平衡【整数划分dp】

其实就是-n~n中求选k个不同的数,和为0的方案数 学到了新姿势叫整数划分,具体实现是dp 详见:https://blog.csdn.net/Vmurder/article/details/42551603 设f[i][j]为j个数和为i的方案数,然后因为互不相同,所以转移的话有两种,就是当前j个数全部+1,和当前j个数全部+1并且多填一个1出来,也就是f[i][j]=f[i-j][j]+f[i-j][j-1] 但是这里要求选的数不能超过n,我们考虑i>n的f中一定有一个大于n的数,我们把这种情

BZOJ 3612: [Heoi2014]平衡( dp )

枚举Fl, 就变成一个整数划分的问题了...f(i,j) = f(i-j,j-1)+f(i-j,j)-f(i-N-1,j-1)递推.f(i,j)表示数i由j个不同的数组成,且最大不超过N的方案数 -------------------------------------------------------------------------- #include<cstdio> #include<cstring> #include<algorithm> using nam

整数划分递归模板

/* 整数划分问题是算法中的一个经典命题之一,有关这个问题的讲述在讲解到递归时基本都将涉及. 所谓整数划分,是指把一个正整数n写成如下形式: n=m1+m2+...+mi; (其中mi为正整数,并且1 <= mi <= n),则{m1,m2,...,mi}为n的一个划分. 如果{m1,m2,...,mi}中的最大值不超过m,即max(m1,m2,...,mi)<=m,则称它属于n的一个m划分.这里我们记n的m划分的个数为f(n,m); 例如但n=4时,他有5个划分,{4},{3,1},{

NYOJ746——整数划分(四)

整数划分(四) 时间限制:1000 ms  |  内存限制:65535 KB 难度:3 描述 暑假来了,hrdv 又要留学校在参加ACM集训了,集训的生活非常Happy(ps:你懂得),可是他最近遇到了一个难题,让他百思不得其解,他非常郁闷..亲爱的你能帮帮他吗? 问题是我们经常见到的整数划分,给出两个整数 n , m ,要求在 n 中加入m - 1 个乘号,将n分成m段,求出这m段的最大乘积 输入 第一行是一个整数T,表示有T组测试数据 接下来T行,每行有两个正整数 n,m ( 1<= n <

NYOJ-571 整数划分(三)

此题是个非常经典的题目,这个题目包含了整数划分(一)和整数划分(二)的所有情形,而且还增加了其它的情形,主要是用递归或者说是递推式来解,只要找到了递推式剩下的任务就是找边界条件了,我觉得边界也是非常重要的一步,如果找不准边界,这个题也很难做出来,当时我就是找边界找了好长时间,边界得琢磨琢磨.递推步骤如下: 第一行:将n划分成若干正整数之和的划分数.状态转移方程:dp[i][j]:和为i.最大数不超过j的拆分数dp[i][j]可以分为两种情况:1.拆分项至少有一个j 2.拆分项一个j也没有dp[i

算法笔记——整数划分3

题目来源:POJ1664-放苹果 和POJ3014 问题描述: 把m个同样的苹果放在n个同样的盘子里,允许有的盘子空着不放,问共有多少种不同的分法?(用K表示)5,1,1和1,5,1 是同一种分法. 输入: 第一行是测试数据的数目t(0 <= t <= 20).以下每行均包含二个整数m和n,以空格分开.1<=m,n<=10. 输出: 对输入的每组数据m和n,用一行输出相应的K. 分析: 问题描述转换成整数划分形式:把一个正整数m分成至多n个正整数的和,有多少种分法? 假设用f(m,

51nod p1201 整数划分

1201 整数划分 基准时间限制:1 秒 空间限制:131072 KB 分值: 80 难度:5级算法题 将N分为若干个不同整数的和,有多少种不同的划分方式,例如:n = 6,{6} {1,5} {2,4} {1,2,3},共4种.由于数据较大,输出Mod 10^9 + 7的结果即可. Input 输入1个数N(1 <= N <= 50000). Output 输出划分的数量Mod 10^9 + 7. Input示例 6 Output示例 4 分析:这题关键在于不同的整数一个包含数字最多的划分必

hdu 1028 Ignatius and the Princess III 【整数划分】

Ignatius and the Princess III                                                                                       Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submission(s): 15730    Accepted Submission(

算法笔记——整数划分2

题目来源:NYOJ176 问题描述: 把一个正整数m分成n个正整数的和,有多少种分法? 例:把5分成3个正正数的和,有两种分法: 1 1 3 1 2 2 输入: 第一行是一个整数T表示共有T组测试数据(T<=50) 每组测试数据都是两个正整数m,n,其中(1<=n<=m<=100),分别表示要拆分的正数和拆分的正整数的个数. 输出: 输出每组拆分的方法的数目. 分析: 题目可以换种等价描述:把m个同样的苹果放在n个同样的盘子里,不允许有的盘子空着不放,问共有多少种不同的分法,其中n