网络流24题(24/24)

01、飞行员配对方案问题

英国皇家空军从沦陷国征募了大量外籍飞行员。由皇家空军派出的每一架飞机都需要配备在航行技能和语言上能互相配合的2 名飞行员,其中1 名是英国飞行员,另1名是外籍飞行员。在众多的飞行员中,每一名外籍飞行员都可以与其他若干名英国飞行员很好地配合。如何选择配对飞行的飞行员才能使一次 派出最多的飞机。对于给定的外籍飞行员与英国飞行员的配合情况,试设计一个算法找出最佳飞行员配对方案,使皇家空军一次能派出最多的飞机。

对于给定的外籍飞行员与英国飞行员的配合情况,编程找出一个最佳飞行员配对方案,使皇家空军一次能派出最多的飞机。


二分图最大匹配

 1 #include <bits/stdc++.h>
 2 #define min(a, b) ((a) < (b) ? (a) : (b))
 3 using namespace std;
 4 const int MAXN = 100000;
 5 const int INF = 0x3f3f3f3f;
 6 inline void read(int &x)
 7 {
 8     x = 0;char ch = getchar(), c = ch;
 9     while(ch < '0' || ch > '9')c = ch, ch = getchar();
10     while(ch <= '9' && ch >= '0')x = x * 10 + ch - '0', ch = getchar();
11     if(c == '-')x = -x; 
12 }
13 
14 struct Edge
15 {
16     int u,v,w,nxt;
17     Edge(int _u, int _v, int _w, int _nxt){u = _u;v = _v;w = _w;nxt = _nxt;}
18     Edge(){}
19 }edge[100010];
20 int head[MAXN<<1],cnt = 1,q[MAXN<<1],h[MAXN<<1],ans=0,n,m,S,T,fa[MAXN <<1];
21 inline void insert(int a,int b,int c)
22 {
23     edge[++cnt] = Edge(a,b,c,head[a]);
24     head[a] = cnt;
25     edge[++cnt] = Edge(b,a,0,head[b]);
26     head[b] = cnt;
27 }
28 bool bfs()
29 {
30     int he = 0, ta = 1;
31     memset(h, -1, sizeof(h));
32     q[he] = S, h[S] = 0;
33     while(he < ta)
34     {
35         int now = q[he ++];
36         for(int pos = head[now];pos;pos = edge[pos].nxt)
37         {
38             int v = edge[pos].v;
39             if(h[v] == -1 && edge[pos].w)
40             {
41                 h[v] = h[now] + 1;
42                 q[ta ++] = v;
43             }
44         }
45     }
46     return h[T] != -1;
47 }
48 
49 int dfs(int x, int f)
50 {
51     if(x == T) return f;
52     int w, used = 0;
53     for(int pos = head[x];pos;pos = edge[pos].nxt)
54     {
55         int v = edge[pos].v;
56         if(h[v] == h[x] + 1)
57         {
58             w = dfs(v, min(f - used, edge[pos].w));
59             if(w) fa[x] = v;
60             edge[pos].w -= w;
61             edge[pos ^ 1].w += w;
62             used += w;
63             if(used == f) return f;
64         }
65     }
66     if(!used) h[x] = -1;
67     return used;
68 }
69 
70 void dinic()
71 {
72     while(bfs()) ans += dfs(S, INF);
73 }
74 
75 int vis[MAXN << 1];
76 
77 int main()
78 {
79     read(m), read(n);
80     int tmp1, tmp2;
81     S = n + m + 2, T = n + m + 3;
82     while(scanf("%d %d", &tmp1, &tmp2) != EOF && tmp1 != -1)
83         insert(tmp1, tmp2, 1);
84     for(register int i = 1;i <= m;++ i)
85         insert(S, i, 1);
86     for(register int i = m + 1;i <= m + n;++ i)
87         insert(i, T, 1);
88     dinic();
89     printf("%d\n", ans);
90     if(ans == 0) printf("No Solution!\n");
91     else
92         for(int i = 1;i <= m;++ i)
93             if(fa[i])
94                 printf("%d %d\n", i, fa[i]);
95     return 0;
96 } 
01

02、太空飞行计划问题

W 教授正在为国家航天中心计划一系列的太空飞行。每次太空飞行可进行一系列商业性实验而获取利润。现已确定了一个可供选择的实验集合E= {E1,E2,…,Em},和进行这些实验需要使用的全部仪器的集合I={I1,I2,…In}。实验Ej需要用到的仪器是I的子集RjÍI。配置仪器 Ik的费用为ck美元。实验Ej的赞助商已同意为该实验结果支付pj美元。W教授的任务是找出一个有效算法,确定在一次太空飞行中要进行哪些实验并因此而 配置哪些仪器才能使太空飞行的净收益最大。这里净收益是指进行实验所获得的全部收入与配置仪器的全部费用的差额。

 

对于给定的实验和仪器配置情况,编程找出净收益最大的试验计划。

最大(小)权闭合子图。S连实验,T连仪器,实验与对应仪器连INF,割与S连的边意味着不做这个实验,割与T连的边意味着用这个仪器

于是S集合代表选择的实验和仪器,求最小割,总收益-最小割即为答案

  1 #include <iostream>
  2 #include <cstdio>
  3 #include <cstring>
  4 #include <cstdlib>
  5 #include <algorithm>
  6 #include <queue>
  7 #include <vector>
  8 #include <cmath> 
  9 #define min(a, b) ((a) < (b) ? (a) : (b))
 10 #define max(a, b) ((a) > (b) ? (a) : (b))
 11 #define abs(a) ((a) < 0 ? (-1 * (a)) : (a))
 12 template <class T>
 13 inline void swap(T& a, T& b)
 14 {
 15     T tmp = a;a = b;b = tmp;
 16 }
 17 inline void read(int &x)
 18 {
 19     x = 0;char ch = getchar(), c = ch;
 20     while(ch < '0' || ch > '9') c = ch, ch = getchar();
 21     while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
 22     if(c == '-') x = -x;
 23 }
 24 
 25 const int INF = 100000;
 26 const int MAXN = 3000 + 10;
 27 const int MAXM = MAXN * MAXN;
 28 
 29 struct Edge
 30 {
 31     int u,v,w,nxt;
 32     Edge(int _u, int _v, int _w, int _nxt){u = _u;v = _v;w = _w;nxt = _nxt;}
 33     Edge(){}
 34 }edge[MAXM << 1];
 35 int head[MAXN], cnt = 1;
 36 inline void insert(int u, int v, int w)
 37 {
 38     edge[++cnt] = Edge(u,v,w,head[u]);
 39     head[u] = cnt;
 40     edge[++cnt] = Edge(v,u,0,head[v]);
 41     head[v] = cnt;
 42 }
 43 
 44 int q[MAXN << 1], h[MAXN << 1], ans, S, T, he, ta;
 45 
 46 bool bfs()
 47 {
 48     int he = 0, ta = 1;
 49     memset(h, -1, sizeof(h));
 50     q[he] = S, h[S] = 0;
 51     while(he < ta)
 52     {
 53         int now = q[he ++];
 54         for(int pos = head[now];pos;pos = edge[pos].nxt)
 55         {
 56             int v = edge[pos].v;
 57             if(h[v] == -1 && edge[pos].w)
 58             {
 59                 h[v] = h[now] + 1;
 60                 q[ta ++] = v;
 61             }
 62         }
 63     }
 64     return h[T] != -1;
 65 }
 66 
 67 int dfs(int x, int f)
 68 {
 69     if(x == T) return f;
 70     int w, used = 0;
 71     for(int pos = head[x];pos;pos = edge[pos].nxt)
 72     {
 73         int v = edge[pos].v;
 74         if(h[v] == h[x] + 1)
 75         {
 76             w = dfs(v, min(f - used, edge[pos].w));
 77             edge[pos].w -= w;
 78             edge[pos ^ 1].w += w;
 79             used += w;
 80             if(used == f) return f;
 81         }
 82     }
 83     if(!used) h[x] = -1;
 84     return used;
 85 }
 86 
 87 void dinic()
 88 {
 89     while(bfs()) ans += dfs(S, INF);
 90 }
 91 
 92 int n,m,sum,vis[MAXN];
 93 
 94 int main()
 95 {
 96     read(m), read(n);S = n + m + 100;T = n + m + 200;
 97     for(int i = 1;i <= m;++ i)
 98     {
 99         char c = 0;int w, tmp = 0;
100         read(w);sum += w;
101         insert(S, i, w);
102         while(c != '\n' && c != '\r')
103         {
104             c = getchar();
105             while(c >= '0' && c <= '9') 
106                 tmp = tmp * 10 + c - '0', c = getchar();
107             if(tmp) insert(i, tmp + m, INF), tmp = 0;
108         }
109     }
110     for(int i = 1;i <= n;++ i)
111     {
112         int tmp;read(tmp);
113         insert(i + m, T, tmp);
114     }
115     dinic();
116     for(int pos = head[S];pos;pos = edge[pos].nxt)
117         if(edge[pos].w)
118             vis[edge[pos].v] = 1;
119     for(int i = 1;i <= m;++ i)    if(h[i] != -1) printf("%d ", i);
120     putchar('\n');
121     for(int i = m + 1;i <= n + m;++ i) if(h[i] != -1) printf("%d ", i - m);putchar('\n');
122     printf("%d", sum - ans);
123     return 0;
124 }
02

03、最小路径覆盖问题

«问题描述:

给定有向图G=(V,E)。设P 是G 的一个简单路(顶点不相交)的集合。如果V 中每个顶点恰好在P 的一条路上,则称P是G 的一个路径覆盖。P 中路径可以从V 的任何一个顶点开始,长度也是任意的,特别地,可以为0。G 的最小路径覆盖是G 的所含路径条数最少的路径覆盖。设计一个有效算法求一个有向无环图G 的最小路径覆盖。提示:设V={1,2,.... ,n},构造网络G1=(V1,E1)如下:

每条边的容量均为1。求网络G1的( 0 x , 0 y )最大流。

«编程任务:

对于给定的给定有向无环图G,编程找出G的一个最小路径覆盖。

 

 

每个点拆成x,x',S->x, x'->T,对于x->y一条边,连x->y'。X集合为点x,Y集合为点x'

正确性:每条路径上的点度数最大为2。x点代表点x的出度,x'点代表点x的入度。一条边会占用一个点的出度和另一个点的入度。对于每条路径,发现起点只有出度,没有入度,那么Y集合中没有被连的点就是某条路径的起点。于是n - 最大匹配 = 最小路径数

  1 #include <iostream>
  2 #include <cstdio>
  3 #include <cstring>
  4 #include <cstdlib>
  5 #include <algorithm>
  6 #include <queue>
  7 #include <vector>
  8 #include <cmath> 
  9 #define min(a, b) ((a) < (b) ? (a) : (b))
 10 #define max(a, b) ((a) > (b) ? (a) : (b))
 11 #define abs(a) ((a) < 0 ? (-1 * (a)) : (a))
 12 template <class T>
 13 inline void swap(T& a, T& b)
 14 {
 15     T tmp = a;a = b;b = tmp;
 16 }
 17 inline void read(int &x)
 18 {
 19     x = 0;char ch = getchar(), c = ch;
 20     while(ch < '0' || ch > '9') c = ch, ch = getchar();
 21     while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
 22     if(c == '-') x = -x;
 23 }
 24 
 25 const int INF = 100000;
 26 const int MAXN = 300 + 10;
 27 const int MAXM = MAXN * MAXN;
 28 
 29 struct Edge
 30 {
 31     int u,v,w,nxt;
 32     Edge(int _u, int _v, int _w, int _nxt){u = _u;v = _v;w = _w;nxt = _nxt;}
 33     Edge(){}
 34 }edge[MAXM << 1];
 35 int head[MAXN], cnt = 1;
 36 inline void insert(int u, int v, int w)
 37 {
 38     edge[++cnt] = Edge(u,v,w,head[u]);
 39     head[u] = cnt;
 40     edge[++cnt] = Edge(v,u,0,head[v]);
 41     head[v] = cnt;
 42 }
 43 
 44 int q[MAXN << 1], h[MAXN << 1], ans, S, T, he, ta, fa[MAXN << 1],vis[MAXN];
 45 std::vector<int> node[MAXN << 1]; 
 46 
 47 bool bfs()
 48 {
 49     int he = 0, ta = 1;
 50     memset(h, -1, sizeof(h));
 51     q[he] = S, h[S] = 0;
 52     while(he < ta)
 53     {
 54         int now = q[he ++];
 55         for(int pos = head[now];pos;pos = edge[pos].nxt)
 56         {
 57             int v = edge[pos].v;
 58             if(h[v] == -1 && edge[pos].w)
 59             {
 60                 h[v] = h[now] + 1;
 61                 q[ta ++] = v;
 62             }
 63         }
 64     }
 65     return h[T] != -1;
 66 }
 67 
 68 int dfs(int x, int f)
 69 {
 70     if(x == T) return f;
 71     int w, used = 0;
 72     for(int pos = head[x];pos;pos = edge[pos].nxt)
 73     {
 74         int v = edge[pos].v;
 75         if(h[v] == h[x] + 1)
 76         {
 77             w = dfs(v, min(f - used, edge[pos].w));
 78             if(w) fa[x] = v, node[v].push_back(x);
 79             edge[pos].w -= w;
 80             edge[pos ^ 1].w += w;
 81             used += w;
 82             if(used == f) return f;
 83         }
 84     }
 85     if(!used) h[x] = -1;
 86     return used;
 87 }
 88 
 89 void dinic()
 90 {
 91     while(bfs()) ans += dfs(S, INF);
 92 }
 93 
 94 int n,m; 
 95 
 96 void dfs(int x)
 97 {
 98     vis[x] = 1;
 99     printf("%d ", x);
100     if(fa[x]) dfs(fa[x] - n);
101 }
102 
103 int main()
104 {
105     read(n), read(m);
106     S = n * 2 + 1, T = n * 2 + 2;
107     for(int i = 1;i <= n;++ i) insert(S, i, 1);
108     for(int i = 1;i <= m;++ i)
109     {
110         int tmp1,tmp2;
111         read(tmp1), read(tmp2);
112         insert(tmp1, tmp2 + n, 1);
113     }
114     for(int i = 1;i <= n;++ i) insert(i + n, T, 1);
115     dinic();
116     for(int i = 1;i <= n;++ i)
117         if(fa[i])
118             vis[fa[i] - n] = 1;
119     for(int i = 1;i <= n;++ i)
120         if(!vis[i])
121             dfs(i), putchar('\n');
122     printf("%d", n - ans);
123     return 0;
124 }
03

04、魔术球问题

«问题描述:

假设有n根柱子,现要按下述规则在这n根柱子中依次放入编号为1,2,3,...的球。

(1)每次只能在某根柱子的最上面放球。

(2)在同一根柱子中,任何2个相邻球的编号之和为完全平方数。

试设计一个算法,计算出在n根柱子上最多能放多少个球。例如,在4 根柱子上最多可放11 个球。

