题目链接

题目描述

小店的优惠方案十分简单有趣:

一次消费过程中,如您在本店购买了精制油的话,您购买香皂时就可以享受2.00元/块的优惠价;如果您在本店购买了香皂的话,您购买可乐时就可以享受1.50元/听的优惠价......诸如此类的优惠方案可概括为:如果您在本店购买了商品A的话,您就可以以P元/件的优惠价格购买商品B(购买的数量不限)。

有趣的是,你需要购买同样一些商品,由于不同的买卖顺序,老板可能会叫你付不同数量的钱。比如你需要一块香皂(原价2.50元)、一瓶精制油(原价10.00元)、一听可乐(原价1.80元),如果你按照可乐、精制油、香皂这样的顺序购买的话,老板会问你要13.80元;而如果你按照精制油、香皂、可乐这样的顺序购买的话,您只需付13.50元。

该处居民发现JSOI集训队的队员均擅长电脑程序设计,于是他们请集训队的队员编写一个程序:在告诉你该小店商品的原价,所有优惠方案及所需的商品后,计算至少需要花多少钱(不允许购买任何不必要的商品,即使这样做可能使花的钱更少)。

Sol

首先不用买的全部丢掉 , 优惠活动也不用管。

考虑一个最优策略 , 显然我们不会蠢到白白的优惠不要 , 那么一定是先所有要买的物品都买一个 , 然后就能取得所有优惠。直接取最小值当做价格即可。

那么问题就是让所有物品买一次后总价格最小。

这就很像一个生成树了。

由于有些物品可能没有优惠 , 我们建立一个超级点 , 向所有其他物品连原价的边 , 相当于买了超级点其他物品都能以原价的优惠购买。

这样就是要求一个以超级点为根的最小树形图了 , 朱刘算法即可。

简单说明朱刘算法的求解步骤:

首先要明确的一点是每一个点只会有一条入边。

  1. 为所有点找到一个最小的入边。
  2. 把这些边的权值加入答案 , 然后判断图中是否存在环。
  3. 不存在环那么算法结束 , 否则缩环(直接往入边方向跳即可,不要tarjan) , 并把所有不在环内边的边的权值减去指向点的最小入边(之前的环不合法 , 其中要去掉一条边 , 这里减去原入边边权就是在之后选择新的边的时候考虑了这个权值的变化)

本题code:

  1. #include<bits/stdc++.h>
  2. using namespace std;
  3. typedef double db;
  4. const int M=6000;
  5. const int N=51;
  6. struct edge{
  7. int u,v;db val;
  8. edge(){u=v=0,val=0.0;}
  9. edge(int a,int b,db c){u=a,v=b,val=c;}
  10. }a[M],b[M];
  11. int In[N],vis[N],bel[N],cnt=0,m,n,bcc;
  12. db cost[N];int ned[N];
  13. db Mi[N];
  14. int rt;db ans=0;
  15. int flag=0;
  16. bool MST(){
  17. for(int i=1;i<=n;++i) In[i]=vis[i]=bel[i]=0;bcc=0,cnt=0;
  18. for(int i=1;i<=m;++i) {
  19. int u=a[i].u,v=a[i].v;db w=a[i].val;
  20. if(!In[v]||a[In[v]].val>w) In[v]=i;
  21. }
  22. for(int i=flag;i<=n;++i) {
  23. if(!flag&&!ned[i]) continue;
  24. if(i!=rt) ans+=a[In[i]].val;
  25. if(vis[i]) continue;
  26. int u=i;
  27. for(;u^rt&&vis[u]!=i&&!bel[u];u=a[In[u]].u) vis[u]=i;
  28. if(u^rt&&!bel[u]) {
  29. ++bcc;while(bel[u]!=bcc) bel[u]=bcc,u=a[In[u]].u;
  30. }
  31. }
  32. if(!bcc) return 0;
  33. for(int i=flag;i<=n;++i) {
  34. if(!flag&&!ned[i]&&i!=rt) continue;
  35. if(!bel[i]) bel[i]=++bcc;
  36. }
  37. for(int i=1;i<=m;++i) {
  38. int u=a[i].u,v=a[i].v;db w=a[i].val;
  39. if(bel[u]==bel[v]) continue;
  40. db goi=a[In[v]].val;b[++cnt]=(edge){bel[u],bel[v],w-goi};
  41. }
  42. for(int i=1;i<=cnt;++i) a[i]=b[i];
  43. n=bcc;m=cnt;
  44. return 1;
  45. }
  46. int main()
  47. {
  48. cin>>n;int tot=n;// !! 注意记录啊
  49. for(int i=1;i<=n;++i) cin>>cost[i]>>ned[i],Mi[i]=cost[i];
  50. cin>>m;
  51. for(int i=1;i<=m;++i) {
  52. int A,B;db P;
  53. cin>>A>>B>>P;
  54. if(!ned[A]||!ned[B]) --i,--m;
  55. else {Mi[B]=min(Mi[B],P);a[i]=edge(A,B,P);}
  56. }
  57. for(int i=1;i<=n;++i) if(ned[i]) a[++m]=edge(0,i,cost[i]);
  58. rt=0;flag=0;
  59. while(MST()) rt=bel[rt],flag=1;
  60. for(int i=1;i<=tot;++i) {
  61. if(ned[i]) {
  62. --ned[i];
  63. ans+=Mi[i]*(db)ned[i];
  64. }
  65. }
  66. printf("%.2lf\n",ans);
  67. }

