Gravatar
Lfc_HeSn
积分:1358
提交:234 / 564

本题三种做法。

第一种:直接跑一遍dfs求最大值,期望得分50分。

第二种:折半搜索,map统计去重,最后求最大值。


#include<bits/stdc++.h>
using namespace std;
#define int long long
int n, m, half, a[50], b[50], num1[1 << 21], num2[1 << 21], cnt1, cnt2, ans;
map<int, int> mp, mp2;
int dfs2(int x, int s, int p) {
	if(x == n + 1) {
		if(!mp2[p]) {
			num2[++ cnt2] = p;
		}
		if(s > mp2[p]) {
			mp2[p] = s;
		}
		return 0;
	}
	dfs2(x + 1, s, p);
	dfs2(x + 1, s + b[x], p ^ a[x]);
}
int dfs(int x, int s, int p) {
	if(x == half + 1) {
		if(!mp[p]) {
			num1[++ cnt1] = p;
		}
		if(s > mp[p]) {
			mp[p] = s;
		}
		return 0;
	}
	dfs(x + 1, s, p);
	dfs(x + 1, s + b[x], p ^ a[x]);
	return 0;
}
signed main() {
    freopen("outsci.in", "r", stdin);
    freopen("outsci.out", "w", stdout);
    cin >> n >> m;
    half = n / 2;
    for(int i = 1; i <= n; i ++) {
    	cin >> a[i];
	}
	for(int i = 1; i <= n; i ++) {
		cin >> b[i];
	}
	dfs(1, 0, 0);
//	for(int i = 1; i <= cnt; i ++) {
//		cout << c[i] << ' ' << mp[c[i]] << endl;
//	}
	dfs2(half + 1, 0, 0);
	for(int i = 1; i <= cnt1; i ++) {
		for(int j = 1; j <= cnt2; j ++) {
			if((num1[i] ^ num2[j]) <= m) {
				ans = max(ans, mp[num1[i]] + mp2[num2[j]]);
			}
		}
	}
	cout << ans << endl;
    return 0;
}

第三种 正解(伪):所有大于零的b求和(

这数据是拿脚捏的吗?这都能水过???

真正的正解还没想到QAQ


题目3790  界外科学 WWWWWWWWWWWW      9      评论
2022-11-06 21:22:10