«编程任务:

对于给定的n,计算在n根柱子上最多能放多少个球。

 

 

最小路径覆盖变种。二分放多少个球,如果x能放到y上面,就有x->y,答案即为图上的最小路径覆盖数。

发现二分放多少个求,还不如直接从1开始枚举放的求数,每多一个球加一个的边,跑一次增广。

  1 // luogu-judger-enable-o2
  2 #include <iostream>
  3 #include <cstdio>
  4 #include <cstring>
  5 #include <cstdlib>
  6 #include <algorithm>
  7 #include <queue>
  8 #include <vector>
  9 #include <cmath> 
 10 #define min(a, b) ((a) < (b) ? (a) : (b))
 11 #define max(a, b) ((a) > (b) ? (a) : (b))
 12 #define abs(a) ((a) < 0 ? (-1 * (a)) : (a))
 13 template <class T>
 14 inline void swap(T& a, T& b)
 15 {
 16     T tmp = a;a = b;b = tmp;
 17 }
 18 inline void read(int &x)
 19 {
 20     x = 0;char ch = getchar(), c = ch;
 21     while(ch < '0' || ch > '9') c = ch, ch = getchar();
 22     while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
 23     if(c == '-') x = -x;
 24 }
 25 
 26 const int INF = 100000;
 27 const int MAXN = 100000;
 28 const int MAXM = 100000;
 29 
 30 struct Edge
 31 {
 32     int u,v,w,nxt;
 33     Edge(int _u, int _v, int _w, int _nxt){u = _u;v = _v;w = _w;nxt = _nxt;}
 34     Edge(){}
 35 }edge[MAXM << 1];
 36 int head[MAXN << 1], cnt = 1, S, T, ans, fa[MAXN << 1], h[MAXN << 1], q[MAXN << 1], he, ta;
 37 inline void insert(int a, int b, int c)
 38 {
 39     edge[++cnt] = Edge(a,b,c,head[a]);
 40     head[a] = cnt;
 41     edge[++cnt] = Edge(b,a,0,head[b]);
 42     head[b] = cnt;
 43 }
 44 bool bfs()
 45 {
 46     memset(h, -1, sizeof(h));
 47     he = 0, ta = 1, q[0] = S, h[S] = 0;
 48     while(he < ta)
 49     {
 50         int now = q[he ++];
 51         for(int pos = head[now];pos;pos = edge[pos].nxt)
 52         {
 53             int v = edge[pos].v;
 54             if(h[v] == -1 && edge[pos].w) q[ta ++] = v, h[v] = h[now] + 1;
 55         }
 56     }
 57     return h[T] != -1;
 58 }
 59 int dfs(int x, int f)
 60 {
 61     if(x == T) return f; 
 62     int used = 0, w;
 63     for(int pos = head[x];pos;pos = edge[pos].nxt)
 64     {
 65         int v = edge[pos].v;
 66         if(h[v] == h[x] + 1)
 67         {
 68             w = dfs(v, min(f - used, edge[pos].w));
 69             if(w) fa[x] = v;
 70             edge[pos].w -= w;
 71             edge[pos ^ 1].w += w;
 72             used += w;
 73             if(used == f) return f;
 74         }
 75     }
 76     if(!used) h[x] = -1;
 77     return used;
 78 }
 79 void Dinic()
 80 {
 81     while(bfs()) ans += dfs(S, INF);
 82 }
 83 
 84 int n, ma, vis[MAXN]; 
 85 
 86 void dfs(int x)
 87 {
 88     vis[x] = 1;
 89     printf("%d ", x);
 90     if(fa[x]) dfs(fa[x] - 2000);
 91 }
 92 
 93 int main()
 94 {
 95     read(n);S = 10000, T = S + 1;
 96     for(ma = 1;;++ ma)
 97     {
 98         insert(S, ma, 1); insert(ma + 2000, T, 1);
 99         for(int j = sqrt(ma) + 1;j * j - ma < ma;++ j)
100             insert(j * j - ma, ma + 2000, 1);
101         Dinic();
102         if(ma - ans == n + 1) break;
103     }
104     -- ma;
105     printf("%d\n", ma);
106     memset(head, 0, sizeof(head)), cnt = 0, memset(fa, 0, sizeof(fa)), ans = 0;
107     for(int i = 1;i <= ma;++ i)
108     {
109         insert(S, i, 1); insert(i + 2000, T, 1);
110         int t = sqrt(i); 
111         for(int j = sqrt(i) + 1;j * j - i < i;++ j)
112             insert(j * j - i, i + 2000, 1);
113         Dinic();
114     }
115     for(int i = 1;i <= ma;++ i)
116         if(fa[i]) vis[fa[i] - 2000] = 1;
117     for(int i = 1;i <= ma;++ i)    if(!vis[i]) dfs(i), putchar('\n');
118     return 0;
119 }
04

 

05、圆桌问题

假设有来自m 个不同单位的代表参加一次国际会议。每个单位的代表数分别为ri (i =1,2,……,m)。

会议餐厅共有n 张餐桌,每张餐桌可容纳ci (i =1,2,……,n)个代表就餐。

为了使代表们充分交流,希望从同一个单位来的代表不在同一个餐桌就餐。试设计一个算法,给出满足要求的代表就餐方案。

对于给定的代表数和餐桌数以及餐桌容量,编程计算满足要求的代表就餐方案。

 

有点裸

  1 #include <iostream>
  2 #include <cstdio>
  3 #include <cstring>
  4 #include <cstdlib>
  5 #include <algorithm>
  6 #include <queue>
  7 #include <vector>
  8 #include <cmath> 
  9 #define min(a, b) ((a) < (b) ? (a) : (b))
 10 #define max(a, b) ((a) > (b) ? (a) : (b))
 11 #define abs(a) ((a) < 0 ? (-1 * (a)) : (a))
 12 template <class T>
 13 inline void swap(T& a, T& b)
 14 {
 15     T tmp = a;a = b;b = tmp;
 16 }
 17 inline void read(int &x)
 18 {
 19     x = 0;char ch = getchar(), c = ch;
 20     while(ch < '0' || ch > '9') c = ch, ch = getchar();
 21     while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
 22     if(c == '-') x = -x;
 23 }
 24 
 25 const int INF = 100000;
 26 const int MAXN = 10000;
 27 const int MAXM = 100000;
 28 
 29 struct Edge
 30 {
 31     int u,v,w,nxt;
 32     Edge(int _u, int _v, int _w, int _nxt){u = _u;v = _v;w = _w;nxt = _nxt;}
 33     Edge(){}
 34 }edge[MAXM << 1];
 35 int head[MAXN << 1], cnt = 1, S, T, ans, h[MAXN << 1], q[MAXN << 1], he, ta;
 36 inline void insert(int a, int b, int c)
 37 {
 38     edge[++cnt] = Edge(a,b,c,head[a]);
 39     head[a] = cnt;
 40     edge[++cnt] = Edge(b,a,0,head[b]);
 41     head[b] = cnt;
 42 }
 43 bool bfs()
 44 {
 45     memset(h, -1, sizeof(h));
 46     he = 0, ta = 1, q[0] = S, h[S] = 0;
 47     while(he < ta)
 48     {
 49         int now = q[he ++];
 50         for(int pos = head[now];pos;pos = edge[pos].nxt)
 51         {
 52             int v = edge[pos].v;
 53             if(h[v] == -1 && edge[pos].w)
 54             {
 55                 q[ta ++] = v, h[v] = h[now] + 1;
 56             }
 57         }
 58     }
 59     return h[T] != -1;
 60 }
 61 int dfs(int x, int f)
 62 {
 63     if(x == T) return f; 
 64     int used = 0, w;
 65     for(int pos = head[x];pos;pos = edge[pos].nxt)
 66     {
 67         int v = edge[pos].v;
 68         if(h[v] == h[x] + 1)
 69         {
 70             w = dfs(v, min(f - used, edge[pos].w));
 71             edge[pos].w -= w;
 72             edge[pos ^ 1].w += w;
 73             used += w;
 74             if(used == f) return f;
 75         }
 76     }
 77     if(!used) h[x] = -1;
 78     return used;
 79 }
 80 void Dinic()
 81 {
 82     while(bfs()) ans += dfs(S, INF);
 83 }
 84 int n,m,tmp,sum,vis[MAXN];
 85 
 86 int main()
 87 {
 88     read(m), read(n); S = n + m + 1, T = S + 1;
 89     for(int i = 1;i <= m;++ i) read(tmp), sum += tmp, insert(S, i, tmp);
 90     for(int i = 1;i <= n;++ i) read(tmp), insert(i + m, T, tmp);
 91     for(int i = 1;i <= m;++ i)
 92         for(int j = 1;j <= n;++ j)
 93             insert(i, j + m, 1);
 94     Dinic();
 95     if(ans != sum){printf("0\n");return 0;}
 96     printf("1");
 97     for(int i = 1;i <= m;++ i)
 98     {
 99         int flag = 1;
100         for(int pos = head[i];pos;pos = edge[pos].nxt)
101             if(edge[pos].v == S) continue;
102             else if(!edge[pos].w)
103                 if(flag) printf("\n%d ", edge[pos].v - m), flag = 0;
104                 else printf("%d ", edge[pos].v - m); 
105     }
106     return 0;
107 }
05

 

06、最长不下降子序列问题

«问题描述:

给定正整数序列x1,...,xn 。

(1)计算其最长不下降子序列的长度s。

(2)计算从给定的序列中最多可取出多少个长度为s的不下降子序列。

(3)如果允许在取出的序列中多次使用x1和xn,则从给定序列中最多可取出多少个长度为s的不下降子序列。

«编程任务:

设计有效算法完成(1)(2)(3)提出的计算任务。

 

第一、二个问题都有nlogn解法,详见http://www.cnblogs.com/huibixiaoxing/p/7398826.html第三题

这里讨论网络流解法

第一问直接暴力dp即可,设答案为k

第二问,相当于每个点限流1,用最大流模拟转移,考虑拆点。对于点x,拆成x和x',x - > x',容量1, 对于i和j,i < j,若有s[j] >= s[i],就从i’ - > j,容量为1。对于f[i] = k的点i,连i' - > T,容量为1,可以证明每一个序列都可以通过一个流表示出来。跑最大流即可

第三问,相当于x1和xn可以取无数次,那就把x1->x1',xn->xn',S->x1',S->xn',xn'->T(如果存在),x1'->T(如果存在)改成正无穷即可

  1 #include <iostream>
  2 #include <cstdio>
  3 #include <cstring>
  4 #include <cstdlib>
  5 #include <algorithm>
  6 #include <queue>
  7 #include <vector>
  8 #include <cmath> 
  9 #define min(a, b) ((a) < (b) ? (a) : (b))
 10 #define max(a, b) ((a) > (b) ? (a) : (b))
 11 #define abs(a) ((a) < 0 ? (-1 * (a)) : (a))
 12 template <class T>
 13 inline void swap(T& a, T& b)
 14 {
 15     T tmp = a;a = b;b = tmp;
 16 }
 17 inline void read(int &x)
 18 {
 19     x = 0;char ch = getchar(), c = ch;
 20     while(ch < '0' || ch > '9') c = ch, ch = getchar();
 21     while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
 22     if(c == '-') x = -x;
 23 }
 24 
 25 const int INF = 100000;
 26 const int MAXN = 10000;
 27 const int MAXM = 100000;
 28 
 29 struct Edge
 30 {
 31     int u,v,w,nxt;
 32     Edge(int _u, int _v, int _w, int _nxt){u = _u;v = _v;w = _w;nxt = _nxt;}
 33     Edge(){}
 34 }edge[MAXM << 1];
 35 int head[MAXN << 1], cnt = 1, S, T, ans, h[MAXN << 1], q[MAXN << 1], he, ta;
 36 inline void insert(int a, int b, int c)
 37 {
 38     edge[++cnt] = Edge(a,b,c,head[a]);
 39     head[a] = cnt;
 40     edge[++cnt] = Edge(b,a,0,head[b]);
 41     head[b] = cnt;
 42 }
 43 bool bfs()
 44 {
 45     memset(h, -1, sizeof(h));
 46     he = 0, ta = 1, q[0] = S, h[S] = 0;
 47     while(he < ta)
 48     {
 49         int now = q[he ++];
 50         for(int pos = head[now];pos;pos = edge[pos].nxt)
 51         {
 52             int v = edge[pos].v;
 53             if(h[v] == -1 && edge[pos].w)
 54             {
 55                 q[ta ++] = v, h[v] = h[now] + 1;
 56             }
 57         }
 58     }
 59     return h[T] != -1;
 60 }
 61 int dfs(int x, int f)
 62 {
 63     if(x == T) return f; 
 64     int used = 0, w;
 65     for(int pos = head[x];pos;pos = edge[pos].nxt)
 66     {
 67         int v = edge[pos].v;
 68         if(h[v] == h[x] + 1)
 69         {
 70             w = dfs(v, min(f - used, edge[pos].w));
 71             edge[pos].w -= w;
 72             edge[pos ^ 1].w += w;
 73             used += w;
 74             if(used == f) return f;
 75         }
 76     }
 77     if(!used) h[x] = -1;
 78     return used;
 79 }
 80 void Dinic()
 81 {
 82     while(bfs()) ans += dfs(S, INF);
 83 }
 84 
 85 int n, f[MAXN], num[MAXN], ma;
 86 
 87 int main()
 88 {
 89     read(n);
 90     for(register int i = 1;i <= n;++ i)
 91     {
 92         insert(i, i + n, 1);
 93         read(num[i]);
 94         f[i] = 1;
 95         for(register int j = 1;j < i;++ j)
 96             if(num[j] <= num[i]) f[i] = max(f[i], f[j] + 1);
 97         ma = max(ma, f[i]);
 98     }
 99     printf("%d\n", ma);
100     if(ma == 1)
101     {
102         printf("%d\n%d", n, n);
103         return 0;
104     } 
105     S = n + n + 1, T = S + 1; 
106     for(register int i = 1;i <= n;++ i) 
107         if(f[i] == 1) insert(S, i, 1); 
108         else if(f[i] == ma) insert(i + n, T, 1);
109     for(register int i = 1;i <= n;++ i)
110         for(register int j = 1;j < i;++ j)
111             if(num[j] <= num[i] && f[i] == f[j] + 1)
112                 insert(j + n, i, 1);
113     Dinic();
114     printf("%d\n", ans);
115     insert(1, n + 1, INF), insert(n, n + n, INF);
116     if(f[1] == 1) insert(S, 1, INF);
117     if(f[1] == ma) insert(1 + n, T, INF);
118     if(f[n] == 1) insert(S, n, INF);
119     if(f[n] == ma) insert(n + n, T, INF);
120     Dinic();
121     printf("%d", ans); 
122     return 0;
123 }
06

 

07、试题库问题

«问题描述:

假设一个试题库中有n道试题。每道试题都标明了所属类别。同一道题可能有多个类别属性。现要从题库中抽取m 道题组成试卷。并要求试卷包含指定类型的试题。试设计一个满足要求的组卷算法。

