题意:

\(n\)个物品,每个物品有一个价格,买一个高价格的物品,可以选择免费得到一个价格严格低于这个物品的物品。求得到\(n\)个物品的最小代价。

题解:

神仙贪心……

题目要求求出最小代价,相当于求最多能免费拿的价格。

先考虑一个\(n^2\)的DP:将物品按价格从高到低排序。把相同价格的物品放在一起处理。\(f[i][j]\)表示DP到第\(i\)位,免费拿了\(j\)个物品,最大能节约的价格。

那么已经原价买了但还没有送东西的物品有\(i - 2 * j\)个,我们称这些物品为可用物品。

有2种转移:

1,使用可用物品,免费拿当前物品(买一送一)

2,买走一个之前免费拿的物品,用腾出来的可用物品和新买的物品免费拿2个当前物品(买一送二)

但复杂度\(n^2\),无法通过此题。

现在来考虑优化:

我们对原来的DP数组进行差分得到\(g[i][j] = f[i][j] - f[i][j - 1]\).

也就是说\(g[i][j]\)表示的不是实际上的值,而是与上一次相比的增量。

既然对于每次转移,我们可以知道相比与上次的增量,那么我们考虑是否可以通过贪心来解决这个问题。

可以发现,\(g[i][j]\)一定单调不增,我们考虑通过3种操作来依次更新\(g\)数组.(2,3步贪心的正确性是基于已经完成了操作1的基础上的)

1,如果还有多余的可用物品,那么先把当前物品中能免费拿的都拿走。

2,获取\(g[i][j - 1]\)的增量\(x\),如果\(x < val_{now}\)那么选择免费拿走带来这个增量\(x\)的物品\(y\),我们买走它,因为\(x\)和腾出来的那个可用物品的价格都严格大于当前物品,因此我们可用最多免费拿2个当前物品放入\(g\)数组来更新增量,但也有可能只能放入一个,因为当前循环枚举的上限是将之前的所有物品都买下,能够免费拿走多少当前物品,而这个上限可能是奇数,所以在最后面如果拿2个可能就超过上限了。增量即为 免费拿的个数*当前物品价格。

3,增量\(x > val_now\),那么\(x\)现在还比较优秀,我们继续选择在\(g[i][j - 1]\)中免费拿走它,但对于\(g[i][j]\),如果在这个状态中,如果我们选择买走\(y\),然后用腾出来的物品和\(y\)免费拿走2个当前物品,是有可能更优的(因为不这么做就无法拿更多的物品),这样操作带来的增量可以表示为\(2 * s[now] - x\),如果这个增量大于0,那么我们选择用这个增量来更新\(g[i][j]\)。

用堆来维护即可。最后的答案即为总价格 - \(\sum{g[n][j]}\)(因为在上述转移中,每次转移都只有增量大于0我们才去更新\(g\)数组,即将真实的\(g\)数组对0取max了,因此\(\sum{g[n][j]} = max(f[n][j])\))

有一个容易产生疑惑的地方:为什么2中的增量是直接用数量×价格,而3中却要减去上一步的增量\(x\)呢?

可以观察到,这2个有一个本质上的区别:2虽然修改了\(g[i][j - 1]\),但从拿物品的方案上而言,是承接了\(g[i][j - 2]\)的,所以不必因为替换了一些物品而减小增量,毕竟\(g[i][j - 2]\)的那个增量是还在的。

而3在更新\(g[i][j]\)的过程中,所使用的方案,和\(g[i][j - 1]\)的方案并不一样,在\(g[i][j - 1]\)中,我们免费拿走了物品\(y\),带来了增量\(x\),而在\(g[i][j]\)中,我们买了物品\(y\),失去了增量\(x\),这样的话,虽然我们带来了\(2 * s[now]\)的增量,但相对于\(f[i][j - 1]\),我们的\(f[i][j]\)并没有增加这么多,因此总的增量为\(2 * s[now] - x\)

如果还有不懂的就看看官方题解 or 代码吧。

