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\n", 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;
}
时间: 2024-11-08 22:03:55

BZOJ2245 [SDOI2011]工作安排的相关文章

BZOJ2245 [SDOI2011]工作安排 【费用流】

题目 你的公司接到了一批订单.订单要求你的公司提供n类产品,产品被编号为1~n,其中第i类产品共需要Ci件.公司共有m名员工,员工被编号为1~m员工能够制造的产品种类有所区别.一件产品必须完整地由一名员工制造,不可以由某名员工制造一部分配件后,再转交给另外一名员工继续进行制造. 我们用一个由0和1组成的m*n的矩阵A来描述每名员工能够制造哪些产品.矩阵的行和列分别被编号为1~m和1~n,Ai,j为1表示员工i能够制造产品j,为0表示员工i不能制造产品j. 如果公司分配了过多工作给一名员工,这名员

[bzoj2245][SDOI2011]工作安排(费用流)

题目:http://www.lydsy.com:808/JudgeOnline/problem.php?id=2245 分析: 要注意到题目下面说的w是单增的 明显的费用流: 弄个源点S,汇点T S连向每种产品,流量是这种产品所需个数,费用是0 每种产品连向能制作它的人,流量为inf,费用是0 每个人向T连Si+1条边,流量t[i][j]-t[i][j-1],费用w[i][j] (因为w是单增的,所以就保证了每个人连向T的Si+1条边肯定是上面的边填满之后再填下面的边,保证了符合题意,如果没有这

BZOJ2245 SDOI2011 工作安排 费用流

题意:有n类产品,其中第i类产品共需要Ci件.有m名员工,员工能够制造的产品种类有所区,一件产品必须完整地由一名员工制造,对于员工i,他的愤怒值与产品数量之间的函数是一个Si+1段的分段函数.当他制造第1~Ti,1件产品时,每件产品会使他的愤怒值增加Wi,1,当他制造第Ti,1+1~Ti,2件产品时,每件产品会使他的愤怒值增加Wi,2……设Ti,0=0,Ti,si+1=+∞,那么当他制造第Ti,j-1+1~Ti,j件产品时,每件产品会使他的愤怒值增加Wi,j, 1≤j≤Si+1.求在订单满足的前

【BZOJ2245】[SDOI2011]工作安排 拆边费用流

[BZOJ2245][SDOI2011]工作安排 Description 你的公司接到了一批订单.订单要求你的公司提供n类产品,产品被编号为1~n,其中第i类产品共需要Ci件.公司共有m名员工,员工被编号为1~m员工能够制造的产品种类有所区别.一件产品必须完整地由一名员工制造,不可以由某名员工制造一部分配件后,再转交给另外一名员工继续进行制造. 我们用一个由0和1组成的m*n的矩阵A来描述每名员工能够制造哪些产品.矩阵的行和列分别被编号为1~m和1~n,Ai,j为1表示员工i能够制造产品j,为0

bzoj 2245: [SDOI2011]工作安排(费用流)

2245: [SDOI2011]工作安排 Time Limit: 20 Sec  Memory Limit: 512 MB Submit: 1446  Solved: 692 [Submit][Status][Discuss] Description 你的公司接到了一批订单.订单要求你的公司提供n类产品,产品被编号为1~n,其中第i类产品共需要Ci件.公司共有m名员工,员工被编号为1~m员工能够制造的产品种类有所区别.一件产品必须完整地由一名员工制造,不可以由某名员工制造一部分配件后,再转交给另

【bzoj2245】[SDOI2011]工作安排 费用流

题目描述 你的公司接到了一批订单.订单要求你的公司提供n类产品,产品被编号为1~n,其中第i类产品共需要Ci件.公司共有m名员工,员工被编号为1~m员工能够制造的产品种类有所区别.一件产品必须完整地由一名员工制造,不可以由某名员工制造一部分配件后,再转交给另外一名员工继续进行制造. 我们用一个由0和1组成的m*n的矩阵A来描述每名员工能够制造哪些产品.矩阵的行和列分别被编号为1~m和1~n,Ai,j为1表示员工i能够制造产品j,为0表示员工i不能制造产品j. 如果公司分配了过多工作给一名员工,这

【BZOJ 2245】[SDOI2011]工作安排

Description 你的公司接到了一批订单.订单要求你的公司提供n类产品,产品被编号为1~n,其中第i类产品共需要Ci件.公司共有m名员工,员工被编号为1~m员工能够制造的产品种类有所区别.一件产品必须完整地由一名员工制造,不可以由某名员工制造一部分配件后,再转交给另外一名员工继续进行制造. 我们用一个由0和1组成的m*n的矩阵A来描述每名员工能够制造哪些产品.矩阵的行和列分别被编号为1~m和1~n,Ai,j为1表示员工i能够制造产品j,为0表示员工i不能制造产品j. 如 果公司分配了过多工

[SDOI2011]工作安排

Description 你的公司接到了一批订单.订单要求你的公司提供n类产品,产品被编号为1~n,其中第i类产品共需要Ci件.公司共有m名员工,员工被编号为1~m员工能够制造的产品种类有所区别.一件产品必须完整地由一名员工制造,不可以由某名员工制造一部分配件后,再转交给另外一名员工继续进行制造. 我们用一个由0和1组成的m*n的矩阵A来描述每名员工能够制造哪些产品.矩阵的行和列分别被编号为1~m和1~n,Ai,j为1表示员工i能够制造产品j,为0表示员工i不能制造产品j. 如果公司分配了过多工作

P2488 [SDOI2011]工作安排 费用流

\(\color{#0066ff}{ 题目描述 }\) 你的任务是制定出一个产品的分配方案,使得订单条件被满足,并且所有员工的愤怒值之和最小.由于我们并不想使用Special Judge,也为了使选手有更多的时间研究其他两道题目,你只需要输出最小的愤怒值之和就可以了. \(\color{#0066ff}{输入格式}\) \(\color{#0066ff}{输出格式}\) 仅输出一个整数,表示最小的愤怒值之和. \(\color{#0066ff}{输入样例}\) 2 3 2 2 2 1 1 0 0