«编程任务:

对于给定的组卷要求,计算满足要求的组卷方案。

 

有点裸

  1 #include <iostream>
  2 #include <cstdio>
  3 #include <cstring>
  4 #include <cstdlib>
  5 #include <algorithm>
  6 #include <queue>
  7 #include <vector>
  8 #include <cmath> 
  9 #define min(a, b) ((a) < (b) ? (a) : (b))
 10 #define max(a, b) ((a) > (b) ? (a) : (b))
 11 #define abs(a) ((a) < 0 ? (-1 * (a)) : (a))
 12 template <class T>
 13 inline void swap(T& a, T& b)
 14 {
 15     T tmp = a;a = b;b = tmp;
 16 }
 17 inline void read(int &x)
 18 {
 19     x = 0;char ch = getchar(), c = ch;
 20     while(ch < '0' || ch > '9') c = ch, ch = getchar();
 21     while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
 22     if(c == '-') x = -x;
 23 }
 24 
 25 const int INF = 100000;
 26 const int MAXN = 100000 + 10;
 27 const int MAXM = 1000000 + 10;
 28 
 29 struct Edge
 30 {
 31     int u,v,w,nxt;
 32     Edge(int _u, int _v, int _w, int _nxt){u = _u;v = _v;w = _w;nxt = _nxt;}
 33     Edge(){}
 34 }edge[MAXM << 1];
 35 int head[MAXN << 1], cnt = 1, S, T, ans, h[MAXN << 1], fa[MAXN << 1], q[MAXN << 1], he, ta;
 36 inline void insert(int a, int b, int c)
 37 {
 38     edge[++ cnt] = Edge(a,b,c,head[a]);
 39     head[a] = cnt;
 40     edge[++ cnt] = Edge(b,a,0,head[b]);
 41     head[b] = cnt;
 42 }
 43 bool bfs()
 44 {
 45     he = 0, ta = 1, q[he] = S, memset(h, -1, sizeof(h)), h[S] = 0;
 46     while(he < ta)
 47     {
 48         int now = q[he ++];
 49         for(int pos = head[now];pos;pos = edge[pos].nxt)
 50         {
 51             int v = edge[pos].v;
 52             if(h[v] == -1 && edge[pos].w)
 53                 h[v] = h[now] + 1, q[ta ++] = v;
 54         }
 55     }
 56     return h[T] != -1;
 57 }
 58 int dfs(int x, int f)
 59 {
 60     if(x == T) return f;
 61     int w,used = 0;
 62     for(int pos = head[x];pos;pos = edge[pos].nxt)
 63     {
 64         int v = edge[pos].v;
 65         if(h[v] == h[x] + 1)
 66         {
 67             w = dfs(v, min(f - used, edge[pos].w));
 68             edge[pos].w -= w;
 69             edge[pos ^ 1].w += w;
 70             if(w) fa[x] = v;
 71             used += w;
 72             if(used == f) return f;
 73         }
 74     }
 75     if(!used) h[x] = -1;
 76     return used;
 77 }
 78 void Dinic()
 79 {
 80     while(bfs()) ans += dfs(S, INF);
 81 }
 82 int k,n,tmp,tmp2,sum;
 83 int main()
 84 {
 85     read(k), read(n);
 86     S = n + k + 1, T = S + 1;
 87     for(int i = 1;i <= n;++ i) insert(S, i, 1);
 88     for(int i = 1;i <= k;++ i) read(tmp), sum += tmp, insert(n + i, T, tmp);
 89     for(int i = 1;i <= n;++ i)
 90     {
 91         read(tmp);
 92         for(int j = 1;j <= tmp;++ j) read(tmp2), insert(i, tmp2 + n, 1);
 93     }
 94     Dinic();
 95     if(ans != sum){printf("No Solution!\n");return 0;}
 96     for(int i = 1;i <= k;++ i) 
 97     {
 98         printf("%d: ", i);
 99         for(int j = 1;j <= n;++ j) if(fa[j] == i + n) printf("%d ", j);
100         putchar('\n');
101     }
102     return 0;
103 }
07

08、机器人路径规划问题

机器人 Rob 可在一个树状路径上自由移动。给定树状路径 T 上的起点 s 和终点 t,机器人 Rob 要从 s 运动到 t。树状路径 T 上有若干可移动的障碍物。由于路径狭窄,任何时刻在路径的任何位置不能同时容纳 2 个物体。每一步可以将障碍物或机器人移到相邻的空顶点上。设计一个有效算法用最少移动次数使机器人从 s 运动到 t。对于给定的树 T,以及障碍物在树 T 中的分布情况。计算机器人从起点 s 到终点 t 的最少移动次数。

 

据说此题网络流不可做。其他做法也不太好,复杂度很高网络流24题:我们之中出了个叛徒!

 

09、方格取数问题

在一个有 m*n 个方格的棋盘中,每个方格中有一个正整数。现要从方格中取数,使任意 2 个数所在方格没有公共边,且取出的数的总和最大。试设计一个满足要求的取数算法。对于给定的方格棋盘,按照取数要求编程找出总和最大的数。

 

看见方格就黑白染色。发现不能放在相邻的异色格子上。想到最小割。

对于白色格点(i,j),向黑色格点(i + 1, j)和(i,j + 1)连边,容量INF,S向白色格点连权值,黑色格点向T连权值,每一条S->T的路径都对应一个不满足条件的放法,因此要使得取出的数S->T不存在,割掉一条边意味着不取出这个方格中的数,求最小割即为留下的最少,取出的用总和减

  1 // luogu-judger-enable-o2
  2 #include <iostream>
  3 #include <cstdio>
  4 #include <cstring>
  5 #include <cstdlib>
  6 #include <algorithm>
  7 #include <queue>
  8 #include <vector>
  9 #include <cmath> 
 10 #define min(a, b) ((a) < (b) ? (a) : (b))
 11 #define max(a, b) ((a) > (b) ? (a) : (b))
 12 #define abs(a) ((a) < 0 ? (-1 * (a)) : (a))
 13 #define hao(a, b) ((a) * m + (b)) 
 14 template <class T>
 15 inline void swap(T& a, T& b)
 16 {
 17     T tmp = a;a = b;b = tmp;
 18 }
 19 inline void read(int &x)
 20 {
 21     x = 0;char ch = getchar(), c = ch;
 22     while(ch < '0' || ch > '9') c = ch, ch = getchar();
 23     while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
 24     if(c == '-') x = -x;
 25 }
 26 
 27 const int INF = 100000;
 28 const int MAXN = 100000 + 10;
 29 const int MAXM = 1000000 + 10;
 30 
 31 struct Edge
 32 {
 33     int u,v,w,nxt;
 34     Edge(int _u, int _v, int _w, int _nxt){u = _u, v = _v, w = _w, nxt = _nxt;}
 35     Edge(){}
 36 }edge[MAXM << 1];
 37 int head[MAXN << 1], cnt = 1, q[MAXN << 1], h[MAXN << 1], S, T, ans, he, ta;
 38 inline void insert(int a, int b, int c)
 39 {
 40     edge[++cnt] = Edge(a,b,c,head[a]);
 41     head[a] = cnt;
 42     edge[++cnt] = Edge(b,a,0,head[b]);
 43     head[b] = cnt;
 44 }
 45 bool bfs()
 46 {
 47     memset(h, -1, sizeof(h)), he = 0, ta = 1, q[0] = S, h[S] = 0;
 48     while(he < ta)
 49     {
 50         int now = q[he ++];
 51         for(int pos = head[now];pos;pos = edge[pos].nxt)
 52         {
 53             int v = edge[pos].v;
 54             if(h[v] == -1 && edge[pos].w)
 55                 h[v] = h[now] + 1, q[ta ++] = v;
 56         }
 57     }
 58     return h[T] != -1;
 59 }
 60 int dfs(int x, int f)
 61 {
 62     if(x == T) return f;
 63     int used = 0, w;
 64     for(int pos = head[x];pos;pos = edge[pos].nxt)
 65     {
 66         int v = edge[pos].v;
 67         if(h[v] == h[x] + 1)
 68         {
 69             w = dfs(v, min(f - used, edge[pos].w));
 70             edge[pos].w -= w;
 71             edge[pos ^ 1].w += w;
 72             used += w;
 73             if(used == f) return f; 
 74         }
 75     }
 76     if(!used) h[x] = -1;
 77     return used;
 78 }
 79 void Dinic()
 80 {
 81     while(bfs()) ans += dfs(S, INF);
 82 }
 83 
 84 int n,m,g[150][150],sum;
 85 
 86 int main()
 87 {
 88     read(n), read(m);
 89     S = hao(n, m) + 1, T = S + 1;
 90     for(int i = 1;i <= n;++ i) 
 91         for(int j = 1;j <= m;++ j) 
 92         {
 93             int tmp; read(tmp);sum += tmp;
 94             if((i + j) & 1) 
 95             {
 96                 insert(S, hao(i, j), tmp);
 97                 if(i + 1 <= n) insert(hao(i, j), hao(i + 1, j), INF);
 98                 if(j + 1 <= m) insert(hao(i, j), hao(i, j + 1), INF);
 99             }
100             else 
101             {
102                 insert(hao(i, j), T, tmp);
103                 if(i + 1 <= n) insert(hao(i + 1, j), hao(i, j), INF);
104                 if(j + 1 <= m) insert(hao(i, j + 1), hao(i, j), INF);
105             }
106         }
107     Dinic();
108     printf("%d", sum - ans);
109     return 0;
110 }
09

 

10、餐巾计划问题

一个餐厅在相继的 NNN 天里,每天需用的餐巾数不尽相同。假设第 iii 天需要 rir_iri块餐巾( i=1,2,...,N)。餐厅可以购买新的餐巾,每块餐巾的费用为 ppp 分;或者把旧餐巾送到快洗部,洗一块需 m 天,其费用为 f 分;或者送到慢洗部,洗一块需 nnn 天(n>mn>mn>m),其费用为 sss 分(s<fs<fs<f)。

每天结束时,餐厅必须决定将多少块脏的餐巾送到快洗部,多少块餐巾送到慢洗部,以及多少块保存起来延期送洗。但是每天洗好的餐巾和购买的新餐巾数之和,要满足当天的需求量。

试设计一个算法为餐厅合理地安排好 NNN 天中餐巾使用计划,使总的花费最小。编程找出一个最佳餐巾使用计划。

 

每天拆成两个点x(X集合),x(Y集合)',x表示第x天用完剩下的餐巾,x'表示第x天要用的餐巾,让x'连向T,容量为当天需要的餐巾数,把S连向x,容量为当天需要的餐巾数,考虑如何用X集合补满Y集合的流。

对于购买,从S连向Y集合,容量+∞,费用p

对于快洗,x->x + m,容量+∞,费用f

对于慢洗,x->x + nnn,容量+∞,费用sss

还有可能上一天的餐巾留到下一天,从x -> x + 1容量+∞,费用0(容易漏掉)

 

  1 #include <iostream>
  2 #include <cstdio>
  3 #include <cstring>
  4 #include <cstdlib>
  5 #include <algorithm>
  6 #include <queue>
  7 #include <vector>
  8 #include <cmath> 
  9 #define min(a, b) ((a) < (b) ? (a) : (b))
 10 #define max(a, b) ((a) > (b) ? (a) : (b))
 11 #define abs(a) ((a) < 0 ? (-1 * (a)) : (a))
 12 template <class T>
 13 inline void swap(T& a, T& b)
 14 {
 15     T tmp = a;a = b;b = tmp;
 16 }
 17 inline void read(long long &x)
 18 {
 19     x = 0;char ch = getchar(), c = ch;
 20     while(ch < '0' || ch > '9') c = ch, ch = getchar();
 21     while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
 22     if(c == '-') x = -x;
 23 }
 24 
 25 const long long INF = 0x3f3f3f3f3f3f3f3f;
 26 const long long MAXN = 200000 + 10;
 27 const long long MAXM = 1000000 + 10;
 28 
 29 struct Edge
 30 {
 31     long long u,v,w,c,nxt;
 32     Edge(long long _u, long long _v, long long _w, long long _c, long long _nxt){u = _u;v = _v;w = _w;c = _c;nxt = _nxt;}
 33     Edge(){}
 34 }edge[MAXN << 1];
 35 long long head[MAXN << 1], cnt = 1, q[MAXN << 1], he, ta, d[MAXN << 1], b[MAXN << 1], from[MAXN << 1], S, T, ans, nn;
 36 inline void insert(long long a, long long b, long long c, long long d)
 37 {
 38     edge[++cnt] = Edge(a,b,c,d,head[a]);
 39     head[a] = cnt;
 40     edge[++cnt] = Edge(b,a,0,-d,head[b]);
 41     head[b] = cnt;
 42 }
 43 bool bfs()
 44 {
 45     he = 0, ta = 1, memset(d, 0x3f3f3f3f, sizeof(d)), d[S] = 0, b[S] = 1, q[0] = S;
 46     while(he < ta)
 47     {
 48         long long now = q[he ++];
 49         if(he > MAXN) he = 0; 
 50         for(long long pos = head[now];pos;pos = edge[pos].nxt)
 51         {
 52             long long v = edge[pos].v;
 53             if(edge[pos].w && d[v] > d[now] + edge[pos].c)
 54             {
 55                 d[v] = d[now] + edge[pos].c, from[v] = pos;
 56                 if(!b[v])
 57                 {
 58                     b[v] = 1, q[ta ++] = v;
 59                     if(ta > MAXN) ta = 0;
 60                 }
 61             }
 62         }
 63         b[now] = 0;
 64     }
 65     return d[T] != INF;
 66 }
 67 void flow()
 68 {
 69     long long x = INF;
 70     for(long long i = from[T];i;i = from[edge[i].u]) x = min(x, edge[i].w);
 71     for(long long i = from[T];i;i = from[edge[i].u]) edge[i].w -= x, edge[i ^ 1].w += x, ans += edge[i].c * x;
 72 }
 73 void mcf()
 74 {
 75     while(bfs()) 
 76         flow();
 77 }
 78 
 79 long long num[MAXN],p,m,f,n,s;
 80 
 81 int main()
 82 {
 83     read(nn);S = nn + nn + 1, T = S + 1;
 84      for(long long i = 1;i <= nn;++ i) read(num[i]);
 85     read(p), read(m), read(f), read(n), read(s);
 86     for(long long i = 1;i <= nn;++ i) 
 87         insert(S, i, num[i], 0);
 88     for(long long i = 1;i < nn;++ i)  
 89         insert(i, i + 1, INF, 0);
 90     for(long long i = 1;i <= nn - m;++ i) 
 91         insert(i, i + m + nn, INF, f);
 92     for(long long i = 1;i <= nn - n;++ i) 
 93         insert(i, i + n + nn, INF, s);
 94     for(long long i = 1;i <= nn;++ i) 
 95         insert(S, i + nn, INF, p);
 96     for(long long i = 1;i <= nn;++ i) 
 97         insert(i + nn, T, num[i], 0);
 98     mcf();
 99     printf("%lld", ans);
100     
101     return 0;
102 }
10

 

