传送门

题意:

  有 n 个数 (1 ~ n) 分给了三个人 a, b, c; 其中 a 有 k1 个, b 有 k2 个, c 有 k3 个。

  现在问最少需要多少操作,使得 a 中所有数 是 1 ~ n 的一个前缀;

  c 中所有数 是 1 ~ n 的一个后缀。 剩下的都在 b 手上。

  每次操作可以让一个人手上的一个数给另一个人。

解: 简单 DP ; 显然就是问你 把  1 ~ n 分成三段的最少花费。

  你把 第一个人 最初拥有的数, 所在的桶 定义为 0;

  第二个人的为1, 第三个人的为2。

  然后你就可以 DP 了;

  dp[ i ][ 0 ] 就代表, 你的 第一个人取 1 ~ i 的花费。

  dp[ i ][ 1 ] 就表示, 第一个人 拿了 1 ~ i 的前缀 1 ~ pos 部分, 第二个人拿了, 剩下的 pos ~ i 部分的最少花费。

  dp[ i ][ 2 ] 类似。 看代码的式子应该就懂了。

#include <bits/stdc++.h>
#define LL long long
#define rep(i, j, k) for(int i = j; i <= k; i++)
#define dep(i, j, k) for(int i = k; i >= j; i--)
#define INF 0x3f3f3f3f
#define inf 0x3f3f3f3f3f3f3f3f
#define mem(i, j) memset(i, j, sizeof(i))
#define pb push_back
using namespace std; const int N = 2e5 + ;
int A[N];
int dp[N][];
int main() { int a, b, c; scanf("%d %d %d", &a, &b, &c);
int n = a + b + c;
for(int i = ; i <= a; i++) {
int x; scanf("%d", &x); A[x] = ;
}
for(int i = ; i <= b; i++) {
int x; scanf("%d", &x); A[x] = ;
}
for(int i = ; i <= c; i++) {
int x; scanf("%d", &x); A[x] = ;
}
for(int i = ; i <= n; i++) {
dp[i][] = dp[i - ][] + (A[i] != ? : );
dp[i][] = min(dp[i - ][], dp[i - ][]) + (A[i] != ? : );
dp[i][] = min(dp[i - ][], min(dp[i - ][], dp[i - ][])) + (A[i] != ? : );
}
printf("%d\n", min(dp[n][], min(dp[n][], dp[n][])));
return ;
}

官方解, 思维 + 贪心。

  我们假设三个人的数编号为 1 2 3

  首先, 假设我们定了 L, R。 就是说, 1 ~ L 这一段是 第一个人的, L ~ R 这一段是第二个人的。

  R ~ n 这一段是第三个人的。 那么, 你若确定了 L, R。  那么你的 答案, 就是固定的。

  我们假设  cnt[ L ][ i ] 表示 1 ~ L 这一段 是 i 个人的数的个数。

  我们假设 1 ~ L 这一段为 L,   L ~ R 这一段为 M 。  R ~ n 这一段为 R

  那么答案就是 cnt[ L ][ 2 ] + cnt[ L ][ 3 ] + cnt[ M ][ 1 ] + cnt[ M ][ 3 ] + cnt[ R ][ 1] + cnt[ R ][ 2 ];

  对吧。 那现在, 要是我们确定了 R 这个点。 然后, L 应该在什么位置 才是最优的呢。

  假设R 确定了。 那么 cnt[ L ][ 3 ] + cnt[ M ][ 3 ] , cnt[ R ][ 1 ] , cnt[ R ][ 2 ] 全都确定了嘛。

  那么还差 cnt[ L ][ 2 ] 和, cnt[ M ][ 1 ] 没有确定嘛。 那你想要 这两个值 尽可能的小嘛。

  那么就是 你希望, 1 ~ L 这一段中 2 的数 的个数尽可能少, L ~ R 这一段中 1 的数的个数尽可能少。

  那也可以说成, 你 1 ~ L 这一段中, 1 的个数尽可能的多, 2的个数尽可能的少嘛。

  因为你 1 ~ L 中 1 的个数尽可能多的话呢, 你 L ~ R 这一段中 1 的个数就自然小了嘛。

  那不就是 cnt[ L ][ 1 ] - cnt[ L ][ 2 ] 最大的时候 的位置, 就是 L 的位置吗。

  那么, 我们枚举  R, 再维护一个 cnt[ L ][ 1 ] - cnt[ L ][ 2 ] 的最大值。

  就可以确定 R 在这个点时的  最小答案了。

  我们假设 cnt[ L ][ 1 ] - cnt[ L ][ 2 ] 的最大值为 ma

  那么我们现在 让 R ~ n 这一段为 R, 1 ~ R 这一段为 LL 。

  那么答案就是 cnt[ R ][ 1 ] + cnt[ R ][ 2 ] + cnt[ LL ][ 3 ] + cnt[ LL ][ 1 ] - ma;

  就是答案了。 cnt[ LL ][ 1 ]  -  ma。  这一段代表的东西就是。

  你 ma 这个值是在最优的 L 处取得的嘛。

  那就是, 1 ~ R 这一段的所有的 1 减去,你 1 ~ L 这一段的不需要动的1嘛 。

  那得到的就是 L ~ R 这一段 1 的个数。

  再加上 你 1 ~ L 这一段的 2 的个数嘛。  那就刚好是 前面的 cnt[ L ][ 2 ] + cnt[ M ][ 1 ]。

  