【LuoguP2792 】[JSOI2008]小店购物(最小树形图)的更多相关文章

  1. Luogu2792 JSOI2008 小店购物 最小树形图

    传送门 被题意杀 本以为一个种类的物品一定要一起买 看了题解才知道可以先把所有要买的物品买一个,剩下要买的物品就可以得到这个种类的物品能够得到的最大优惠-- 所以现在只需要知道:第一次买所有物品一遍时 ...

  2. 洛谷P2792 [JSOI2008]小店购物(最小树形图)

    题意 题目链接 Sol 一开始的思路:新建一个虚点向每个点连边,再加上题面中给出的边,边权均为大小*需要购买的数量 然后发现死活都过不去 看了题解才发现题目中有个细节--买了\(A\)就可以买\(B\ ...

  3. Luogu2792 [JSOI2008]小店购物

    Luogu2792 [JSOI2008]小店购物 重题 bzoj4349 最小树形图 有 \(n\) 个物品,每个物品有价格 \(c_i\) 和所需个数 \(k_i\) ,所有物品必须恰好买 \(k_ ...

  4. [JSOI2008]小店购物 & bzoj4349:最小树形图 最小树形图

    ---题面(洛谷)--- ---题面(bzoj)--- 其实是同一道题,,,样例都一模一样 题解: 一开始看想了好久,,,还想到了最短路和最小生成树,,然而写的时候才意识到最小生成树应该要用无向边 其 ...

  5. JSOI2008 小店购物

    https://www.luogu.org/problem/show?pid=2792 题目背景 JSOI集训队的队员发现,在他们经常活动的集训地,有一个小店因为其丰富的经营优惠方案深受附近居民的青睐 ...

  6. 最小树形图--朱刘算法([JSOI2008]小店购物)

    题面 luogu Sol 首先设一个 \(0\) 号点,向所有点连边,表示初始价值 显然这个图的一个 \(0\) 为根的最小有向生成树的边权和就是每个买一次的最小价值 再买就一定能优惠(包含 \(0\ ...

  7. 【Bzoj2260】【Bzoj4349】商店购物 & 最小树形图

    目录 List Bzoj 2260 商店购物 Description Input Output Sample Input Sample Output Bzoj 4349 最小树形图 Descripti ...

  8. bzoj2260: 商店购物 && 4349: 最小树形图

    Description Grant是一个个体户老板,他经营的小店因为其丰富的优惠方案深受附近居民的青睐,生意红火.小店的优惠方案十分简单有趣.Grant规定:在一次消费过程中,如果您在本店购买了精制油 ...

  9. bzoj2260: 商店购物&&4349: 最小树形图

    最小树形图问题啊 最小树形图是撒哩,就是给你一个有向图,确定一个根,要你到达所有点,那棵最短路径树的总边权 做这个用的是朱(jv)刘(lao)算法. 首先假如有多个联通块就无解啦 对应每个点(除了根) ...

随机推荐

  1. robot framework 接口自动化之登录

    网络不便,好久没更了,颓废好久,惭愧 目录 1.安装必须的库 2.固定格式介绍 3.完成一个登录 1.安装必须的库 requestsLibrary.requests安装 1.pip install r ...

  2. c# AES128 加解密算法

    using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.D ...

  3. 将Unix时间戳转换为Date、Json属性动态生成反序列化、序列化指定属性

    实体类 public class Test { [JsonIgnore] public string GetDate { get { return GetTime.ToString("yyy ...

  4. excel 导入导出测试点

    目前,为方便操作,很多系统都会增加批量导入导出的功能.文件导入导出一般格式都是excel.由于用户直接在excel在填写内容,无法控制填写的格 式,加上excel解析比较困难,所以一般涉及到excel ...

  5. Android Studio编译报错Could not reserve enough space for 2097152KB object heap解决方法

    环境变量中添加

  6. 【Qt开发】【Linux开发】QT设置环境变量QWS_DISPLAY

    QT设置环境变量QWS_DISPLAY 当应用程序./myQtApp -qws启动时,会去检测QWS_DISPLAY这个环境变量, 判断界面最终显示在哪个framebuffer中, 如果是虚拟的fra ...

  7. Spring Security Session Time Out

    最近在用Spring Security做登录管理,登陆成功后,页面长时间无操作,超过session的有效期后,再次点击页面操作,页面无反应,需重新登录后才可正常使用系统. 为了优化用户体验,使得在se ...

  8. axios入门使用

    vue项目中axios的基本使用和简单封装 axios中文文档官网 http://www.axios-js.com/docs/ 一:不封装直接使用 npm install axios 在main.js ...

  9. 004 gcc 编译 C/C++ 默认使用哪个标准

    0. 前言 我挺久没碰 C,不想就这么忘了,最近重温了一些相关知识 1. C 语言的几种"方言" 简单地说,有这么几种常见的 年份 名称 1983 ANSI C 1987 C87 ...

  10. JavaScript ES6 class指南

    前言 EcmaScript 2015 (又称ES6)通过一些新的关键字,使类成为了JS中一个新的一等公民.但是目前为止,这些关于类的新关键字仅仅是建立在旧的原型系统上的语法糖,所以它们并没有带来任何的 ...