BZOJ 3721: PA2014 Final Bazarek【乱搞】
有n件商品,选出其中的k个,要求它们的总价为奇数,求最大可能的总价。
Input
第一行一个整数n(1<=n<=1000000),表示商品数量。
接下来一行有n个整数,表示每件商品的价格,范围在[1,10^9]。
接下来一行有一个整数m(1<=m<=1000000),表示询问数量。
接下来m行,每行一个整数k[i](1<=k[i]<=n)。
Output
对于每个询问,输出一行表示保证奇数的情况下最大的总价。若无法满足要求,输出-1。
Sample Input
4
4 2 1 3
3
2
3
4
Sample Output
7
9
-1
思路:如果不要是奇数,那么..........排个序 输出前K个。。。。。。。。。。。。。
如果前K个已经是奇数的话直接输出即可
如果不是呢?
思考熊。。。
果断猜了个命题:设第K大的数为X,将X替换成不在前K大的和X奇偶性不一样的最大的数,这样一定是最优的。
仔细思考了后发现这个命题不够完善 例如下面这个数据:
5
2 99 101 102 133
1000
1
2
3
唔 K=3时用上面那个算法做出来的是133+102+2 但最优的是133+101+99
于是完善了上面的算法:设在前K大的数中与X奇偶性相同的最小的数为Y(如果存在的话),那么把Y当成X替换成不在前K大中的奇偶性与Y不同的最大的那个数 这样算出的数和上面算法一算出的数取个最大值就是答案了
然后YY了个证明:设Bi表示i个数不在前k大数中的K个数的和的最大值,则Bi>=B(i+1),即i+1个数不在前K大里的答案绝对不优于i个数不在前K大里的答案
因此优先考虑0个数不在前K大里 即就是前K大的和 如果不为奇数,那么考虑只有1个数不在前K大里,显然通过以上方法一定能将和调整为奇数,或者题目无解。
接下来考虑为何用以上算法是调整一个数时的最大值:显然由于此时前K大的和为偶数,因此调整一个数要么把奇数调整成一个偶数,要么把偶数调整成一个奇数,那显然就是将最小的偶数调整成最大的奇数 或者最小的奇数调整成最大的偶数就可以了
//bzoj2464
#include <stdio.h>
#include <string.h>
#include <algorithm>
#include <iostream>
#define maxn 1000009
using namespace std;
int a[maxn],foredif[maxn],same[maxn],badif[maxn];
long long sum[maxn];
int cmp(int x, int y){return x>y;}
long long mmax(long long x,long long y)
{
return x>y?x:y;
}
int main()
{
long long n,k,lasto=-1,laste=-1,m;
scanf("%lld",&n);
for(int i=1;i<=n;i++)scanf("%lld",&a[i]);
sort(a+1,a+1+n,cmp);
for(int i=1;i<=n;i++)sum[i]=sum[i-1]+a[i];
for(int i=1;i<=n;i++)
{
if((a[i]&1)==1)//odd
{
foredif[i]=laste;
lasto=i;
}
else
{
foredif[i]=lasto;
laste=i;
}
}
lasto=laste=-1;
for(int i=n;i>=1;i--)
{
if((a[i]&1)==1)//odd
{
same[i]=lasto;
badif[i]=laste;
lasto=i;
}
else
{
same[i]=laste;
badif[i]=lasto;
laste=i;
}
}
scanf("%lld",&m);
for(int i=1;i<=m;i++)
{
scanf("%lld",&k);
long long s=sum[k];
if((s&1)==1){printf("%lld\n",s);continue;}
if(badif[k]==-1)
{
if(same[k]==-1||foredif[k]==-1)s=-1;else
s=(long long)s+(long long)a[same[k]]-(long long)a[foredif[k]];
}
else
{
if(same[k]==-1||foredif[k]==-1)s=s-(long long)a[k]+(long long)a[badif[k]];
else
{
s=mmax((long long)s+(long long)a[same[k]]-(long long)a[foredif[k]],(long long)s-(long long)a[k]+(long long)a[badif[k]]);
}
}
printf("%lld\n",s);
}
return 0;
}
BZOJ 3721: PA2014 Final Bazarek【乱搞】的更多相关文章
- BZOJ 3721: PA2014 Final Bazarek
3721: PA2014 Final Bazarek Time Limit: 20 Sec Memory Limit: 128 MBSubmit: 645 Solved: 261[Submit][ ...
- bzoj 3721: PA2014 Final Bazarek 贪心
如果没有限制,直接取前 $k$ 大即可. 有限制,则只有几种可能:奇换偶,偶换奇. 维护奇数偶数的前缀最小值和后缀最大值即可. code: #include <bits/stdc++.h> ...
- 【BZOJ3721】PA2014 Final Bazarek 贪心
[BZOJ3721]PA2014 Final Bazarek Description 有n件商品,选出其中的k个,要求它们的总价为奇数,求最大可能的总价. Input 第一行一个整数n(1<=n ...
- bzoj 3722: PA2014 Final Budowa
3722: PA2014 Final Budowa Time Limit: 1 Sec Memory Limit: 128 MBSubmit: 303 Solved: 108[Submit][St ...
- BZOJ 3727 PA2014 Final Zadanie 树形DP
题目大意:给定一棵树,令一个点到全部点的距离与点权的乘积之和为b[i].求每一个点的权值a[i] 首先假设给定a[i]我们能够非常轻松的求出b[i] 可是反过来怎么搞?高斯消元?30W? 考虑已知a[ ...
- bzoj3721 [PA2014 Final] Bazarek
Description 有n件商品,选出其中的k个,要求它们的总价为奇数,求最大可能的总价. Input 第一行一个整数n(1<=n<=1000000),表示商品数量.接下来一行有n个整数 ...
- 【BZOJ】2456 mode(乱搞)
Description 给你一个n个数的数列,其中某个数出现了超过n div 2次即众数,请你找出那个数. Input 第1行一个正整数n.第2行n个正整数用空格隔开. Output 一行一个正整数表 ...
- bzoj 4900 [CTSC2017]密钥 模拟+乱搞
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4900 #include<cstring> #include<cmath&g ...
- 【贪心】bzoj3721 PA2014 Final Bazarek
考虑不限制奇偶的情况,那就是直接排序取前k个的和. 加上奇偶限制:若排序后的前k个的和是偶数,则“显然地”:将其中的最小的奇数替换成未被选择的数中最大的偶数 或者 将其中的最小的偶数替换成未被选择的数 ...
随机推荐
- [转]MVC 检测用户是否已经登录
本文转自:http://blog.csdn.net/jayzai/article/details/41252137 当我们访问某个网站的时候需要检测用户是否已经登录(通过Session是否为null) ...
- 关于发布WP 8.1应用信息不匹配问题的解决办法
错误提示: 与此更新关联的程序包标识符与已上传程序包中的标识符不匹配: The package identity associated with this update doesn't match ...
- [ POI 2017 ] Sabota?
Description 题目链接 Solution 因为一个节点染黑了子树就都被染黑了,所以最后染黑的点集必然是一棵子树. 可以得出的结论是,如果被染黑的节点在节点 \(a\) 的子树中,而 \(a\ ...
- 【学习笔记】深入理解js原型和闭包(14)——从【自由变量】到【作用域链】
先解释一下什么是“自由变量”. 在A作用域中使用的变量x,却没有在A作用域中声明(即在其他作用域中声明的),对于A作用域来说,x就是一个自由变量.如下图 如上程序中,在调用fn()函数时,函数体中第6 ...
- 10个顶级的CSS3代码生成器
新出来的在线工具和web应用允许开发人员快速创建网站,而无需手动一行一行地编写代码.当前,不断有新的框架和代码库涌现在前端开发这个领域里. 但是,这也让许多开发人员忘记了代码生成器以及它们在构建网站时 ...
- HDU 5414 CRB and String (字符串,模拟)
题意:给两个字符串s和t,如果能插入一些字符使得s=t,则输出yes,否则输出no.插入规则:在s中选定一个字符c,可以在其后面插入一个字符k,只要k!=c即可. 思路:特殊的情况就是s和t的最长相同 ...
- 在Windows 10 系统上启用Hyper V遇到的错误:0x800f0831
Hyper-V是微软的一款虚拟化技术,是微软第一个采用类似Vmware和Citrix开源Xen一样的基于hypervisor的技术. 在Windows 10的powershell命令里,输入如下的命令 ...
- 一行命令杀掉defunct进程
一行命令杀掉defunct进程 今天在杀掉defunc过程中一直搞不完,索性写一行命令,注意先看懂谨慎使用 ps -ef|grep defunct|awk '{print " ps -ef| ...
- Java后端学习路线_备战
SpringCloud项目实战 Dubbo项目实战 项目实战应涵括哪些技术: 缓存.消息队列 WEB应用服务器(Weblogic.Jetty.JBoss.WebSphere) NoSQL(MongoD ...
- react开启一个项目 webpack版本出错
npx create-react-app my-app cd my-app npm start 在命令行里执行以上语句就可(前两天刚刚发现,最新版的react对webpack的版本要了新要求,大概是他 ...