#include <bits/stdc++.h>
#define LL long long
using namespace std;
const int N = 2e5 + ;
int pos[N];
int cntR[];
int cntL[];
int main() {
int a, b, c; scanf("%d %d %d", &a, &b, &c);
int n = a + b + c;
for(int i = ; i <= a; i++) {
int x; scanf("%d", &x);
pos[x] = ;
}
for(int i = ; i <= b; i++) {
int x; scanf("%d", &x);
pos[x] = ;
}
for(int i = ; i <= c; i++) {
int x; scanf("%d", &x);
pos[x] = ;
}
for(int i = ; i <= n; i++) {
cntR[pos[i]]++;
}
/// 整段都是 3 的。
int ans = cntR[] + cntR[];
int ma = ;
for(int i = ; i <= n; i++) {
cntL[pos[i]]++;
cntR[pos[i]]--;
ma = max(ma, cntL[] - cntL[]);
int tmp = cntR[] + cntR[] + cntL[] + cntL[] - ma;
ans = min(ans, tmp);
}
printf("%d\n", ans);
return ;
}

  

E. The Contest ( 简单DP || 思维 + 贪心)的更多相关文章

  1. hdu 1257 && hdu 1789(简单DP或贪心)

    第一题;http://acm.hdu.edu.cn/showproblem.php?pid=1257 贪心与dp傻傻分不清楚,把每一个系统的最小值存起来比较 #include<cstdio> ...

  2. codeforce1029B B. Creating the Contest(简单dp,简单版单调栈)

    B. Creating the Contest time limit per test 1 second memory limit per test 256 megabytes input stand ...

  3. codeforces Gym 100500H A. Potion of Immortality 简单DP

    Problem H. ICPC QuestTime Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/gym/100500/a ...

  4. 『简单dp测试题解』

    这一次组织了一场\(dp\)的专项考试,出了好几道经典的简单\(dp\)套路题,特开一篇博客写一下题解. Tower(双向dp) Description 信大家都写过数字三角形问题,题目很简单求最大化 ...

  5. luoguP1281 书的复制 DP,贪心

    luoguP1281 书的复制 链接 https://www.luogu.org/problemnew/show/P1281 思路 简单dp,输出方案. 很明显dp记录路径对不对? 恭喜你死了. 求出 ...

  6. Codeforces Round #302 (Div. 2) C. Writing Code 简单dp

    C. Writing Code Time Limit: 20 Sec  Memory Limit: 256 MB 题目连接 http://codeforces.com/contest/544/prob ...

  7. 洛谷 - P1004 - 方格取数 - 简单dp

    https://www.luogu.org/problemnew/show/P1004 这道题分类到简单dp但是感觉一点都不简单……这种做两次的dp真的不是很懂怎么写.假如是贪心做两次,感觉又不能证明 ...

  8. HDU 5375 Gray code (简单dp)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5375 题面: Gray code Time Limit: 2000/1000 MS (Java/Oth ...

  9. 4.15 每周作业 —— 简单DP

    免费馅饼 Time Limit : 2000/1000ms (Java/Other)   Memory Limit : 65536/32768K (Java/Other) Total Submissi ...

随机推荐

  1. pandas之数据IO笔记

    pandas在进行数据存储与输出时会做一些相应的操作 1.*索引:将一个列或多个列读取出来构成DataFrame,其中涉及是否从文件中读取索引以及列名 2 *类型推断和数据转换:包括用户自定义的转换以 ...

  2. 解析spring启动加载dubbo过程

    一:简单配置 web.xml <context-param> <param-name>contextConfigLocation</param-name> < ...

  3. 请问IOS中做一个手机网站的app壳复杂吗?

    公司开发了一个平台,手机网站已经做出来了,想开发一个苹果应用app,但公司没人会IOS开发,为了减小成本,现在想直接做一个壳来加载手机网站,请问在ios中复杂吗?是否有相应的控件直接加载url就行? ...

  4. 一次腾讯云centos服务器被入侵的处理

    昨天一大早,我还没到公司呢,就收到腾讯云安全中心发来的服务器异常登录告警,登录控制台一看,ip还是美国的,一脸懵逼.由于本人之前也没有过处理服务器入侵的经验,而且这台服务器目前还没有部署商用系统,所以 ...

  5. 一个简单的window.onscroll实例

    鉴于better-scroll实现这个效果很复杂,想用最原生的效果来实现吸顶效果 一个简单的window.onscroll实例,可以应用于移动端 demo 一个简单的window.onscroll实例 ...

  6. ifeq ifneq ifdef ifndef

    条件语句中使用到了三个关键字:“ifeq”.“else”和“endif”.其中: 1.        “ifeq”表示条件语句的开始,并指定了一个比较条件(相等).之后是用圆括号括包围的.使用逗号“, ...

  7. 4.解析配置文件 redis.conf

    将原始的redis.conf拷贝,得到一个myRedis.conf文件,修改配置文件时,就修改这个文件,不对原始的配置文件进行修改 redis配置文件中主要有以下内容: 1.units单位 a)配置大 ...

  8. 一个基于Scrapy框架的pixiv爬虫

    源码 https://github.com/vicety/Pixiv-Crawler,功能什么的都在这里介绍了 说几个重要的部分吧 登录部分 困扰我最久的部分,网上找的其他pixiv爬虫的登录方式大多 ...

  9. 机器学习 三剑客 之 pandas + numpy

    机器学习 什么是机器学习? 机器学习是从数据中自动分析获得规律(模型),并利用规律对未知数据进行预测 机器学习存在的目的和价值领域? 领域: 医疗.航空.教育.物流.电商 等... 目的: 让机器学习 ...

  10. Python自动化测试PO模式

    页面元素定位信息 页面元素定位信息文件 [leadscloud_login] input_user_name = xpath>//*[@id='main']/div/div[2]/div[2]/ ...