NOIP2007矩阵取数[DP|高精度]

题目描述

帅帅经常跟同学玩一个矩阵取数游戏:对于一个给定的n*m的矩阵,矩阵中的每个元素aij均为非负整数。游戏规则如下:

1.每次取数时须从每行各取走一个元素,共n个。m次后取完矩阵所有元素;

2.每次取走的各个元素只能是该元素所在行的行首或行尾;

3.每次取数都有一个得分值,为每行取数的得分之和,每行取数的得分 = 被取走的元素值*2^i,其中i表示第i次取数(从1开始编号);

4.游戏结束总得分为m次取数得分之和。

帅帅想请你帮忙写一个程序,对于任意矩阵,可以求出取数后的最大得分。

输入输出格式

输入格式:

输入文件game.in包括n+1行:

第1行为两个用空格隔开的整数n和m。

第2~n+1行为n*m矩阵,其中每行有m个用单个空格隔开的非负整数。

数据范围:

60%的数据满足:1<=n, m<=30,答案不超过10^16

100%的数据满足:1<=n, m<=80,0<=aij<=1000

输出格式:

输出文件game.out仅包含1行,为一个整数,即输入矩阵取数后的最大得分。

输入输出样例

输入样例#1:

2 3
1 2 3
3 4 2

输出样例#1:

82

说明

NOIP 2007 提高第三题

-------------------------------------

DP

行与行之间相互独立,各行单独求解;一定先清空

f(i,j)左选了i个,右选了j个的最大得分

状态转移很好想,注意边界处理

高精度[这次从数组1开始]

高加高,高乘低

输出时一定小心0

建议先写出long long 的方便调试

#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
const int N=85;
const int B=1e4,L=100;

struct big{
    int size,d[L];
    big(int a=1):size(a){memset(d,0,sizeof(int)*L);}
};

void jia(big &a,big &b){
    int g=0,i;
    for(i=1;;i++){
        if(g==0&&i>a.size&&i>b.size) break;
        int tmp=g;
        if(i<=a.size) tmp+=a.d[i];
        if(i<=b.size) tmp+=b.d[i];
        a.d[i]=tmp%B;
        g=tmp/B;
    }
    a.size=i-1;
}

void chengInt(big &a,int k){
    int g=0,i;
    for(i=1;i<=a.size;i++){
        int tmp=a.d[i]*k;
        a.d[i]=(tmp+g)%B;
        g=(tmp+g)/B;
    }
    while(g){
        a.d[++a.size]=g%B;
        g/=B;
    }
}

void copy(big &t,big &s){
    t.size=s.size;
    for(int i=1;i<=s.size;i++) t.d[i]=s.d[i];
}

big max(big &a,big &b){
    if(a.size>b.size) return a;
    if(a.size<b.size) return b;
    for(int i=a.size;i>=1;i--){
        if(a.d[i]>b.d[i]) return a;
        if(a.d[i]<b.d[i]) return b;
    }
    return a;
}

void print(big &a){
    for(int i=a.size;i>=1;i--){
        if(i==a.size);
        else if(a.d[i]<10) cout<<"0000";
        else if(a.d[i]<100) cout<<"00";
        else if(a.d[i]<1000) cout<<"0";
        cout<<a.d[i];
    }
    cout<<"\n";
}

int n,m;
int a[N][N];
big f[N][N];
big ans,fin=0;

