【NOIP2007】矩阵取数
因为傻逼写错高精度搞了一下午浪费好多时间,好想哭qaq
原题:
帅帅经常更同学玩一个矩阵取数游戏:对于一个给定的n*m的矩阵,矩阵中的每个元素aij据为非负整数。游戏规则如下:
1. 每次取数时须从每行各取走一个元素,共n个。m次后取完矩阵所有的元素;
2. 每次取走的各个元素只能是该元素所在行的行首或行尾;
3. 每次取数都有一个得分值,为每行取数的得分之和;每行取数的得分 = 被取走的元素值*2^i,其中i表示第i次取数(从1开始编号);
4. 游戏结束总得分为m次取数得分之和。
帅帅想请你帮忙写一个程序,对于任意矩阵,可以求出取数后的最大得分。
1<=n, m<=80,0<=a[i][j]<=1000
首先每一行怎么取互不影响,就可以分开来搞,最后加到一起,下面说的f,a都是某一行里的
然后每一行中区间DP,f[i][j]表示i到j这个区间最大值多少,f[i][j]=max(f[i+1][j]+a[i],f[i][j-1]+a[j])*2
在求f[i][j]的时候直接在后面*2,这样子就不用计算2^i的高精度运算
高精度傻逼了一下午,能力会随着时间的推移降低
记录傻逼的高精度错误:
//for(int i=1;i<=x[0];i++) x[i]=f[_left][_right][i]+z;高精度+单精度不是这么写的qaq
正确做法:x[1]+=z
while(x[x[0]+1]){ x[0]++; x[x[0]+1]+=x[x[0]]/ss,x[x[0]]%=ss;}//如果没有每次都清空的话,会因为上一次遗留下来的数继续往后推
//for(int i=1;i<=nleft[0];i++)if(nleft[i]!=nright[i]) return nleft[i]>nright[i];高精度比较应该先比高位qaq
代码:
//因为高精度傻逼了搞了一下午,好想哭qaq
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
using namespace std;
const int ss=;
int m,n,a[][];
int ans[];
int f[][][];
int nleft[],nright[];
void getn(int *x,int _left,int _right,int z){
x[]=f[_left][_right][];
//for(int i=1;i<=x[0];i++) x[i]=f[_left][_right][i]+z;傻逼了qaq
for(int i=;i<=x[];i++) x[i]=f[_left][_right][i];
x[]+=z;
for(int i=;i<=x[];i++) x[i+]+=x[i]/ss,x[i]%=ss;
while(x[x[]+]){ x[]++; x[x[]+]+=x[x[]]/ss,x[x[]]%=ss;}//注意这里,如果没有每次都清空的话,会因为上一次遗留下来的数继续往后推
}
bool getmax(){
if(nleft[]>nright[]) return true;
if(nleft[]<nright[]) return false;
//for(int i=1;i<=nleft[0];i++)if(nleft[i]!=nright[i]) return nleft[i]>nright[i];第二次傻逼qaq
for(int i=nleft[];i>=;i--)if(nleft[i]!=nright[i]) return nleft[i]>nright[i];
return true;
}
void fan(int *x){
for(int i=;i<=x[];i++) x[i]<<=;
for(int i=;i<=x[];i++) x[i+]+=x[i]/ss,x[i]%=ss;
while(x[x[]+]){ x[]++; x[x[]+]+=x[x[]]/ss,x[x[]]%=ss;}
}
void jia(int *x,int *y){
x[]=max(x[],y[]);
for(int i=;i<=x[];i++) x[i]+=y[i];
for(int i=;i<=x[];i++) x[i+]+=x[i]/ss,x[i]%=ss;
while(x[x[]+]){ x[]++; x[x[]+]+=x[x[]]/ss,x[x[]]%=ss;}
}
void copy(int *x,int *y){ y[]=x[]; for(int i=;i<=x[];i++) y[i]=x[i];}
int main(){//freopen("ddd.in","r",stdin);
//freopen("ddd.out","w",stdout);
cin>>m>>n;
for(int i=;i<=m;i++) for(int j=;j<=n;j++) scanf("%d",&a[i][j]);
for(int k=;k<=m;k++){
memset(f,,sizeof(f));
for(int i=;i<=n;i++) f[i][i][f[i][i][]=]=a[k][i]*;//因为输入数据保证a[i][j]<=1000所以直接塞进去就行了
for(int l=;l<=n;l++)
for(int i=,j=i+l-;j<=n;i++,j++){
memset(nleft,,sizeof(nleft)),memset(nright,,sizeof(nright));
getn(nleft,i+,j,a[k][i]),getn(nright,i,j-,a[k][j]);
//cout<<nleft[0]<<endl;
copy((getmax() ? nleft : nright),f[i][j]);
fan(f[i][j]);
/*cout<<f[i][j][f[i][j][0]];
for(int t=f[i][j][0]-1;t>=1;t--) printf("%04d",f[i][j][t]);
cout<<endl;*/
}
jia(ans,f[][n]);
}
cout<<ans[ans[]];
for(int i=ans[]-;i>=;i--) printf("%04d",ans[i]);
cout<<endl;
return ;
}
【NOIP2007】矩阵取数的更多相关文章
- NOIP2007 矩阵取数游戏
题目描述 帅帅经常跟同学玩一个矩阵取数游戏:对于一个给定的n*m的矩阵,矩阵中的每个元素aij均为非负整数.游戏规则如下: 1.每次取数时须从每行各取走一个元素,共n个.m次后取完矩阵所有元素: 2. ...
- NOIP2007矩阵取数[DP|高精度]
题目描述 帅帅经常跟同学玩一个矩阵取数游戏:对于一个给定的n*m的矩阵,矩阵中的每个元素aij均为非负整数.游戏规则如下: 1.每次取数时须从每行各取走一个元素,共n个.m次后取完矩阵所有元素: 2. ...
- [JZYZOJ 1288][洛谷 1005] NOIP2007 矩阵取数 dp 高精度
https://www.luogu.org/problem/show?pid=1005 dp好想,高精度练手题,有点不舒服的是前后取数位置的计算,代码量太少才会写题这么慢,noip之前虽然重点放在 ...
- NOIP2007 矩阵取数游戏(区间DP)
传送门 这道题第一眼看上去可能让人以为是贪心……不过贪心并不行,因为每次的操作是有2的幂次方的权值的.这样的话直接每次贪心最小的就目光短浅.所以那我们自然想到了DP. 据说这是一道很正常的区间DP? ...
- [P1005][NOIP2007] 矩阵取数游戏 (DP+高精)
我不会高精…… 也不会DP…… 这道题即考高精又考DP…… 我要死了 给一个不是高精的代码(当然不能满分) #include<cstdio> #include<iostream> ...
- [NOIP2007] 提高组 洛谷P1005 矩阵取数游戏
题目描述 帅帅经常跟同学玩一个矩阵取数游戏:对于一个给定的n*m的矩阵,矩阵中的每个元素aij均为非负整数.游戏规则如下: 1.每次取数时须从每行各取走一个元素,共n个.m次后取完矩阵所有元素: 2. ...
- 洛谷1005 【NOIP2007】矩阵取数游戏
问题描述 帅帅经常跟同学玩一个矩阵取数游戏:对于一个给定的n*m的矩阵,矩阵中的每个元素aij均为非负整数.游戏规则如下: 1.每次取数时须从每行各取走一个元素,共n个.m次后取完矩阵所有元素: 2. ...
- 矩阵取数游戏 NOIP 2007
2016-05-31 17:26:45 题目链接: NOIP 2007 矩阵取数游戏(Codevs) 题目大意: 给定一个矩阵,每次在每一行的行首或者行尾取一个数乘上2^次数,求取完最多获得的分数 解 ...
- TYVJ 矩阵取数 Label:高精度+dp
题目描述 帅帅经常跟同学玩一个矩阵取数游戏:对于一个给定的n*m的矩阵,矩阵中的每个元素aij均为非负整数.游戏规则如下: 1.每次取数时须从每行各取走一个元素,共n个.m次后取完矩阵所有元素: 2. ...
随机推荐
- iOS开发之通知使用总结
通知中心(NSNotificationCenter) 每一个应用程序都有一个通知中心(NSNotificationCenter)实例,专门负责协助不同对象之间的消息通信 任何一个对象都可以向通知中心发 ...
- html<textarea>标签
最近在项目中页面回显<textarea>的值,可是设置了value属性怎么也回显不出来,后来才弄清楚,原来想要设置<textarea>的文本,不是使用value,而是如下方式: ...
- ubuntu 输入时弹出剪切板候选项
fcitx很坑的把这个功能的快捷键设置成了ctrl + ;结果我在用vim的时候怎么也 没法输入command 不知道是哪次更新引入的,简直是坑人! 我找了半天系统设置都没找到这个快捷键是在哪设置的. ...
- ubuntu命令chmod755
使用方式 : 在终端切换到文件目录 输入 chmod775 hello.py 这样就将hello.py变成了可执行文件 当然作为 python文件 还需要再开头加上 #!/usr/bin/env py ...
- socket编程(Linux)
“一切皆Socket!” 话虽些许夸张,但是事实也是,现在的网络编程几乎都是用的socket. ——有感于实际编程和开源项目研究. 我们深谙信息交流的价值,那网络中进程之间如何通信,如我们每天打开浏览 ...
- XAMPP Apache 配置多端口和多域名方法
我们在工作中经常遇到同时调试多个网站的情况,那么如何配置呢?就像平时访问网站一样,网站 a.com 与网站 b.com 截然不同.这都是常见现象,如果在局域网中要访问另外一台电脑上的多个网站,就需要使 ...
- WPF Step By Step -基础知识介绍
回顾 上一篇我们介绍了WPF基本的知识.并且介绍了WPF与winform传统的cs桌面应用编程模式上的变化,这篇,我们将会对WPF的一些基础的知识做一个简单的介绍,关于这些基础知识更深入的应用则在后续 ...
- [转Go-简洁的并发 ]
http://www.yankay.com/go-clear-concurreny/ Posted on 2012-11-28by yankay 多核处理器越来越普及.有没有一种简单的办法,能够让我们 ...
- LinearLayout 控件
LinearLayout 控件,垂直显示各控件一行一个显示,比较好控件. 用RelativeLayout多个控件会堆在一起 <LinearLayout xmlns:android="h ...
- 【转】oracle job相关内容
每天凌晨2点执行是这样的 dbms_job.submit(v_job,'lv;',TRUNC(sysdate+1)+2/24,'TRUNC(sysdate+1)+2/24'); 还有定义JOB最好是这 ...