11、航空路线问题

 

给定一张航空图,图中顶点代表城市,边代表 2 城市间的直通航线。现要求找出一条满足下述限制条件的且途经城市最多的旅行路线。

(1)从最西端城市出发,单向从西向东途经若干城市到达最东端城市,然后再单向从东向西飞回起点(可途经若干城市)。

(2)除起点城市外,任何城市只能访问 1 次。

对于给定的航空图,试设计一个算法找出一条满足要求的最佳航空旅行路线。

 

直接想成从起点到终点,找两条不相交最路径即可

每个点限流一次,还是考虑拆点x,x',x - > x'容量为1,费用为0,特别的起点和终点可以流两次,S - S‘和T - > T',容量为2,费用为0

对于a -> b,连a' -> b,容量为1,费用为1(环上点数等于边数)

求最大费用最大流即可

  1 #include <iostream>
  2 #include <cstdio>
  3 #include <cstring>
  4 #include <cstdlib>
  5 #include <algorithm>
  6 #include <queue>
  7 #include <vector>
  8 #include <map>
  9 #include <string> 
 10 #include <cmath> 
 11 #define min(a, b) ((a) < (b) ? (a) : (b))
 12 #define max(a, b) ((a) > (b) ? (a) : (b))
 13 #define abs(a) ((a) < 0 ? (-1 * (a)) : (a))
 14 template <class T>
 15 inline void swap(T& a, T& b)
 16 {
 17     T tmp = a;a = b;b = tmp;
 18 }
 19 inline void read(int &x)
 20 {
 21     x = 0;char ch = getchar(), c = ch;
 22     while(ch < '0' || ch > '9') c = ch, ch = getchar();
 23     while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
 24     if(c == '-') x = -x;
 25 }
 26 
 27 const int INF = 0x3f3f3f3f;
 28 const int MAXN = 100000 + 10;
 29 const int MAXM = 1000000 + 10;
 30 
 31 struct Edge
 32 {
 33     int u,v,w,c,nxt;
 34     Edge(int _u, int _v, int _w, int _c, int _nxt){u = _u;v = _v;w = _w;c = _c;nxt = _nxt;}
 35     Edge(){}
 36 }edge[MAXM << 1];
 37 int head[MAXN << 1], he, ta, cnt = 1, ans, anss, q[MAXN << 1], from[MAXN << 1], d[MAXN << 1], b[MAXN << 1], S, T;
 38 void insert(int a, int b, int c, int d)
 39 {
 40     edge[++cnt] = Edge(a,b,c,d,head[a]); head[a] = cnt;
 41     edge[++cnt] = Edge(b,a,0,-d,head[b]); head[b] = cnt;
 42 }
 43 bool spfa()
 44 {
 45     memset(d, 0x3f, sizeof(d));
 46     int he = 0, ta = 1;
 47     d[S] = 0, b[S] = 1, q[he] = S;
 48     while(he != ta)
 49     {
 50         int now = q[he ++];
 51         if(he > MAXN) he = 0;
 52         for(int pos = head[now];pos;pos = edge[pos].nxt)
 53             if(edge[pos].w && d[edge[pos].v] > d[now] + edge[pos].c)
 54             {
 55                 d[edge[pos].v] = d[now] + edge[pos].c;
 56                 from[edge[pos].v] = pos;
 57                 if(!b[edge[pos].v])
 58                 {
 59                     b[edge[pos].v]=1;
 60                     q[ta ++] = edge[pos].v;
 61                     if(ta > MAXN)ta = 0;
 62                 } 
 63             }
 64         b[now] = 0; 
 65     }
 66     if(d[T] == INF)return 0;
 67     return 1;
 68 }
 69 
 70 void flow()
 71 {
 72     int x = INF;
 73     for(int i = from[T];i;i = from[edge[i].u])
 74         x = min(x, edge[i].w);
 75     for(int i = from[T];i;i = from[edge[i].u])
 76     {
 77         edge[i].w -= x;
 78         edge[i^1].w += x;
 79         ans += edge[i].c * x;
 80     }
 81     anss += x;
 82 }
 83 
 84 void mcf()
 85 {
 86     while(spfa()) flow();
 87 }
 88 
 89 int n,m,vis[MAXN];
 90 std::string s[MAXN];
 91 std::map<std::string, int> mp;
 92 
 93 void dfs(int x)
 94 {
 95     if(x == n) return;
 96     vis[x] = 1;
 97     std::cout << s[x] << std::endl;
 98     x += n;
 99     for(int pos = head[x];pos;pos = edge[pos].nxt)
100     {
101         int v = edge[pos].v;
102         if(v > n || vis[v] || edge[pos].w) continue;
103         dfs(v);return;
104     }
105 }
106 void redfs(int x)
107 {
108     vis[x] = 1;
109     if(x == n)
110     {
111         std::cout << s[n] << std::endl;
112         return;
113     }
114     x += n;
115     for(int pos = head[x];pos;pos = edge[pos].nxt)
116     {
117         int v = edge[pos].v;
118         if(v > n || vis[v] || edge[pos].w) continue;
119         redfs(v);break;
120     }
121     std::cout << s[x - n] << std::endl;
122 }
123 
124 int main()
125 {
126     read(n), read(m);S = 1, T = n + n;
127     for(int i = 1;i <= n;++ i)
128     {
129         std::cin >> s[i];
130         mp[s[i]] = i;
131     }
132     insert(1, 1 + n, 2, 0);
133     insert(n, n + n, 2, 0);
134     for(int i = 2;i < n;++ i) 
135         insert(i, i + n, 1, 0);
136     for(int i = 1;i <= m;++ i)
137     {
138         std::string tmp1,tmp2;
139         std::cin >> tmp1 >> tmp2;
140         int ma1, ma2;
141         ma1 = mp[tmp1], ma2 = mp[tmp2];
142         if(ma1 > ma2) swap(ma1, ma2);
143         if(ma1 == 1 && ma2 == n) insert(ma1 + n, ma2, 2, -1);
144         else insert(ma1 + n, ma2, 1, -1);
145     }
146     mcf();
147     if(anss != 2) printf("No Solution!\n");
148     else 
149     {
150         printf("%d\n", -ans);
151         dfs(1);
152         redfs(1);
153     }
154     return 0;
155 }
11

12、软件补丁问题

T 公司发现其研制的一个软件中有 n 个错误,随即为该软件发放了一批共 m 个补丁程序。每一个补丁程序都有其特定的适用环境,某个补丁只有在软件中包含某些错误而同时又不包含另一些错误时才可以使用。一个补丁在排除某些错误的同时,往往会加入另一些错误。

换句话说,对于每一个补丁 i,都有 2 个与之相应的错误集合 B1[i]和 B2[i],使得仅当软件包含 B1[i]中的所有错误,而不包含 B2[i]中的任何错误时,才可以使用补丁 i。补丁 i 将修复软件中的某些错误 F1[i],而同时加入另一些错误 F2[i]。另外,每个补丁都耗费一定的时间。

试设计一个算法,利用 T 公司提供的 m 个补丁程序将原软件修复成一个没有错误的软件,并使修复后的软件耗时最少。对于给定的 n 个错误和 m 个补丁程序,找到总耗时最少的软件修复方案。

 

把每个状态看做节点,状态与状态之间的转移看做边,形成一张“决策图”(根据决策树yy的名词),spfa/dij即可

 1 #include <iostream>
 2 #include <cstdio>
 3 #include <cstring>
 4 #include <cstdlib>
 5 #include <algorithm>
 6 #include <queue>
 7 #include <vector>
 8 #include <map>
 9 #include <string> 
10 #include <cmath> 
11 inline int abs(int x){return x < 0 ? -x : x;}
12 inline int max(int a, int b){return a > b ? a : b;}
13 inline int min(int a, int b){return a < b ? a : b;}
14 inline int swap(int& a, int& b){int tmp = a;a = b;b = tmp;}
15 inline void read(int &x)
16 {
17     x = 0;char ch = getchar(), c = ch;
18     while(ch < '0' || ch > '9') c = ch, ch = getchar();
19     while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
20     if(c == '-') x = -x;
21 }
22 
23 const int INF = 0x3f3f3f3f;
24 const int MAXN = 21;
25 const int MAXM = 100 + 5;
26 
27 int n,m,t[MAXM];
28 int B1[MAXM], B2[MAXM], F1[MAXM], F2[MAXM]; 
29 char s[MAXM];
30 
31 struct Node
32 {
33     int v,w;
34     Node(int _v, int _w){v = _v, w = _w;}
35     Node(){v = w = 0;}
36 };
37 struct cmp
38 {
39     bool operator()(Node a, Node b)
40     {
41         return a.w > b.w;
42     }    
43 };
44 std::priority_queue<Node, std::vector<Node>, cmp> q;
45 int d[1 << MAXN], vis[1 << MAXN];
46 
47 void dij()
48 {
49     memset(d, 0x3f, sizeof(d));
50     int S = (1 << n) - 1;
51     d[S] = 0;
52     q.push(Node(S, 0));
53     Node now;
54     while(q.size())
55     {
56         now = q.top();q.pop();
57         if(vis[now.v]) continue;
58         vis[now.v] = 1;
59         for(int i = 1;i <= m;++ i)
60             if((now.v | B1[i]) == now.v && !(now.v & B2[i]))
61             {
62                 int tmp = ((now.v | F1[i]) ^ F1[i]) | F2[i];
63                 if(d[tmp] > d[now.v] + t[i])
64                 {
65                     d[tmp] = d[now.v] + t[i];
66                     q.push(Node(tmp, d[tmp]));
67                 }
68             }
69     }
70 }
71 
72 int main()
73 {
74     read(n), read(m);
75     for(int i = 1;i <= m;++ i)
76     {
77         read(t[i]);scanf("%s", s + 1);
78         for(int j = 1;j <= m;++ j)
79             if(s[j] == '+') 
80                 B1[i] |= 1 << (j - 1);
81             else if(s[j] == '-') 
82                 B2[i] |= 1 << (j - 1);
83         scanf("%s", s + 1);
84         for(int j = 1;j <= m;++ j)
85             if(s[j] == '-') 
86                 F1[i] |= 1 << (j - 1);
87             else if(s[j] == '+') 
88                 F2[i] |= 1 << (j - 1);
89     }
90     dij();
91     if(d[0] == INF) d[0] = 0;
92     printf("%d", d[0]);
93     return 0;
94 }
14

 

13、星际转移问题

由于人类对自然资源的消耗,人们意识到大约在 2300 年之后,地球就不能再居住了。于是在月球上建立了新的绿地,以便在需要时移民。令人意想不到的是,2177 年冬由于未知的原因,地球环境发生了连锁崩溃,人类必须在最短的时间内迁往月球。现有 n 个太空站位于地球与月球之间,且有 m 艘公共交通太空船在其间来回穿梭。每个太空站可容纳无限多的人,而每艘太空船 i 只可容纳 H[i]个人。每艘太空船将周期性地停靠一系列的太空站,例如:(1,3,4)表示该太空船将周期性地停靠太空站 134134134…。每一艘太空船从一个太空站驶往任一太空站耗时均为 1。人们只能在太空船停靠太空站(或月球、地球)时上、下船。初始时所有人全在地球上,太空船全在初始站。试设计一个算法,找出让所有人尽快地全部转移到月球上的运输方案。

对于给定的太空船的信息,找到让所有人尽快地全部转移到月球上的运输方案。

 

考虑把太空站按照天数拆点,连边即可。

二分天数需要重新建图跑dinic,复杂度堪忧。不如枚举天数,每次在原图的基础上增广即可。

  1 #include <iostream>
  2 #include <cstdio>
  3 #include <cstring>
  4 #include <cstdlib>
  5 #include <algorithm>
  6 #include <queue>
  7 #include <vector>
  8 #include <map>
  9 #include <string> 
 10 #include <cmath> 
 11 inline int abs(int x){return x < 0 ? -x : x;}
 12 inline int max(int a, int b){return a > b ? a : b;}
 13 inline int min(int a, int b){return a < b ? a : b;}
 14 inline int swap(int& a, int& b){int tmp = a;a = b;b = tmp;}
 15 inline void read(int &x)
 16 {
 17     x = 0;char ch = getchar(), c = ch;
 18     while(ch < '0' || ch > '9') c = ch, ch = getchar();
 19     while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
 20     if(c == '-') x = -x;
 21 }
 22 const int INF = 0x3f3f3f3f;
 23 const int MAXN = 13 + 5;
 24 const int MAXM = 20 + 5;
 25 const int MAXK = 50 + 5;
 26 struct Edge
 27 {
 28     int v,w,nxt;
 29     Edge(int _v, int _w, int _nxt){v = _v;w = _w;nxt = _nxt;}
 30     Edge(){}
 31 }edge[400010];
 32 int head[400010], cnt = 1;
 33 inline void insert(int a, int b, int c)
 34 {
 35     edge[++ cnt] = Edge(b, c, head[a]), head[a] = cnt;
 36     edge[++ cnt] = Edge(a, 0, head[b]), head[b] = cnt;
 37 }
 38 int S, T, h[400010], q[400010], he, ta, ans;
 39 bool bfs()
 40 {
 41     memset(h, -1, sizeof(h));
 42     he = ta = 0, q[ta ++] = S, h[S] = 0;
 43     while(he < ta)
 44     {
 45         int now = q[he ++];
 46         for(int pos = head[now];pos;pos = edge[pos].nxt)
 47         {
 48             int v = edge[pos].v;
 49             if(h[v] == -1 && edge[pos].w) 
 50                 h[v] = h[now] + 1, q[ta ++] = v; 
 51         }
 52     }
 53     return h[T] != -1;
 54 }
 55 int dfs(int x, int f)
 56 {
 57     if(x == T) return f;
 58     int used = 0, w;
 59     for(int pos = head[x];pos;pos = edge[pos].nxt)
 60     {
 61         int v = edge[pos].v;
 62         if(h[v] == h[x] + 1)
 63         {
 64             w = dfs(v, min(f - used, edge[pos].w));
 65             edge[pos].w -= w, edge[pos ^ 1].w += w;
 66             used += w;
 67             if(used == f) return f;
 68         }
 69     }
 70     if(!used) h[x] = -1;
 71     return used;
 72 }
 73 void dinic()
 74 {
 75     while(bfs()) ans += dfs(S, INF);
 76 }
 77 int n,m,k,hh[MAXM],r[MAXM], sta[MAXM][MAXN + 2], tot[MAXN + 2], pre[MAXN + 2];
 78 int fa[MAXN];
 79 int find(int x)
 80 {
 81     return x == fa[x] ? x : fa[x] = find(fa[x]);
 82 }
 83 int main()
 84 {
 85     read(n), read(m), read(k);
 86     for(int i = 1;i <= n + 2;++ i) fa[i] = i;
 87     for(int i = 1;i <= m;++ i)
 88     {
 89         read(hh[i]), read(r[i]);
 90         int f1,f2;
 91         for(int j = 1;j <= r[i];++ j)
 92         {
 93             read(sta[i][j]);
 94             if(sta[i][j] == 0) 
 95                 sta[i][j] = n + 1;
 96             else if(sta[i][j] == -1) 
 97                 sta[i][j] = n + 2;
 98         }
 99         for(int j = 2;j <= r[i];++ j)
100         {
101             f1 = find(sta[i][j - 1]), f2 = find(sta[i][j]);
102             if(f1 != f2) fa[f1] = f2;
103         }
104     }
105     if(find(n + 1) != find(n + 2)) printf("0");
106     else
107     {
108         S = 399999, T = 400000;
109         int now = n + 2, pnow = 0, day;
110         for(int i = 1;i <= m;++ i)
111         {
112             if(r[i] == 1) continue;
113             tot[i] = 2, pre[i] = 1;
114         }
115         insert(S, pnow + n + 1, INF);
116         insert(pnow + n + 2, T, INF);
117         for(day = 1;;++ day)
118         {
119             insert(S, now + n + 1, INF);
120             insert(now + n + 2, T, INF);
121             for(int i = 1;i <= n + 2;++ i)
122                 insert(pnow + i, now + i, INF);
123             for(int i = 1;i <= m;++ i)
124             {
125                 if(r[i] == 1) continue;
126                 insert(sta[i][pre[i]] + pnow, sta[i][tot[i]] + now, hh[i]);
127                 pre[i] = tot[i];
128                 tot[i] = tot[i] % r[i] + 1;
129             }
130             dinic();
131             if(ans >= k) break;
132                 pnow += n + 2, now += n + 2;
133         }
134         printf("%d", day);
135     }
136     return 0;
137 }
13

 

 

