tags:

  • 贪心

    date: 2019-4-4

jag2017autumnJ Farm Village

题面

题目链接

翻译

数轴上有 n 个村庄,每个村庄可以生产两个单位的粮食。在每个村庄生产一单位粮食有一定的代价。运输粮食的代价就是运输的距离。求最小代价使得每个村庄都有 一个单位的粮食。


题解

虽然在叶老的课件里这题是贪心优化费用流,但是讲起来和写起来和费用流没有毛关系

首先先把问题抽象一下,每个村庄可以有两个出口,需要一个入口,自给自足就可以看作是自己出口给自己,然后

根据一般套路准备两个堆,一个出口一个进口。

————叶老

首先预处理一下,把 d 取前缀和,方便计算距离

那么我们先贪心的想(也就是不计之后的代价),把这一次的出口与之前的入口需求相匹配,如果可以优化答案(也就是他们之间的距离 + \(G_i\)比相匹配的节点自给自足更优),那么就直接使用

然后就有反悔操作,就是更加后面的节点的出口,匹配前面我们匹配上的这个节点,所以每当我们根据贪心取完一次后(贪心每次是从入口堆里取最小值与当前的出口相匹配),就再往入口堆里面丢一个\(-(当前给答案的贡献) - dist[i] + g[i]\),方便之后反悔(说实话只有这里有点像网络流)

然后还有一个反悔就是这一次的出口不给之前的,给之后的,那么就往出口堆里丢一个\(-(给答案造成的贡献) - dist[i] + g[i]\),以便之后反悔

做完上述的发现自己的入口需求还没有解决,那么就从出口堆里取出一个最优的,与自己匹配

然后再往入口堆里丢一个\(-(对答案造成的贡献) - dist[j]\)以便之后反悔(用后面的出口来满足自己的入口)

代码

/**************************
* Author : Leo101
* Problem : Farm Village
* Tags : 贪心
**************************/
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <functional>
#define _FILE(s) freopen(#s".in", "r", stdin); freopen(#s".out", "w", stdout)
#define gi get_int()
typedef long long LL;
#define int LL
const int MAXN = 200010;
int get_int()
{
int x = 0, y = 1; char ch = getchar();
while (!isdigit(ch) && ch != '-') ch = getchar();
if (ch == '-') y = -1, ch = getchar();
while (isdigit(ch)) x = x * 10 + ch - '0', ch = getchar();
return x * y;
} int heap1[MAXN], heap2[MAXN], dist[MAXN], cnt1, cnt2;
int topHeap(int* heap, int &cnt)
{
return heap[0];
}
int deleteHeap(int* heap, int &cnt)
{
std :: pop_heap(heap, heap + cnt, std :: greater<int>());
return heap[--cnt];
}
void insertHeap(int* heap, int val, int &cnt)
{
heap[cnt++] = val;
std :: push_heap(heap, heap + cnt, std :: greater<int>());
} signed main()
{
_FILE(code); int n = gi, ans = 0;
for (int i = 1; i < n; i++) dist[i] = gi + dist[i - 1];
for (int i = 0; i < n; i++) {
int cost = gi;
for (int j = 0; j < 2; j++) {
int tmp = topHeap(heap1, cnt1) + dist[i] + cost;
if (cnt1 != 0 && tmp < 0) {
deleteHeap(heap1, cnt1); ans += tmp;
insertHeap(heap1, -dist[i] - cost, cnt1);
insertHeap(heap2, -dist[i] + cost - tmp, cnt2);
} else {
insertHeap(heap2, -dist[i] + cost, cnt2);
}
}
int tmp = deleteHeap(heap2, cnt2) + dist[i];
ans += tmp;
insertHeap(heap1, -tmp - dist[i], cnt1);
} printf("%lld\n", ans); return 0;
}

leo101的更多相关文章

  1. Luogu P3757 [CQOI2017]老C的键盘

    题目描述 老C的键盘 题解 显然对于每个数 x 都有唯一对应的 \(x/2\) , 然而对于每个数 x 却可以成为 \(x*2\) 和 \(x*2+1\) 的对应数 根据这一特性想到了啥??? 感谢l ...

随机推荐

  1. ElasticSearch 简单的crud查询

    //数据库和es的对应关系(学习文档可以参考https://es.xiaoleilu.com/010_Intro/35_Tutorial_Aggregations.html) //如下接口调用都是使用 ...

  2. Spring Boot 系统启动任务定义

    前言 系统任务:在项目启动阶段要做一些数据初始化操作,这些操作有一个共同的特点,只在项目启动时进行,以后都不再执行. 应用场景:例如配置文件加载,数据库初始化等操作 Spring Boot出现之前 解 ...

  3. 【小白学PyTorch】17 TFrec文件的创建与读取

    [新闻]:机器学习炼丹术的粉丝的人工智能交流群已经建立,目前有目标检测.医学图像.时间序列等多个目标为技术学习的分群和水群唠嗑的总群,欢迎大家加炼丹兄为好友,加入炼丹协会.微信:cyx64501661 ...

  4. Web前后端:如何分离,如何解耦?

    摘要:在本文中我们一起来探讨一下为什么要在软件开发中进行前后端的分离,如何做到前后端分离,如何解耦. 简单地说,就是要把复杂的问题简单化,把一个从0到N的问题转化为N个0到1的问题.另一个相近的说法就 ...

  5. 一文带你定制unittest测试用例的名称

    在之前的文章中,我在之前的文章中提到过,这里呢,考虑后,感觉之前的写法不够优雅,于是乎呢,我自己抽空去研究了下,主要是新写方法,这样呢,以后的要使用的时候,可以直接去使用,而不是每次换个环境就要修改环 ...

  6. xshell选项卡不见了

    最近使用xshell的时候发现建立多个选项卡的时候,因为没有选项卡,所以不能切换. 弄了好一会儿才弄出来 点击会话选项卡或者Ctrl + Shift + T可以调出来

  7. 工信部今日向三大运营商和中国广电发放5G商用牌照

    央视快讯:工信部向中国电信.中国移动.中国联通.中国广电发放5G商用牌照. 2016年5月5日,工信部向中国广播电视网络有限公司颁发了<基础电信业务经营许可证>,批准中国广播电视网络有限公 ...

  8. vue 异步提交php 两种方式传值

    1.首先要在php的入口文件写上一条代码,允许异步提交 header("ACCESS-CONTROL-ALLOW-ORIGIN:*"); 2.在vue有两种方式将数据异步提交到ph ...

  9. 多测师_肖sir_性能测试之性能测试了解001(jmeter)

    一.了解jmeter 1.Jmeter的概念? JMeter是Apache组织开发的基于Java的压力测试工具.具有开源免费.框架灵活.多平台支持等优势.除了压力测试外,JMeter在接口测试方面也有 ...

  10. 多测师讲解python_oo1基本操作

    1.什么是Python? Python是一门面向对象,解释型的动态类型的编程语言,有Guido van Rossunm于1989年发明,第一个公开发行版发行于1991年: Guido van Ross ...