• BZOJ2245 [SDOI2011]工作安排


    题意:自行脑补,看懂分段函数是什么即可。


    思路:显然是最小费用最大流。

    对于每一个工作人员的每一段,从原点到工作人员相应的点连一条费用与流量与这一段其相适应的边。

    对于每一个部件。从其相应的点到汇点连一条流量为须要的数目,费用为0的边。

    然后就能够出解了。

    建模还是非常显然的。

    还有这题我写spfa的多路增广TLE了,反倒是不加上多路增广能过。

    不知道为什么。。。


    Code:

    #include <cstdio>
    #include <cstring>
    #include <algorithm>
    #include <climits>
    #include <iostream>
    #include <queue>
    using namespace std;
    
    #define N 260
    #define M 260
    
    #define INF 0x3f3f3f3f
    
    queue<int> q;
    struct Solver {
    	int head[M*7+N],next[N*M*6],end[N*M*6],flow[N*M*6],cost[N*M*6],ind;
    	int dis[M*7+N], ld[M*7+N], lb[M*7+N];
    	bool inq[M*7+N];
    	void reset() {
    		ind = 0;
    		memset(head, -1, sizeof(head));
    	}
    	void addedge(int a, int b, int _flow, int _cost) {
    		//printf("%d %d flow=%d cost=%d
    ", a, b, _flow, _cost);
    		int q = ind++;
    		flow[q] = _flow;
    		cost[q] = _cost;
    		end[q] = b;
    		next[q] = head[a];
    		head[a] = q;
    	}
    	void make(int a, int b, int _flow, int _cost) {
    		addedge(a, b, _flow, _cost);
    		addedge(b, a, 0, -_cost);
    	}
    	bool spfa(int S, int T) {
    		inq[S] = 1;
    		q.push(S);
    		memset(dis, 0x3f, sizeof dis);
    		dis[S] = 0;
    		int i, j;
    		while(!q.empty()) {
    			i = q.front();
    			q.pop();
    			inq[i] = 0;
    			for(j = head[i]; j != -1; j = next[j]) {
    				if (flow[j] && dis[end[j]] > dis[i] + cost[j]) {
    					ld[end[j]] = i;
    					lb[end[j]] = j;
    					dis[end[j]] = dis[i] + cost[j];
    					if (!inq[end[j]]) {
    						inq[end[j]] = 1;
    						q.push(end[j]);
    					}
    				}
    			}
    		}
    		return dis[T] != 0x3f3f3f3f;
    	}
    	long long Mincost(int S, int T) {
    		long long res = 0;
    		int Min, i;
    		while(spfa(S, T)) {
    			Min = INF;
    			for(i = T; i != S; i = ld[i])
    				Min = Min > flow[lb[i]] ? flow[lb[i]] : Min;
    			res += (long long)Min * dis[T];
    			for(i = T; i != S; i = ld[i])
    				flow[lb[i]] -= Min, flow[lb[i] ^ 1] += Min;
    		}
    		return res;
    	}
    }G;
    
    int s[M][N];
    
    int num[N], size[M], ins[M][7], w[M][7], id;
    
    int main() {
    	int m, n;
    	scanf("%d%d", &m, &n);
    	
    	register int i, j, k;
    	for(i = 1; i <= n; ++i)
    		scanf("%d", &num[i]);
    	for(i = 1; i <= m; ++i)
    		for(j = 1; j <= n; ++j)
    			scanf("%d", &s[i][j]);
    	
    	id = n;
    	for(i = 1; i <= m; ++i) {
    		scanf("%d", &size[i]);
    		for(j = 1; j <= size[i]; ++j)
    			scanf("%d", &ins[i][j]);
    		for(j = 1; j <= size[i] + 1; ++j)
    			scanf("%d", &w[i][j]);
    	}
    	
    	G.reset();
    	for(i = 1; i <= n; ++i)
    		G.make(i, n + m + 1, num[i], 0);
    	for(i = 1; i <= m; ++i) {
    		for(j = 1; j <= size[i]; ++j)
    			G.make(0, n + i, ins[i][j] - ins[i][j - 1], w[i][j]);
    		G.make(0, n + i, INF, w[i][size[i] + 1]);
    		for(j = 1; j <= n; ++j)
    			if (s[i][j])
    				G.make(n + i, j, INF, 0);
    	}
    	
    	printf("%lld", G.Mincost(0, n + m + 1));
    	
    	return 0;
    }


  • 相关阅读:
    梦心日记本V2.0终于要完工了
    上班半年大总结
    真有趣
    搞定设计模式1之策略模式
    利用GDI+制作背景颜色淡入淡出效果的按钮
    浏览器之争
    学习自定义控件
    搞定设计模式2之代理模式
    (转)学习asp.net比较完整的流程
    搞定设计模式3之中介者模式
  • 原文地址:https://www.cnblogs.com/yfceshi/p/7246630.html
Copyright © 2020-2023  润新知