Lightoj1205——Palindromic Numbers(数位dp+回文数)

A palindromic number or numeral palindrome is a ‘symmetrical’ number like 16461 that remains the same when its digits are reversed. In this problem you will be given two integers i j, you have to find the number of palindromic numbers between i and j (inclusive).

Input 
Input starts with an integer T (≤ 200), denoting the number of test cases.

Each case starts with a line containing two integers i j (0 ≤ i, j ≤ 1017).

Output 
For each case, print the case number and the total number of palindromic numbers between i and j (inclusive).

Sample Input 

1 10 
100 1 
1 1000 
1 10000 
Output for Sample Input 
Case 1: 9 
Case 2: 18 
Case 3: 108 
Case 4: 198

代码:

 1 #include<bits/stdc++.h>
 2 #define ll long long
 3 using namespace std;
 4 int a[40],tmp[40];
 5 ll dp[40][100][100];
 6 ll dfs(int start,int pos,int ok,bool limit)
 7 {
 8     if(pos<0)
 9         return ok;
10     if(!limit&&dp[pos][ok][start]!=-1)
11         return dp[pos][ok][start];
12     ll ans=0;
13     int up=limit?a[pos]:9;
14     for(int d=0; d<=up; ++d)
15     {
16         tmp[pos]=d;
17         if(start==pos&&d==0)
18             ans+=dfs(start-1,pos-1,ok,limit&&d==up);
19         else if(ok&&pos<(start+1)/2)
20             ans+=dfs(start,pos-1,tmp[start-pos]==d,limit&&d==up);
21         else
22             ans+=dfs(start,pos-1,ok,limit&&d==up);
23     }
24     if(!limit)
25         dp[pos][ok][start]=ans;
26     return ans;
27 }
28 ll solve(ll x)
29 {
30     memset(a,0,sizeof(a));
31     int cnt=0;
32     while(x!=0)
33     {
34         a[cnt++]=x%10;
35         x/=10;
36     }
37     return dfs(cnt-1,cnt-1,1,1);
38 }
39 int main()
40 {
41     memset(dp,-1,sizeof(dp));
42     int t,cnt=1;
43     scanf("%d",&t);
44     while(t--)
45     {
46         ll x,y;
47         scanf("%lld%lld",&x,&y);
48         if(x>y)
49             swap(x,y);
50         printf("Case %d: %lld\n",cnt++,solve(y)-solve(x-1));
51     }
52     return 0;
53 }

 

posted @ 2017-08-20 16:33  thges  阅读(349)  评论(0编辑  收藏  举报