2018-2019 ACM-ICPC, Asia Nanjing Regional Contest

Contest Info


[Practice Link](https://codeforc.es/gym/101981)
Solved A B C D E F G H I J K L M
7/13 O ! - O O - O ! O O O - Ø
  • O 在比赛中通过
  • Ø 赛后通过
  • ! 尝试了但是失败了
  • - 没有尝试

Solutions


A. Adrien and Austin

题意:
\(n\)个石头,标号为\(1, \cdots, n\),每一次可以移走连续的\([1, k]\)个石头,问先手必胜还是后手必胜。

思路:
先手必输的状态为:

  • \(n = 0\)
  • \(n \% 2 = 1, k = 1\)

为什么其他情况先手必胜?

代码:

#include <bits/stdc++.h>
using namespace std;
 
int main() {
	int n, k;
	char *fi = "Adrien";
	char *se = "Austin";
	while (scanf("%d%d", &n, &k) != EOF) {
		if (n == 0 || (n % 2 == 0 && k == 1)) {
			puts(se);
		} else {
			puts(fi);
		}
	}
	return 0;
}

D. Country Meow

题意:
三维里有\(n\)个点,找一个最小的球将所有点覆盖。

思路:
模拟退火求最小球覆盖。

代码:

#include <bits/stdc++.h>
using namespace std;
 
#define db double
#define N 110
const db eps = 1e-7;
int n;
struct node {
	db x, y, z;
	node() {
		x = y = z = 0;
	}
	void scan() {
		scanf("%lf%lf%lf", &x, &y, &z);
	}
}a[N];
db dis(node a, node b) {
	return sqrt(1.0 * (a.x - b.x) * (a.x - b.x) + 1.0 * (a.y - b.y) * (a.y - b.y) + 1.0 * (a.z - b.z) * (a.z - b.z));
}
 
db solve() {
	db step = 10000, ans = 1e30, mt;
	node c = node();
	int s = 1;
	while (step > eps) {
		for (int i = 1; i <= n; ++i) {
			if (dis(c, a[s]) < dis(c, a[i])) {
				s = i;
			}
		}
		mt = dis(c, a[s]);
		ans = min(ans, mt);
		c.x += (a[s].x - c.x) / mt * step;
		c.y += (a[s].y - c.y) / mt * step;
		c.z += (a[s].z - c.z) / mt * step;
		step *= 0.98;
	}
	return ans;
}
 
int main() {
	while (scanf("%d", &n) != EOF) {
		for (int i = 1; i <= n; ++i) {
			a[i].scan();
		}
		printf("%.16f\n", solve());
	}
	return 0;
}

E.Eva and Euro coins

题意:
有两个01串\(s、t\),问能够通过有限步的操作将\(s\)变成\(t\)
操作为:
每次选择连续\(k\)个相同的字符,将其翻转,\(0 \rightarrow 1, 1 \rightarrow 0\)

思路:
我们考虑连续的\(k\)个相同字符,那么它可以任意移动。
比如说,\(k = 4\)的时候:
00001 -> 11111 -> 10000

这三步操作可以理解为把\(1\)从最后一位移动了第一位。
那么既然可以随便移动的话,我们可以认为直接把它们移出去了,而剩下的字符的相对位置是不会变的。
所以将所有连续的\(k\)个字符移走,判断剩下的字符组成的字符串是否相同,即表示\(s\)能否变到\(t\)

代码:

#include <bits/stdc++.h>
using namespace std;
 
#define N 1000010
char s[N], t[N], sta[N];
int cnt[N], top;
int n, k;
 
void work(char *s) {
	memset(cnt, 0, sizeof cnt);
	top = 0; 
	for (int i = 1; i <= n; ++i) {
		if (top == 0) {
			sta[++top] = s[i];
			cnt[top] = 1;
		} else {
			if (sta[top] == s[i]) {
				cnt[top + 1] = cnt[top] + 1;
			} else {
				cnt[top + 1] = 1;
			}
			sta[++top] = s[i]; 
		}
		if (cnt[top] == k) {
			top -= k; 
		}
	}
	for (int i = 1; i <= top; ++i) {
		s[i] = sta[i]; 
	}
	s[top + 1] = 0;
}
 
int main() {
	while (scanf("%d%d", &n, &k) != EOF) {
		scanf("%s%s", s + 1, t + 1);
		work(s); work(t);
		puts(strcmp(s + 1, t + 1) == 0 ? "Yes" : "No");
	}
	return 0;
}

G.Pyramid

题意:
询问类似于这样的三角形中:

里面正三角形的个数是多少。

思路:
找了个规律,,答案是\({n + 3 \choose 4}\)

代码:

    #include <bits/stdc++.h>
    using namespace std;
     
    #define ll long long
    const ll p = 1e9 + 7;
    ll qmod(ll base, ll n) {
    	ll res = 1;
    	while (n) {
    		if (n & 1) {
    			res = res * base % p;
    		}
    		base = base * base % p;
    		n >>= 1;
    	}
    	return res;
    }
     
     
    int main() {
    	ll inv = qmod(24, p - 2);
    	int T; scanf("%d", &T);	
    	while (T--) {
    		int n; scanf("%d", &n);
    		printf("%lld\n", 1ll * n * (n + 1) % p * (n + 2) % p * (n + 3) % p * inv % p);
    	}
    	return 0;
    }

I.Magic Potion

题意:
\(n\)个士兵,\(m\)个怪兽,每个士兵初始的时候可以打一个怪兽,但是有\(k\)瓶药,每个士兵最多喝一瓶药,喝一瓶药可以多打一个怪兽。
士兵和怪兽之间有边,表示某个士兵可以打哪些怪兽,问最多打我的怪兽数量最多是多少?

思路:
网络流。
如果没有药,那么就是个简单的二分匹配。
但是这个和二分图-多重匹配有不一样。因为不知道把药给谁喝。
但是可以这么建图,跑网络流:

  • \(S\)到所有士兵加一条流量为\(1\)的边。
  • 所有士兵到它能打的怪兽加一条流量为\(1\)的边。
  • 所有怪兽到\(T\)加一条流量为\(1\)的边。
  • 再加一个点\(C\),和\(S\)相连,流量为\(k\)
  • \(C\)到所有士兵加一条流量为\(1\)的边。

代码:

#include <bits/stdc++.h>
using namespace std;
 
#define INF 0x3f3f3f3f
#define N 510
#define M 1000010
 
struct Edge {
	int to, flow, nxt;
	Edge() {}
	Edge(int to, int nxt, int flow) : to(to), nxt(nxt), flow(flow) {}
}edge[M << 2];
 
int head[N << 2], tot, dep[N << 2];
int S, T, n, m, k;
void init() {
	memset(head, -1, sizeof head);
	tot = 0 ;	
}
 
void add(int u, int v, int w, int rw = 0) {
	edge[tot] = Edge(v, head[u], w); head[u] = tot++;
	edge[tot] = Edge(u, head[v], rw); head[v] = tot++; 
}
 
bool BFS() {
	memset(dep, -1, sizeof dep);
	queue <int> q;
	q.push(S);
	dep[S] = 1;
	while (!q.empty()) {
		int u = q.front();
		q.pop();
		for (int i = head[u]; ~i; i = edge[i].nxt) {
			if (edge[i].flow && dep[edge[i].to] == -1) {
				dep[edge[i].to] = dep[u] + 1;
				q.push(edge[i].to);
			}
		}
	}
	return dep[T] >= 0;
}
 
int DFS(int u, int f) {
	if (u == T || f == 0) return f;
	int w, used = 0;
	for (int i = head[u]; ~i; i = edge[i].nxt) {
		if (edge[i].flow && dep[edge[i].to] == dep[u] + 1) {
			w = DFS(edge[i].to, min(f - used, edge[i].flow));
			edge[i].flow -= w;
			edge[i ^ 1].flow += w;
			used += w;
			if (used == f) return f;
		}
	}
	if (!used) dep[u] = -1;
	return used;
}
 
int Dicnic() {
	int ans = 0;
	while (BFS()) {
		ans += DFS(S, INF);
	}
	return ans;
}
 
int main() {
	while (scanf("%d%d%d", &n, &m, &k) != EOF) {
		init(); S = 0; T = n + m + 1; int C = n + m + 2;
		add(S, C, k);
		for (int i = 1; i <= n; ++i) {
			add(S, i, 1);
			add(C, i, 1);
		}
		for (int i = 1, sze, x; i <= n; ++i) {
			scanf("%d", &sze);
			while (sze--) {
				scanf("%d", &x);
				add(i, x + n, 1);
			}
		}
		for (int i = 1; i <= m; ++i) add(i + n, T, 1);
		printf("%d\n", Dicnic());
	}
	return 0;
}

J.Prime Game

题意:
给出一个序列\(a_1, \cdots, a_n\)
定义:

\[\begin{eqnarray*} mul(l, r) = \prod\limits_{i = l}^r a_i \end{eqnarray*} \]

\(fac(l, r)\)\(mul(l, r)\)中不同质因数的个数。
请计算:

\[\begin{eqnarray*} \sum\limits_{i = 1}^n \sum\limits_{j = i}^n fac(l, r) \end{eqnarray*} \]

思路:
筛出\(10^6\)每个数的质因子,然后从右往左扫。
考虑在知道\(fac(i, j)\)的答案,推出\(fac(i - 1, j)\)的答案。
那么对于\(a_{i - 1}\)中每个质因子,它产生贡献的区间是\([i - 1, nx - 1]\)\(nx\)表示这个质因子下一个出现的位置
那么倒着处理的时候顺带处理一下\(nx\)数组,再更新答案就好了。

代码:

#include <bits/stdc++.h>
using namespace std;
 
#define ll long long
#define N 1000010
int n, a[N], nx[N]; 
int prime[N], tot, check[N];
vector <vector<int>> vec;
void init() {
	memset(check, 0, sizeof check);
	tot = 0;
	for (int i = 2; i < N; ++i) {
		if (!check[i]) {
			prime[++tot] = i;
		}
		for (int j = 1; j <= tot; ++j) {
			if (1ll * i * prime[j] >= N) break;
			check[i * prime[j]] = 1;
			if (i % prime[j] == 0) break;
		}
	}
	vec.clear(); vec.resize(N);
	for (int i = 1; i <= tot; ++i) {
		for (int j = prime[i]; j < N; j += prime[i]) {
			vec[j].push_back(prime[i]);
		}
	}
}
 
int main() {
	init();
	while (scanf("%d", &n) != EOF) {
		for (int i = 1; i <= n; ++i) {
			scanf("%d", a + i);
		}
		for (int i = 1; i <= 1000000; ++i) {
			nx[i] = n + 1; 
		}
		ll res = 0, base = 0;		
		for (int i = n; i >= 1; --i) {
			for (auto it : vec[a[i]]) {
				base += (nx[it] - i);
				nx[it] = i;
			}
			res += base;
		}
		printf("%lld\n", res);
	}
	return 0;
}

K.Kangaroo Puzzle

题意:
在一个\(20 \times 20\)的地图上,\(1\)表示有袋鼠,\(0\)表示有障碍物,边界外和障碍物上不能走。
要求给出一个\(50000\)步以内的操作,每一步操作为'L', 'R', 'U', 'D', 表示所有袋鼠一起动的方向,如果某个袋鼠下一个地方是不能走的,那么它那一步会忽略,使得所有袋鼠都聚集在一起。

思路:
随机输出\(50000\)个字符就好了,为什么呢?

代码:

    #include <bits/stdc++.h>
    using namespace std;
     
    #define N 50
    char G[N][N];
    int n, m;
     
    int main() {
    	srand(time(NULL));
    	char *s = "LRUD";
    	while (scanf("%d%d", &n, &m) != EOF) {
    		for (int i = 1; i <= n; ++i) {
    			scanf("%*s");
    		}
    		for (int i = 1; i <= 50000; ++i) {
    			putchar(s[rand() % 4]);
    		}
    		puts("");
    	}
    	return 0;
    }

M.Mediocre String Problem

题意:
给出两个字符串\(S\)\(T\),问有多少个三元组\((i, j, k)\)满足以下要求:

  • \(1 \leq i \leq j \leq |s|\)
  • \(1 \leq k \leq |t|\)
  • \(j - i +1 > k\)
  • \(s_i \cdots s_j\)\(t_1 \cdots t_k\)拼起来是一个回文串。

思路:
首先我们解析以下这四个条件:

  • 相当于要从\(s\)串找一个子串,从\(t\)串中找一个前缀,并且\(s\)串中的子串长度要大于\(t\)串的子串长度
  • 并且满足\(s\)串的子串的长度和\(t\)串子串长度相同的那段前缀刚好是\(t\)串子串的翻转
  • 并且\(s\)串子串的剩余部分是个回文串
    那么我们把\(s\)串翻转,求每个后缀\(i\)\(t\)串的\(LCP\),那么这部分再翻转回去,肯定存在一个\(k\)使得这两部分回文。
    那么\(LCP\)的长度就是选择数量,那么左边还要再接一段回文串,用\(Manacher\)处理出每个端点结尾的回文串数量即可。

代码:

#include <bits/stdc++.h>
using namespace std;
#define ll long long
#define N 1000010
char s[N], t[N];

struct ExKMP {
	int Next[N];
	int extend[N];
	void get_Next(char *s) { 
		int lens = strlen(s + 1), p = 1, pos;
		Next[1] = lens;
		while (p + 1 <= lens && s[p] == s[p + 1]) ++p;
		Next[pos = 2] = p - 1;
		for (int i = 3; i <= lens; ++i) {
			int len = Next[i - pos + 1];
			if (len + i < p + 1) Next[i] = len;
			else {
				int j = max(p - i + 1, 0);
				while (i + j <= lens && s[j + 1] == s[i + j]) ++j;
				p = i + (Next[pos = i] = j) - 1; 
			}
		}
	}

	void work(char *s, char *t) {
		get_Next(t);
		int lens = strlen(s + 1), lent = strlen(t + 1), p = 1, pos;
		while (p <= lent && s[p] == t[p]) ++p;
		p = extend[pos = 1] = p - 1;
		for (int i = 2; i <= lens; ++i) {
			int len = Next[i - pos + 1];
			if (len + i < p + 1) extend[i] = len;
			else {
				int j = max(p - i + 1, 0);
				while (i + j <= lens && j <= lent && t[j + 1] == s[i + j]) ++j;
				p = i + (extend[pos = i] = j) - 1;
			}
		}
	}
}exkmp;

struct Manacher {
	char Ma[N << 1];
	int Mp[N << 1];
	int num[N << 1];
//字符串从0开始
	void work(char *s) {
		int l = 0, len = strlen(s);
		Ma[l++] = '$';
		Ma[l++] = '#';
		for (int i = 0; i < len; ++i) {
			Ma[l++] = s[i];
			Ma[l++] = '#';  
		}    
		Ma[l] = 0;
		int mx = 0, id = 0;
		for (int i = 0; i < l; ++i) {
			Mp[i] = mx > i ? min(Mp[2 * id - i], mx - i) : 1;
			while (Ma[i + Mp[i]] == Ma[i - Mp[i]]) Mp[i]++;
			if (i + Mp[i] > mx) {
				mx = i + Mp[i];
				id = i;
			}
		}
		for (int i = 0; i < l; ++i) num[i] = 0;
			for (int i = 2; i < l; ++i) {
				int r = i + Mp[i] - 1;
				++num[i];
				--num[r + 1];
			}
			for (int i = 2; i < l; ++i) num[i] += num[i - 1];
		}
}man;

int main() {
	while (scanf("%s%s", s + 1, t + 1) != EOF) {
		int lens = strlen(s + 1);
		reverse(s + 1, s + 1 + lens);
		exkmp.work(s, t);
		man.work(s + 1);
		ll res = 0;
		for (int i = 2; i <= lens; ++i) {
			res += 1ll * (man.num[2 * (i - 1)]) * exkmp.extend[i];
		}
		printf("%lld\n", res);
	}
	return 0;
}   
posted @ 2019-07-03 08:34  Dup4  阅读(757)  评论(0编辑  收藏  举报