14、汽车加油行驶问题

给定一个 N×NN \times NN×N 的方形网格,设其左上角为起点◎,坐标(1,1)(1,1)(1,1)XXX 轴向右为正, YYY 轴向下为正,每个方格边长为 111 ,如图所示。

一辆汽车从起点◎出发驶向右下角终点▲,其坐标为 (N,N)(N,N)(N,N)

在若干个网格交叉点处,设置了油库,可供汽车在行驶途中加油。汽车在行驶过程中应遵守如下规则:

  1. 汽车只能沿网格边行驶,装满油后能行驶 KKK 条网格边。出发时汽车已装满油,在起点与终点处不设油库。

  2. 汽车经过一条网格边时,若其 XXX 坐标或 YYY 坐标减小,则应付费用 BBB ,否则免付费用。

  3. 汽车在行驶过程中遇油库则应加满油并付加油费用 AAA

  4. 在需要时可在网格点处增设油库,并付增设油库费用 CCC (不含加油费用AAA )。

  5. N,K,A,B,CN,K,A,B,CN,K,A,B,C 均为正整数, 且满足约束: 2≤N≤100,2≤K≤102\leq N\leq 100,2 \leq K \leq 102N100,2K10

设计一个算法,求出汽车从起点出发到达终点所付的最小费用。

 

有一个坑点,满油到加油站就不用加油了。

  1 #include <iostream>
  2 #include <cstdio>
  3 #include <cstdlib>
  4 #include <cstring>
  5 #include <algorithm>
  6 #include <queue>
  7 #include <vector>
  8 #include <stack>
  9 #include <cmath>
 10 inline int max(int a, int b){return a > b ? a : b;}
 11 inline int min(int a, int b){return a < b ? a : b;}
 12 inline int abs(int x){return x < 0 ? -x : x;}
 13 inline void swap(int &x, int &y){int tmp = x;x = y;y = tmp;}
 14 inline void read(int &x)
 15 {
 16     x = 0;char ch = getchar(), c = ch;
 17     while(ch < '0' || ch > '9') c = ch, ch = getchar();
 18     while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
 19     if(c == '-') x = -x;
 20 }
 21 const int INF = 0x3f3f3f3f;
 22 const int MAXN = 200 + 10;
 23 const int MAXK = 100 + 2;
 24 const int dx[4] = {-1, 0, 1, 0};
 25 const int dy[4] = {0, -1, 0, 1}; 
 26 int n,k,a,b[4],c,g[MAXN][MAXN],num[MAXN][MAXN][MAXK];
 27 struct Edge
 28 {
 29     int v,w,nxt;
 30     Edge(int _v, int _w, int _nxt){v = _v, w = _w, nxt = _nxt;}
 31     Edge(){}
 32 }edge[1000010];
 33 int head[1000010], cnt;
 34 inline void insert(int a, int b, int c)
 35 {
 36     edge[++ cnt] = Edge(b, c, head[a]), head[a] = cnt;
 37 } 
 38 struct Node
 39 {
 40     int v, w;
 41     Node(int _v, int _w){v = _v;w = _w;}
 42     Node(){v = w = 0;}
 43 };
 44 struct cmp
 45 {
 46     bool operator()(Node a, Node b)
 47     {
 48         return a.w > b.w;
 49     }
 50 };
 51 std::priority_queue<Node, std::vector<Node>, cmp> q;
 52 int d[1000010], vis[1000010];
 53 void dij()
 54 {
 55     memset(d, 0x3f, sizeof(d)), d[num[1][1][k]] = 0;
 56     q.push(Node(num[1][1][k], 0));Node now;
 57     while(q.size())
 58     {
 59         now = q.top(), q.pop();
 60         if(vis[now.v]) continue;
 61         vis[now.v] = 1;
 62         for(int pos = head[now.v];pos;pos = edge[pos].nxt)
 63         {
 64             int v = edge[pos].v;
 65             if(d[v] > d[now.v] + edge[pos].w)
 66             {
 67                 d[v] = d[now.v] + edge[pos].w;
 68                 q.push(Node(v, d[v]));
 69             }
 70         }
 71     }
 72 }
 73 int main()
 74 {
 75     read(n), read(k), read(a), read(b[0]), b[1] = b[0], read(c);
 76     for(int i = 1;i <= n;++ i)
 77         for(int j = 1;j <= n;++ j)
 78         {
 79             read(g[i][j]);
 80             for(int l = 0;l <= k;++ l)
 81                 num[i][j][l] = (i - 1) * (n *(k + 1)) + (j - 1) * (k + 1) + l + 1;
 82         }
 83     /*
 84     如果 在加油站  连向满油,费用为a
 85     如果 不在加油站 
 86         如果 油空 连向油满, 费用为a + c
 87         如果 油不空 向四周连边 ,费用为0/b 
 88     */
 89     for(int i = 1;i <= n;++ i) 
 90         for(int j = 1;j <= n;++ j)
 91                 for(int l = 0;l <= k;++ l)
 92                     if(g[i][j] && l < k) 
 93                         insert(num[i][j][l], num[i][j][k], a);
 94                     else if(!g[i][j] && !l) 
 95                         insert(num[i][j][l], num[i][j][k], a + c);
 96                     else
 97                         for(int p = 0;p < 4;++ p)
 98                         {
 99                             int xx = i + dx[p], yy = j + dy[p];
100                             if(xx <= 0 || yy <= 0 || xx > n || yy > n) continue;
101                             insert(num[i][j][l], num[xx][yy][l - 1], b[p]);
102                         }
103     dij();int ans = INF;
104     for(int l = 0;l <= k;++ l) ans = min(ans, d[num[n][n][l]]);
105     printf("%d", ans);
106     return 0;
107 }
14

 

 

15、孤岛营救问题

1944 年,特种兵麦克接到国防部的命令,要求立即赶赴太平洋上的一个孤岛,营救被敌军俘虏的大兵瑞恩。瑞恩被关押在一个迷宫里,迷宫地形复杂,但幸好麦克得到了迷宫的地形图。迷宫的外形是一个长方形,其南北方向被划分为 NNN 行,东西方向被划分为 MMM 列,于是整个迷宫被划分为 N×MN\times MN×M 个单元。每一个单元的位置可用一个有序数对(单元的行号,单元的列号)来表示。南北或东西方向相邻的 222 个单元之间可能互通,也可能有一扇锁着的门,或者是一堵不可逾越的墙。迷宫中有一些单元存放着钥匙,并且所有的门被分成PPP 类,打开同一类的门的钥匙相同,不同类门的钥匙不同。

大兵瑞恩被关押在迷宫的东南角,即 (N,M)(N,M)(N,M) 单元里,并已经昏迷。迷宫只有一个入口,在西北角。也就是说,麦克可以直接进入 (1,1)(1,1)(1,1) 单元。另外,麦克从一个单元移动到另一个相邻单元的时间为 111 ,拿取所在单元的钥匙的时间以及用钥匙开门的时间可忽略不计。

试设计一个算法,帮助麦克以最快的方式到达瑞恩所在单元,营救大兵瑞恩。

 

决策图最短路。

  1 #include <iostream>
  2 #include <cstdio>
  3 #include <cstdlib>
  4 #include <cstring>
  5 #include <algorithm>
  6 #include <queue>
  7 #include <vector>
  8 #include <stack>
  9 #include <cmath>
 10 inline int max(int a, int b){return a > b ? a : b;}
 11 inline int min(int a, int b){return a < b ? a : b;}
 12 inline int abs(int x){return x < 0 ? -x : x;}
 13 inline void swap(int &x, int &y){int tmp = x;x = y;y = tmp;}
 14 inline void read(int &x)
 15 {
 16     x = 0;char ch = getchar(), c = ch;
 17     while(ch < '0' || ch > '9') c = ch, ch = getchar();
 18     while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
 19     if(c == '-') x = -x;
 20 }
 21 const int INF = 0x3f3f3f3f;
 22 const int MAXN = 10 + 1;
 23 const int MAXK = 150 + 1;
 24 const int MAXS = 14;
 25 const int dx[4] = {1, -1, 0, 0};
 26 const int dy[4] = {0, 0, 1, -1};
 27 int n,m,p,k,s,g[MAXN][MAXN][MAXN][MAXN],key[MAXN][MAXN][MAXN + 2],ans = INF;
 28 
 29 struct Node
 30 {
 31     int x, y, S, w;
 32     Node(int _x, int _y, int _S, int _w){x = _x, y = _y, S = _S, w = _w;}
 33     Node(){}
 34 }q[MAXN * MAXN * (1 << MAXN)];
 35 int he, ta, vis[MAXN][MAXN][1 << MAXN];
 36 
 37 void bfs()
 38 {
 39     for(int i = 1;i <= p;++ i)
 40         if(key[1][1][i])
 41             q[0].S |= 1 << (i - 1);
 42     q[0].x = q[0].y = 1;q[0].w = 0;
 43     he = 0, ta = 1, vis[1][1][0] = 1;
 44     Node tmp;
 45     while(he < ta)
 46     {
 47         tmp = q[he ++];
 48         int x = tmp.x, y = tmp.y, S = tmp.S, w = tmp.w;
 49         
 50     //    printf("%d %d %d %d\n", x, y, S, w);
 51         
 52         if(x == n && y == m)
 53         {
 54             ans = min(ans, w);
 55             continue;
 56         }
 57         for(int i = 0;i < 4;++ i)
 58         {
 59             int xx = x + dx[i], yy = y + dy[i], t = g[x][y][xx][yy];
 60             S = tmp.S;
 61             if(xx <= 0 || yy <= 0 || xx > n || yy > m || t == 0) continue;
 62             if(t == -1)
 63             {
 64                 for(int i = 1;i <= p;++ i)
 65                     if(key[xx][yy][i])
 66                         S |= 1 << (i - 1);
 67                 if(!vis[xx][yy][S])
 68                     q[ta ++] = Node(xx, yy, S, w + 1), vis[xx][yy][S] = 1;
 69             }
 70             else if((S & (1 << (t - 1))))
 71             {
 72                 for(int i = 1;i <= p;++ i)
 73                     if(key[xx][yy][i])
 74                         S |= 1 << (i - 1);
 75                 if(!vis[xx][yy][S])
 76                     q[ta ++] = Node(xx, yy, S, w + 1), vis[xx][yy][S] = 1;
 77             }
 78         }
 79     }
 80 }
 81 
 82 int main()
 83 {
 84     read(n), read(m), read(p), read(k);
 85     memset(g, -1, sizeof(g));
 86     for(int i = 1;i <= k;++i)
 87     {
 88         int tmp1,tmp2,tmp3,tmp4;
 89         read(tmp1), read(tmp2), read(tmp3), read(tmp4);
 90         read(g[tmp1][tmp2][tmp3][tmp4]);
 91         g[tmp3][tmp4][tmp1][tmp2] = g[tmp1][tmp2][tmp3][tmp4];
 92     }
 93     read(s);
 94     for(int i = 1;i <= s;++ i)
 95     {
 96         int tmp1,tmp2,tmp3;read(tmp1), read(tmp2), read(tmp3);
 97         ++ key[tmp1][tmp2][tmp3];
 98     }
 99     bfs();
100     if(ans == INF) ans = -1;
101     printf("%d", ans); 
102     return 0;
103 }
15

 

16、数字梯形问题

给定一个由 nnn 行数字组成的数字梯形如下图所示。

梯形的第一行有 mmm 个数字。从梯形的顶部的 mmm 个数字开始,在每个数字处可以沿左下或右下方向移动,形成一条从梯形的顶至底的路径。

分别遵守以下规则:

  1. 从梯形的顶至底的 mmm 条路径互不相交;

  2. 从梯形的顶至底的 mmm 条路径仅在数字结点处相交;

  3. 从梯形的顶至底的 mmm 条路径允许在数字结点相交或边相交。

 

第一个,拆点限流即可。

第二个,正常连边,边上限流即可。

