传送门:http://codeforces.com/problemset/problem/609/D

(如需转载,请注明出处,谢谢O(_)O

题意:

Nura想买k个小玩意,她手上有 s 个burles(一种货币),有m个小玩意供她选择购买,但每个小玩意只能用dollars或者pounds来购买,所以每次购买的时候Nura都要通过汇率将她手上的burles换成dollars或者pounds,而每一天的汇率又不一样,给出n天,并且这n天中burles换成dollars和pounds的比率,问是否能在这n天中用手上的burles买下k个小玩意,如果不能,输出-1;如果能,输出最小的天数(即能使得Nura买下k个小玩意的最少天数),并在接下来的m行中,依次输出两个数字,分别是买下的小玩意的编号(按给出的顺序编号)以及买下该小玩意的日期(即第几天)。——在同一天中可以买任意多个小玩意。

解题思路与分析:

注意到数据范围是2e5,而显然需要通过某种方式来比较“天数”,直接枚举肯定很虚,所以要用到二分。

(这里给出同学写的二分的正确姿势:

int L,R,M;

while(R-L>1){

M=((R-L)>>1)+L;

if() R=M;

else L=M;

}

确定了用二分来枚举天数之后,就需要确定二分里面的内容。因为一天可以买多个,所以显然应该找兑换率最高的一天,即区间最小值,这里用了ST算法,O(nlogn)预处理,O(1)查询——获得了区间里burles兑换dollars和pounds最“划算”的一天后,就是确定买下k个小玩意的最小花费了,这里没有想到更简便的方法,只能用了O(mlogm)的预处理(排序),每次O(k)获取花费。用结构体存储小玩意的type,cost以及num(编号,后面输出用到),然后排序——重载小于号,先按type排,再按cost排,记录type==2的第一个小玩意对应的下标ter,之后只需要每次计算的时候比较一下,就能保证获取最小花费了(细节见代码)——获取最小花费后,将之与s比较一下即可知该区间是否可行。

代码实现过程与反思:

感觉基本上坑被踩了个尽。。

1、计算最小花费的时候,会造成int溢出,要用long long。

2、不是坑的坑:原题output部分中的”number of gadget”,是指小玩意的编号,不是数目(gadget没有s),醉。

4、细节:计算最小花费的时候,while循环应为j<m而不为j<n(j为ter为起点的下标,i从0开始),又wa一发。。

3、二分的时候,因为while中的是R-L>1,所以当n为1时,将直接跳过二分判断。并且,如果存在可行的答案,其必然为二分循环结束后的L或者R,先判L,如果L不行就判断R;如果都不行,说明不存在正确结果。这里又wa了几发,,被二分教做人。

代码如下:

 #include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
using namespace std;
typedef long long ll;
const int N=2e5+,M=;
struct gadget{
int type,cost,num;
bool operator < (const gadget g) const{
return type==g.type?cost<g.cost:type<g.type;
}
}G[N];
int n,m,k,s,ter,a[N],b[N],da[N][M],db[N][M];
void init(){
for(int i=;i<n;i++) da[i][]=a[i],db[i][]=b[i];
for(int j=;(<<j)<=n;j++)
for(int i=;i+(<<j)-<n;i++){
da[i][j]=min(da[i][j-],da[i+(<<(j-))][j-]);
db[i][j]=min(db[i][j-],db[i+(<<(j-))][j-]);
}
}
int RMQ(int L,int R,int d[N][M]){
int k=;
while((<<(k+))<=R-L+) k++;
return min(d[L][k],d[R-(<<k)+][k]);
}
bool check(ll p1,ll p2){
int cnt=;
ll tot=;
int i=,j=ter;
while(j<m){
if(i==ter||cnt==k) break;
if(G[i].cost*p1<=G[j].cost*p2)
tot+=G[i++].cost*p1;
else tot+=G[j++].cost*p2;
cnt++;
}
if(cnt<k){
if(i<ter) swap(i,j),swap(p1,p2);
while(cnt<k) tot+=p2*G[j++].cost,cnt++;
}
return tot<=s;
}
int main(){
//freopen("in.txt","r",stdin);
while(~scanf("%d%d%d%d",&n,&m,&k,&s)){
for(int i=;i<n;i++)
scanf("%d",a+i);
for(int i=;i<n;i++)
scanf("%d",b+i);
ter=;
for(int i=;i<m;i++){
scanf("%d%d",&G[i].type,&G[i].cost);
if(G[i].type==) ter++;
G[i].num=i+;
}
init();
sort(G,G+m);
bool flag=false;
int L=,R=n-,mid;
ll p1,p2;
while(R-L>){
mid=((R-L)>>)+L;
p1=RMQ(,mid,da),p2=RMQ(,mid,db);
if(check(p1,p2)) R=mid;
else L=mid;
}
p1=RMQ(,L,da),p2=RMQ(,L,db);
if(check(p1,p2)) flag=true;
else{
p1=RMQ(,R,da),p2=RMQ(,R,db);
if(check(p1,p2)) flag=true;
}
if(flag){
int d1,d2;
for(int i=;i<n;i++) if(a[i]==p1){
d1=i+;break;
}
for(int i=;i<n;i++) if(b[i]==p2){
d2=i+;break;
}
printf("%d\n",max(d1,d2));
int cnt=,tot=;
int i=,j=ter;
while(j<m){
if(i==ter||cnt==k) break;
if(G[i].cost*p1<=G[j].cost*p2){
cnt++;
printf("%d %d\n",G[i++].num,d1);
}
else{
cnt++;
printf("%d %d\n",G[j++].num,d2);
}
}
if(cnt<k){
if(i<ter) swap(d1,d2),swap(i,j);
while(cnt<k) printf("%d %d\n",G[j++].num,d2),cnt++;
}
}
else puts("-1");
}
return ;
}

Codeforces 609D 被二分教做人的更多相关文章

  1. Codeforces #590 D 二维树状数组

    题意 给一个10^5之内的字符串(小写字母)时限2s 输入n,有n个操作  (n<10^5) 当操作是1的时候,输入位置x和改变的字母 当操作是2的时候,输入区间l和r,有多少不同的字母 思路 ...

  2. C - Monitor CodeForces - 846D (二维前缀和 + 二分)

    Recently Luba bought a monitor. Monitor is a rectangular matrix of size n × m. But then she started ...

  3. CodeForces 609D Gadgets for dollars and pounds

    二分天数+验证 #include<cstdio> #include<cstring> #include<cmath> #include<algorithm&g ...

  4. Gadgets for dollars and pounds CodeForces - 609D

    Nura wants to buy k gadgets. She has only sburles for that. She can buy each gadget for dollars or f ...

  5. Candies CodeForces - 991C(二分水题)

    就是二分暴力就好了 为什么要记下来 呵呵....emm你说为什么... 行吧 好吧 我一直以为我的二分出问题了 原来不是 依旧很帅 统计的时候求的减了多少次  然后用次数乘了mid 这样做会使那个人获 ...

  6. Codeforces 719E (线段树教做人系列) 线段树维护矩阵

    题面简洁明了,一看就懂 做了这个题之后,才知道怎么用线段树维护递推式.递推式的递推过程可以看作两个矩阵相乘,假设矩阵A是初始值矩阵,矩阵B是变换矩阵,求第n项相当于把矩阵B乘了n - 1次. 那么我们 ...

  7. codeforces 609D D. Gadgets for dollars and pounds(二分+贪心)

    题目链接: D. Gadgets for dollars and pounds time limit per test 2 seconds memory limit per test 256 mega ...

  8. Codeforces 1165F2(二分内的check)

    要点 二分答案,内部喜闻乐见的拖延策略:对于某个打折玩具,就选最晚的打折时间买,答案并不会变劣,只是购买时间的平移. 注意最晚时间不是预处理的东西,是二分内部的.在mid以内的最晚时间. #inclu ...

  9. Codeforces Round #353 (Div. 2) D. Tree Construction 二叉搜索树

    题目链接: http://codeforces.com/contest/675/problem/D 题意: 给你一系列点,叫你构造二叉搜索树,并且按输入顺序输出除根节点以外的所有节点的父亲. 题解: ...

随机推荐

  1. iptables详解(5):iptables匹配条件总结之二(常用扩展模块)

    所属分类:IPtables  Linux基础 在本博客中,从理论到实践,系统的介绍了iptables,如果你想要从头开始了解iptables,可以查看iptables文章列表,直达链接如下 iptab ...

  2. How To:防火墙规则去重

    主要命令 iptables-save| awk ' !x[$0]++ | iptables-restore 演示: [root@testname ~]# iptables -vL Chain INPU ...

  3. uva-679 Dropping Balls UVA - 679

    题目大意 总共有一个深度为D的满二叉树,I个小球,每个节点具有开关,小球经过节点后节点开关会有变化,初始都关闭,若关闭往左右否则往右走 只需要循环一下每层的情况即可 代码 #include <b ...

  4. JSONP代码收藏

    摘抄自jQuery,用于JSONP请求. var callback = 'callback_' + (new Date() - 0), url = 'http://localhost/', scrip ...

  5. 腾讯云,搭建python开发环境

    准备工作 任务时间:5min ~ 10min Python是一种解释型.面向对象.动态数据类型的高级程序设计语言.首先我们来看看系统中是否已经存在 Python ,并安装一些开发工具包: 安装前准备 ...

  6. Linq表达式写法

    Linq表达式,实现按照某个字段排序的简单写法. 做项目的时候遇到的一个简单问题,于是记下来. 列举一个例子: <T> model=new <T>(); 加入model中有要根 ...

  7. webstorm+nodejs环境中安装淘宝镜像

    用过nodejs的人都知道,从node的官方模板库下载依赖包的时候,经常会遇到“假死”(页面静止不动)的状态,这种速度简直要逼死焦急地等待下班的人.还好咱们万能的淘宝提供了淘宝镜像这么一个不要更好用的 ...

  8. 【ACM】NYOJ_288_Time_20130725

    Time时间限制:1000 ms  |  内存限制:65535 KB 难度:2描述 Digital clock use 4 digits to express time, each digit is ...

  9. 子序列 NYOJ (尺取法+队列+hash) (尺取法+离散化)

    子序列 时间限制:3000 ms  |  内存限制:65535 KB 难度:5   描述 给定一个序列,请你求出该序列的一个连续的子序列,使原串中出现的所有元素皆在该子序列中出现过至少1次. 如2 8 ...

  10. 《MySQL 5.7 Replication新特性》分享之互动问题解答

    原创 2016-07-21 宋利兵 MySQL中文网 分享主题 <MySQL 5.7 Replication新特性> 嘉宾介绍 宋利兵,MySQL研发工程师.2009年加入MySQL全球研 ...