big two[N];
void PRE(){
    two[0].d[1]=1;
    for(int i=1;i<=m+1;i++){
        copy(two[i],two[i-1]);
        chengInt(two[i],2);
    }
}
big sol(int num){
    ans=big();
    for(int i=0;i<=n;i++)
        for(int j=0;j<=n;j++) f[i][j]=big();

    f[1][0].d[1]=a[num][1]*2;
    f[0][1].d[1]=a[num][m]*2;
    for(int i=2;i<=m;i++){
//        f[i][0]=f[i-1][0]+a[num][i]*((ll)1<<i);
//        f[0][i]=f[0][i-1]+a[num][m-i+1]*((ll)1<<i);

        copy(f[i][0],f[i-1][0]);
        big tmp; copy(tmp,two[i]); chengInt(tmp,a[num][i]);
        jia(f[i][0],tmp);

        copy(f[0][i],f[0][i-1]);
        big tmp2; copy(tmp2,two[i]); chengInt(tmp2,a[num][m-i+1]);
        jia(f[0][i],tmp2);

    }
    for(int i=1;i<=m;i++)
        for(int j=1;j<=m-i;j++){
//            f[i][j]=max(f[i-1][j]+a[num][i]*((ll)1<<(i+j)),
//                        f[i][j-1]+a[num][m-j+1]*((ll)1<<(i+j)));

            big x,y;
            copy(x,f[i-1][j]);
            big tmp; copy(tmp,two[i+j]); chengInt(tmp,a[num][i]);
            jia(x,tmp);

            copy(y,f[i][j-1]);
            big tmp2; copy(tmp2,two[i+j]); chengInt(tmp2,a[num][m-j+1]);
            jia(y,tmp2);

            f[i][j]=max(x,y);
        }

    for(int i=0;i<=m;i++) ans=max(ans,f[i][m-i]);
    return ans;
}

int main(){//system("pause");
    cin>>n>>m;
    PRE();
    for(int i=1;i<=n;i++)
        for(int j=1;j<=m;j++)
            cin>>a[i][j];

    for(int i=1;i<=n;i++) {big tmp=sol(i);jia(fin,tmp);}
    print(fin);

}
时间: 2024-08-02 06:58:34

NOIP2007矩阵取数[DP|高精度]的相关文章

NOIP2007 矩阵取数游戏

题目描述 Description [问题描述] 帅帅经常跟同学玩一个矩阵取数游戏:对于一个给定的n*m 的矩阵,矩阵中的每个元素aij均 为非负整数.游戏规则如下: 1. 每次取数时须从每行各取走一个元素,共n个.m次后取完矩阵所有元素: 2. 每次取走的各个元素只能是该元素所在行的行首或行尾: 3. 每次取数都有一个得分值,为每行取数的得分之和,每行取数的得分= 被取走的元素值*2i, 其中i 表示第i 次取数(从1 开始编号): 4. 游戏结束总得分为m次取数得分之和. 帅帅想请你帮忙写一个

TYVJ 矩阵取数 Label:高精度+dp

题目描述 帅帅经常跟同学玩一个矩阵取数游戏:对于一个给定的n*m的矩阵,矩阵中的每个元素aij均为非负整数.游戏规则如下: 1.每次取数时须从每行各取走一个元素,共n个.m次后取完矩阵所有元素: 2.每次取走的各个元素只能是该元素所在行的行首或行尾: 3.每次取数都有一个得分值,为每行取数的得分之和,每行取数的得分 = 被取走的元素值*2^i,其中i表示第i次取数(从1开始编号): 4.游戏结束总得分为m次取数得分之和. 帅帅想请你帮忙写一个程序,对于任意矩阵,可以求出取数后的最大得分. 输入输

NOIP2007矩阵取数游戏

眼看就要刷完NOIP2007了,然而我忽然想放弃了,咱家图论崩的一B,最后一题可能要推迟放出了,咱们返回正题:矩阵取数 题目简述:给你一个n*m的矩阵,每次从每一行的行头或行末取出一个数,这个数乘上2 ^ i (i 是 第几次取数) 加到answer中,求answer的最大值 你是不是有一种贪心的冲动?我就无脑写了一个,对于每次的两个选择中取较小的数,加到answer中,然而三组样例只能过一组. 贪心是不可行的,证明呢,由程序完成.让我们来想想正解应该怎么写吧,具体的分析过后我们可以发现一个隐含

Noip2007矩阵取数游戏题解

