Mark一下, dp状态转移方程写对,可是写代码都错,poj 1651 poj 1179
dp题:
1、写状态转移方程;
2、考虑初始化边界,有意义的赋定值。还没计算的赋边界值;
3、怎么写代码自底向上计算最优值
今天做了几个基础dp,所有是dp方程写对可是初始化以及计算写错
先是poj 1651 事实上就是个赤裸裸的矩阵连乘。dp方程非常easy写出
dp[i][j]=min(dp[i][k]+dp[k+1][j]+r[i]*c[k]*c[j],dp[i][j]);
先贴两个个二逼的代码,mark下自己多么的二逼:
二逼一:在计算的时候使用了还没有算出来的值,模拟下就知道第一重循环里算dp[0][2]=dp[0][0]+dp[1][2];
!!!!!dp[1][2]这时候根本没有计算出来还,Tmd我就用了这个值!!
!
。!!!!
!。!
。!
!。!
。。!。。!
!
二逼啊!
!!!
!。。!
!
#include <cstdio>
#include <cstring>
#include <algorithm> using namespace std; const int SIZE = 100+10;
const int INF = 10000000;
int dp[SIZE][SIZE];
int r[SIZE],c[SIZE]; int main()
{
int n;
while(~scanf("%d",&n))
{ for(int i=0;i<n-1;i++)
{
scanf("%d",&r[i]);
if(i>0)c[i-1]=r[i];
}
scanf("%d",&c[n-2]);
n--;
int ans=0;
for(int i=0;i<n;i++)dp[i][i]=0; for(int i=0;i<=n;i++)
for(int j=i;j<n;j++)
{ for(int k=i;k<j;k++)
dp[i][j]=min(dp[i][k]+dp[k+1][j]+r[i]*c[k]*c[j],dp[i][j]);
}
for(int i=0;i<n;i++)
{
for(int j=0;j<n;j++)
printf("%d ",dp[i][j]);
putchar('\n');
} printf("%d\n",dp[0][n-1]); }
return 0;
}
二逼二:来回换思路的时候各种思路弄混。看凝视
#include <cstdio>
#include <cstring>
#include <algorithm> using namespace std; const int SIZE = 100+10;
const int INF = 10000000;
int dp[SIZE][SIZE];
int r[SIZE],c[SIZE]; int main()
{
int n;
while(~scanf("%d",&n))
{ for(int i=0;i<n-1;i++)
{
scanf("%d",&r[i]);
if(i>0)c[i-1]=r[i];
}
scanf("%d",&c[n-2]);
n--;
int ans=0;
for(int i=0;i<n;i++)dp[i][i]=0;
int j;
for(int l=2;l<=n;l++)
for(int i=0;i<l;i++)//此处i<l及其傻逼,请勿模仿
{
j=i+l-1;
dp[i][j]=INF;
for(int k=i;k<j;k++)
dp[i][j]=min(dp[i][k]+dp[k+1][j]+r[i]*c[k]*c[j],dp[i][j]);
}
/*for(int i=0;i<n;i++)
{
for(int j=0;j<n;j++)
printf("%d ",dp[i][j]);
putchar('\n');
}*/ printf("%d\n",dp[0][n-1]); }
return 0;
}
正确代码:
#include <cstdio>
#include <cstring>
#include <algorithm> using namespace std;
#define ll long long
const int SIZE = 100+10;
const int INF = 200000000;
ll dp[SIZE][SIZE];
int r[SIZE],c[SIZE]; int main()
{
int n;
while(~scanf("%d",&n))
{ for(int i=0;i<n-1;i++)
{
scanf("%d",&r[i]);
if(i>0)c[i-1]=r[i];
}
scanf("%d",&c[n-2]);
n--;
int ans=0;
for(int i=0;i<=n;i++)dp[i][i]=0;
int j;
for(int l=2;l<=n;l++)
for(int i=0;i<n-l+1;i++)
{
j=i+l-1;
dp[i][j]=INF;
for(int k=i;k<j;k++)
dp[i][j]=min(dp[i][k]+dp[k+1][j]+r[i]*c[k]*c[j],dp[i][j]);
}
printf("%lld\n",dp[0][n-1]); }
return 0;
}
在做这个poj 1651之前 先做的poj 1179,然后过了poj 1651之后改了poj1179原来的循环次序,事实上就是类比这个poj1651,第一重循环弄成长度就可以
注意 负数乘以负数是正数 考虑最大最小值得时候一定注意
只是又一次检查下发现代码在初始化的时候冗余,精简了下。
贴代码:
#include <cstdio>
#include <cstring>
#include <algorithm> using namespace std;
#define ll long long const int SIZE =55;
const int INF = (-2147483647 ) ; ll dp[SIZE][SIZE],dpmin[SIZE][SIZE];
int pos[SIZE];
bool op[SIZE]; int main()
{
//freopen("poj1179.txt","r",stdin);
int n;
ll tmp1,tmp2;
while(scanf("%d",&n)!=EOF)
{
for(int i=0;i<n;i++)dp[i][0]=dpmin[i][0]=0;
for(int i=0;i<n;i++)
{
scanf(" %c %lld",&op[i],&dp[i][1]);
dpmin[i][1]=dp[i][1];
}
for(int j=1;j<=n;j++)
for(int i=0;i<n;i++)
{
if(j>1)dp[i][j]=INF,dpmin[i][j]=-INF;
for(int k=1;k<j;k++)
{
if(op[(i+k)%n ] == 't')
{
dp[i%n][j]=max(dp[i%n][k]+dp[(i+k)%n ][j-k],dp[i%n][j]);
dpmin[i%n][j]=min(dpmin[i%n][k]+dpmin[(i+k)%n ][j-k], dpmin[i%n][j]);
} else
{
tmp1=max(dp[i%n][k]*dp[(i+k)%n ][j-k],dpmin[i%n][k]*dpmin[(i+k)%n ][j-k]);
tmp2=max(dp[i%n][k]*dpmin[(i+k)%n ][j-k], dpmin[i%n][k]*dp[(i+k)%n ][j-k]);
tmp1=max(tmp1,tmp2);
dp[i%n][j]=max(tmp1,dp[i%n][j]);
tmp2=min(dp[i%n][k]*dp[(i+k)%n ][j-k],dpmin[i%n][k]*dpmin[(i+k)%n ][j-k]);
tmp1=min(dp[i%n][k]*dpmin[(i+k)%n ][j-k], dpmin[i%n][k]*dp[(i+k)%n ][j-k]);
tmp2=min(tmp1,tmp2);
dpmin[i%n][j]=min(dpmin[i%n][j],tmp2);
} }
}
int cnt=0;
ll ans=INF;
for(int i=0;i<n;i++)//開始的时候,这里i<=n。于是初始化的时候必须优dp[n][...]=INF,才干AC,奇怪的是计算过程中第一个下标我都%n了,最后才看到这里的问题
{
ans=max(ans,dp[i][n]);
}
for(int i=0;i<n;i++)
if(ans == dp[i][n])
pos[cnt++]=i+1; printf("%lld\n",ans);
for(int i=0;i<cnt;i++)
if(i == cnt-1)printf("%d\n",pos[i]);
else printf("%d ",pos[i]);
} return 0;
}
Mark一下, dp状态转移方程写对,可是写代码都错,poj 1651 poj 1179的更多相关文章
- poj2385 Apple Catching(dp状态转移方程推导)
https://vjudge.net/problem/POJ-2385 猛刷简单dp的第一天的第一题. 状态:dp[i][j]表示第i秒移动j次所得的最大苹果数.关键要想到移动j次,根据j的奇偶判断人 ...
- DP 状态 DP 转移方程 动态规划解题思路
如何学好动态规划(2) 原创 Gene_Liu LeetCode力扣 今天 算法萌新如何学好动态规划(1) https://mp.weixin.qq.com/s/rhyUb7d8IL8UW1IosoE ...
- hdu 动态规划(46道题目)倾情奉献~ 【只提供思路与状态转移方程】(转)
HDU 动态规划(46道题目)倾情奉献~ [只提供思路与状态转移方程] Robberies http://acm.hdu.edu.cn/showproblem.php?pid=2955 背包 ...
- 【转载】 HDU 动态规划46题【只提供思路与状态转移方程】
1.Robberies 连接 :http://acm.hdu.edu.cn/showproblem.php?pid=2955 背包;第一次做的时候把概率当做背包(放大100000倍化为整数) ...
- dp状态压缩
dp状态压缩 动态规划本来就很抽象,状态的设定和状态的转移都不好把握,而状态压缩的动态规划解决的就是那种状态很多,不容易用一般的方法表示的动态规划问题,这个就更加的难于把握了.难点在于以下几个方面:状 ...
- 1113: [视频]树形动态规划(TreeDP)8:树(tree)(树形dp状态设计总结)
根据最近做的几道树形dp题总结一下规律.(从这篇往前到洛谷 P1352 ) 这几道题都是在一颗树上,然后要让整棵树的节点或边 满足一种状态.然后点可以影响到相邻点的这种状态 然后求最小次数 那么要从两 ...
- Codeforces 889E - Mod Mod Mod(dp+状态设计)
Codeforces 题目传送门 & 洛谷题目传送门 题目名称 hopping 我们记 \(x_i=X\bmod a_1\bmod a_2\bmod\dots\bmod a_i\),也就是 \ ...
- HDU 4628 Pieces(DP + 状态压缩)
Pieces 题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4628 题目大意:给定一个字符串s,如果子序列中有回文,可以一步删除掉它,求把整个序列删除 ...
- P2157 [SDOI2009]学校食堂 (dp+状态压缩)
题目链接:传送门 题目: 题目描述 小F 的学校在城市的一个偏僻角落,所有学生都只好在学校吃饭.学校有一个食堂,虽然简陋,但食堂大厨总能做出让同学们满意的菜肴.当然,不同的人口味也不一定相同,但每个人 ...
随机推荐
- 【转载】zookeeper使用和原理探究(一)
最近开始看到一些公司在使用zookeeper,本身对此了解的很少,这里看到一篇非常好的文章,因此转载 原贴地址:http://www.blogjava.net/BucketLi/archive/201 ...
- python中 in, any 和 all用法
in if x == 1 or y == 1 or z == 1: print('passed') if 1 in (x, y, z): print('passed') any if x or y o ...
- tomcat源码分析一
废话少说,拉代码,导入eclipse开干,具体步骤可以参考http://hi.baidu.com/hateeyes/blog/item/7f44942a20ad8f9d023bf66d.html 下面 ...
- CSA Round 84 Mahattan Center
题目 题目大意 给定平面上的 $n$ 个点和常数 $k$,求 $x$ 轴上的点 $p$ 到 $n$ 个点中距其最近的 $k$ 个点的距离之和的最小值.两点之间的距离定义为曼哈顿距离. 数据范围 $1\ ...
- Newton 插值法
定义 $f(x)$ 关于 $x_0, x_1, \dots, x_k$ 的 $k$ 阶均差(差商)记做 $ f [x_0, x_1, \dots, x_k] $,均差是递归定义的,有两种等价定义 \b ...
- 永琳的竹林迷径(path)
永琳的竹林迷径(path) 题目描述 竹林可以看作是一个n 个点的树,每个边有一个边长wi,其中有k 个关键点,永琳需要破坏这些关键点才能走出竹林迷径. 然而永琳打算将这k 个点编号记录下来,然后随机 ...
- Vue的this.$root.Bus.$on事件被多次触发、多次监听的问题
前端vue项目中,各个组件(非父子关系也可)之间可以通过Bus进行事件通信. main.js中: import Vue from 'vue' const Bus = new Vue(); const ...
- 外星千足虫(bzoj 1923)
Description Input 第一行是两个正整数 N, M. 接下来 M行,按顺序给出 Charles 这M次使用“点足机”的统计结果.每行 包含一个“01”串和一个数字,用一个空格隔开.“01 ...
- 【BZOJ1030】文本生成器(容斥原理,AC自动机,计数DP)
题意:给出n个字符串,求长为m至少包含n个里其中一个的串的字符串一共有多少个,字符集为A到Z,答案对10007取模 n<=60,len<=100 思路:将至少一个转化为所有个数减去没有出现 ...
- LOJ#2083. 「NOI2016」优秀的拆分
$n \leq 30000$的字符串,问其所有子串的所有AABB形式的拆分有多少种.$t \leq 10$组询问. $n^3$过80,$n^2$过95,鬼去写正解.. $n^2$:先枚举一次算每个位置 ...