我感觉这题的代码应该是比较清晰的。(不喜欢注释的可以把注释删掉再看QWQ)

(如果您觉得我有地方理解错了,还请指出……毕竟这题比较神仙)

#include<bits/stdc++.h>
using namespace std;
#define R register int
#define AC 501000
#define LL long long int n, tot, sum, rnt;
int cnt[AC], s[AC], tmp[AC];
LL ans; struct cmp1{bool operator() (int a, int b){return a > b;}};//堆要用小根堆,因为f[j]必定递减,且优先替换便宜的肯定要优一些
inline bool cmp(int a, int b){return a > b;} priority_queue<int, vector<int>, cmp1> q; inline int read()
{
int x = 0;char c = getchar();
while(c > '9' || c < '0') c = getchar();
while(c >= '0' && c <= '9') x = x * 10 + c - '0', c = getchar();
return x;
} void pre()
{
n = read();
for(R i = 1; i <= n; i ++) s[i] = read(), ans += s[i];
sort(s + 1, s + n + 1, cmp);
for(R i = 1; i <= n; i ++)//去重
if(s[i] != s[i + 1]) s[++ tot] = s[i], ++ cnt[tot];
else ++ cnt[tot + 1];
} LL t[AC], top;
inline void check()
{
top = 0;
while(!q.empty()) t[++ top] = q.top(), q.pop();
for(int i = top; i; i --) printf("%lld ", t[i]), q.push(t[i]);
if(top) printf("\n");
else printf("0\n");
} void work()
{
for(R i = 1; i <= tot; i ++)
{
int have = sum - 2 * q.size();//获取现在不做任何修改可以免费送的物品
have = min(have, cnt[i]);//只能放下来取min,因为q.size默认是unsigned int……
for(R j = 1; j <= have; j ++) tmp[++ rnt] = s[i];//先把可以送的直接送了,单独放在一个数组里,防止和之前那些严格比当前大的混在一起
have = min(sum, cnt[i]) - have;//获取最多可以送几个(在当前已经送出一部分的前提下做修改)
for(R j = 1; j <= have; j += 2)//have重新赋值为最多还可以送的个数,因为每次会塞进来2个数,所以每次加2个
{//(min(sum, cnt[i])是把之前的全都买了可以送当前物品的个数,再减去已经送了的)
LL x = q.top(); q.pop();
if(x < s[i])//如果替换掉这一位更优,那么就换掉
{
tmp[++ rnt] = s[i];
if(j < have) tmp[++ rnt] = s[i];//如果j = have,只能说明是奇数,所以不能放第2个进来,,,
}
else//否则的话就维持之前的方案,并且由于无法替换(下一位置是0),只能尝试买下x,送2个s[j]
{
tmp[++ rnt] = x;//这个也要放到临时数组,,,不然就永远取不出更高的了
if(j < have && 2 * s[i] - x > 0) tmp[++ rnt] = 2 * s[i] - x;//下一状态就买下x,送2 个 s[i],不过如果没有增益的话,还不如不送
}
}
for( ; rnt; -- rnt) q.push(tmp[rnt]);
sum += cnt[i];//获取下一个位置(已处理总数)
// check();
}
while(!q.empty()) ans -= q.top(), q.pop();
printf("%lld\n", ans);
} int main()
{
freopen("in.in", "r", stdin);
pre();
work();
fclose(stdin);
return 0;
}

