题目链接:点击打开链接
题意:给你n个火柴棍, 要求你恰好用完, 来组成一个一个等式, 等式的形式是a - b = c 。求可以组成的等式个数。
思路:很明显的数位DP, 不过巧妙的是, 该题利用了手动模拟大数相加的过程,首先, 我们不妨将等式改成b + c = a, 用d[res][a][b][c] 表示还剩res根火柴, 当前对应位相加之后有没有进位, b和c是否已经停止放火柴棒 的方法数。
细节参见代码:
#include<cstdio> #include<cstring> #include<algorithm> #include<iostream> #include<string> #include<vector> #include<stack> #include<bitset> #include<cstdlib> #include<cmath> #include<set> #include<list> #include<deque> #include<map> #include<queue> #define Max(a,b) ((a)>(b)?(a):(b)) #define Min(a,b) ((a)<(b)?(a):(b)) using namespace std; typedef long long ll; typedef long double ld; const ld eps = 1e-6, PI = 3.1415926535897932384626433832795; const int mod = 1000000000 + 7; const int INF = 0x3f3f3f3f; // & 0x7FFFFFFF const int seed = 131; const ll INF64 = ll(1e18); const int maxn = 500 + 10; int T,kase = 0,vis[maxn][2][2][2]; int table[] = {6, 2, 5, 5, 4, 5, 6, 3, 7, 6}; ll m,n,d[maxn][2][2][2]; inline void add(ll& a, ll b) { a += b; if(a >= m) a -= m; } ll dp(int res, int A, int B, int C) { ll& ans = d[res][A][B][C]; if(!B && !C) { if(res == 0 && !A) return 1; else if(A && table[1] == res) return 1; else return 0; } if(vis[res][A][B][C] == kase) return ans; vis[res][A][B][C] = kase; ans = 0; if(B && C) { for(int i=0;i<10;i++) { for(int j=0;j<10;j++) { ll sum = table[i] + table[j] + table[(i + j + A) % 10]; if(sum > res) continue; bool nxt = i + j + A >= 10; add(ans, dp(res - sum, nxt, 1, 1)); if(j) add(ans, dp(res - sum, nxt, 1, 0)); if(i) add(ans, dp(res - sum, nxt, 0, 1)); if(i && j) add(ans, dp(res - sum, nxt, 0, 0)); } } } else if(B) { for(int i=0;i<10;i++) { ll sum = table[i] + table[(i + A) % 10]; if(sum > res) continue; bool nxt = i + A >= 10; add(ans, dp(res - sum, nxt, 1, 0)); if(i) add(ans, dp(res - sum, nxt, 0, 0)); } } else if(C) { for(int i=0;i<10;i++) { ll sum = table[i] + table[(i + A) % 10]; if(sum > res) continue; bool nxt = i + A >= 10; add(ans, dp(res - sum, nxt, 0, 1)); if(i) add(ans, dp(res - sum, nxt, 0, 0)); } } return ans; } int main() { scanf("%d",&T); while(T--) { scanf("%I64d%I64d",&n,&m); printf("Case #%d: ",++kase); ll ans = dp(n - 3, 0, 1, 1); printf("%I64d\n",ans); } return 0; }