题目描述 Description [问题描述] 帅帅经常跟同学玩一个矩阵取数游戏:对于一个给定的n?m的矩阵,矩阵中的每个元素ai,j均 为非负整数.游戏规则如下: 每次取数时须从每行各取走一个元素,共n个.m次后取完矩阵所有元素: 每次取走的各个元素只能是该元素所在行的行首或行尾: 每次取数都有一个得分值,为每行取数的得分之和,每行取数的得分=被取走的元素值?2i, 其中i表示第i 次取数(从1 开始编号): 游戏结束总得分为m次取数得分之和. 帅帅想请你帮忙写一个程序,对于任意矩阵,可以求出

51Nod 1083 矩阵取数问题(矩阵取数dp,基础题)

1083 矩阵取数问题 基准时间限制:1 秒 空间限制:131072 KB 分值: 5 难度:1级算法题 一个N*N矩阵中有不同的正整数,经过这个格子,就能获得相应价值的奖励,从左上走到右下,只能向下向右走,求能够获得的最大价值. 例如:3 * 3的方格. 1 3 3 2 1 3 2 2 1 能够获得的最大价值为:11. Input 第1行:N,N为矩阵的大小.(2 <= N <= 500) 第2 - N + 1行:每行N个数,中间用空格隔开,对应格子中奖励的价值.(1 <= N[i] 

dp+高精度(洛谷1005 矩阵取数游戏NOIP 2007 提高第三题)

帅帅经常跟同学玩一个矩阵取数游戏:对于一个给定的n*m的矩阵,矩阵中的每个元素aij均为非负整数.游戏规则如下: 1.每次取数时须从每行各取走一个元素,共n个.m次后取完矩阵所有元素: 2.每次取走的各个元素只能是该元素所在行的行首或行尾: 3.每次取数都有一个得分值,为每行取数的得分之和,每行取数的得分 = 被取走的元素值*2^i,其中i表示第i次取数(从1开始编号): 4.游戏结束总得分为m次取数得分之和. 帅帅想请你帮忙写一个程序,对于任意矩阵,可以求出取数后的最大得分. 输入格式: 输入

1166 矩阵取数游戏[区间dp+高精度]

1166 矩阵取数游戏 2007年NOIP全国联赛提高组 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 黄金 Gold 题解 题目描述 Description [问题描述]帅帅经常跟同学玩一个矩阵取数游戏:对于一个给定的n*m 的矩阵,矩阵中的每个元素aij均为非负整数.游戏规则如下:1. 每次取数时须从每行各取走一个元素,共n个.m次后取完矩阵所有元素:2. 每次取走的各个元素只能是该元素所在行的行首或行尾:3. 每次取数都有一个得分值,为每行取数的得分之和,每行取数的得分

[LuoguP1005]矩阵取数游戏 (DP+高精度)

题面 传送门:https://www.luogu.org/problemnew/show/P1005 Solution 我们可以先考虑贪心 我们每一次都选左右两边尽可能小的数,方便大的放在后面 听起来挺OK的 实则并不OK 反例: 3 1 1 1 1 1 2 2 2 2 2 2 2 2 2 2 如果贪心的话,我们会优先把右边那一串2先选了,再去选3和1 但是正确答案显然是先把3和1选了,再去选那一串2 既然贪心不成,我们可以考虑一下DP 然后我们考虑这样一个状态: f[i][j][k] 表示第i

【NOIP2007】矩阵取数

因为傻逼写错高精度搞了一下午浪费好多时间,好想哭qaq 原题: 帅帅经常更同学玩一个矩阵取数游戏:对于一个给定的n*m的矩阵,矩阵中的每个元素aij据为非负整数.游戏规则如下: 1. 每次取数时须从每行各取走一个元素,共n个.m次后取完矩阵所有的元素: 2. 每次取走的各个元素只能是该元素所在行的行首或行尾: 3. 每次取数都有一个得分值,为每行取数的得分之和:每行取数的得分 = 被取走的元素值*2^i,其中i表示第i次取数(从1开始编号):  4. 游戏结束总得分为m次取数得分之和. 帅帅想请