数轴上n<=1000个点,从p出发以任意顺序走到所有的点,求到达每个点的时间之和的最小值。

好题!看起来水水的实际易错!

显然的结论是经过一个区间点之后肯定落在左端点或右端点上,谁没事最后还往中间跑呢!那就可以拍个序然后设计dp状态了,一个区间dp,f[i,j,0/1]表示走了区间i~j,最后落在左/右端点。

一个小技巧是把p算成一个点,初始化时之后p这个状态为0,其他都inf。那么问题来了!

方法一:记到状态的时间t[i,j,0/1],那么,相应更新t,其他同理。

 #include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<algorithm>
//#include<iostream>
using namespace std; int n,p;
#define maxn 1011
int a[maxn],f[maxn][maxn][],t[maxn][maxn][];
const int inf=0x3f3f3f3f;
int min(int a,int b) {return a<b?a:b;}
int main()
{
scanf("%d%d",&n,&p);
for (int i=;i<=n;i++) scanf("%d",&a[i]);
sort(a+,a++n);
for (int i=;i<=n;i++) f[i][i][]=f[i][i][]=t[i][i][]=t[i][i][]=inf;
a[n+]=inf;int pos;
for (pos=;pos<=n+;pos++) if (a[pos]>p) break;
if (pos==n+) t[n][n][]=t[n][n][]=f[n][n][]=f[n][n][]=p-a[n];
else if (pos==) t[][][]=t[][][]=f[][][]=f[][][]=a[]-p;
else f[pos][pos][]=f[pos][pos][]=a[pos]-p,f[pos-][pos-][]=f[pos-][pos-][]=p-a[pos-],
t[pos][pos][]=t[pos][pos][]=a[pos]-p,t[pos-][pos-][]=t[pos-][pos-][]=p-a[pos-];
for (int len=;len<n;len++)
for (int i=;i<=n-len;i++)
{
const int j=i+len;
if (f[i+][j][]+t[i+][j][]+a[i+]-a[i]<f[i+][j][]+t[i+][j][]+a[j]-a[i])
{
t[i][j][]=t[i+][j][]+a[i+]-a[i];
f[i][j][]=f[i+][j][]+t[i][j][];
}
else
{
t[i][j][]=t[i+][j][]+a[j]-a[i];
f[i][j][]=f[i+][j][]+t[i][j][];
}
if (f[i][j-][]+t[i][j-][]+a[j]-a[i]<f[i][j-][]+t[i][j-][]+a[j]-a[j-])
{
t[i][j][]=t[i][j-][]+a[j]-a[i];
f[i][j][]=f[i][j-][]+t[i][j][];
}
else
{
t[i][j][]=t[i][j-][]+a[j]-a[j-];
f[i][j][]=f[i][j-][]+t[i][j][];
}
f[i][j][]=min(f[i][j][],inf);
f[i][j][]=min(f[i][j][],inf);
}
printf("%d\n",min(f[][n][],f[][n][]));
return ;
}

错误!递推式不成立,没有考虑当前决策对后续状态的影响。错误样例?自己找个标程对拍吧!

方法二:直接把状态对后面的影响算出来,,保证了无后效性。

 #include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<algorithm>
//#include<iostream>
using namespace std; int n,p;
#define maxn 1011
int a[maxn],f[maxn][maxn][];
const int inf=0x3f3f3f3f;
int min(int a,int b) {return a<b?a:b;}
int main()
{
scanf("%d%d",&n,&p);
for (int i=;i<=n;i++) scanf("%d",&a[i]);
a[++n]=p;
sort(a+,a++n);
for (int i=;i<=n;i++)
{
if (a[i]!=p) f[i][i][]=f[i][i][]=inf;
else f[i][i][]=f[i][i][]=;
}
for (int len=;len<n;len++)
for (int i=;i<=n-len;i++)
{
const int j=i+len;
f[i][j][]=min(inf,min(f[i+][j][]+(n-j+i)*(a[i+]-a[i]),f[i+][j][]+(n-j+i)*(a[j]-a[i])));
f[i][j][]=min(inf,min(f[i][j-][]+(n-j+i)*(a[j]-a[i]),f[i][j-][]+(n-j+i)*(a[j]-a[j-])));
}
printf("%d\n",min(f[][n][],f[][n][]));
return ;
}

此题在九月份错过一次。

BZOJ1694一样哦!然而又错了,一看原题一激动忘排序了。

