zoukankan      html  css  js  c++  java
  • 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;
    }


  • 相关阅读:
    TypesScript+Webpack
    TypeScript 类型
    git操作
    kafka
    java: cannot find symbol symbol: variable log
    Angular结构型指令,模块和样式
    Angular 自定义拖拽指令
    Angular changeDetction
    Angular 依赖注入
    RXJS Observable的冷,热和Subject
  • 原文地址:https://www.cnblogs.com/yfceshi/p/7246630.html
Copyright © 2011-2022 走看看