题面

luogu

Sol

首先设一个 \(0\) 号点,向所有点连边,表示初始价值

显然这个图的一个 \(0\) 为根的最小有向生成树的边权和就是每个买一次的最小价值

再买就一定能优惠(包含 \(0\) 的边)

有向图最小生成树???

朱刘算法

其实正确性不会理论。。

可以说是一个不断调整的过程,从而得到最优解

时间复杂度 \(O(VE)\)

流程:

1.去掉自环

2.先给每个点选择一条最小的入边,并记录连过来的点

3.如果此时有点没有入边(除根以外),那么显然无解

4.算上每个点入边贡献,加入答案

5.如果没有有向环,那么做完结束

6.如果有,缩点,并且把连出去的边都减去连出去那个点的入边,因为贡献算过了

7.存储新图,对新图重复所有操作

Code

# include <bits/stdc++.h>
# define IL inline
# define RG register
# define Fill(a, b) memset(a, b, sizeof(a))
using namespace std;
typedef long long ll; const int maxn(100005);
const double inf(1e9); int n, tot, m, cnt, need[maxn], pre[maxn], vis[maxn], id[maxn];
double ans, cost[maxn], prize, inw[maxn]; struct Edge{
int u, v;
double w;
} e[maxn]; IL void DirectedMST(){
RG int num = n, rt = 1, idx;
while(true){
// 初始化
for(RG int i = 1; i <= num; ++i) id[i] = vis[i] = pre[i] = -1, inw[i] = inf;
// 选入边
for(RG int i = 1; i <= cnt; ++i)
if(inw[e[i].v] > e[i].w && e[i].u != e[i].v) inw[e[i].v] = e[i].w, pre[e[i].v] = e[i].u;
pre[rt] = rt, idx = inw[rt] = 0;
// 缩环,统计贡献
for(RG int i = 1; i <= num; ++i){
ans += inw[i];
if(vis[i] == -1){
RG int nw = i;
while(vis[nw] == -1) vis[nw] = i, nw = pre[nw];
if(vis[nw] == i && nw != rt){
id[nw] = ++idx;
for(RG int j = pre[nw]; j != nw; j = pre[j]) id[j] = idx;
}
}
}
// 没有环结束
if(!idx) return;
// 重标号,记录新图
for(RG int i = 1; i <= num; ++i) if(id[i] == -1) id[i] = ++idx;
for(RG int i = 1; i <= cnt; ++i)
e[i].w -= inw[e[i].v], e[i].u = id[e[i].u], e[i].v = id[e[i].v];
num = idx, rt = id[rt];
}
} int main(){
scanf("%d", &tot), n = 2;
for(RG int i = 1; i <= tot; ++i){
scanf("%lf%d", &cost[n], &need[n]);
if(need[n]) e[++cnt] = (Edge){1, n, cost[n]}, vis[i] = n++;
}
--n, scanf("%d", &m);
for(RG int i = 1, a, b; i <= m; ++i){
scanf("%d%d%lf", &a, &b, &prize);
a = vis[a], b = vis[b];
if(a && b){
cost[b] = min(cost[b], prize);
e[++cnt] = (Edge){a, b, prize};
}
}
for(RG int i = 2; i <= n; ++i) ans += (need[i] - 1) * cost[i];
DirectedMST();
printf("%.2lf\n", ans);
return 0;
}