第三个,dp(划掉)边上不限流即可

  1 #include <iostream>
  2 #include <cstdio>
  3 #include <cstdlib>
  4 #include <cstring>
  5 #include <algorithm>
  6 #include <queue>
  7 #include <vector>
  8 #include <stack>
  9 #include <cmath>
 10 inline int max(int a, int b){return a > b ? a : b;}
 11 inline int min(int a, int b){return a < b ? a : b;}
 12 inline int abs(int x){return x < 0 ? -x : x;}
 13 inline void swap(int &x, int &y){int tmp = x;x = y;y = tmp;}
 14 inline void read(int &x)
 15 {
 16     x = 0;char ch = getchar(), c = ch;
 17     while(ch < '0' || ch > '9') c = ch, ch = getchar();
 18     while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
 19     if(c == '-') x = -x;
 20 }
 21 const int INF = 0x3f3f3f3f;
 22 const int MAXN = 20 + 5;
 23 int n,m,num[MAXN][MAXN], hao[MAXN][MAXN], tot;
 24 
 25 struct Edge
 26 {
 27     int u,v,w,c,nxt;
 28     Edge(int _u, int _v, int _w, int _c, int _nxt){u = _u;v = _v;w = _w;c = _c;nxt = _nxt;}
 29     Edge(){}
 30 }edge[10000010];
 31 int head[1000010], cnt = 1, d[1000010], vis[1000010], from[1000010], q[1000010], he, ta, S, T, ans;
 32 inline void insert(int a, int b, int c, int d)
 33 {
 34     edge[++ cnt] = Edge(a, b, c, d, head[a]), head[a] = cnt;
 35     edge[++ cnt] = Edge(b, a, 0, -d, head[b]), head[b] = cnt;
 36 }
 37 bool spfa()
 38 {
 39     memset(d, 0x3f, sizeof(d)), d[S] = 0, he = 0, ta = 1, q[0] = S, vis[S] = 1;
 40     while(he != ta)
 41     {
 42         int now = q[he ++];if(he > 1000000) he = 0;
 43         for(int pos = head[now];pos;pos = edge[pos].nxt)
 44         {
 45             int v = edge[pos].v;
 46             if(edge[pos].w && d[v] > d[now] + edge[pos].c)
 47             {
 48                 d[v] = d[now] + edge[pos].c, from[v] = pos;
 49                 if(!vis[v])
 50                 {
 51                     q[ta ++] = v, vis[v] = 1;
 52                     if(ta > 1000000) ta = 0;
 53                 }
 54             }
 55         }
 56         vis[now] = 0;
 57     }
 58     return d[T] != INF;
 59 }
 60 void flow()
 61 {
 62     int mi = INF;
 63     for(int i = from[T];i;i = from[edge[i].u]) mi = min(mi, edge[i].w);
 64     for(int i = from[T];i;i = from[edge[i].u]) edge[i].w -= mi, edge[i ^ 1].w += mi, ans += mi * edge[i].c;
 65 }
 66 void mcf()
 67 {
 68     while(spfa()) flow();
 69 }
 70 void build1()
 71 {
 72     int SS = (tot << 1) + 1;S = SS + 1, T = S + 1;
 73     insert(S, SS, m, 0);
 74     for(int i = 1;i <= m;++ i) insert(SS, hao[1][i], 1, -num[1][i]);
 75     for(int i = 1;i <= m + n - 1;++ i) insert(hao[n][i] + tot, T, 1, 0);
 76     for(int i = 1;i <= n;++ i)
 77         for(int j = 1;j <= m + i - 1;++ j)
 78             insert(hao[i][j], tot + hao[i][j], 1, 0);
 79     for(int i = 1;i < n;++ i)
 80         for(int j = 1;j <= m + i - 1;++ j)
 81         {
 82             insert(tot + hao[i][j], hao[i + 1][j], 1, -num[i + 1][j]);
 83             insert(tot + hao[i][j], hao[i + 1][j + 1], 1, -num[i + 1][j + 1]);
 84         }
 85     mcf();
 86 }
 87 void build2()
 88 {
 89     memset(head, 0, sizeof(head)), cnt = 1, ans = 0, memset(from, 0, sizeof(from));
 90     int SS = tot + 1;S = SS + 1, T = S + 1;
 91     insert(S, SS, m, 0);
 92     for(int i = 1;i <= m;++ i) insert(SS, hao[1][i], 1, -num[1][i]);
 93     for(int i = 1;i <= m + n - 1;++ i) insert(hao[n][i], T, m, 0);
 94     for(int i = 1;i < n;++ i)
 95         for(int j = 1;j <= m + i - 1;++ j)
 96         {
 97             insert(hao[i][j], hao[i + 1][j], 1, -num[i + 1][j]);
 98             insert(hao[i][j], hao[i + 1][j + 1], 1, -num[i + 1][j + 1]);
 99         }
100     mcf();
101 }
102 void build3()
103 {
104     memset(head, 0, sizeof(head)), cnt = 1, ans = 0, memset(from, 0, sizeof(from));
105     int SS = tot + 1;S = SS + 1, T = S + 1;
106     insert(S, SS, m, 0);
107     for(int i = 1;i <= m;++ i) insert(SS, hao[1][i], 1, -num[1][i]);
108     for(int i = 1;i <= m + n - 1;++ i) insert(hao[n][i], T, m, 0);
109     for(int i = 1;i < n;++ i)
110         for(int j = 1;j <= m + i - 1;++ j)
111         {
112             insert(hao[i][j], hao[i + 1][j], m, -num[i + 1][j]);
113             insert(hao[i][j], hao[i + 1][j + 1], m, -num[i + 1][j + 1]);
114         }
115     mcf();
116 }
117 int main()
118 {
119     read(m), read(n);
120     for(int i = 1;i <= n;++ i)
121         for(int j = 1;j <= m + i - 1;++ j)
122             read(num[i][j]), hao[i][j] = ++ tot;
123     build1();printf("%d\n", -ans);
124     build2();printf("%d\n", -ans);
125     build3();printf("%d\n", -ans);
126     return 0;
127 }
16

 

17、运输问题

公司有 mm 个仓库和 nn 个零售商店。第 iii 个仓库有 aia_i 个单位的货物;第 jj 个零售商店需要 bjb_jb 个单位的货物。

货物供需平衡,即∑i=1mai=∑j=1nbj\sum\limits_{i=1}^{m}a_i=\sum\limits_{j=1}^{n}b_j

从第 iii 个仓库运送每单位货物到第 jj个零售商店的费用为 cijc_{ij}​​。

试设计一个将仓库中所有货物运送到零售商店的运输方案,使总运输费用最少。

 

板子题。

 1 #include <iostream>
 2 #include <cstdio>
 3 #include <cstdlib>
 4 #include <cstring>
 5 #include <algorithm>
 6 #include <queue>
 7 #include <vector>
 8 #include <stack>
 9 #include <cmath>
10 inline int max(int a, int b){return a > b ? a : b;}
11 inline int min(int a, int b){return a < b ? a : b;}
12 inline int abs(int x){return x < 0 ? -x : x;}
13 inline void swap(int &x, int &y){int tmp = x;x = y;y = tmp;}
14 inline void read(int &x)
15 {
16     x = 0;char ch = getchar(), c = ch;
17     while(ch < '0' || ch > '9') c = ch, ch = getchar();
18     while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
19     if(c == '-') x = -x;
20 }
21 const int INF = 0x3f3f3f3f;
22 const int MAXN = 100 + 10;
23 struct Edge
24 {
25     int u, v, w, c, nxt;
26     Edge(int _u, int _v , int _w, int _c, int _nxt){u = _u, v = _v, w = _w, c = _c, nxt = _nxt;}
27     Edge(){}
28 }edge[MAXN * MAXN << 1];
29 int head[MAXN << 2], cnt = 1, S, T, q[MAXN << 2], he, ta, from[MAXN << 2], d[MAXN << 2], vis[MAXN << 2], ans;
30 inline void insert(int a, int b, int c, int d)
31 {
32     edge[++ cnt] = Edge(a, b, c, d, head[a]), head[a] = cnt;
33     edge[++ cnt] = Edge(b, a, 0, -d, head[b]), head[b] = cnt;
34 }
35 bool spfa()
36 {
37     memset(d, 0x3f, sizeof(d)), d[S] = 0, vis[S] = 1, he = 0, ta = 1, q[0] = S;
38     while(he < ta)
39     {
40         int now = q[he ++];if(he > (MAXN << 2) - 2) he = 0;
41         for(int pos = head[now];pos;pos = edge[pos].nxt)
42         {
43             int v = edge[pos].v;
44             if(edge[pos].w && d[v] > d[now] + edge[pos].c)
45             {
46                 d[v] = d[now] + edge[pos].c, from[v] = pos;
47                 if(!vis[v])
48                 {
49                     q[ta ++] = v, vis[v] = 1;
50                     if(ta > (MAXN << 2) - 2) ta = 0;
51                 }
52             }
53         }
54         vis[now] = 0;
55     }
56     return d[T] != INF;
57 }
58 void flow()
59 {
60     int mi = INF;
61     for(int i = from[T];i;i = from[edge[i].u]) mi = min(mi, edge[i].w);
62     for(int i = from[T];i;i = from[edge[i].u]) edge[i].w -= mi, edge[i ^ 1].w += mi, ans += mi * edge[i].c;
63 }
64 void mcf()
65 {
66     while(spfa()) flow();
67 }
68 int n,m,a[MAXN], b[MAXN], g[MAXN][MAXN];
69 int main()
70 {
71     read(m), read(n);
72     S = m + n + 1, T = S + 1;
73     for(int i = 1;i <= m;++ i)
74         read(a[i]), insert(S, i, a[i], 0);
75     for(int i = 1;i <= n;++ i)
76         read(b[i]), insert(m + i, T, b[i], 0);
77     for(int i = 1;i <= m;++ i)
78         for(int j = 1;j <= n;++ j)
79             read(g[i][j]), insert(i, m + j, INF, g[i][j]); 
80     mcf();
81     printf("%d\n", ans);
82     
83     memset(head, 0, sizeof(head)), memset(from, 0, sizeof(from)), ans = 0;
84     for(int i = 1;i <= m;++ i)
85         insert(S, i, a[i], 0);
86     for(int i = 1;i <= n;++ i)
87         insert(m + i, T, b[i], 0);
88     for(int i = 1;i <= m;++ i)
89         for(int j = 1;j <= n;++ j)
90             insert(i, m + j, INF, -g[i][j]); 
91     mcf();
92     printf("%d\n", -ans);
93     return 0;
94 }
17

 

18、分配问题

nn 件工作要分配给 nn个人做。第 iii 个人做第 jj 件工作产生的效益为 cijc_{ij} 。试设计一个将 nn件工作分配给 nn个人做的分配方案,使产生的总效益最大。

 

KM算法板子题。

 1 #include <iostream>
 2 #include <cstdio>
 3 #include <cstdlib>
 4 #include <cstring>
 5 #include <algorithm>
 6 #include <queue>
 7 #include <vector>
 8 #include <stack>
 9 #include <cmath>
10 inline int max(int a, int b){return a > b ? a : b;}
11 inline int min(int a, int b){return a < b ? a : b;}
12 inline int abs(int x){return x < 0 ? -x : x;}
13 inline void swap(int &x, int &y){int tmp = x;x = y;y = tmp;}
14 inline void read(int &x)
15 {
16     x = 0;char ch = getchar(), c = ch;
17     while(ch < '0' || ch > '9') c = ch, ch = getchar();
18     while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
19     if(c == '-') x = -x;
20 }
21 const int INF = 0x3f3f3f3f;
22 const int MAXN = 100 + 5;
23 int g[MAXN][MAXN], n1, n2, lab1[MAXN], lab2[MAXN], lk1[MAXN], lk2[MAXN], pre[MAXN], sla[MAXN], vis[MAXN]; 
24 void fac(int x)
25 {
26     memset(pre, 0, sizeof(pre)), memset(vis, 0, sizeof(vis)), memset(sla, 0x3f, sizeof(sla)), vis[0] = 1;
27     int y;
28     do
29     {
30         y = 0;
31         for(int i = 1;i <= n2;++ i)
32         {
33             if(vis[i]) continue;
34             if(lab1[x] + lab2[i] - g[x][i] < sla[i]) sla[i] = lab1[x] + lab2[i] - g[x][i], pre[i] = x;
35             if(sla[i] < sla[y]) y = i;
36         }
37         int d = sla[y];
38         for(int i = 1;i <= n1;++ i) if(vis[lk1[i]]) lab1[i] -= d;
39         for(int i = 1;i <= n2;++ i) if(vis[i]) lab2[i] += d; else sla[i] -= d;
40         vis[y] = 1;
41     }while(x = lk2[y]);
42     for(;y;swap(y, lk1[lk2[y] = pre[y]]));
43 }
44 int KM()
45 {
46     for(int i = 1;i <= n1;++ i) fac(i);
47     int ans = 0;
48     for(int i = 1;i <= n1;++ i) ans += g[i][lk1[i]];
49     return ans;
50 }
51 int main()
52 {
53     read(n1), n2 = n1;
54     for(int i = 1;i <= n1;++ i)
55         for(int j = 1;j <= n2;++ j)
56             read(g[i][j]), g[i][j] = -g[i][j];
57     printf("%d\n", -KM()); 
58     
59     memset(lab1, 0, sizeof(lab1)), memset(lab2, 0, sizeof(lab2));
60     memset(lk1, 0, sizeof(lk1)), memset(lk2, 0, sizeof(lk2));
61     for(int i = 1;i <= n1;++ i)
62         for(int j = 1;j <= n2;++ j)
63             g[i][j] = -g[i][j], lab1[i] = max(lab1[i], g[i][j]);
64     printf("%d", KM());
65     return 0;
66 }
18

 

19、负载平衡问题

G 公司有 nn 个沿铁路运输线环形排列的仓库,每个仓库存储的货物数量不等。如何用最少搬运量可以使 nn个仓库的库存数量相同。搬运货物时,只能在相邻的仓库之间搬运。

 

可数学做法(环形纸牌加强版)

也可最小费用最大流

 1 #include <iostream>
 2 #include <cstdio>
 3 #include <cstdlib>
 4 #include <cstring>
 5 #include <algorithm>
 6 #include <queue>
 7 #include <vector>
 8 #include <stack>
 9 #include <cmath>
10 inline int max(int a, int b){return a > b ? a : b;}
11 inline int min(int a, int b){return a < b ? a : b;}
12 inline int abs(int x){return x < 0 ? -x : x;}
13 inline void swap(int &x, int &y){int tmp = x;x = y;y = tmp;}
14 inline void read(int &x)
15 {
16     x = 0;char ch = getchar(), c = ch;
17     while(ch < '0' || ch > '9') c = ch, ch = getchar();
18     while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
19     if(c == '-') x = -x;
20 }
21 const int INF = 0x3f3f3f3f;
22 const int MAXN = 100 + 10;
23 struct Edge
24 {
25     int u, v, w, c, nxt;
26     Edge(int _u, int _v , int _w, int _c, int _nxt){u = _u, v = _v, w = _w, c = _c, nxt = _nxt;}
27     Edge(){}
28 }edge[MAXN * MAXN << 1];
29 int head[MAXN << 2], cnt = 1, S, T, q[MAXN << 2], he, ta, from[MAXN << 2], d[MAXN << 2], vis[MAXN << 2], ans;
30 inline void insert(int a, int b, int c, int d)
31 {
32     edge[++ cnt] = Edge(a, b, c, d, head[a]), head[a] = cnt;
33     edge[++ cnt] = Edge(b, a, 0, -d, head[b]), head[b] = cnt;
34 }
35 bool spfa()
36 {
37     memset(d, 0x3f, sizeof(d)), d[S] = 0, vis[S] = 1, he = 0, ta = 1, q[0] = S;
38     while(he < ta)
39     {
40         int now = q[he ++];if(he > (MAXN << 2) - 2) he = 0;
41         for(int pos = head[now];pos;pos = edge[pos].nxt)
42         {
43             int v = edge[pos].v;
44             if(edge[pos].w && d[v] > d[now] + edge[pos].c)
45             {
46                 d[v] = d[now] + edge[pos].c, from[v] = pos;
47                 if(!vis[v])
48                 {
49                     q[ta ++] = v, vis[v] = 1;
50                     if(ta > (MAXN << 2) - 2) ta = 0;
51                 }
52             }
53         }
54         vis[now] = 0;
55     }
56     return d[T] != INF;
57 }
58 void flow()
59 {
60     int mi = INF;
61     for(int i = from[T];i;i = from[edge[i].u]) mi = min(mi, edge[i].w);
62     for(int i = from[T];i;i = from[edge[i].u]) edge[i].w -= mi, edge[i ^ 1].w += mi, ans += mi * edge[i].c;
63 }
64 void mcf()
65 {
66     while(spfa()) flow();
67 }
68 int n, num[MAXN],sum,ave;
69 int main()
70 {
71     read(n);for(int i = 1;i <= n;++ i) read(num[i]), sum += num[i];
72     S = n + 1, T = S + 1, ave = sum / n;
73     for(int i = 1;i <= n;++ i)
74         if(num[i] > ave) insert(S, i, num[i] - ave, 0);
75         else if(num[i] < ave) insert(i, T, ave - num[i], 0);
76     insert(n, 1, INF, 1), insert(1, n, INF, 1);
77     for(int i = 1;i < n;++ i) insert(i, i + 1, INF, 1), insert(i + 1, i, INF, 1);
78     mcf();
79     printf("%d", ans);
80     return 0;
81 }
19

 

 

