300iq Contest 1

A 一般图匹配
B Best Subsequence
双向链表+优先队列


const int maxn = 1e5 + 10;
int F[maxn];

int Find(int x) {
	return x == F[x] ? x : F[x] = Find(F[x]);
}
int main(void)
{
	int n, m; cin >> n >> m;
	for (int i = 1; i <= n; ++i) F[i] = i;
	for (int i = 1; i <= m; ++i) {
		int u, v;
		scanf("%d%d", &u, &v);
		int xx = Find(u);
		int yy = Find(v);
		F[xx] = F[yy];
	}
	int cnt = 0;
	for (int i = 1; i <= n; ++i)
		if (i == F[i])
			cnt++;

	cout << m - (n - cnt) << endl;

	return 0;
}

C Cool Pairs
题意:给定两个1…n的排列,要求求出两个数列A,B ,-n<=A[i],B[i]<=+n,并且 A[a[i]] <= A[a[i+1]],B[b[i]] <= B[b[i+1]],并且 a[i] + b[j] < 0 (i 分析:
突破点1:排列,与排列相对应的就是位置,这一题也不能免俗,当正着想无处下手的时候,反其道而行之,往往能取到奇效,我们不能从一个挨一个位置放,按照排列放
突破点2:正负,构造问题往往需要一个一个结论,这个结论一锤定音,决定问题的解决方法,这个题目暗示非常明显,包括 a[i]+b[j] < 0 和 数字的范围,那我们A数组放负数,B数组放正数,方案肯定够
不存在不满足条件的情况
突破点3:特殊情况,k = 0,构造的时候,极值,边界值,都是可以作为初始化的值的,我们将 A[a[i]],初始化为i-n-1,如果k = n*(n-1)/2 那么B全取0 ,k = 0, B全取n,中间的值怎么考虑呢,
考虑插入第x位,如果第x位放0,那么它的贡献是x-1,如果k>=x-1,我们就一直放0,当0 完毕

#include 
using namespace std;
/*

*/
const int maxn = 3e5 + 10;
int a[maxn], b[maxn], A[maxn], B[maxn];
int main(void)
{
	int n; long long  k;
	cin >> n >> k;
	for (int i = 1; i <= n; ++i) {
		scanf("%d", &a[i]); A[a[i]] = i - n - 1; B[i] = n;
	}
	for (int i = 1; i <= n; ++i) {
		scanf("%d", &b[i]);
		if (k >= b[i] - 1) {
			B[b[i]] = 0; k -= b[i] - 1; continue;
		}
		else {
			for (int j = 1; j < b[i]; ++j)
				a[j] = A[j];
			sort(a + 1, a + b[i]);
			B[b[i]] = -a[k + 1];
			break;
		}
	}
	cout << "Yes" << endl;
	for (int i = 1; i <= n; ++i) printf("%d ", A[i]); puts("");
	for (int i = 1; i <= n; ++i) printf("%d ", B[i]); puts("");

	return 0;
}

D Dates
霍尔定理
E Expected Value
国家集训队论文2019 wxh
F Free Edges
m − ( n − c n t ) m-(n-cnt) m(ncnt) c n t cnt cnt是连通块个数
H Hall’s Theorem
一开始不连边的时候 k = 2 n − 1 k = 2^n-1 k=2n1,将左边1号点与右边1号点相连,集合 < ( 1 ) , ( 1 ) > <(1),(1)> <(1),(1)>不再符合条件,将左边1号点与右边2号点相连 < ( 1 , x ) , ( 1 , 2 > ( x = 2 , . . . n ) <(1,x),(1,2> (x = 2,...n) <(1,x),(1,2>(x=2,...n)这些集合不再符合条件,这样不停的加点,每次减少的是 C ( n − 1 , j − 1 ) C(n-1,j-1) C(n1,j1),考虑将2号点与左边1号点相连,那么 < ( 2 ) , ( 1 ) > <(2),(1)> <(2),(1)>不再满足条件,与2号点相连 < ( 2 , x ) , ( 2 , 2 ) > ( x = 3 , . . . n ) <(2,x),(2,2)> (x=3,...n) <(2,x),(2,2)>(x=3,...n)不再符合条件,每次减少的是 C ( n − 2 , j − 1 ) C(n-2,j-1) C(n2,j1),这样算法就出来了

const int maxn = 21;
int A[1 << maxn], B[1 << maxn];
int C[maxn][maxn];
vector<P> vec;
int main(void)
{
	int n, k; cin >> n >> k;
	k = (1 << n) - 1 - k;
	C[0][0] = 1;
	for (int i = 1; i < maxn; ++i) {
		C[i][0] = 1;
		for (int j = 1; j <= i; ++j)
			C[i][j] = C[i - 1][j - 1] + C[i - 1][j];
	}
	for (int i = n - 1; i >= 0; --i) {
		for (int j = 0; j <= i; ++j) {
			if (k >= C[i][j]) {
				k -= C[i][j], vec.Pb(P(n - i, j + 1));
			}
			else break;
		}
	}
	cout << vec.size() << endl;
	for (auto p : vec)
		printf("%d %d\n", p.first, p.second);

	return 0;
}

你可能感兴趣的:(比赛总结)