题目链接:https://zhixincode.com/contest/3/problem/J?problem_id=43

样例输入 1

4 11
10 1
1 1
10 2
1 2
10 3
1 3
15 4
15 4
15 4
15 4
15 4

样例输出 1

28

题解:

首先是有个简单的想法,假设wls买完后,$n$ 个居民他们的各自的宝物数目最大不超过 $k$,因此wls手里的宝物数目至少要大于 $k$。

所以暴力枚举 $k$,然后再暴力地对所有宝物数目超过 $k$ 的居民,将他们买到不超过 $k$;然后如果此时wls手里的宝物数 $\leq k$ 则再从所有居民的宝物中挑最便宜的买,直到wls的宝物数目大于 $k$ 为止(假设要再买 $e$ 个宝物才行)。

如果是div2的话,由于 $n,m$ 范围小,想到这里就可以直接写了,div1的话还需要考虑优化。

我们知道,如果把居民按宝物数目从大到小排序,并且每个居民的宝物按价值排序(小的放在上面),那么从大到小枚举 $k$ 的时候,就像一把刀一层层的往下压,那么每个居民的宝物局如同一个不规则的楼梯上一层层的被削去,由于宝物数目最多 $m$ 个,因此把这些宝物一点点收入wls的囊中只需要 $O(m)$ 的时间复杂度。

那么,再考虑怎么从居民手里还剩的宝物中,再买 $e$ 个宝物以使wls的宝物数最多,可以用线段树来做,每次已经确定收入wls囊中的宝物都在线段树中标成 $0$,然后在线段树中用数量 $e$ 去查询得到相应的花费。最后维护每个 $k$ 对应的wls总花费的最小值即可。

这样一来,时间复杂度就是 $O(m \log m)$ 的,就不会超时了。

AC代码:

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef pair<ll,int> P;
#define val(p) (p.first)
#define idx(p) (p.second)
#define mk(x,y) make_pair(x,y)
const ll INF=1e16;
const int maxn=1e5+, maxm=1e5+; int n,m;
vector<P> peo[maxn]; //按居民存储宝物
int rk[maxn]; //按宝物数目存储居民编号
bool cmp(int a,int b) {
return peo[a].size()>peo[b].size();
} P tre[maxm]; //存储所有宝物并排序,并构建相应的线段树
int pos[maxm]; //指出宝物在线段树中的位置 #define ls (rt<<1)
#define rs (rt<<1|1)
struct Node{
int l,r;
ll v; int c;
}o[maxm<<];
void pushup(int rt)
{
o[rt].v=o[ls].v+o[rs].v;
o[rt].c=o[ls].c+o[rs].c;
}
void build(int rt,int l,int r)
{
o[rt].l=l, o[rt].r=r;
if(l==r)
{
o[rt].v=val(tre[l]), o[rt].c=;
return;
}
int mid=(l+r)>>;
build(ls,l,mid), build(rs,mid+,r);
pushup(rt);
}
void update(int rt,int pos)
{
if(o[rt].l==o[rt].r)
{
o[rt].v=, o[rt].c=;
return;
}
int mid=(o[rt].l+o[rt].r)>>;
pos<=mid?update(ls,pos):update(rs,pos);
pushup(rt);
}
ll query(int rt,int k)
{
if(o[rt].l==o[rt].r) return o[rt].v;
if(o[ls].c>=k) return query(ls,k);
else return o[ls].v+query(rs,k-o[ls].c);
} int main()
{
ios::sync_with_stdio();
cin.tie(), cout.tie(); cin>>n>>m;
for(int i=;i<=n;i++) peo[i].clear();
for(int i=;i<=m;i++)
{
ll a; int c; cin>>a>>c;
peo[c].push_back(tre[i]=mk(a,i));
}
for(int i=;i<=n;i++) sort(peo[i].begin(),peo[i].end(),greater<P>());
for(int i=;i<=n;i++) rk[i]=i; sort(rk+,rk+n+,cmp); sort(tre+,tre+m+);
for(int i=;i<=m;i++) pos[idx(tre[i])]=i; //根据原编号得到在线段树中的位置
build(,,m); int ed=;
ll ans=INF;
int acc_tot=; ll acc_cost=;
for(int k=peo[rk[]].size();k>=;k--)
{
for(;ed<=n && peo[rk[ed]].size()>k;ed++);
for(int i=;i<ed;i++)
{
while(peo[rk[i]].size()>k)
{
acc_cost+=val(peo[rk[i]].back());
acc_tot++;
update(,pos[idx(peo[rk[i]].back())]);
peo[rk[i]].pop_back();
}
} ll cost; //wls的总花费
if(acc_tot<=k) cost=acc_cost+query(,min(m,k+-acc_tot));
else cost=acc_cost; ans=min(ans,cost);
}
cout<<ans<<endl;
}