20、深海机器人问题

深海资源考察探险队的潜艇将到达深海的海底进行科学考察。

潜艇内有多个深海机器人。潜艇到达深海海底后,深海机器人将离开潜艇向预定目标移动。

深海机器人在移动中还必须沿途采集海底生物标本。沿途生物标本由最先遇到它的深海机器人完成采集。

每条预定路径上的生物标本的价值是已知的,而且生物标本只能被采集一次。

本题限定深海机器人只能从其出发位置沿着向北或向东的方向移动,而且多个深海机器人可以在同一时间占据同一位置。

用一个 P×QP\times QP×Q 网格表示深海机器人的可移动位置。西南角的坐标为 (0,0)(0,0)(0,0) ,东北角的坐标为 (Q,P)(Q,P)(Q,P) 。

给定每个深海机器人的出发位置和目标位置,以及每条网格边上生物标本的价值。

计算深海机器人的最优移动方案, 使深海机器人到达目的地后,采集到的生物标本的总价值最高。

 

裸。

 1 #include <iostream>
 2 #include <cstdio>
 3 #include <cstdlib>
 4 #include <cstring>
 5 #include <algorithm>
 6 #include <queue>
 7 #include <vector>
 8 #include <stack>
 9 #include <cmath>
10 inline int max(int a, int b){return a > b ? a : b;}
11 inline int min(int a, int b){return a < b ? a : b;}
12 inline int abs(int x){return x < 0 ? -x : x;}
13 inline void swap(int &x, int &y){int tmp = x;x = y;y = tmp;}
14 inline void read(int &x)
15 {
16     x = 0;char ch = getchar(), c = ch;
17     while(ch < '0' || ch > '9') c = ch, ch = getchar();
18     while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
19     if(c == '-') x = -x;
20 }
21 const int INF = 0x3f3f3f3f;
22 struct Edge
23 {
24     int u, v, w, c, nxt;
25     Edge(int _u, int _v , int _w, int _c, int _nxt){u = _u, v = _v, w = _w, c = _c, nxt = _nxt;}
26     Edge(){}
27 }edge[10000010];
28 int head[1000010], cnt = 1, S, T, q[1000010], he, ta, from[1000010], d[1000010], vis[1000010], ans;
29 inline void insert(int a, int b, int c, int d)
30 {
31     edge[++ cnt] = Edge(a, b, c, d, head[a]), head[a] = cnt;
32     edge[++ cnt] = Edge(b, a, 0, -d, head[b]), head[b] = cnt;
33 }
34 bool spfa()
35 {
36     memset(d, 0x3f, sizeof(d)), d[S] = 0, vis[S] = 1, he = 0, ta = 1, q[0] = S;
37     while(he < ta)
38     {
39         int now = q[he ++];if(he > 1000010) he = 0;
40         for(int pos = head[now];pos;pos = edge[pos].nxt)
41         {
42             int v = edge[pos].v;
43             if(edge[pos].w && d[v] > d[now] + edge[pos].c)
44             {
45                 d[v] = d[now] + edge[pos].c, from[v] = pos;
46                 if(!vis[v])
47                 {
48                     q[ta ++] = v, vis[v] = 1;
49                     if(ta > 1000010) ta = 0;
50                 }
51             }
52         }
53         vis[now] = 0;
54     }
55     return d[T] != INF;
56 }
57 void flow()
58 {
59     int mi = INF;
60     for(int i = from[T];i;i = from[edge[i].u]) mi = min(mi, edge[i].w);
61     for(int i = from[T];i;i = from[edge[i].u]) edge[i].w -= mi, edge[i ^ 1].w += mi, ans += mi * edge[i].c;
62 }
63 void mcf()
64 {
65     while(spfa()) flow();
66 }
67 int a, b, p, qq, tmp, tmp1, tmp2, ma;
68 inline int num(int x, int y)
69 {
70     ma = max(ma, (x - 1) * p + y);
71     return (x - 1) * p + y;
72 }
73 int main()
74 {
75     read(a), read(b), read(p), read(qq);
76     ++ p, ++ qq;
77     S = p * qq + 1, T = S + 1;
78     for(int i = 1;i <= p;++ i)
79         for(int j = 1;j < qq;++ j)
80             read(tmp), insert(num(j, i), num(j + 1, i), 1, -tmp), insert(num(j, i), num(j + 1, i), INF, 0); 
81     for(int i = 1;i <= qq;++ i)
82         for(int j = 1;j < p;++ j)
83             read(tmp), insert(num(i, j), num(i, j + 1), 1, -tmp), insert(num(i, j), num(i, j + 1), INF, 0);
84     for(int i = 1;i <= a;++ i) 
85         read(tmp), read(tmp1), read(tmp2), insert(S, num(tmp2 + 1, tmp1 + 1), tmp, 0);
86     for(int i = 1;i <= b;++ i)
87         read(tmp), read(tmp1), read(tmp2), insert(num(tmp2 + 1, tmp1 + 1), T, tmp, 0);
88     mcf();
89     printf("%d", -ans);
90     return 0;
91 }
20

 

 

21、最长k可重线段集问题

给定平面 x−O−yx-O-yxOy 上 nnn 个开线段组成的集合 III ,和一个正整数 kkk 。试设计一个算法,从开线段集合 III 中选取出开线段集合 S⊆IS\subseteq ISI ,使得在 xxx 轴上的任何一点 ppp ,SSS 中与直线 x=px=px=p 相交的开线段个数不超过 kkk ,且∑z∈S∣z∣\sum\limits_{z\in S}|z|zSz∣ 达到最大。这样的集合 SSS 称为开线段集合 III 的最长 kkk 可重线段集。∑z∈S∣z∣\sum\limits_{z\in S}|z|zSz∣ 称为最长 kkk 可重线段集的长度。

对于任何开线段 zzz ,设其断点坐标为 (x0,y0)(x_0,y_0)(x0,y0) 和 (x1,y1)(x_1,y_1)(x1,y1) ,则开线段 zzz 的长度 ∣z∣|z|z∣ 定义为:∣z∣=⌊(x1−x0)2+(y1−y0)2⌋ |z|=\lfloor\sqrt{(x_1-x_0)^2+(y_1-y_0)^2}\rfloorz=(x1x0)2+(y1y0)2

对于给定的开线段集合 III 和正整数 kkk ,计算开线段集合 III 的最长 kkk 可重线段集的长度。

 

考虑线段不垂直于x轴的情况。

把横坐标离散化,然后S连向1,流量为k,费用0,之后所有的i连向i + 1,流量INF,费用0,最后一个点连向T,流量k,费用0

对于一条线段,将两段的x连起来,小的连向大的,费用为线段长。

跑最大流即可。

为什么是对的呢?

跑一个流相当于找到了几段不相交的区间。

这样,从S流出的流动过程中有费用流量为i时,一定有i个区间重叠。

如果没有。。那我强行加区间不就好了吗。。答案不就更优了吗。。。

还需要说明得到的流是最优的。

如果得到的不是最优的,说明还能选一些线段进来,这时S->1的流一定流满了(没流慢不就把没选的选进来了吗。。)

考虑增加流,一个一个加,先加一个流,如果多流区间,那么设这个区间与p个区间相交,p = k时条件不满足,p < k时,说明存在已经流了的k - p条流可以流没选的区间但却没有流

感性理解一下。。

 

还有一个问题,这里没考虑线段垂直于x轴的情况。

拆点就好了。因为离散化写炸调了很久。

  1 #include <iostream>
  2 #include <cstdio>
  3 #include <cstdlib>
  4 #include <cstring>
  5 #include <algorithm>
  6 #include <map>
  7 #include <cmath>
  8 inline long long max(long long a, long long b){return a > b ? a : b;}
  9 inline long long min(long long a, long long b){return a < b ? a : b;}
 10 inline long long abs(long long x){return x < 0 ? -x : x;}
 11 inline void swap(long long &x, long long &y){long long tmp = x;x = y;y = tmp;}
 12 inline void read(long long &x)
 13 {
 14     x = 0;char ch = getchar(), c = ch;
 15     while(ch < '0' || ch > '9') c = ch, ch = getchar();
 16     while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
 17     if(c == '-') x = -x;
 18 }
 19 const long long INF = 0x3f3f3f3f3f3f3f3f;
 20 struct Edge
 21 {
 22     long long u, v, w, c, nxt;
 23     Edge(long long _u, long long _v , long long _w, long long _c, long long _nxt){u = _u, v = _v, w = _w, c = _c, nxt = _nxt;}
 24     Edge(){}
 25 }edge[2000010];
 26 long long head[2000010], cnt = 1, S, T, q[2000010], he, ta, from[2000010], d[2000010], vis[2000010], ans;
 27 inline void insert(long long a, long long b, long long c, long long d)
 28 {
 29     edge[++ cnt] = Edge(a, b, c, d, head[a]), head[a] = cnt;
 30     edge[++ cnt] = Edge(b, a, 0, -d, head[b]), head[b] = cnt;
 31 }
 32 bool spfa()
 33 {
 34     memset(d, 0x3f, sizeof(d)), d[S] = 0, vis[S] = 1, he = 0, ta = 1, q[0] = S;
 35     while(he < ta)
 36     {
 37         long long now = q[he ++];if(he > 2000000) he = 0;
 38         for(long long pos = head[now];pos;pos = edge[pos].nxt)
 39         {
 40             long long v = edge[pos].v;
 41             if(edge[pos].w && d[v] > d[now] + edge[pos].c)
 42             {
 43                 d[v] = d[now] + edge[pos].c, from[v] = pos;
 44                 if(!vis[v])
 45                 {
 46                     q[ta ++] = v, vis[v] = 1;
 47                     if(ta > 2000000) ta = 0;
 48                 }
 49             }
 50         }
 51         vis[now] = 0;
 52     }
 53     return d[T] != INF;
 54 }
 55 void flow()
 56 {
 57     long long mi = INF;
 58     for(long long i = from[T];i;i = from[edge[i].u]) mi = min(mi, edge[i].w);
 59     for(long long i = from[T];i;i = from[edge[i].u]) edge[i].w -= mi, edge[i ^ 1].w += mi, ans += mi * edge[i].c;
 60 }
 61 void mcf()
 62 {
 63     while(spfa()) flow();
 64 }
 65 long long n, k;
 66 struct Node
 67 {
 68     long long x1, y1, x2, y2;
 69 }node[20010];
 70 std::map<long long, long long> mp;
 71 std::map<long long, long long> mpp;
 72 long long num[20010], tot;
 73 long long dis(long long x1, long long y1, long long x2, long long y2)
 74 {
 75     return sqrt((x1 - x2) * (x1 - x2) + (y1 - y2) * (y1 - y2));
 76 }
 77 int main()
 78 {
 79     read(n), read(k);
 80     for(long long i = 1;i <= n;++ i)
 81     {
 82         read(node[i].x1), num[++ tot] = node[i].x1,
 83         read(node[i].y1), num[++ tot] = node[i].y1,
 84         read(node[i].x2), num[++ tot] = node[i].x2,
 85         read(node[i].y2), num[++ tot] = node[i].y2;
 86         if(node[i].x1 > node[i].x2)
 87             swap(node[i].x1, node[i].x2), swap(node[i].y1, node[i].y2);
 88     }
 89     std::sort(num + 1, num + 1 + tot);
 90     tot = std::unique(num + 1, num + 1 + tot) - num - 1;
 91     for(long long i = 1;i < tot;++ i)
 92         mp[num[i]] = i;
 93     int tmp = tot;
 94     mp[num[tot]] = tot;
 95     for(long long i = 1;i <= n;++ i)
 96         if(node[i].x1 == node[i].x2 && !mpp.count(mp[node[i].x1]))
 97             mpp[mp[node[i].x1]] = ++ tot;
 98 
 99     S = tot + 1, T = S + 1;
100     insert(S, 1, k, 0), insert(tmp, T, k, 0); 
101     for(int i = 1;i < tmp;++ i)
102     {
103         if(mpp.count(i)) insert(i, mpp[i], INF, 0), insert(mpp[i], i + 1, INF, 0);
104         else insert(i, i + 1, INF, 0);
105     }
106     for(long long i = 1;i <= n;++ i)
107         if(node[i].x1 == node[i].x2) 
108             insert(mp[node[i].x1], mpp[mp[node[i].x1]], 1, -abs(node[i].y1 - node[i].y2));
109         else if(mpp.count(mp[node[i].x1]))
110             insert(mpp[mp[node[i].x1]], mp[node[i].x2], 1, -dis(node[i].x1, node[i].y1, node[i].x2, node[i].y2));
111         else 
112             insert(mp[node[i].x1], mp[node[i].x2], 1, -dis(node[i].x1, node[i].y1, node[i].x2, node[i].y2));
113     mcf();
114     printf("%lld", -ans);
115     return 0;
116 }
21

 

22、最长k可重区间集问题

对于给定的开区间集合 I 和正整数 k,计算开区间集合 I 的最长 k可重区间集的长度。

 