最小树形图--朱刘算法([JSOI2008]小店购物)的更多相关文章

  1. 最小树形图——朱刘算法(Edmonds)

    定义:一个有向图,存在从某个点为根的,可以到达所有点的一个最小生成树,则它就是最小树形图. 朱刘算法实现过程: [在选出入边集后(看步骤1),若有向图中不存在有向环,说明该图就是最小树形图] 1,选入 ...

  2. POJ 3164 Command Network 最小树形图 朱刘算法

    =============== 分割线之下摘自Sasuke_SCUT的blog============= 最 小树形图,就是给有向带权图中指定一个特殊的点root,求一棵以root为根的有向生成树T, ...

  3. POJ 3164 Command Network ( 最小树形图 朱刘算法)

    题目链接 Description After a long lasting war on words, a war on arms finally breaks out between littlek ...

  4. poj3164(最小树形图&朱刘算法模板)

    题目链接:http://poj.org/problem?id=3164 题意:第一行为n, m,接下来n行为n个点的二维坐标, 再接下来m行每行输入两个数u, v,表点u到点v是单向可达的,求这个有向 ...

  5. 洛谷P4716 【模板】最小树形图(朱刘算法)

    题意 题目链接 Sol 朱刘算法?感觉又是一种神仙贪心算法 大概就是每次贪心的用每个点边权最小的入边更新答案,如果不行的话就缩起来找其他的边 不详细说了,丢链接走人.. #include<bit ...

  6. POJ - 3164-Command Network 最小树形图——朱刘算法

    POJ - 3164 题意: 一个有向图,存在从某个点为根的,可以到达所有点的一个最小生成树,则它就是最小树形图. 题目就是求这个最小的树形图. 参考资料:https://blog.csdn.net/ ...

  7. bzoj 4349 最小树形图——朱刘算法

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4349. 学习博客:http://www.cnblogs.com/xzxl/p/7243466 ...

  8. Luogu2792 [JSOI2008]小店购物

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

  9. 【LuoguP2792 】[JSOI2008]小店购物(最小树形图)

    题目链接 题目描述 小店的优惠方案十分简单有趣: 一次消费过程中,如您在本店购买了精制油的话,您购买香皂时就可以享受2.00元/块的优惠价:如果您在本店购买了香皂的话,您购买可乐时就可以享受1.50元 ...

随机推荐

  1. Leetcode 88 合并两个有序数组 Python

    合并两个有序数组 给定两个有序整数数组 nums1 和 nums2,将 nums2 合并到 nums1 中,使得 num1 成为一个有序数组. 说明: 初始化 nums1 和 nums2 的元素数量分 ...

  2. springboot自定义配置文件

    前言:如果你一点spring的基础没有,建议你不要学习springboot,至少先有一个spring的项目经验或者自己搭建过spring的项目再学习springboot,这样你会发现在spring中搞 ...

  3. random 模块常用方法学习

    >>> import random#随机小数>>> random.random() # 大于0且小于1之间的小数0.7664338663654585>> ...

  4. PlistBuddy简单使用

    PlistBuddy简单使用 由于PlistBuddy并不在Mac默认的Path里,所以我们得通过绝对路径来引用这个工具: 查看帮助 /usr/libexec/PlistBuddy --help 下面 ...

  5. OmniMarkupPreviewer 404

    Here is the answer: Sublime Text > Preferences > Package Settings > OmniMarkupPreviewer > ...

  6. vue封装插件并发布到npm上

    vue封装插件并发布到npm上 项目初始化 首先,要创建项目,封装vue的插件用webpack-simple很合适,vue init webpack-simple 项目名称此命令创建我们的项目的目录, ...

  7. openproject安装与使用

    思路: 1.生成config配置文件 2.导出配置文件,修改配置文件,删除容器,重新部署容器 3.登录后配置, 4.配置git,可以从openproject里查看gitlab上的代码库 第一次安装: ...

  8. linux如何安装和启动mongdb

    1.下载安装包 下载地址: https://www.mongodb.com/dr/fastdl.mongodb.org/linux/mongodb-linux-x86_64-4.0.9.tgz/dow ...

  9. js字符串去重

    js字符串去重: 1.  去掉字符串前后所有空格: function Trim(str) { return str.replace(/(^\s*)|(\s*$)/g, ""); } ...

  10. Spring Boot中自动执行sql脚本

    说明:所有的代码基于SpringBoot 2.0.3版本 背景 在应用程序启动后,可以自动执行建库.建表等SQL脚本.下文中以要自动化执行people.sql脚本为例说明,脚本在SpringBoot工 ...