码迷,mamicode.com
首页 > 其他好文 > 详细

hdu 5576 dp

时间:2018-07-19 10:49:14      阅读:140      评论:0      收藏:0      [点我收藏+]

标签:strlen   ons   lld   乘法   需要   思路   cond   include   void   

题目大意:给你一个长度为 n 的 字符串表示一个乘法,一次操作随机选两个字符进行交换,进行m次操作,让你求出所有可能操作

的答案和。  (1 <= n, m <= 50)

 

思路:巨难。。 对于固定位置的三个字符来说,把中间的看成*, 两边的为a,b, 这个产生的贡献为,a * b * C,C只与a, b和 * 的位置有关,

所以我们只要求出a*b的贡献就好啦。

我们定义dp[ a ][ b ][ c ][ k ] 表示三个字符分别是a,b,c, b在中间,还需要交换 k次的答案。

最后枚举三个位置求答案。

#include<bits/stdc++.h>
#define LL long long
#define fi first
#define se second
#define mk make_pair
#define pii pair<int, int>
#define y1 skldjfskldjg
#define y2 skldfjsklejg

using namespace std;

const int N = 100 + 7;
const int M = 1e5 + 7;
const int inf = 0x3f3f3f3f;
const LL INF = 0x3f3f3f3f3f3f3f3f;
const int mod = 1e9 +7;

char s[56];
int n, m, a[56], cnt[11];
LL f[11][11][11][51], ten[56];

void add(LL &a, LL b) {
    a += b; if(a >= mod) a -= mod;
}

LL dp(int a, int b, int c, int m) {
//    printf("%d %d %d %d $$\n", a, b, c, m);
    if(!m) return b == 10 ? a * c : 0;
    if(a > c) swap(a, c);
    LL &ans = f[a][b][c][m];
    if(~ans) return ans;
    ans = 0;

    add(ans, dp(c, b, a, m - 1));   // a <-> c
    add(ans, dp(b, a, c, m - 1));   // a <-> b
    add(ans, dp(a, c, b, m - 1));   // b <-> c

    LL num = (n - 3) * (n - 4) / 2;

    if(num) add(ans, num * dp(a, b, c, m - 1) % mod);  // a, b, c 之外的两个数相互交换

    int t[11];
    memcpy(t, cnt, sizeof(t));
    t[a]--; t[b]--; t[c]--;
    for(int i = 0; i <= 10; i++) { // a, b, c 之外的一个数与 a 或 b 或 c 交换
        if(!t[i]) continue;
        add(ans, t[i] * dp(i, b, c, m - 1) % mod); //挑一种数与a交换
        add(ans, t[i] * dp(a, i, c, m - 1) % mod); //挑一种数与b交换
        add(ans, t[i] * dp(a, b, i, m - 1) % mod); //挑一种数与c交换
    }

    return ans;
}

int main() {

    ten[0] = 1;
    for(int i = 1; i <= 55; i++) ten[i] = ten[i - 1] * 10 % mod;

    int T; scanf("%d", &T);

    for(int cas = 1; cas <= T; cas++) {
        memset(f, -1, sizeof(f));
        memset(cnt, 0, sizeof(cnt));
        scanf("%d%s", &m, s);
        n = strlen(s);
        printf("Case #%d: ", cas);

        for(int i = 0; i < n; i++) {
            a[i] = (s[i] == * ? 10 : s[i] - 0);
            cnt[a[i]]++;
        }

        LL ans = 0;

        for(int i = 0; i < n; i++) {
            for(int j = i + 1; j < n; j++) {
                for(int k = j + 1; k < n; k++) {
                    add(ans, ten[j - i - 1] * ten[n - k - 1] % mod * dp(a[i], a[j], a[k], m) % mod);
                }
            }
        }

        printf("%lld\n", ans);
    }
    return 0;
}


/*
2
1
1*2
2
1*2
*/

 

hdu 5576 dp

标签:strlen   ons   lld   乘法   需要   思路   cond   include   void   

原文地址:https://www.cnblogs.com/CJLHY/p/9334070.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!