同上题。

  1 #include <iostream>
  2 #include <cstdio>
  3 #include <cstdlib>
  4 #include <cstring>
  5 #include <algorithm>
  6 #include <map>
  7 #include <cmath>
  8 inline long long max(long long a, long long b){return a > b ? a : b;}
  9 inline long long min(long long a, long long b){return a < b ? a : b;}
 10 inline long long abs(long long x){return x < 0 ? -x : x;}
 11 inline void swap(long long &x, long long &y){long long tmp = x;x = y;y = tmp;}
 12 inline void read(long long &x)
 13 {
 14     x = 0;char ch = getchar(), c = ch;
 15     while(ch < '0' || ch > '9') c = ch, ch = getchar();
 16     while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
 17     if(c == '-') x = -x;
 18 }
 19 const long long INF = 0x3f3f3f3f3f3f3f3f;
 20 struct Edge
 21 {
 22     long long u, v, w, c, nxt;
 23     Edge(long long _u, long long _v , long long _w, long long _c, long long _nxt){u = _u, v = _v, w = _w, c = _c, nxt = _nxt;}
 24     Edge(){}
 25 }edge[2000010];
 26 long long head[2000010], cnt = 1, S, T, q[2000010], he, ta, from[2000010], d[2000010], vis[2000010], ans;
 27 inline void insert(long long a, long long b, long long c, long long d)
 28 {
 29     edge[++ cnt] = Edge(a, b, c, d, head[a]), head[a] = cnt;
 30     edge[++ cnt] = Edge(b, a, 0, -d, head[b]), head[b] = cnt;
 31 }
 32 bool spfa()
 33 {
 34     memset(d, 0x3f, sizeof(d)), d[S] = 0, vis[S] = 1, he = 0, ta = 1, q[0] = S;
 35     while(he < ta)
 36     {
 37         long long now = q[he ++];if(he > 2000000) he = 0;
 38         for(long long pos = head[now];pos;pos = edge[pos].nxt)
 39         {
 40             long long v = edge[pos].v;
 41             if(edge[pos].w && d[v] > d[now] + edge[pos].c)
 42             {
 43                 d[v] = d[now] + edge[pos].c, from[v] = pos;
 44                 if(!vis[v])
 45                 {
 46                     q[ta ++] = v, vis[v] = 1;
 47                     if(ta > 2000000) ta = 0;
 48                 }
 49             }
 50         }
 51         vis[now] = 0;
 52     }
 53     return d[T] != INF;
 54 }
 55 void flow()
 56 {
 57     long long mi = INF;
 58     for(long long i = from[T];i;i = from[edge[i].u]) mi = min(mi, edge[i].w);
 59     for(long long i = from[T];i;i = from[edge[i].u]) edge[i].w -= mi, edge[i ^ 1].w += mi, ans += mi * edge[i].c;
 60 }
 61 void mcf()
 62 {
 63     while(spfa()) flow();
 64 }
 65 long long n, k;
 66 struct Node
 67 {
 68     long long x1, x2;
 69 }node[20010];
 70 std::map<long long, long long> mp;
 71 long long num[20010], tot;
 72 long long dis(long long x1, long long x2)
 73 {
 74     return abs(x1 - x2);
 75 }
 76 int main()
 77 {
 78     read(n), read(k);
 79     for(long long i = 1;i <= n;++ i)
 80     {
 81         read(node[i].x1), num[++ tot] = node[i].x1,
 82         read(node[i].x2), num[++ tot] = node[i].x2;
 83         if(node[i].x1 > node[i].x2)
 84             swap(node[i].x1, node[i].x2);
 85     }
 86     std::sort(num + 1, num + 1 + tot);
 87     tot = std::unique(num + 1, num + 1 + tot) - num - 1;
 88     for(long long i = 1;i < tot;++ i)
 89         mp[num[i]] = i;
 90     mp[num[tot]] = tot;
 91     S = tot + 1, T = S + 1;
 92     insert(S, 1, k, 0), insert(tot, T, k, 0); 
 93     for(int i = 1;i < tot;++ i)
 94         insert(i, i + 1, INF, 0);
 95     for(long long i = 1;i <= n;++ i)
 96         insert(mp[node[i].x1], mp[node[i].x2], 1, -dis(node[i].x1, node[i].x2));
 97     mcf();
 98     printf("%lld", -ans);
 99     return 0;
100 }
22

 

 

23、火星探险问题

火星探险队的登陆舱将在火星表面着陆,登陆舱内有多部障碍物探测车。登陆舱着陆后,探测车将离开登陆舱向先期到达的传送器方向移动。探测车在移动中还必须采集岩石标本。每一块岩石标本由最先遇到它的探测车完成采集。每块岩石标本只能被采集一次。岩石标本被采集后,其他探测车可以从原来岩石标本所在处通过。探测车不能通过有障碍的地面。本题限定探测车只能从登陆处沿着向南或向东的方向朝传送器移动,而且多个探测车可以在同一时间占据同一位置。如果某个探测车在到达传送器以前不能继续前进,则该车所采集的岩石标本将全部损失。

用一个 P·Q 网格表示登陆舱与传送器之间的位置。登陆舱的位置在(X1,Y1)处,传送器

的位置在(XP ,YQ)处。

X 1,Y 1 X 2 , Y 1 X 3 , Y 1 ... X P-1, Y 1 X P , Y 1

X 1,Y 2 X 2 , Y 2 X 3 , Y 2 ... X P-1, Y 2 X P , Y 2

X 1, Y 3 X 2 , Y 3 X 3 ,Y 3 ... X P-1, Y 3 X P , Y 3

... ...

X 1 ,Y Q-1 X 2 , Y Q-1 X 3 , Y Q-1 ... X P-1, Y Q-1 X P , Y Q-1

X 1,Y Q X 2 , Y Q X 3 , Y Q ... X P-1, Y Q X P ,Y Q

给定每个位置的状态,计算探测车的最优移动方案,使到达传送器的探测车的数量最多,

而且探测车采集到的岩石标本的数量最多

 

拆点即可。

输出路径注意细节。

 

  1 #include <iostream>
  2 #include <cstdio>
  3 #include <cstdlib>
  4 #include <cstring>
  5 #include <algorithm>
  6 #include <map>
  7 #include <cmath>
  8 inline int max(int a, int b){return a > b ? a : b;}
  9 inline int min(int a, int b){return a < b ? a : b;}
 10 inline int abs(int x){return x < 0 ? -x : x;}
 11 inline void swap(int &x, int &y){int tmp = x;x = y;y = tmp;}
 12 inline void read(int &x)
 13 {
 14     x = 0;char ch = getchar(), c = ch;
 15     while(ch < '0' || ch > '9') c = ch, ch = getchar();
 16     while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
 17     if(c == '-') x = -x;
 18 }
 19 const int INF = 0x3f3f3f3f;
 20 struct Edge
 21 {
 22     int u, v, w, c, nxt;
 23     Edge(int _u, int _v, int _w, int _c, int _nxt){u = _u;v = _v;w = _w;c = _c;nxt = _nxt;}
 24     Edge(){}
 25 }edge[1000010];
 26 int head[100010], cnt = 1, q[100010], he, ta, vis[100010], d[100010], from[100010], ans1, ans2, S, T;
 27 
 28 inline void insert(int a, int b, int c, int d)
 29 {
 30     edge[++ cnt] = Edge(a, b, c, d, head[a]), head[a] = cnt;
 31     edge[++ cnt] = Edge(b, a, 0, -d, head[b]), head[b] = cnt;
 32 }
 33 bool spfa()
 34 {
 35     memset(d, 0x3f, sizeof(d)), d[S] = 0, vis[S] = 1, he = 0, ta = 1, q[0] = S;
 36     while(he != ta)
 37     {
 38         int now = q[he ++];if(he > 100000) he = 0;
 39         for(int pos = head[now];pos;pos = edge[pos].nxt)
 40         {
 41             int v = edge[pos].v;
 42             if(edge[pos].w && d[v] > d[now] + edge[pos].c)
 43             {
 44                 d[v] = d[now] + edge[pos].c, from[v] = pos;
 45                 if(!vis[v]) 
 46                 {
 47                     vis[v] = 1, q[ta ++] = v; 
 48                     if(ta > 100000) ta = 0;
 49                 }
 50             }
 51         }
 52         vis[now] = 0;
 53     }
 54     return d[T] != INF;
 55 }
 56 void flow()
 57 {
 58     int mi = INF;
 59     for(int i = from[T];i;i = from[edge[i].u]) mi = min(mi, edge[i].w);
 60     for(int i = from[T];i;i = from[edge[i].u]) edge[i].w -= mi, edge[i ^ 1].w += mi, ans2 += mi * edge[i].c;
 61     ans1 += mi;
 62 }
 63 void mcf()
 64 {
 65     while(spfa()) flow();
 66 }
 67 int n, p, qq, tot, num[1010][1010][2], g[1010][1010], x[100010], y[100010];
 68 int cntt;
 69 void dfs(int from, int u, int tag)
 70 {
 71     for(int pos = head[u];pos;pos = edge[pos].nxt)
 72     {
 73         if(!(pos & 1) && edge[pos ^ 1].w)
 74         {
 75             -- edge[pos ^ 1].w;
 76             int v = edge[pos].v;
 77             if(!tag)
 78             {
 79                 dfs(from, v, tag ^ 1);
 80                 return;
 81             }
 82             printf("%d ", cntt);
 83             if(x[v] - x[from] == 1) printf("0\n");
 84             else printf("1\n");
 85             dfs(v, v, tag ^ 1);
 86             return;
 87         }
 88     }
 89 }
 90 //笔  本子  投影仪上的高考题//
 91 int main()
 92 {
 93     read(n), read(p), read(qq);
 94     swap(p, qq);
 95     for(int i = 1;i <= p;++ i)
 96         for(int j = 1;j <= qq;++ j)
 97         {
 98             read(g[i][j]), 
 99             num[i][j][0] = ++ tot, x[tot] = i, y[tot] = j;
100             num[i][j][1] = ++ tot, x[tot] = i, y[tot] = j;
101             if(g[i][j] == 2) insert(num[i][j][0], num[i][j][1], 1, -1);
102             if(g[i][j] != 1) insert(num[i][j][0], num[i][j][1], INF, 0);
103         }
104     S = tot + 1, T = S + 1;
105     insert(S, num[1][1][0], n, 0), insert(num[p][qq][1], T, n, 0);
106     for(int i = 1;i <= p;++ i)
107         for(int j = 1;j <= qq;++ j)
108         {
109             if(j + 1 <= qq) insert(num[i][j][1], num[i][j + 1][0], INF, 0);
110             if(i + 1 <= p) insert(num[i][j][1], num[i + 1][j][0], INF, 0);
111         }
112     mcf();
113     for(int i = 1;i <= ans1;++ i)
114         ++ cntt, dfs(num[1][1][1], num[1][1][1], 0);
115     return 0;
116 }
23

 

24、骑士共存问题

在一个 n*n个方格的国际象棋棋盘上,马(骑士)可以攻击的棋盘方格如图所示。棋盘上某些方格设置了障碍,骑士不得进入

对于给定的 n*n 个方格的国际象棋棋盘和障碍标志,计算棋盘上最多可以放置多少个骑士,使得它们彼此互不攻击

 

黑白染色,最大独立集

 

之前写过一个二分图最大独立集的题解,这次写一个最小割的。

 

首先把方格黑白染色,对于每个黑格,他能跳到的地方都是白格,反之亦然。

于是每个黑格向能跳到的白格连边。每条边所连的两个点只能选一个,另一个必须割掉。

最小割。

 

  1 #include <iostream>
  2 #include <cstdio>
  3 #include <cstdlib>
  4 #include <cstring>
  5 #include <algorithm>
  6 #include <map>
  7 #include <cmath>
  8 inline int max(int a, int b){return a > b ? a : b;}
  9 inline int min(int a, int b){return a < b ? a : b;}
 10 inline int abs(int x){return x < 0 ? -x : x;}
 11 inline void swap(int &x, int &y){int tmp = x;x = y;y = tmp;}
 12 inline void read(int &x)
 13 {
 14     x = 0;char ch = getchar(), c = ch;
 15     while(ch < '0' || ch > '9') c = ch, ch = getchar();
 16     while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
 17     if(c == '-') x = -x;
 18 }
 19 const int INF = 0x3f3f3f3f;
 20 const int dx[8] = {1, 1, 2, 2, -1, -1, -2, -2};
 21 const int dy[8] = {2, -2, 1, -1, 2, -2, 1, -1};
 22 struct Edge
 23 {
 24     int u,v,w,nxt;
 25     Edge(int _u, int _v, int _w, int _nxt){u = _u;v = _v;w = _w;nxt = _nxt;}
 26     Edge(){}
 27 }edge[1000010];
 28 int head[100010], cnt = 1, S, T, q[100010], he, ta, h[100010], ans;
 29 inline void insert(int a, int b, int c)
 30 {
 31     edge[++ cnt] = Edge(a, b, c, head[a]), head[a] = cnt;
 32     edge[++ cnt] = Edge(b, a, 0, head[b]), head[b] = cnt;
 33 }
 34 bool bfs()
 35 {
 36     memset(h, -1, sizeof(h)), h[S] = 0, he = ta = 0, q[ta ++] = S;
 37     while(he < ta)
 38     {
 39         int now = q[he ++];
 40         for(int pos = head[now];pos;pos = edge[pos].nxt)
 41         {
 42             int v = edge[pos].v;
 43             if(edge[pos].w && h[v] == -1)
 44                 h[v] = h[now] + 1, q[ta ++] = v;
 45         }
 46     }
 47     return h[T] != -1;
 48 }
 49 int dfs(int x, int f)
 50 {
 51     if(x == T) return f;
 52     int used = 0, w;
 53     for(int pos = head[x];pos;pos = edge[pos].nxt)
 54     {
 55         int v = edge[pos].v;
 56         if(h[v] == h[x] + 1)
 57         {
 58             w = dfs(v, min(edge[pos].w, f - used));
 59             edge[pos].w -= w;
 60             edge[pos ^ 1].w += w;
 61             used += w;
 62             if(used == f) return f;
 63         }
 64     }
 65     if(!used) h[x] = -1;
 66     return used;
 67 }
 68 void dinic()
 69 {
 70     while(bfs()) ans += dfs(S, INF);
 71 }
 72 int n, m, n1, n2, g[505][505], num[505][505], tot;
 73 int main()
 74 {
 75     read(n), read(m);
 76     for(int i = 1;i <= m;++ i)
 77     {
 78         int tmp1, tmp2;
 79         read(tmp1), read(tmp2);
 80         g[tmp1][tmp2] = 1;
 81     }
 82     S = n * n + 1, T = S + 1;
 83     for(int i = 1;i <= n;++ i)
 84         for(int j = 1;j <= n;++ j)
 85             num[i][j] = ++ tot;
 86     for(int i = 1;i <= n;++ i)
 87         for(int j = 1;j <= n;++ j)
 88         {
 89             if(g[i][j]) continue;
 90             if((i + j) & 1) insert(num[i][j], T, 1);
 91             else
 92             {
 93                 insert(S, num[i][j], 1);
 94                 for(int k = 0;k < 8;++ k)
 95                 {
 96                     int x = i + dx[k], y = j + dy[k];
 97                     if(x < 1 || y < 1 || x > n || y > n || g[x][y]) continue;
 98                     insert(num[i][j], num[x][y], INF);
 99                 }
100             }
101         }
102     dinic();
103     printf("%d", n * n - m - ans);
104     return 0;
105 }
24

 

 

The end.

 

posted @ 2018-01-29 08:16  嘒彼小星  阅读(591)  评论(0编辑  收藏  举报