CCPC-Wannafly Winter Camp Day1 Div1 - 夺宝奇兵 - [贪心+线段树]的更多相关文章

  1. CCPC-Wannafly Winter Camp Day4 Div1 - 夺宝奇兵 - [简单思维题]

    题目链接:https://zhixincode.com/contest/18/problem/A?problem_id=259 题目描述 wls正在玩一个寻宝游戏. 宝藏一共有 $n$ 种,都藏在一个 ...

  2. 2020 CCPC Wannafly Winter Camp Day1 C. 染色图

    2020 CCPC Wannafly Winter Camp Day1 C. 染色图 定义一张无向图 G=⟨V,E⟩ 是 k 可染色的当且仅当存在函数 f:V↦{1,2,⋯,k} 满足对于 G 中的任 ...

  3. 2020 CCPC Wannafly Winter Camp Day1 Div.1&amp F

    #include<bits/stdc++.h> #define forn(i, n) for (int i = 0; i < int(n); i++) #define fore(i, ...

  4. 2020 CCPC Wannafly Winter Camp Day1 - I. K小数查询(分块)

    题目链接:K小数查询 题意:给你一个长度为$n$序列$A$,有$m$个操作,操作分为两种: 输入$x,y,c$,表示对$i\in[x,y] $,令$A_{i}=min(A_{i},c)$ 输入$x,y ...

  5. Wannafly Winter Camp Day8(Div1,onsite) E题 Souls-like Game 线段树 矩阵乘法

    目录 Catalog Solution: (有任何问题欢迎留言或私聊 && 欢迎交流讨论哦 Catalog @ Problem:传送门  Portal  原题目描述在最下面.  简单的 ...

  6. CCPC-Wannafly Winter Camp Day1 Div1 - 爬爬爬山 - [最短路][堆优化dijkstra]

    题目链接:https://zhixincode.com/contest/3/problem/F?problem_id=39 样例输入 1  4 5 1 1 2 3 4 1 2 1 1 3 1 1 4 ...

  7. CCPC Wannafly Winter Camp Div2 部分题解

    Day 1, Div 2, Prob. B - 吃豆豆 题目大意 wls有一个\(n\)行\(m\)列的棋盘,对于第\(i\)行第\(j\)列的格子,每过\(T[i][j]\)秒会在上面出现一个糖果, ...

  8. Wannafly Winter Camp Day5 Div1 E题 Fast Kronecker Transform 转化为NTT或FFT

    目录 Catalog Solution: (有任何问题欢迎留言或私聊 && 欢迎交流讨论哦 Catalog @ Problem:传送门  原题目描述在最下面.  对给定的式子算解.   ...

  9. 2020 CCPC Wannafly Winter Camp Day2-K-破忒头的匿名信

    题目传送门 sol:先通过AC自动机构建字典,用$dp[i]$表示长串前$i$位的最小代价,若有一个单词$s$是长串的前$i$项的后缀,那么可以用$dp[i - len(s)] + val(s)$转移 ...

随机推荐

  1. cmd adb批量安装与卸载

    批量安装: SET dir=%~dp0echo dir is: %dir%cd /d %dir%for /R %dir% %%i in (*.apk) do adb install %%i 批量卸载: ...

  2. WEBAPI 的简单示例

    一.webapi 1.在webapiconfig中移除xml的返回格式,返回格式就自动使用Json格式 config.Formatters.Remove(config.Formatters.XmlFo ...

  3. IDEA环境设置

    设置SDK:https://blog.csdn.net/y999666/article/details/51893348 打开模板使用说明,找到Maven本地安装目录, 备份E:\Program Fi ...

  4. IT? 挨踢

    中国的IT,是最憋屈的IT. 他们掌握着正常人看不懂的英文+字母+标点符号组成的各类代码语言 他们像作者一样从无到有,从空白的白纸上敲出上千上万条华丽的计算机语言 但是他们承受着正常人的鄙视: 我的需 ...

  5. 菜鸟教程之工具使用(七)——从GIt上导出Maven项目

    今天继续我们的工具教程,公司用Git作为版本控制工具,所以最近一直在跟Git打交道.也是一边学习一边使用,于是想做一些入门教程,一来自己总结一下,二来还能帮助一些刚刚接触Git的朋友.一举两得,何乐而 ...

  6. 【Big Data - Hadoop - MapReduce】通过腾讯shuffle部署对shuffle过程进行详解

    摘要: 通过腾讯shuffle部署对shuffle过程进行详解 摘要:腾讯分布式数据仓库基于开源软件Hadoop和Hive进行构建,TDW计算引擎包括两部分:MapReduce和Spark,两者内部都 ...

  7. PX4/PixHawk无人机飞控应用开发

    最近做的一个国防背景的field UAV项目,细节不能多谈,简单写点技术体会. 1.PX4/Pixhawk飞控软件架构简介 PX4是目前最流行的开源飞控板之一.PX4的软件系统实际上就是一个firmw ...

  8. java中的数据加密3 非对称加密

    非对称加密也加公钥加密,不对称算法使用一对密钥对,一个公钥,一个私钥,使用公钥加密的数据,只有私钥能解开(可用于加密):同时,使用私钥加密的数据,只有公钥能解开(签名).但是速度很慢(比私钥加密慢10 ...

  9. Dapper的基本使用,Insert、Update、Select、Delete

    简介 Dapper是.NET下一个micro的ORM,它和Entity Framework或Nhibnate不同,属于轻量级的,并且是半自动的.也就是说实体类都要自己写.它没有复杂的配置文件,一个单文 ...

  10. Python 字符串转JSON; 先装字典在转JSON; json.dumps(d)

    #-*- coding:UTF-8 -*- import os; import json class MysqlUtil(): def __init__(self): pass if __name__ ...