【jzoj1732】【DP】【10.5NOIP普及模拟】count(count.cpp/pas)

57 篇文章 0 订阅
题目描述

小x开发了一个奇怪的游戏,这个游戏的是这样的:一个长方形,被分成 N N N M M M 列的格子,第 i i i行第 j j j 列的格子记为 ( i , j ) (i, j) (ij),就是说,左上角的格子是 ( 1 , 1 ) (1, 1) (11),右下角的格子是 ( N , M ) (N, M) (NM)。开始的时候,小y在 ( 1 , 1 ) (1, 1) (11),他需要走到 ( N , M ) (N, M) (NM)。每一步,小y可以走到正右方或者正下方的一个格子。具体地说,如小y现在在 ( x , y ) (x, y) (xy),那么他可以走到 ( x , y + 1 ) (x, y + 1) (xy+1) ( x + 1 , y ) (x + 1, y) (x+1y)。当然,小y不能走出离开这个长方形。

每个格子有积分,用一个 1 1 1~ 10 10 10 的整数表示。经过这个格子,就会获取这个格子的积分(起点和终 点的积分也计算)。通过的方法是:到达 ( N , M ) (N, M) (NM) 的时候,积分恰好为 P P P

现在给出这个长方形每个格子的积分,你需要帮助小y,求出从起点走到终点,积分为 P P P 的线路有多少条。

输入

第 1 行3 个整数 N , M , P N, M, P N,M,P

接下来 N N N 行,每行 M M M 个整数 A [ i ] [ j ] A[i][j] A[i][j] ,表示格子 ( i , j ) (i, j) (ij) 的积分。

输出

1 行1 个整数,表示积分为 P P P 线路的数量。

因为数值太大,你只需要输出结果除以 ( 1 0 9 + 7 ) (10^9 + 7) (109+7) 的 余数。

样例输入

3 3 9
2 2 1
2 2 2
1 2 2

样例输出

2

数据范围限制

对于 50 50 50% 的数据: 1 ≤ N , M ≤ 10 1 ≤ N, M ≤ 10 1NM10

对于 100 100 100% 的数据: 1 ≤ N , M ≤ 100 , 0 ≤ A [ i ] [ j ] ≤ 10 1 ≤ N, M ≤ 100,0 ≤ A[i][j] ≤ 10 1NM1000A[i][j]10


解题思路

f [ i ] [ j ] [ k ] f[i][j][k] f[i][j][k]为走到第 i i i行第 j j j列和为 k k k的方案数。

  • i − 1 , j i-1,j i1,j走到 i , j i,j i,j,因为走到 i , j i,j i,j的和是k,那么走到 i − 1 , j i-1,j i1,j的和就是 k − a [ i ] [ j ] k-a[i][j] ka[i][j] f [ i ] [ j ] = f [ i − 1 ] [ j ] [ k − a [ i ] [ j ] ] f[i][j]=f[i-1][j][k-a[i][j]] f[i][j]=f[i1][j][ka[i][j]]
  • 同上, f [ i ] [ j ] = f [ i − 1 ] [ j ] [ k − a [ i ] [ j ] ] + f [ i ] [ j − 1 ] [ k − a [ i ] [ j ] ] f[i][j]=f[i-1][j][k-a[i][j]]+f[i][j-1][k-a[i][j]] f[i][j]=f[i1][j][ka[i][j]]+f[i][j1][ka[i][j]]

那么答案就是 f [ n ] [ m ] [ p ] f[n][m][p] f[n][m][p]


#include<iostream>
#include<cstdio>
using namespace std;
const int Mod=1000000007;
int a[105][105],f[105][105][5050],n,m,p;
int main(){
//	freopen("count.in","r",stdin);
//	freopen("count.out","w",stdout);
	scanf("%d%d%d",&n,&m,&p);
	for(int i=1;i<=n;i++)
	    for(int j=1;j<=m;j++)
	        scanf("%d",&a[i][j]);
	f[1][1][a[1][1]]=1;//初始值
	for(int i=1;i<=n;i++)
	    for(int j=1;j<=m;j++)
	        for(int k=a[i][j];k<=p;k++){
	            if(i-1>0)//注意边界
				   if(j-1>0)
				      f[i][j][k]=(f[i-1][j][k-a[i][j]]+f[i][j-1][k-a[i][j]])%Mod;	
				    else
					  f[i][j][k]=f[i-1][j][k-a[i][j]]%Mod;
				else
				   if(j-1>0)
				       f[i][j][k]=f[i][j-1][k-a[i][j]]%Mod;
			}
	printf("%d",f[n][m][p]);
} 
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值