2019杭电多校一 C. Milk (dp)
大意: $n*m$棋盘, 初始位置$(1,1)$, 横坐标为$\frac{m+1}{2}$时可以向下走, 否则只能左右走, 每走一步花费$1$秒. 有$k$管奶, 第$i$罐位置$(r_i,c_i)$, 要花费$t_i$的时间去喝. 对于所有的$1\le i\le k$, 求出喝完$i$管奶最短用时.
实现略复杂的$dp$题, 直接按照官方题解写了.
主要思路是对每一行求出向左/向右喝$x$罐奶的最少用时, 然后$dp$合并答案.
#include <iostream>
#include <algorithm>
#include <cstdio>
#include <queue>
#define REP(i,a,n) for(int i=a;i<=n;++i)
#define PER(i,a,n) for(int i=n;i>=a;--i)
#define x first
#define y second
#define pb push_back
using namespace std;
typedef long long ll;
typedef pair<int,int> pii;
const ll INF = 1e15;
const int N = 1e4+10;
int n, m, k, b[N];
struct {int x,y,t;} a[N];
ll ans[N];
vector<pii> v[N];
vector<ll> solve(vector<pii> a, int s, int x) {
a.insert(a.begin(),pii(s,0));
vector<ll> ret(a.size(),INF), t(ret);
ret[0] = x==-1?0:abs(x-s);
t[0] = 0;
for (int i=1; i<a.size(); ++i) {
int len = abs(a[i].x-a[i-1].x);
REP(j,0,i-1) t[j]+=len;
PER(j,1,i) t[j]=min(t[j],t[j-1]+a[i].y);
REP(j,1,i) ret[j]=min(ret[j],t[j]+(x==-1?0:abs(x-a[i].x)));
}
return ret;
}
vector<ll> Merge(const vector<ll> &L, const vector<ll> &R) {
vector<ll> ret(L.size()+R.size()-1,INF);
for (int i=0;i<L.size();++i) {
for (int j=0;j<R.size();++j) {
ret[i+j] = min(ret[i+j], L[i]+R[j]);
}
}
return ret;
} void work() {
scanf("%d%d%d", &n, &m, &k);
*b = 0;
REP(i,1,k) {
scanf("%d%d%d",&a[i].x,&a[i].y,&a[i].t);
b[++*b] = a[i].x;
}
b[++*b] = 1;
sort(b+1,b+1+*b),*b=unique(b+1,b+1+*b)-b-1;
REP(i,1,*b) v[i].clear();
REP(i,1,k) {
a[i].x=lower_bound(b+1,b+1+*b,a[i].x)-b;
v[a[i].x].pb(pii(a[i].y,a[i].t));
ans[i] = INF;
}
REP(i,1,*b) sort(begin(v[i]),end(v[i]));
auto g = solve(v[1],1,-1);
for (int i=0;i<g.size();++i) {
ans[i] = min(ans[i], g[i]);
}
int cur = 0;
vector<ll> f[2];
f[0] = solve(v[1],1,(m+1)/2);
REP(i,2,*b) {
cur ^= 1;
auto p = lower_bound(begin(v[i]),end(v[i]),pii((m+1)/2,0));
vector<pii> L(begin(v[i]),p), R(p,end(v[i]));
reverse(begin(L),end(L));
auto f0 = solve(L,(m+1)/2,(m+1)/2), f1 = solve(R,(m+1)/2,(m+1)/2);
auto g0 = solve(L,(m+1)/2,-1), g1 = solve(R,(m+1)/2,-1);
g0 = Merge(f1,g0), g1 = Merge(f0,g1);
auto g = g0;
for (int j=0;j<g.size();++j) {
g[j] = min(g[j], g1[j]);
}
g = Merge(f[!cur],g);
f[cur] = Merge(f[!cur],Merge(f0,f1));
for (int j=0;j<g.size();++j) {
ans[j] = min(ans[j], g[j]+b[i]-b[i-1]);
f[cur][j] += b[i]-b[i-1];
}
}
REP(i,1,k) {
if (i==k) printf("%lld\n",ans[i]);
else printf("%lld ",ans[i]);
}
} int main() {
int t;
scanf("%d", &t);
while (t--) work();
}
2019杭电多校一 C. Milk (dp)的更多相关文章
- 2019杭电多校一 A. Blank (dp)
大意: 长为$n$的数组, 每个位置范围$[0,3]$, $m$个限制$(l,r,x)$表示$[l,r]$内有$x$种数, 求方案数. 维护每个数字最后一次出现位置, 暴力$DP$ 实现时有个技巧是把 ...
- [2019杭电多校第八场][hdu6667]Roundgod and Milk Tea
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6667 题目大意是说n个班级,每个班级有ai人和bi杯茶,每个人只能喝其他班的茶并且只能喝一杯.问最多有 ...
- 2019杭电多校&CCPC网络赛&大一总结
多校结束了, 网络赛结束了.发现自己还是太菜了,多校基本就是爆零和签到徘徊,第一次打这种高强度的比赛, 全英文,知识点又很广,充分暴露了自己菜的事实,发现数学还是很重要的.还是要多刷题,少玩游戏. 网 ...
- 2019杭电多校第一场hdu6581 Vacation
Vacation 题目传送门 update(O(n)) 看了那个O(n)的方法,感觉自己想的那个O(nlogn)的好傻,awsl. 0车最终通过停车线的时候,状态一定是某个车堵住后面的所有车(这个车也 ...
- 2019杭电多校第二场hdu6601 Keen On Everything But Triangle
Keen On Everything But Triangle 题目传送门 解题思路 利用主席树求区间第k小,先求区间内最大的值,再求第二大,第三大--直到找到连续的三个数可以构成一个三角形.因为对于 ...
- 2019杭电多校第二场hdu6602 Longest Subarray(线段树)
Longest Subarray 题目传送门 解题思路 本题求一个最大的子区间,满足区间内的数字要么出现次数大于等于k次,要么没出现过.给定区间内的数字范围是1~c. 如果r为右边界,对于一种数字x, ...
- Rikka with Game[技巧]----2019 杭电多校第九场:1005
Rikka with Game Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 524288/524288 K (Java/Othe ...
- 2019杭电多校 hdu6662 Acesrc and Travel (树形dp
http://acm.hdu.edu.cn/showproblem.php?pid=6662 题意:有两个人在树上博弈,每个点节点有两个分数a[i]和b[i],先手先选择一个点,后手在先手选的点的相邻 ...
- 2019杭电多校 hdu6659 Acesrc and Good Numbers
http://acm.hdu.edu.cn/showproblem.php?pid=6659 题意:给你d,x,让求满足f(d,n)=n的最大n(n<=x),其中f(d,n)表示数字d在从1到n ...
随机推荐
- Hibernate 关系配置
表之间关系 1. 一对多 一个部门有多个员工,一个员工只能属于某一个部门 一个班级有多个学生,一个学生只能属于一个班级 2. 多对多 一个老师教多个学生,一个学生可以被多个老师教 一个学生可以先择多门 ...
- [转]解决Git报错:error: You have not concluded your merge (MERGE_HEAD exists).
Git fetch和git pull的区别: 都可以从远程获取最新版本到本地 1.Git fetch:只是从远程获取最新版本到本地,不会merge(合并) $:git fetch origin mas ...
- D.Dwarf Tower
Vasya在玩一个叫做"Dwarf Tower"的游戏,这个游戏中有n个不同的物品, 它们的编号为1到n.现在Vasya想得到编号为1的物品. 获得一个物品有两种方式: 直接购买该 ...
- [Java]某日期时间加上若干分钟得到新的日期时间
使用Java自带类库实现日期时间增减还是比自己人工拆分编写要牢靠,代码也简洁多了. 下面代码实现了在原有日期时间上加上一些分钟得到新的日期时间的功能,稍加改造还可以实现逆向运算. 代码: packag ...
- PyMouse、PyKeyboard用python操作鼠标和键盘
1.PyUserInput 简介 PyUserInput是一个使用python的跨平台的操作鼠标和键盘的模块,非常方便使用.支持的平台及依赖如下: Linux - Xlib Mac - Quartz, ...
- log4j:ERROR setFile(null,true) call failed.错误解决
首先说明,我是用hive执行bin/hiveserver2时出现的这个错误.如下图所示,红框中的内容也清晰的告诉我们出错的原因和文件路径. 之后,我查看了一下该路径.发现我用的是beifeng的用户, ...
- ehcache的heap、off-heap、desk浅谈
ehcache的heap.off-heap.desk浅谈 答: 从读取速度上比较:heap > off-heap > disk heap堆内内存: heap表示使用堆内内存,heap( ...
- List根据多个字段分组
List<ClassEntity> distinctClass = classEntities.stream().collect(Collectors.collectingAndThen( ...
- 30分钟让你学会 Spring事务管理属性
Spring是一个Java开源框架,是为了解决企业应用程序开发复杂性由Rod Johnson创建的.框架的主要优势之一就是其分层架构,分层架构允许使用者选择使用哪一个组件,同时为 J2EE 应用程序开 ...
- EM算法之不同的推导方法和自己的理解
EM算法之不同的推导方法和自己的理解 一.前言 EM算法主要针对概率生成模型解决具有隐变量的混合模型的参数估计问题. 对于简单的模型,根据极大似然估计的方法可以直接得到解析解:可以在具有隐变量的复杂模 ...