题意:

有N个任务M个机器,给每个任务i完成所花费的时间Pi且每个任务要在第Si天后开始,在第Ei天前结束,保证任务在(S,E)之间一定能完成。

每个机器在一天里只能运行一个任务,一个任务可以在中途更换机器继续执行,问有没有一种安排时间的方式使得全部任务都完成。

(第一时间看不出来是最大流啊!!)

思路:

分析之后可以这样想,设立一个超级源点S=0,让S与每个任务i建立双向边S->i,权重为Pi,然后让每个任务向每一天d(d就是Si到Ei的每一天)

建立双向边i->d,权重是1(因为每个机器一天只能运行1个任务),然后让每一天d向一个超级汇点T建立双向边d->T,权重为M(因为从任务开

始到所有任务完成,这一天d最多会有M个机器运转,也就是说最多有M个任务在进行着)。然后跑最大流,如果最大流maxflow和sigma(Pi)相同,

那么输出“Yes”,else输出“No”。

!!!值得注意的是,要把任务i和天数d当做二分图,(因为可能任务i和任务j可能会连接一条边,因为所有[Si, Ei]区间一定有可能包含任务下标)

那就意味着要把任务序号或者天数序号进行映射,我的代码是把天数d映射到了d+n。

#include <bits/stdc++.h>
using namespace std; const int maxn = + ;
const int inf = 0x3f3f3f3f; //INF
const int ainf = 0xcfcfcfcf; //-INF
int n, m, T, d[maxn*], s, t, maxflow;
int head[maxn*], tot, sum;
struct edge{
int to, w, next;
} ed[maxn*maxn*+maxn*];
//边的个数 = 任务节点的到天数节点连边上界为n*n,双向乘2 + 超级源点与任务节点连双向边 + 超级汇点与天数节点连双向边。
inline void add( int u, int v, int w ){
ed[++tot].to = v; ed[tot].w = w; ed[tot].next = head[u]; head[u] = tot;
ed[++tot].to = u; ed[tot].w = ; ed[tot].next = head[v]; head[v] = tot;
} inline bool bfs(){
queue<int> q;
memset( d, , sizeof(d) );
d[s] = ;
q.push(s);
while( !q.empty() ){
int x = q.front();
q.pop();
for( int i=head[x]; i!=-; i=ed[i].next ){
int y = ed[i].to;
if( ed[i].w && !d[y] ){
d[y] = d[x] + ;
q.push(y);
if( y==t ) return ;
}
}
}
return ;
} inline int min( int a, int b ){
return a<b ?a:b;
} inline int max( int a, int b ){
return a>b? a:b;
} inline int dfs( int x, int flow ){
if(x==t) return flow;
int res = flow, k;
for( int i=head[x]; i!=- && res; i=ed[i].next ){
int y = ed[i].to;
if( ed[i].w && d[y]==d[x]+ ){
k = dfs(y, min(res, ed[i].w) );
if(!k) d[y] = ;
ed[i].w -= k;
ed[i^].w += k;
res -= k;
}
}
return flow-res;
} inline void dinic(){
int flow = maxflow = ;
while( bfs() )
while( flow=dfs(s, inf) ) maxflow += flow;
} int main(){
// freopen("in.txt", "r", stdin);
scanf("%d", &T);
for( int k=; k<=T; k++ ){
memset( head, -, sizeof(head) );
tot = ; sum = ;
scanf("%d%d", &n, &m);
s = ;
int l = inf, r = ainf; //ainf==0xcfcfcfcf 即-INF
for( int i=; i<=n; i++ ){
int a, b, p;
scanf("%d%d%d", &p, &a, &b);
l = min( a, l ); //找到天数的下界
r = max( b, r ); //找到天数的上界
add( s, i, p ); //超级源点与任务i连边
for( int j=a; j<=b; j++ ) add( i, j+n, ); //任务与映射过得天数序号连边
sum += p; //sigma(Pi)
}
t = r+n+; //超级汇点,也可以让t普遍选择为2*maxn-1,没啥区别的
for( int i=l; i<=r; i++ ) add( i+n, t, m ); //天数向超级汇点t连边
dinic();
printf("Case %d: ", k);
if( maxflow==sum ) puts("Yes\n"); //puts()自带换行,再加一个\n来控制输出格式
else puts("No\n");
} return ;
}

