5432. 【NOIP2017提高A组集训10.28】三元组
题目
题目大意
给你\(X+Y+Z\)个三元组\((x_i,y_i,z_i)\)。
然后选\(X\)个\(x_i\),选\(Y\)个\(y_i\),选\(Z\)个\(z_i\)。
每个三元组只能选择其中一个。
问最大的和。
思考历程
想不到贪心……
于是只能\(DP\)了……
\(DP\)就不用说了吧……
正解
首先考虑\(X=0\)的情况:
按照\(z-y\)排个序,前面\(Z\)个选择\(z\),后面\(Y\)个选择\(y\)。
这就是一个可撤销贪心的思路,可以看成先全部选\(y\),然后选\(Z\)个\(z-y\)最大的。
然后就是普通的情况。首先强制所有选\(x\),然后按照\(z-y\)排个序。
枚举\(z-y\)的分界点,在前面的选\(z\)或\(x\),在后面的选\(y\)或\(x\)。
那么就变成了上面的问题:在\(x\)和\(z\)中选择,显然是\(z\)选\(z-x\)最大的\(Z\)个。
这个东西可以用数据结构维护,只不过会TLE。
于是可以搞个桶,用一个指针\(l\)表示当前选的最小的数在桶中的位置。
新加进来一个树的时候,用它在同种的位置和\(l\)比较一下,如果更大,说明\(l\)废了,于是就将它加进桶中,然后\(l\)往后找下一个最小的。
很显然,随着分界点朝右延伸,\(l\)一定是越来越大。
右边的同理。
如果排序也用桶排序,那就可以达到真正的\(O(n)\)
代码
然而我懒得打桶排序,就直接用自带的快排过去了……
using namespace std;
#include <cstdio>
#include <cstring>
#include <algorithm>
#define N 1500010
#define ll long long
inline int input(){
char ch=getchar();
while (ch<'0' || '9'<ch)
ch=getchar();
int x=0;
do{
x=x*10+ch-'0';
ch=getchar();
}
while ('0'<=ch && ch<='9');
return x;
}
int n,X,Y,Z;
struct Triple{
int x,y,z;
} _t[N],t[N];
int q[N],r[N];
int c[N];
inline bool cmp1(int a,int b){return _t[a].z-_t[a].y>_t[b].z-_t[b].y;}
inline bool cmp2(int a,int b){return c[a]<c[b];}
bool used[N];
ll ans1[N],ans2[N];
inline void work(int Z,int Y,ll *ans){
memset(used,0,sizeof(int)*(n+1));
for (int i=1;i<=n;++i)
q[i]=i;
sort(q+1,q+n+1,cmp2);
for (int i=1;i<=n;++i)
r[q[i]]=i;
ll sum=0;
int l=n+1;
for (int i=1;i<=Z;++i){
sum+=c[i];
l=min(l,r[i]);
used[r[i]]=1;
}
for (int i=Z;i<n-Y+1;++i){
ans[i]=sum;
if (l<r[i+1]){
sum+=c[i+1]-c[q[l]];
used[l]=0;
used[r[i+1]]=1;
while (!used[l])
++l;
}
}
}
int main(){
freopen("triple.in","r",stdin);
freopen("triple.out","w",stdout);
X=input(),Y=input(),Z=input();
n=X+Y+Z;
ll sumx=0;
for (int i=1;i<=n;++i)
_t[i]={input(),input(),input()},sumx+=_t[i].x;
for (int i=1;i<=n;++i)
q[i]=i;
sort(q+1,q+n+1,cmp1);
for (int i=1;i<=n;++i)
t[i]=_t[q[i]];
for (int i=1;i<=n;++i)
c[i]=t[i].z-t[i].x;
work(Z,Y,ans1);
reverse(t+1,t+n+1);
for (int i=1;i<=n;++i)
c[i]=t[i].y-t[i].x;
work(Y,Z,ans2);
ll res=0;
for (int i=Z;i<n-Y+1;++i)
res=max(res,sumx+ans1[i]+ans2[n-i]);
printf("%lld\n",res);
return 0;
}
总结
贪心要靠大胆地猜想……
有时候可以通过强制、分界点、可撤销贪心的方式转化成一个更加简单的问题。
5432. 【NOIP2017提高A组集训10.28】三元组的更多相关文章
- 5433. 【NOIP2017提高A组集训10.28】图
题目描述 Description 有一个n个点A+B条边的无向连通图,有一变量x,每条边的权值都是一个关于x的简单多项式,其中有A条边的权值是k+x,另外B条边的权值是k-x,如果只保留权值形如k+x ...
- 【JZOJ5428】【NOIP2017提高A组集训10.27】查询
题目 给出一个长度为n的序列a[] 给出q组询问,每组询问形如\(<x,y>\),求a序列的所有区间中,数字x的出现次数与数字y的出现次数相同的区间有多少个. 分析 我们可以维护一个前缀和 ...
- 【JZOJ5439】【NOIP2017提高A组集训10.31】Calculate
题目 分析 对于\[\sum_{i=1}^{n}\lfloor\dfrac{T-B_i}{A_i}\rfloor\] 我们考虑拆开处理,得到 \[\sum_{i=1}^{n}(\lfloor\dfra ...
- 【JZOJ5430】【NOIP2017提高A组集训10.27】图
题目 有一个n个点的无向图,给出m条边,每条边的信息形如\(<x,y,c,r>\) 给出q组询问形如\(<u,v,l,r>\) 接下来解释询问以及边的意义 询问表示,一开始你在 ...
- [JZOJ 5437] [NOIP2017提高A组集训10.31] Sequence 解题报告 (KMP)
题目链接: http://172.16.0.132/senior/#main/show/5437 题目: 题解: 发现满足上述性质并且仅当A序列的子序列的差分序列与B序列的差分序列相同 于是我们把A变 ...
- 【JZOJ5434】【NOIP2017提高A组集训10.30】Matrix
题目 分析 假设答案为ans, 发现\[k=\sum_{i=1}^{min(n,k)}\lfloor \dfrac{ans}{i} \rfloor\] 于是可以对ans进行二分, 用分块来求出上面的式 ...
- 【NOIP2017提高A组集训10.21】Fantasy
题目 Y sera 陷入了沉睡,幻境中它梦到一个长度为N 的序列{Ai}. 对于这个序列的每一个子串,定义其幻境值为这个子串的和,现在Y sera 希望选择K 个不同的子串并使得这K 个子串的幻境值之 ...
- 【NOIP2017提高A组模拟10.7】Adore
题目 小w 偶然间见到了一个DAG. 这个DAG 有m 层,第一层只有一个源点,最后一层只有一个汇点,剩下的每一层都有k 个节点. 现在小w 每次可以取反第i(1 < i < n - 1) ...
- NOIP2017提高A组模拟10.6】Biology
题目 trie 暴力就是对于每个询问的T个字符串 第i个和第i+1个直接个从后暴力枚举每位是否相同, 但这个方法TLE 我们考虑是否可以用更快的方法来求出两个字符串的最长公共后缀. 我们把所有的字符串 ...
随机推荐
- 使用Hystrix实现断路器处理
在之前的架构的基础上我们会发现,一旦级别低的服务宕了,会导致调用它的服务也挂掉,这样容易产生级联效应(雪崩效应),为了防止这种情况的出现,我引入了Hystrix来处理,先介绍ribbon使用Hystr ...
- Gym 102007I 二分 网络流
题意:给你一张图,每个城市有一些人,有不超过10个城市有避难所,避难所有容量上限,问最快多久可以让所有人进入避难所? 思路:二分时间,对于每个时间跑一遍最大流,判断最大流是不是人数即可.我们还需要用二 ...
- PHP-两数相除
给定两个整数,被除数 dividend 和除数 divisor.将两数相除,要求不使用乘法.除法和 mod 运算符. 返回被除数 dividend 除以除数 divisor 得到的商. 示例 1: 输 ...
- Centos7命令行安装Tomcat以及配置防火墙开放端口
[转载]Centos 7 yum安装tomcat 命令: 系统环境CentOS Linux release 7.2.1511 (Core) 一.搭建准备:1.先到tomcat官网https://tom ...
- RabbitMQ:从零开始
目录 一.介绍 二.安装 三.基本配置 四.Java Demo 五.基础API使用 六.ACK机制 七.消息的持久化 八.消息的公平分发 九.消息的优先级 十.消息的路由分发 十一.Spring集成 ...
- mysql-一行分隔成多行数据
mysql将某个字段有分隔符号分隔成多行数据 SELECT a.id, a. NAME, substring_index( substring_index( a.name, ',', b.help_t ...
- 【leetcode】667. Beautiful Arrangement II
题目如下: Given two integers n and k, you need to construct a list which contains ndifferent positive in ...
- 中位数+暴力——cf433C
/* 中位数到所有数的距离之和最小 因为只能改一个数,所以我们找一个数,将其改为和其相邻的数的中位数,使答案最小 先求一次原答案 把每个数相邻的数用vector存下来,然后排序找中位数,计算减小的量 ...
- css布局方面小结
1 ####css选择器 1 .left-word.moreinfor{} 这样是找不到选择器的.中间需要一个空格 但是div.moreinfor 是可以的. 2 max-width的作用: p元素只 ...
- jmeter 参数化3种
一.利用函数助手获取参数值 选项->函数助手对话框 __CSVRead, __Random, 生成的函数字符串:${__Random(,,)}第一个参数为随机数的下限,第二个参数为随机数的上限, ...