post on 14 Nov 2020 about 13155words require 44min
CC BY 4.0 (除特别声明或转载文章外)
如果这篇博客帮助到你,可以请我喝一杯咖啡~
预计难度:CF Div2-
Sun Yat-sen University, originally known as National Guangdong University, was founded in 1924 by Dr. Sun Yat-sen (also called Sun Zhongshan), a great democratic revolutionary leader of the 20th century. The University is located in Guangdong Province, an area neighboring Hong Kong and Macao, which is at the forefront of China’s reform and opening up.
Being one of the leading universities in the People’s Republic of China, Sun Yat-sen University is a comprehensive multi-disciplinary university, including the humanities, social sciences, natural sciences, technical sciences, medical sciences, pharmacology, and management science. At present, Sun Yat-sen University covers a total area of 6.17 square kilometers and has 4 campuses: Guangzhou South Campus, Guangzhou North Campus, Guangzhou East Campus, and Zhuhai Campus.
The input contains a positive integers n.
$1\le n\le 12$
You should output the lexicographically N-th string with the same letters(2*S
+1*Y
+1*U
) as SYSU
.
1
1
1
SSUY
1
5
1
SYSU
吴坎
题目大意:输出与 SYSU
字母相同的字典序第 $n$ 小的串。
熟悉 STL 的同学可以直接使用 next_permutation
解决.
1
2
3
4
5
6
7
8
9
10
11
#include <bits/stdc++.h>
using namespace std;
int main()
{
string s("SYSU");
sort(s.begin(), s.end());
int n;
for (scanf("%d", &n); --n;)
next_permutation(s.begin(), s.end());
printf("%s", s.c_str());
}
不熟的同学直接手算然后打表也是可以的啦~
欢迎报考中山大学!
This formula is written as an equation about the length of edges A, B and C, which is usually called Pythagorean law.
$A^2 + B^2 = C^2$
Where $C$ is the length of the slanted side and $A$ and $B$ are the lengths of the other two sides.
Given $n$, you need to compute right triangles with side lengths $A$, $B$ and $C$ satisfying inequality $1 \le A \le B \le C \le n$.
The first line of input contains an integer $n~(1\leq n\leq 10000)$.
The first line of input contains an integer donating the answer.
1
5
1
1
3, 4, 5
范俭豪
暴力枚举 a,b 即可。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
#include <bits/stdc++.h>
using namespace std;
int main()
{
int n, ans = 0;
scanf("%d", &n);
for (int a = 1; a <= n; ++a)
for (int b = a; b <= n; ++b)
{
double cc = sqrt(a * a + b * b);
int c = cc;
if (b <= c && c <= n && c == cc)
++ans;
}
printf("%d", ans);
}
如果担心时间被卡的话打表也是毫无问题的。
A tour group is trapped on an island. The island can be regarded as a matrix of equal side length.
Each grid is either an obstacle (character #
), a tourist (character O
), or an open space (character .
), visitors can walk up, down, left, right(to neighboring grid), but can’t walk over obstacles.
Ask how many tourists can never get out of the island.
First line of the input is an integer $T$($1\le T \le 10$) indicating the number of test cases.
For each case, the first line contains an integer $n$ ($1\le n \le 50$) indicating the side length of island, each of the following $n$ lines contains a string with exact n charactors(obstacle #
, tourist O
or space .
).
For each case output one integer per line, the number of tourist(s) cannot get out of the island.
1
2
3
4
5
6
1
4
O.##
.#O#
#O.#
.###
1
2
梁赛波
题目大意:给定一个 $n\times n$ 矩阵,每个点要么是游客,要么是空地,要么是障碍物,每个人每次可以朝上下左右任一方向走一格,问有多少人最终可以走出矩阵。
暴力的做法是,每个游客分别以他初始位置为起点,搜索(BFS/DFS)出到达矩阵之外的路径,时间复杂度 $O(Tn^4)$
可以发现如果某个格子可以最终走出矩阵,那么所有路过该格子的人都可以最终走出矩阵。
上述搜索过程一旦走到最终可以逃出的格子,就可以立即返回。每个格子只会被遍历一次,时间复杂度 $O(Tn^2)$。
同样的时间复杂度,也可以从矩阵的非障碍边界开始向内搜索,能到达的游客格子肯定可以沿着相反的路逃出去。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
#include <bits/stdc++.h>
using namespace std;
const pair<int, int> dir[4] = {{-1, 0}, {0, 1}, {1, 0}, {0, -1}};
const int maxn = 105;
char a[maxn][maxn];
bool vis[maxn][maxn];
int main()
{
int _;
cin >> _;
while (_--)
{
int n;
cin >> n;
memset(vis, 0, sizeof(vis));
for (int i = 0; i < n; i++)
cin >> a[i];
queue<pair<int, int>> q;
for (int i = 0; i < n; i++)
{
for (int j = 0; j < n; j++)
{
if ((i == 0 || j == 0 || i == n - 1 || j == n - 1) && a[i][j] != '#')
{
vis[i][j] = true;
q.emplace(i, j);
}
}
}
while (q.size())
{
auto it = q.front();
q.pop();
for (int i = 0; i < 4; i++)
{
int r = it.first + dir[i].first;
int c = it.second + dir[i].second;
if (r < 0 || r >= n || c < 0 || c >= n)
continue;
if (vis[r][c] || a[r][c] == '#')
continue;
vis[r][c] = true;
q.emplace(r, c);
}
}
int ans = 0;
for (int i = 0; i < n; i++)
{
for (int j = 0; j < n; j++)
{
if (!vis[i][j] && a[i][j] == 'O')
ans++;
}
}
cout << ans << "\n";
}
}
After a long thought, Liz made a tough decision. When the girl came back home from picking raspberries with delight. Liz said goodbye to the girl with smile. A basket of raspberries dropped, scattering on the floor. 「You are ought to be free and use your lightsome wings to fly to everywhere you want.」 「Why? I just want to stay with you forever. Only by your side can I taste the sweet of happiness.」 「I am the cage that traps you. You have wings and vast sky to explore. I can’t deprive you of your wings,」 Liz opened the door, smiling mildly. 「So just leave here and fly high. Please let me watch your beautiful figure go away,」 Liz held out her hand, 「That’s the way I convey my feelings.」 She paused and took a deep breath 「I love you.」 The girl sobbed, stepped slowly towards door. She looked back, with love, with unwillingness, and turned into a blue bird, flying away.
After the blue bird flew away, Liz still missed her very much. Worrying that she and other small animals might be caught in a storm again, Liz decided to transplant trees in the woods to cover as much area as possible. There were $n$ trees in the woods, and each tree $T_i$ can be represented as a closed interval $[a_i,b_i]$. Liz wanted to find a solution so that all intervals covered $[0,10000]$, and the displacement of the interval with the largest displacement was minimized. Specifically, suppose $T_i$ was moved to $[a_i+c_i, b_i+c_i]$, and Liz wanted to minimize $\max_i\lvert c_i \rvert$.
First line of the input is an integer $n$ indicating the number of trees.
The following $n$ lines, each line contains 2 integers $a_i,b_i$.
$1<n<10000, 0\le a_i<b_i\le 10000, \sum_{i=1}^n(b_i-a_i)\ge 10000$
Output a real number that represents the answer.
If the answer is an integer, output the integer directly; otherwise, keep 1 decimal place.
If you still have any questions, you can refer to the sample output.
1
2
3
2
10 5010
4980 9980
1
20
1
2
3
4
5
4
0 4000
3000 5000
5001 8000
7000 10000
1
0.5
The answer must be a semi-integer, that is, twice the number must be an integer.
吴坎
二分答案 $\Delta$,然后判断是否有移动方案使得所有区间的移动量都不超过 $\Delta$。令 $[S,T]=[0,10000]$ 表示我们要覆盖的区域,当 $\Delta$ 已经给定时,可以使用如下贪心算法。
算法正确性证明思路如下。假定在某个状态下存在对未使用区间的一个放法(记作 $\text{SOL}$)去覆盖 $[A,T]$ 这个区域,那么按上述贪心标准运行下去也一定能找到一个放法去覆盖 $[A,T]$ 这个区域。使用归纳法证明这个结论。显然上述命题对 $K=1$ 是成立的,下面考虑 $K>1$的情况。
假定 $D_1$ 是根据贪心算法选择的那个区间。如果 $\text{SOL}$ 中 $D_1$ 被放置的位置能 cover 住 $A$。那么很明显结论是正确的。因此,我们假定 $\text{SOL}$ 用了另一个区间,例如 $D_2$,来 cover 点 $A$。不妨假定 $D_2$ 在 $\text{SOL}$ 中被移动到了尽量靠右的位置。否则可以调整 SOL。假设 $D_2$ 在 $\text{SOL}$ 中移动后的右坐标为 $A’$。分两种情况讨论:
根据以上两种情况我们知道,有解的情况下我们的贪心标准也能找到一个解。算法的正确性得证。
根据算法流程,给定 $\Delta$,很容易在 $O(n^2)$ 的时间内实现判定,不过这会超时。利用堆这一数据结构,我们可以将算法的效率提升到 $O(n\log n)$。将未使用的区间分为三类:
由于算法过程中 A 是不断变大的。所以 Dead 的区间永远是 Dead 的。为实现我们的算法,只需一个最小堆来维护 Active 的区间,Key 为区间的右端点。
参考:Danny Z. Chen Yan Gu Jian Li Haitao Wang, Algorithms on Minimizing the Maximum Sensor Movement for Barrier Coverage of a Linear Domain
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
#include <bits/stdc++.h>
using namespace std;
const int N = 1e4 + 9, M = 1e4;
const double EPS = 0.25;
pair<int, int> p[N];
int n;
int ok(double delta)
{
priority_queue<pair<int, int>> q;
int i = 0;
for (double A = 0; A < M - EPS;)
{
for (; i < n && p[i].first <= A + delta + EPS; ++i)
q.push({-p[i].second, i});
int updated = 0;
for (int x, l, r; !q.empty() && !updated; q.pop())
if (x = q.top().second, l = p[x].first, r = p[x].second,
A <= r + delta + EPS)
{
if (A > l + delta)
A = r + delta;
else
A += r - l;
updated = 1;
}
if (!updated)
return 0;
}
return 1;
}
int bs(int b, int e)
{
if (e - b < 2)
return e;
int m = b + (e - b >> 1);
return ok(m * 0.5) ? bs(b, m) : bs(m, e);
}
int main()
{
scanf("%d", &n);
for (int i = 0; i < n; ++i)
scanf("%d%d", &p[i].first, &p[i].second);
sort(p, p + n);
int ans = bs(0, M << 1);
printf("%d", ans >> 1);
if (ans & 1)
printf(".5");
}
完整的背景故事可见 https://www.bilibili.com/read/cv7276984 ,此处已获得原作者的使用许可。
强推一波利兹与青鸟,官方在国内开通了微博,不过感觉离上映还遥遥无期的样子…
In a container terminal, the bottleneck of the traffic is often at the quay. Therefore, the terminal operator has to allocate a limited number of berths of the quay to vessels in an efficient way.
As illustrate in Figure above (An illustrated example, with a quay of $n=5$ berths, $m=7$ vessels, where two vessels are waiting outside the quay), consider a container terminal of $n$ berths and $m$ vessels arrived, where each vessel $i$ (for $i=1,2,\dots,m$) requires a berth to load and unload containers, and the handling time is $t_{i,j}$ minutes if berth $j$ (for $j=1, 2, \dots, n$) is allocated to vessel $i$. For each vessel $i=1,2,\dots,m$, the terminal manager, Brother D, needs to decide on the berth, denoted by $b_i\in\lbrace 1,2,\dots,n\rbrace $, as well on the starting time of berthing, denoted by $s_i\ge 0$. It must be satisfied that no two vessels are allowed to occupy the same berth simultaneously, i.e., for any two different vessels i and j, if $b_i=b_j$, then either $s_i+t_{i,b_i}\le s_j$ or $s_j+t_{j,b_j}\le s_i$ must be satisfied. Your task is to help Brother D to minimize the total completion time of the vessels, i.e., to minimize $\sum_{i=1}^m(s_i+t_{i,b_i})$.
First line of the input is an integer $T$ indicating the number of test cases. For each case, the first line contains two integers $n$ and $m$ (for $1\le n\le 50, 1\le m\le 200$). Each of the following $m$ lines contains $n$ positive integers representing vessel i’s handling times $t_{i,1}, t_{i,2}, \dots, t_{i,n} (t_{i,j} \le 1000)$.
For each case, output one integer, the minimum total weighted completion time of all the vessels.
1
2
3
4
5
6
7
8
9
1
5 7
1 2 3 4 5
2 1 3 4 5
2 3 1 4 5
2 3 4 1 5
2 3 4 5 1
2 2 2 2 2
2 2 2 2 2
1
11
吴坎
菜鸡命题人背锅了…PDF Input 结尾处应该是 $t_{i,n}$ 不是 $t_{i,m}$,此问题间接导致我校一队没有 AK…
题目大意:$n$ 个卸货点,$m$ 条船,船 $i$ 在卸货点 $j$ 卸货的时间是 $t_{i,j}$,同一时刻一个卸货点只能有一条船,现在每条船都要卸货,要最小化 $\sum_{i=1}^m\left(s_i+t_{i,b_i}\right)$,其中船 $i$ 在 $s_i$ 时刻开始在 $b_i$ 卸货点卸货。
容易看出,$s_i$ 一定是若干个 $t_{a,b}$ 的和。因此直接考虑每个 $t_{a,b}$ 对答案的贡献,不难口可口可想出一个费用流模型:
如果你对这个建图方式仍然有疑问,可以看一下这个简单的样例。
1
2
3
4
1
1 2
7
10
对应的建图方式如下(图中边上的数据是「边长,容量」的格式,费用 = 边长 * 流量),容易看出这张图的最小费用是 24,也是最后的答案。
flowchart LR
S--"0,#infin;"-->Q1
Q1--0,1-->R1,1
Q1--0,1-->R1,2
R1,1-.7,1.->P1
R1,1--10,1-->P2
R1,2--14,1-->P1
R1,2-.20,1.->P2
P1--0,1-->T
P2--0,1-->T
由于 $\lvert R\rvert \le 50\times 200=10000$,上述方案时间复杂度难以接受。于是考虑在跑费用流的时候动态建图,一边增广一边扩 $R$ 集合的点,直到汇点的流量达到 $n$。由于每次找到的增广路一定符合 $S\to Q\to R\to P\to T$ 的形式,此时对于这条路上的 $Q$ 点就可以扩张新的一层 $R$ 点,与在完整图上跑费用流在效果上是等价的。
以下是我自己写的标程,用了单路增广的 PD,比 EK 算法快三倍左右(0.740s:0.219s)。由于本题动态扩图的特性,可能不太适合多路增广的费用流算法,感兴趣的同学可以自己尝试一下。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
#include <bits/stdc++.h>
using namespace std;
typedef int ll;
const ll INF = 1e9, M = 255, N = 63;
int t, n, m, tim[M][N];
struct Graph
{
struct Vertex
{
vector<int> o;
};
struct Edge
{
int first, second;
ll len, cap;
};
vector<Vertex> v;
vector<Edge> e;
Graph(int n) : v(n) {}
void add(const Edge &ed)
{
v[ed.first].o.push_back(e.size());
e.push_back(ed);
}
};
struct PrimalDual : Graph
{
ll flow, cost;
vector<ll> f;
PrimalDual(int n) : Graph(n) {}
void add(Edge ed)
{
Graph::add(ed);
swap(ed.first, ed.second), ed.cap = 0, ed.len *= -1;
Graph::add(ed);
}
void ask(int s, int t)
{
vector<int> p(v.size(), -1), now(n, 1);
vector<ll> d(v.size(), INF), h(v.size(), 0);
for (f.assign(e.size(), flow = cost = 0);;)
{
priority_queue<pair<ll, int>> q;
for (q.push(make_pair(d[s] = 0, s)); !q.empty();)
{
ll dis = -q.top().first;
int u = q.top().second;
if (q.pop(), d[u] < dis)
continue;
for (int i = 0, k, to; i < v[u].o.size(); ++i)
if (k = v[u].o[i], to = e[k].second,
e[k].cap > f[k] && d[to] > d[u] + e[k].len + h[u] - h[to])
{
d[to] = d[u] + e[k].len + h[u] - h[to], p[to] = k;
q.push(make_pair(-d[to], to));
}
}
if (d[t] == INF)
return;
for (int i = 0; i < d.size(); ++i)
if (d[i] != INF)
h[i] += d[i], d[i] = INF;
ll _f = INF;
for (int u = t; u != s; u = e[p[u]].first)
_f = min(_f, e[p[u]].cap - f[p[u]]);
for (int u = t; u != s; u = e[p[u]].first)
cost += _f * e[p[u]].len, f[p[u]] += _f, f[p[u] ^ 1] -= _f;
flow += _f;
for (int u = t; u != s; u = e[p[u]].first)
if (1 < u && u < n + 2)
{
int k = u - 2;
++now[k];
p.push_back(-1);
d.push_back(INF);
h.push_back(0);
v.push_back(Vertex());
add({u, v.size() - 1, 0, 1});
for (int i = 0; i < m; ++i)
add({v.size() - 1, 2 + n + i, now[k] * tim[i][k], 1});
f.resize(e.size(), 0);
break;
}
}
}
};
int main()
{
for (scanf("%d", &t); t--;)
{
scanf("%d%d", &n, &m);
PrimalDual g(2 + n + m + n);
for (int j = 0; j < n; ++j)
{
g.add({0, 2 + j, 0, INF});
g.add({2 + j, 2 + n + m + j, 0, 1});
}
for (int i = 0; i < m; ++i)
{
for (int j = 0; j < n; ++j)
{
scanf("%d", &tim[i][j]);
g.add({2 + n + m + j, 2 + n + i, tim[i][j], 1});
}
g.add({2 + n + i, 1, 0, 1});
}
g.ask(0, 1);
printf("%d\n", g.cost);
}
}
D 哥带带我
Related posts