CF335F Buy One, Get One Free 贪心的更多相关文章

  1. 【CF865D】Buy Low Sell High(贪心)

    [CF865D]Buy Low Sell High(贪心) 题面 洛谷 CF 题解 首先有一个\(O(n^2)\)的\(dp\)很显然,设\(f[i][j]\)表示前\(i\)天手中还有\(j\)股股 ...

  2. LEETCODE —— Best Time to Buy and Sell Stock II [贪心算法]

    Best Time to Buy and Sell Stock II Say you have an array for which the ith element is the price of a ...

  3. hdu3438 Buy and Resell(优先队列+贪心)

    Buy and Resell Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)To ...

  4. CF867E: Buy Low Sell High(贪心, STL) (hdu6438)

    Description 有nn个城市,第ii个城市商品价格为aiai​,从11城市出发依次经过这nn个城市到达n n城市,在每个城市可以把手头商品出售也可以至多买一个商品,问最大收益. Input 第 ...

  5. CodeForces - 867E Buy Low Sell High (贪心 +小顶堆)

    https://vjudge.net/problem/CodeForces-867E 题意 一个物品在n天内有n种价格,每天仅能进行买入或卖出或不作为一种操作,可以同时拥有多种物品,问交易后的最大利益 ...

  6. [LeetCode] Best Time to Buy and Sell Stock II 贪心算法

    Say you have an array for which the ith element is the price of a given stock on day i. Design an al ...

  7. Leetcode 122 Best Time to Buy and Sell Stock II 贪心

    用一个数组表示股票每天的价格,数组的第i个数表示股票在第i天的价格.交易次数不限,但一次只能交易一支股票,也就是说手上最多只能持有一支股票,求最大收益. 关键:能赚就赚 class Solution ...

  8. 【leetcode刷题笔记】Best Time to Buy and Sell Stock II

    Say you have an array for which the ith element is the price of a given stock on day i. Design an al ...

  9. LeetCode--Best Time to Buy and Sell Stock (贪心策略 or 动态规划)

    Best Time to Buy and Sell Stock Total Accepted: 14044 Total Submissions: 45572My Submissions Say you ...

随机推荐

  1. hive 空值、NULL判断

    hive中空值判断基本分两种 (1)NULL 与 \N hive在底层数据中如何保存和标识NULL,是由 alter table name SET SERDEPROPERTIES('serializa ...

  2. python全栈开发-前方高能-内置函数2

    python_day_15 一.今日主要内容 1. lambda 匿名函数 语法: lambda 参数:返回值 不能完成复杂的操作 2. sorted() 函数 排序. 1. 可迭代对象 2. key ...

  3. My status

    I haven‘t any one who is strong relationship with me. My skill is normal. I'm not interesting in neg ...

  4. CentOS 下 Java 的下载、安装、配置

    CentOS 下 Java 的下载.安装.配置 系统: CentOS 7 x86_64 Java 版本: 1.8.0_171 本文将 Java 目录放在 /usr/local/java 文件夹下,读者 ...

  5. Python之requests的安装

    在 windows 系统下,只需要输入命令 pip install requests ,即可安装. 在 linux 系统下,只需要输入命令 sudo pip install requests ,即可安 ...

  6. Hbase 教程-安装

    HBase安装 安装前设置 安装Hadoop在Linux环境下之前,需要建立和使用Linux SSH(安全Shell).按照下面设立Linux环境提供的步骤. 创建一个用户 首先,建议从Unix创建一 ...

  7. NUMA 体系架构

    NUMA 体系架构 SMP 体系架构 NUMA 体系架构 NUMA 结构基本概念 Openstack flavor NUMA 策略 Nova 实现 NUMA 流程 1. SMP 体系架构 CPU 计算 ...

  8. how to update product listing price sale price and sale date using mobile App

    Greetings from Amazon Seller Support, Thank you for writing back to us. I have reviewed our previous ...

  9. 兰亭集势股价疯涨背后:物流成外贸B2C发展掣肘

    21世纪经济报道 汤浔芳 北京报道 核心提示:“兰亭集势涨势喜人,这样的增长是这两三年中概股没有出现过的.”一位负责美股投资的基金合伙人告诉记者,此前,中概股比较低迷,持续大幅度上涨,难得一见. 在唯 ...

  10. 使用Scrapy构建一个网络爬虫

    记得n年前项目需要一个灵活的爬虫工具,就组织了一个小团队用Java实现了一个爬虫框架,可以根据目标网站的结构.地址和需要的内容,做简单的配置开发,即可实现特定网站的爬虫功能.因为要考虑到各种特殊情形, ...