P1077 [NOIP2012 普及组] 摆花

[NOIP2012 普及组] 摆花

题目描述

小明的花店新开张,为了吸引顾客,他想在花店的门口摆上一排花,共 m m m 盆。通过调查顾客的喜好,小明列出了顾客最喜欢的 n n n 种花,从 1 1 1 到 n n n 标号。为了在门口展出更多种花,规定第 i i i 种花不能超过 a i a_i ai 盆,摆花时同一种花放在一起,且不同种类的花需按标号的从小到大的顺序依次摆列。

试编程计算,一共有多少种不同的摆花方案。

输入格式

第一行包含两个正整数 n n n 和 m m m,中间用一个空格隔开。

第二行有 n n n 个整数,每两个整数之间用一个空格隔开,依次表示 a 1 , a 2 , ⋯   , a n a_1,a_2, \cdots ,a_n a1,a2,⋯,an。

输出格式

一个整数,表示有多少种方案。注意:因为方案数可能很多,请输出方案数对 1 0 6 + 7 10^6+7 106+7 取模的结果。

样例 #1

样例输入 #1

复制代码
2 4
3 2

样例输出 #1

复制代码
2

提示

【数据范围】

对于 20 % 20\% 20% 数据,有 0 < n ≤ 8 , 0 < m ≤ 8 , 0 ≤ a i ≤ 8 0<n \le 8,0<m \le 8,0 \le a_i \le 8 0<n≤8,0<m≤8,0≤ai≤8。

对于 50 % 50\% 50% 数据,有 0 < n ≤ 20 , 0 < m ≤ 20 , 0 ≤ a i ≤ 20 0<n \le 20,0<m \le 20,0 \le a_i \le 20 0<n≤20,0<m≤20,0≤ai≤20。

对于 100 % 100\% 100% 数据,有 0 < n ≤ 100 , 0 < m ≤ 100 , 0 ≤ a i ≤ 100 0<n \le 100,0<m \le 100,0 \le a_i \le 100 0<n≤100,0<m≤100,0≤ai≤100。

正解

二维 dp。

一、状态:因为有花的种类和花的数量两个参数,所以要用二维 dp,那么 d p [ i ] [ j ] dp[i][j] dp[i][j] 就表示取 i i i 种花, j j j 盆花的方案数。

二、转移:

  1. 初始化:只有摆 i i i 种,但是 0 0 0 盆,没有摆 0 0 0 种, i i i 盆,所以只要将 d p [ i = 1 , 2 , 3 , ... , n ] [ 0 ] dp[i=1,2,3,\dots,n][0] dp[i=1,2,3,...,n][0] 都赋值为 1 1 1 即可。
  2. 转移方程:第一层遍历花的种类数,第二层遍历第 i i i 种花放多少盆,第三层遍历原来放了多少盆,那么 d p [ i ] [ j + k ] = ( d p [ i ] [ j + k ] + d p [ i − 1 ] [ k ] )   m o d   1000007 dp[i][j+k] = (dp[i][j+k] + dp[i-1][k]) \bmod 1000007 dp[i][j+k]=(dp[i][j+k]+dp[i−1][k])mod1000007。

三、答案:题目要求求出 n n n 种 m m m 盆的方案数,即 d p [ n ] [ m ] dp[n][m] dp[n][m]。

代码

cpp 复制代码
#include <bits/stdc++.h>
//#define int long long
using namespace std;

int n,m,a[110];
int dp[110][110];

void solve()
{
	cin >> n >> m;
	for (int i = 1;i <= n;i++)
		cin >> a[i];
	for (int i = 0;i <= n;i++)
		dp[i][0] = 1;
	for (int i = 1;i <= n;i++)
		for (int j = 0;j <= a[i];j++)
			for (int k = 0;k <= m-j;k++)
				if (j || k)
					dp[i][j+k] = (dp[i][j+k] + dp[i-1][k]) % 1000007;
	cout << dp[n][m];
}

signed main()
{
	solve();
	printf("\n");
	return 0;
}
相关推荐
好大哥呀25 分钟前
C++ Web 编程
开发语言·前端·c++
Fuxiao___35 分钟前
C 语言核心知识点讲义(循环 + 函数篇)
算法·c#
Mr_Xuhhh1 小时前
LeetCode hot 100(C++版本)(上)
c++·leetcode·哈希算法
漫随流水1 小时前
c++编程:反转字符串(leetcode344)
数据结构·c++·算法
南境十里·墨染春水2 小时前
C++ 笔记 友元(面向对象)
开发语言·c++·笔记
C++ 老炮儿的技术栈2 小时前
分享一个安全的CString
c语言·c++·windows·git·安全·visual studio
穿条秋裤到处跑2 小时前
每日一道leetcode(2026.03.31):字典序最小的生成字符串
算法·leetcode
桦03 小时前
[C++复习]:STL
开发语言·c++
苏宸啊4 小时前
rbtree封装map和set
c++
汉克老师4 小时前
GESP2025年6月认证C++三级( 第一部分选择题(1-8))
c++·二进制·原码·补码·gesp三级·gesp3级·八进制、