程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> C++ >> C++入門知識 >> HDU--杭電--1978--How many ways--記憶化搜索--這個鳥東西應該能算半個DP吧

HDU--杭電--1978--How many ways--記憶化搜索--這個鳥東西應該能算半個DP吧

編輯:C++入門知識

How many ways
Time Limit: 3000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)
Total Submission(s): 1970    Accepted Submission(s): 1204

 


Problem Description
這是一個簡單的生存游戲,你控制一個機器人從一個棋盤的起始點(1,1)走到棋盤的終點(n,m)。游戲的規則描述如下:
1.機器人一開始在棋盤的起始點並有起始點所標有的能量。
2.機器人只能向右或者向下走,並且每走一步消耗一單位能量。
3.機器人不能在原地停留。
4.當機器人選擇了一條可行路徑後,當他走到這條路徑的終點時,他將只有終點所標記的能量。

 

\


 

如上圖,機器人一開始在(1,1)點,並擁有4單位能量,藍色方塊表示他所能到達的點,如果他在這次路徑選擇中選擇的終點是(2,4)

點,當他到達(2,4)點時將擁有1單位的能量,並開始下一次路徑選擇,直到到達(6,6)點。
我們的問題是機器人有多少種方式從起點走到終點。這可能是一個很大的數,輸出的結果對10000取模。
 

 

Input
第一行輸入一個整數T,表示數據的組數。
對於每一組數據第一行輸入兩個整數n,m(1 <= n,m <= 100)。表示棋盤的大小。接下來輸入n行,每行m個整數e(0 <= e < 20)。
 

 

Output
對於每一組數據輸出方式總數對10000取模的結果.
 

 

Sample Input
1
6 6
4 5 6 6 4 3
2 2 3 1 7 2
1 1 4 6 2 7
5 8 4 3 9 5
7 6 6 2 1 5
3 1 1 3 7 2

 

Sample Output
3948

<SPAN style="BACKGROUND-COLOR: #ffffff"><STRONG>#include <iostream>  
using namespace std;  
int map[111][111],dp[111][111],n,m;  
void DP(int x,int y)  //我的思想就是從這一點往可以走的點走,走到哪裡就把這一點的次數加上去   
{  
    int i,j;  
    for(i=0;i<=map[x][y];i++)  //用i和j來配合成x+i與y+j來選擇能到達的所有點   
        for(j=0;j<=map[x][y]-i;j++)  
            if((i!=0||j!=0)&&x+i<n&&y+j<m)  //i=0且j=0時x+i與y+j就是當前點,總不能你不懂我也把能到自己的次數加給自己吧?   
                dp[x+i][y+j]+=dp[x][y],dp[x+i][y+j]%=10000;  //就是由(x,y)能到達的點都加上從起點到達(x,y)的次數,並取模   
}  
int main (void)  
{  
    int i,j,k,l,t;  
    cin>>t;  
    while(t--&&cin>>n>>m)  
    {  
        for(i=0;i<n;i++)  
            for(j=0;j<m;j++)  
                cin>>map[i][j],dp[i][j]=0;dp[0][0]=1;  
            for(i=0;i<n;i++)  //遍歷懂?好吧,你蠢,那就換個詞,換成全部數據遍歷- -!   
                for(j=0;j<m;j++)  
                DP(i,j);  //把每個點都往能走的點走一遍   
            cout<<dp[n-1][m-1]<<endl;  //最後dp裡面都是記錄的從起點到這裡的方法數   
    }  
    return 0;  
}</STRONG>  
  
</SPAN>  

#include <iostream>
using namespace std;
int map[111][111],dp[111][111],n,m;
void DP(int x,int y)  //我的思想就是從這一點往可以走的點走,走到哪裡就把這一點的次數加上去
{
    int i,j;
    for(i=0;i<=map[x][y];i++)  //用i和j來配合成x+i與y+j來選擇能到達的所有點
        for(j=0;j<=map[x][y]-i;j++)
            if((i!=0||j!=0)&&x+i<n&&y+j<m)  //i=0且j=0時x+i與y+j就是當前點,總不能你不懂我也把能到自己的次數加給自己吧?
                dp[x+i][y+j]+=dp[x][y],dp[x+i][y+j]%=10000;  //就是由(x,y)能到達的點都加上從起點到達(x,y)的次數,並取模
}
int main (void)
{
    int i,j,k,l,t;
    cin>>t;
    while(t--&&cin>>n>>m)
    {
        for(i=0;i<n;i++)
            for(j=0;j<m;j++)
                cin>>map[i][j],dp[i][j]=0;dp[0][0]=1;
            for(i=0;i<n;i++)  //遍歷懂?好吧,你蠢,那就換個詞,換成全部數據遍歷- -!
                for(j=0;j<m;j++)
                DP(i,j);  //把每個點都往能走的點走一遍
            cout<<dp[n-1][m-1]<<endl;  //最後dp裡面都是記錄的從起點到這裡的方法數
    }
    return 0;
}



 能懂?不懂?那你還真蠻笨蛋的咯,我好好來講講?也許看代碼能看懂,但是我要讓你印象深刻,就討論一個點,題目說了只能往右或者下走,那麼就只要找右邊和下面所有的當前的能量可以到達的點,你想啊,我能從A點到B點,說明所有能到A的次數也都能繼續到達B,不是麼?這樣是不是灰常簡單咯?還好意思說不懂啊?妹的別挑戰我的底線,中國人可不是吃醋的,是吃大蒜的- -!dp數組就是記錄當前點中能到的次數,因為只能向右和向下走,所以我從上向下並每行都從左到右不是正好都能求出他們的次數麼?這樣就不用擔心會出現能到A點的次數還沒全部統計下來就用A去計算B了,好好想想,親,你是很聰明的

 

 

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved