洛谷 1373 小a和uim之大逃离

这是一道dp好题

咱们读读题

题意就是求到达某一点使得两人获取的能量一样

本蒟蒻一开始想了半天都没能把两者能量一样转换成两者差为0

若能这么转换,dp方程就能得出来了

dp[i][j][h][1]=(dp[i][j][h][1]+dp[i-1][j][(h+d[i][j])%k][0])%mod;

dp[i][j][h][1]=(dp[i][j][h][1]+dp[i][j-1][(h+d[i][j])%k][0])%mod;

dp[i][j][h][0]=(dp[i][j][h][0]+dp[i-1][j][(h-d[i][j]+k)%k][1])%mod;

dp[i][j][h][0]=(dp[i][j][h][0]+dp[i][j-1][(h-d[i][j]+k)%k][1])%mod;

i,j表示到达的位置坐标为 i,j

h表示两者获得的能量之差(小a-uim)

0表示当前走的人是小a,1表示当前走的人是uim

状态转移就如上

当前是1时,就是由上一个点的差值加上这个点的值

当前是0时,就是由上一个点的差值在减去这个点的值

不要忘记%k

先看看代码,后面会讲解坑点

#include <cstdio>
#include <cstdlib>
#include <iostream>
#include <algorithm>
#include <cstring>
using namespace std;
typedef long long ll;
const int mod=1000000007;
const int N=805;
int dp[N][N][20][2],d[N][N],n,m,k;
int main()
{
    scanf("%d %d %d",&n,&m,&k);
    k++;
    for(int i=1;i<=n;i++)
        for(int j=1;j<=m;j++)
        {
            scanf("%d",&d[i][j]);
            dp[i][j][d[i][j]%k][0]=1;
        }
    for(int i=1;i<=n;i++)
        for(int j=1;j<=m;j++)
        {
            for(int h=0;h<=k;h++)
            {
                dp[i][j][h][1]=(dp[i][j][h][1]+dp[i-1][j][(h+d[i][j])%k][0])%mod;
                dp[i][j][h][1]=(dp[i][j][h][1]+dp[i][j-1][(h+d[i][j])%k][0])%mod;
                dp[i][j][h][0]=(dp[i][j][h][0]+dp[i-1][j][(h-d[i][j]+k)%k][1])%mod;
                dp[i][j][h][0]=(dp[i][j][h][0]+dp[i][j-1][(h-d[i][j]+k)%k][1])%mod;
            }
        }
    int ans=0;
    for(int i=1;i<=n;i++)
        for(int j=1;j<=m;j++)
        {
            ans=(ans+dp[i][j][0][1])%mod;
        }
    printf("%d
",ans);

}

注意:

1.一开始的初始化,应该把小a放在每个点所获得的能量值都求出来

2.%运算是有优先级的

  上述式子就不能写成dp[i][j][h][1]+=dp[i-1][j][(h+d[i][j])%k][0]%mod,因为这是先给后面取模,在进行加法运算,

  这里坑了我很久

3.因为减法会产生负数,所以要加上k在取模

4.这题还卡空间,如果定成long long 就会爆掉

5.别忘记k++,看清楚题

差不多都讲完了,希望对你们有帮助

原文地址:https://www.cnblogs.com/wzrdl/p/9785996.html