tju3243 Blocked Road

本文介绍了一种利用树状数组解决环形岛屿连通性问题的方法,通过输入输出实例详细解释了算法实现过程,并提供了解决方案的C++代码。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

There are N seaside villages on X island, numbered from 1 to NN roads are built to connect all of them, which are also numbered from 1 to N, and the road with number i connects the village i and i % N + 1. Sometimes, for some reasons, some roads are blocked, so some villages are not connected anymore. Now, you are assigned to write a program to offer dynamic information about the connectivity.

At first, all roads are not blocked. The input will tell you the road with number i are blocked or unblocked, or ask you if village i and j are connected. Here two villages are connected means we can reach another village from one via some unblocked road. BTW, all the roads are bidirectional.

Input

The first line of the input contains one integer T, which indicate the number of test cases. The very first line of each case contains two integers, N and MN (where 2 ≤ N ≤ 100000) is the total number of the villages, M (where 1 ≤ M ≤ 100000) is the number of queries. The next M lines each describe one query. For each line, the first integer (0 or 1) indicates the type of the query. If the first integer is 0, there will be another integer i followed, if the road i is blocked at present, it will be unblocked, and vice versa. If the query type is 1, there will be two more integers i and j followed, if the village i and j are connected at present, the answer is 1, otherwise it shall be 0.

Output

For each query of type 1, output its answer in a single line

Sample Input

1
5 10
1 2 5
0 4
1 4 5
0 2
1 3 4
1 1 3
0 1
0 2
1 2 4
1 2 5

Sample Output

1
1
1
0
1
0
一开始以为是并查集,后来想想不能实现,看了别人的思路,发现因为连接的道路很有规律,所以可以用树状数组来实现,这题主要是判断两个点是否是相连的,这里因为是环装,所以两点有两种可能的连接顺序,一种是从小的数到大的数,另一种是从大的数到小的数。
#include<iostream>
#include<stdio.h>
#include<string.h>
#include<math.h>
#include<vector>
#include<map>
#include<queue>
#include<stack>
#include<string>
#include<algorithm>
using namespace std;
int b[100005],n,zhi[100006];
int lowbit(int x){
	return x&(-x);
}
void update(int pos,int num)
{
	while(pos<=n){
		b[pos]+=num;pos+=lowbit(pos);
	}
}
int getsum(int pos)
{
	int num=0;
	while(pos>0){
		num+=b[pos];pos-=lowbit(pos);
	}
	return num;
}

int main()
{
	int m,i,j,T,a,c,d;
	scanf("%d",&T);
	while(T--)
	{
		scanf("%d%d",&n,&m);
		for(i=1;i<=n;i++){
			zhi[i]=1;
			b[i]=lowbit(i);
		}
		for(i=1;i<=m;i++){
			scanf("%d",&a);
			if(a==0){
				scanf("%d",&c);
				if(zhi[c]==1){update(c,-1);zhi[c]=0;}
				else {update(c,1);zhi[c]=1;}
			}
			else{
				scanf("%d%d",&c,&d);
				if(c>d)swap(c,d);
				if( getsum(d-1)-getsum(c-1)==d-c   ||   getsum(n)-getsum(d-1)+getsum(c-1)==c+n-d )printf("1\n");
				else printf("0\n");
			}
		}
	}
	return 0;
}



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值