BZOJ1742: [Usaco2005 nov]Grazing on the Run 边跑边吃草的更多相关文章

  1. bzoj1742[Usaco2005 nov]Grazing on the Run 边跑边吃草*&&bzoj3074[Usaco2013 Mar]The Cow Run*

    bzoj1742[Usaco2005 nov]Grazing on the Run 边跑边吃草 bzoj3074[Usaco2013 Mar]The Cow Run 题意: 数轴上有n棵草,牛初始在L ...

  2. BZOJ 1742: [Usaco2005 nov]Grazing on the Run 边跑边吃草( dp )

    dp... dp( l , r , k )  , 表示 吃了[ l , r ] 的草 , k = 1 表示最后在 r 处 , k = 0 表示最后在 l 处 . ------------------- ...

  3. [Usaco2005 nov]Grazing on the Run 边跑边吃草 BZOJ1742

    分析: 首先,连续选择一段必定最优... 区间DP,f[i][j]表示从i开始,连续j个被吃掉了,并且,牛在i处,g[i][j]则表示在i+j-1处 f[i][j]可以从g[i+1][j]和f[i+1 ...

  4. 【bzoj1742】[Usaco2005 nov]Grazing on the Run 边跑边吃草 区间dp

    题目描述 John养了一只叫Joseph的奶牛.一次她去放牛,来到一个非常长的一片地,上面有N块地方长了茂盛的草.我们可以认为草地是一个数轴上的一些点.Joseph看到这些草非常兴奋,它想把它们全部吃 ...

  5. bzoj 1742: [Usaco2005 nov]Grazing on the Run 边跑边吃草【区间dp】

    挺好的区间dp,状态设计很好玩 一开始按套路设f[i][j],g[i][j]为吃完(i,j)区间站在i/j的最小腐败值,后来发现这样并不能保证最优 实际上是设f[i][j],g[i][j]为从i开始吃 ...

  6. BZOJ1742[Usaco2005 nov]Grazing on the Run

    Description John养了一只叫Joseph的奶牛.一次她去放牛,来到一个非常长的一片地,上面有N块地方长了茂盛的草.我们可 以认为草地是一个数轴上的一些点.Joseph看到这些草非常兴奋, ...

  7. 2018.10.22 bzoj1742: Grazing on the Run 边跑边吃草(区间dp)

    传送门 区间dp入门题. 可以想到当前吃掉的草一定是一个区间(因为经过的草一定会吃掉). 然后最后一定会停在左端点或者右端点. f[i][j][0/1]f[i][j][0/1]f[i][j][0/1] ...

  8. [USACO2005 nov] Grazing on the Run【区间Dp】

    Online Judge:bzoj1742,bzoj1694 Label:区间Dp 题目描述 John养了一只叫Joseph的奶牛.一次她去放牛,来到一个非常长的一片地,上面有N块地方长了茂盛的草.我 ...

  9. BZOJ2023: [Usaco2005 Nov]Ant Counting 数蚂蚁

    2023: [Usaco2005 Nov]Ant Counting 数蚂蚁 Time Limit: 4 Sec  Memory Limit: 64 MBSubmit: 56  Solved: 16[S ...

随机推荐

  1. 【转】彻底解析Android缓存机制——LruCache

    彻底解析Android缓存机制——LruCache 关于Android的三级缓存,其中主要的就是内存缓存和硬盘缓存.这两种缓存机制的实现都应用到了LruCache算法,今天我们就从使用到源码解析,来彻 ...

  2. iOS 网络开发

    http://www.cnblogs.com/kenshincui/p/4042190.html

  3. Android获取本地相册图片、拍照获取图片

    需求:从本地相册找图片,或通过调用系统相机拍照得到图片. 容易出错的地方: 1,当我们指定了照片的uri路径,我们就不能通过data.getData();来获取uri,而应该直接拿到uri(用全局变量 ...

  4. JavaScript也谈内存优化

    相对C/C++ 而言,我们所用的JavaScript 在内存这一方面的处理已经让我们在开发中更注重业务逻辑的编写.但是随着业务的不断复杂化,单页面应用.移动HTML5 应用和Node.js 程序等等的 ...

  5. SPOJ COT2 Count on a tree II (树上莫队,倍增算法求LCA)

    题意:给一个树图,每个点的点权(比如颜色编号),m个询问,每个询问是一个区间[a,b],图中两点之间唯一路径上有多少个不同点权(即多少种颜色).n<40000,m<100000. 思路:无 ...

  6. rmdir

    rmdir——删除空目录 remove empty directories 命令所在路径:bin/rmdir 示例: # rmdir /tmp/japan/longze 删除/tmp/japan/目录 ...

  7. learnpythonthehardway EX41 相关

    str.count() # str.count()方法用于统计字符串里某个字符出现的次数.可选参数为在字符串搜索的开始与结束位置. # str.count(sub, start= 0,end=len( ...

  8. 推荐一个高大上的网易云音乐命令行播放工具:musicbox

    网易云音乐上有很多适合程序猿的歌单,但是今天文章介绍的不是这些适合程序员工作时听的歌,而是一个用Python开发的开源播放器,专门适用于网易云音乐的播放.这个播放器的名称为MusicBox, 特色是用 ...

  9. 聊天室(C++客户端+Pyhton服务器)3.群功能添加

    创建群 数据库 group_table(user, name) grpuser_table(grpname,user) 按下添加群按钮 // 创建群组void CUserDialog::OnBnCli ...

  10. 嵌入式C语言-学习书籍推荐(pdf附上百度云链接)

    先推荐学习视频网站: https://www.bilibili.com/video/av22631677?from=search&seid=800092160484173881 书籍只推荐2本 ...