HDU3572 Task Schedule(最大流+构图思维)的更多相关文章

  1. HDU3572 Task Schedule 【最大流】

    Task Schedule Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) To ...

  2. HDU 3572 Task Schedule (最大流)

    C - Task Schedule Time Limit:1000MS     Memory Limit:32768KB     64bit IO Format:%I64d & %I64u S ...

  3. HDU3572:Task Schedule【最大流】

    上了一天课 心塞塞的 果然像刘老师那么说 如果你有挂科+4级没过 那基本上是WF队 题目大意:有时间补吧 思路:给每个任务向每个时间点连边容量为1 每个时间点向汇点连边 容量为机器的个数 源点向每个任 ...

  4. hdu-3572 Task Schedule---最大流判断满流+dinic算法

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=3572 题目大意: 给N个任务,M台机器.每个任务有最早才能开始做的时间S,deadline E,和持 ...

  5. HDU 3572 Task Schedule(拆点+最大流dinic)

    Task Schedule Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) To ...

  6. hdu 3572 Task Schedule(最大流&amp;&amp;建图经典&amp;&amp;dinic)

    Task Schedule Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) To ...

  7. hdu 3572 Task Schedule

    Task Schedule 题意:有N个任务,M台机器.每一个任务给S,P,E分别表示该任务的(最早开始)开始时间,持续时间和(最晚)结束时间:问每一个任务是否能在预定的时间区间内完成: 注:每一个任 ...

  8. hdu 3572 Task Schedule (dinic算法)

    pid=3572">Task Schedule Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 ...

  9. HDU3572Task Schedule(最大流 ISAP比較快)建图方法不错

    Task Schedule Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) To ...

随机推荐

  1. html中利用flex容器书写的布局样式

    首先页面基本样式见下图: 如有兴趣可以打开https://migloo.gitee.io/front 或者 https://www.igloo.xin/front 进行查看

  2. elasticsearch 常用命令 一直红色 重启不稳定 不停的宕机

    persistent (重启后设置也会存在) or transient (整个集群重启后会消失的设置). 查看集群状态和每个indices状态.搜索到red的,没用就删除 GET /_cluster/ ...

  3. 【C++】内联函数(inline)和宏定义(# define)的优劣及其区别

    一.宏定义:# define 1.为什么要使用宏? 因为调用宏比调用函数更有效率,函数的调用必须要将程序的执行顺序转移到函数所存放的内存地址中,将函数程序内容执行完后,再返回到执行该函数前的地方,这种 ...

  4. Feign实现自定义错误处理

    关键操作 实现ErrorDecoder接口 问题和背景 最近项目中在大量使用Feign和OkHttp作为http客户端使用,开发效率得到显著的提升.但也面临一些问题,比如每个下游系统的异常返回方式不同 ...

  5. Linux C++ Socket 高并发短连接 TIME_WAIT 挥之不去解决方法

    近期遇到一个项目 需要在Linux上建立一个Socket 进行 HTTP_GET , 需要线程高并发的 使用TCP Socket 进行Send 发送HTTP_GET请求到 指定网站 . 而且不需要re ...

  6. FusionInsight大数据开发---MapReduce与YARN应用开发

    MapReduce MapReduce的基本定义及过程 搭建开发环境 代码实例及运行程序 MapReduce开发接口介绍 1. MapReduce的基本定义及过程 MapReduce是面向大数据并行处 ...

  7. Java学习:接口(interface)的使用于注意事项

    接口 接口就是一种公共的规范标准.只要符合规范标准,就可以大家通用. 接口就是多个类的公共规范.接口是一种引用数据类型,最重要的内容就是其中的:抽象方法. 如何定义一个接口的格式 如何定义一个接口的格 ...

  8. 搞清楚一道关于Integer的面试题【华为云技术分享】

    请看题1: public class IntegerDemo { public static void main(String[] args) { Integer a = ; Integer b = ...

  9. UDP比TCP好用的优势

    网络带宽环境变好 在2007年至2015年间,网络的带宽飞速发展,从1.5Mbps的带宽增加到5.1Mbps的带宽,足足增加了4倍,网络环境快速.稳定,所以UDP的丢包率 下降至5%以下,越来越好的网 ...

  10. Smobiler客户端会话

    //客户端会话存值 Client.Session["userid"] = Class1.userid; //客户端会话取值 userid = Client.Session[&quo ...