分金子(奇虎360 2017春招真题)

A、B两伙马贼意外地在一片沙漠中发现了一处金矿,双方都想独占金矿,但各自的实力都不足以吞下对方,经过谈判后,双方同意用一个公平的方式来处理这片金矿。处理的规则如下:他们把整个金矿分成n段,由A、B开始轮流从最左端或最右端占据一段,直到分完为止。 

马贼A想提前知道他们能分到多少金子,因此请你帮忙计算他们最后各自拥有多少金子?(两伙马贼均会采取对己方有利的策略)


输入

测试数据包含多组输入数据。输入数据的第一行为一个正整数T(T<=20),表示测试数据的组数。然后是T组测试数据,每组测试数据的第一行包含一个整数n,下一行包含n个数(n <= 500 ),表示每段金矿的含金量,保证其数值大小不超过1000。


样例输入

6

4 7 2 9 5 2

10

140 649 340 982 105 86 56 610 340 879


输出

对于每一组测试数据,输出一行"Case #id: sc1 sc2",表示第id组数据时马贼A分到金子数量为sc1,马贼B分到金子数量为sc2。详见样例。


样例输出

Case #1: 18 11

Case #2: 3206 981


时间限制

C/C++语言:1000MS

其他语言:3000MS

内存限制

C/C++语言:65536KB

其他语言:589824KB


一道dp,半年不做题,简直菜成辣鸡QAQ

不过本来就不会dp。。说到底还是自己太菜。。。


#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
using namespace std;
const int MAXN = 1e5 + 10;
#define PI acos(-1.0)

int dp[505][505];   
int sum[505];  //前缀和

int solve(int l, int r)
{
    if (l == r) {
        dp[l][r] = sum[r] - sum[l-1];
        return dp[l][r];
    }
    if (dp[l][r] != 0) return dp[l][r];  //最开始没写这行,TLE了还一脸黑人问号??? 说好的记忆化搜索只剩下搜索2333
    dp[l][r] = sum[r] - sum[l-1] - min(solve(l+1, r), solve(l, r-1));
    return dp[l][r];
}

int main()
{
    int cast;
    scanf("%d", &cast);
    for (int casi = 1; casi <= cast; casi++) {
        int n;
        scanf("%d", &n);
        memset(dp, 0, sizeof(dp));
        sum[0] = 0;
        for (int i = 1; i <= n; i++) {
            scanf("%d", &sum[i]);
            if (i == 1) continue;
            sum[i] = sum[i] + sum[i-1];
        }
        int ans = solve(1, n);
        int ans2 = sum[n] - ans;
        printf("Case #%d: %d %d\n", casi, ans, ans2);
    }
    return 0;
}


你可能感兴趣的:(dp)