Codeforces Global Round 5 C2. Balanced Removals (Harder) (并查集)

题目链接

题意:三维空间中有偶数个点,每次你可以选择两个点将其删除,但是需要保证以两点为端点的长方体不能包含未删除的点。问怎样可以将所有点删除。

题解:对x,y,z排序,先x, y相同的点删掉,再x相同的删掉,再把剩下的点删掉。这样保证每次删除的点对中间不会存在别的点。对于c2,因为n比较大,所以我用了并查集维护每个位置后第一个为空的位置。

 

c1:

#include<bits/stdc++.h>
using namespace std;
const int N = 3100;
int vis[N];
struct node{
	int x, y, z, id;
	bool friend operator < (node a, node b){
		if(a.x != b.x) return a.x > b.x;
		if(a.y != b.y) return a.y > b.y;
		return a.z > b.z;
	}
}a[N];
vector<int>ans[N];
int main()
{
	int n, cnt = 0;
	scanf("%d", &n);
	for(int i = 1; i <= n; ++i)
		scanf("%d%d%d", &a[i].x, &a[i].y, &a[i].z), a[i].id = i;
	sort(a + 1, a + 1 + n);
	for(int i = 1; i < n; ++i){
		if(vis[i]) continue;
		if(a[i].x == a[i + 1].x && a[i].y == a[i + 1].y){
			vis[i] = vis[i + 1] = 1;
			++cnt;
			ans[cnt].push_back(a[i].id);
			ans[cnt].push_back(a[i + 1].id);
		}
	}
	for(int i = 1; i < n; ++i){
		if(vis[i]) continue;
		for(int j = i + 1; j <= n; ++j)
		if(!vis[j]){
			if(a[i].x == a[j].x){
				vis[i] = vis[j] = 1;
				++cnt;
				ans[cnt].push_back(a[i].id);
				ans[cnt].push_back(a[j].id);
			}
			break;
		}
	}
	for(int i = 1; i < n; ++i){
		if(vis[i]) continue;
		for(int j = i + 1; j <= n; ++j)
		if(!vis[j]){
			vis[i] = vis[j] = 1;
			++cnt;
			ans[cnt].push_back(a[i].id);
			ans[cnt].push_back(a[j].id);
			break;
		}
	}
	for(int i = 1; i <= cnt; ++i){
		for(auto t : ans[i]) printf("%d ", t);
		puts("");
	}
}

c2:

#include<bits/stdc++.h>
using namespace std;
const int N = 5e4 + 100;
int vis[N], pre[N];
struct node{
	int x, y, z, id;
	bool friend operator < (node a, node b){
		if(a.x != b.x) return a.x > b.x;
		if(a.y != b.y) return a.y > b.y;
		return a.z > b.z;
	}
}a[N];
vector<int>ans[N];
 
inline void find(int &x){
	while(x!=pre[x])
		x=pre[x]=pre[pre[x]];
}
 
int main()
{
	int n, cnt = 0;
	scanf("%d", &n);
	
	for(int i = 1; i <= n; ++i) 
	pre[i] = i;
	
	for(int i = 1; i <= n; ++i)
		scanf("%d%d%d", &a[i].x, &a[i].y, &a[i].z), a[i].id = i;
	sort(a + 1, a + 1 + n);
	for(int i = 1; i < n; ++i){
		if(vis[i]) continue;
		if(a[i].x == a[i + 1].x && a[i].y == a[i + 1].y){
			pre[i] = i + 2;
			pre[i + 1] = i + 2;
			vis[i] = vis[i + 1] = 1;
			++cnt;
			ans[cnt].push_back(a[i].id);
			ans[cnt].push_back(a[i + 1].id);
		}
	}
	for(int i = 1; i < n; ++i){
		if(vis[i]) continue;
		int t1 = i;
		find(t1);
		if(t1 == i) {
			t1 += 1;
			find(t1);
		}
		if(a[i].x == a[t1].x){
			++cnt;
			ans[cnt].push_back(a[i].id);
			ans[cnt].push_back(a[t1].id);
			vis[i] = vis[t1] = 1;
			
			int t2 = (t1 + 1);
			find(t2);
			pre[i] = t2;
			pre[t1] = t2;
		}
	}
	for(int i = 1; i < n; ++i){
		if(vis[i]) continue;
		int t1 = i;
		find(t1);
		if(t1 == i) {
			t1 += 1;
			find(t1);
		}
		++cnt;
		ans[cnt].push_back(a[i].id);
		ans[cnt].push_back(a[t1].id);
		vis[i] = vis[t1] = 1;
		
		int t2 = (t1 + 1);
		find(t2);
		pre[i] = t2;
		pre[t1] = t2;
	}
	for(int i = 1; i <= cnt; ++i){
		for(auto t : ans[i]) printf("%d ", t);
		puts("");
	}
}

 

 

根据提供的引用内容,Codeforces Round 511 (Div. 1)是一个比赛的名称。然而,引用内容中没有提供与这个比赛相关的具体信息或问题。因此,我无法回答关于Codeforces Round 511 (Div. 1)的问题。如果您有关于这个比赛的具体问题,请提供更多的信息,我将尽力回答。 #### 引用[.reference_title] - *1* [Codeforces Round 860 (Div. 2)题解](https://blog.csdn.net/qq_60653991/article/details/129802687)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [Codeforces Round 867 (Div. 3)(A题到E题)](https://blog.csdn.net/wdgkd/article/details/130370975)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [Codeforces Round 872 (Div. 2)(前三道](https://blog.csdn.net/qq_68286180/article/details/130570952)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值