Moi dam cuoi

 avatar
quoc14
c_cpp
5 months ago
3.9 kB
4
Indexable
Backtrack
Level 4
Mời Đám Cưới
Mời đám cưới

Anh Uranus sắp tổ chức đám cưới, hôm nay anh muốn đi phát thiệp mời đến những người bạn trong team. Thấy Uranus đi mời cưới nên Tomoky giả vờ đi ra ngoài có việc để trốn. Uranus rất tức và quyết tâm tìm được Tomoky để mời.

Giả sử đường đi trong công ty tạo thành 1 đồ thị và, giữa hai điểm bất kỳ đều tồn tại đường đi trực tiếp hoặc gián tiếp thông qua một số điểm trung gian.

Do hỏi anh VenG nên anh Uranus biết trước điểm bắt đầu và điểm kết thúc trên đường đi của Tomoky, nhưng anh lại không biết Tomoky sẽ đi đường nào, do đó anh muốn tìm những điểm mà anh Tomoky bắt buộc phải đi qua trong hành trình của mình (trừ điểm đầu và điểm cuối)

Hãy giúp anh Uranus tìm số điểm bắt buộc phải đi qua trên đường đi của anh Tomoky.

Input

Dòng đầu tiên chứa số nguyên dương không lớn hơn 100 là số lượng các bộ dữ liệu. Các dòng tiếp theo chứa các bộ dữ liệu. Mỗi bộ dữ liệu gồm một nhóm dòng theo khuôn dạng: 

  Dòng 1 chứa 4 số nguyên N,M,u,v (u,v,N ≤ 100; M ≤ 1000). Trong đó N là số lượng đỉnh trên đồ thị. M là số đường đi.. u, v lần lượt là đỉnh bắt đầu và đỉnh kết thúc hành trình của anh Tomoky.

  M dòng  sau, mỗi dòng ghi hai  số  i,  j cách nhau một khoảng trống cho biết có đường nối trực tiếp giữa i với j (1≤i,j≤N).

Output

Với mỗi bộ dữ liệu, đưa ra màn hình một số nguyên là số lượng đỉnh bắt buộc phải đi qua khi đi từ u,v.

Example

Input:

2

5 7 1 3

1 2

2 4

2 5

3 1

3 2

4 3

5 4

4 5 1 4

1 2

1 3

2 3

2 4

3 4

Output:

2

0

0
3
2
0
1
Time: 0.00000 s.

5
10 16 1 10
1 2
1 3
2 3
2 7
3 4
3 6
4 1
5 3
5 4
6 5
6 10
7 8
8 10
9 5
9 6
10 9
10 16 10 1
1 2
1 3
2 3
2 7
3 4
3 6
4 1
5 3
5 4
6 5
6 10
7 8
8 10
9 5
9 6
10 9
10 14 1 10
1 2
1 3
2 4
3 4
4 5
5 6
5 7
5 8
5 9
6 7
6 9
7 8
8 10
9 10
10 14 5 10
1 2
1 3
2 4
3 4
4 5
5 6
5 7
5 8
5 9
6 7
6 9
7 8
8 10
9 10
10 14 1 5
1 2
1 3
2 4
3 4
4 5
5 6
5 7
5 8
5 9
6 7
6 9
7 8
8 10
9 10


#include <iostream>
#include <time.h>
using namespace std;

int oo = 2000000000;

int T, n, m, mp[101][101], vs[101], check[101];
int a, b;
int result;

int path[101];

void backtrack(int curr, int index){
	if(curr == b){
		for(int i = 1; i <= n; i++){
			if(!vs[i] || !check[i]) check[i] = 0;
		}
		return;
	}
	for(int choose = 1; choose <= n; choose++){
		if(vs[choose] == 0 && mp[curr][choose] == 1){
			vs[choose]++;
			path[index] = choose;
			backtrack(choose, index + 1);
			vs[choose]--;
		}
	}
}

int main(){

	freopen("input.txt", "r", stdin);

	// Calc clock
	clock_t time_start, time_end; 
	time_start = clock(); 
	
	cin >> T;
	for(int tc = 1; tc <= T; tc++){
		// Initial && Input
		cin >> n >> m;
		cin >> a >> b;
		for(int i = 1; i <= n; i++){
			for(int j = 1; j <= n; j++){
				mp[i][j] = 0;
			}
			vs[i] = 0, check[i] = 1;
		}
		for(int i = 0; i < m; i++){
			int u, v;
			cin >> u >> v;
			mp[u][v] = 1;
		}
		result = 0;
		check[a] = 0, check[b] = 0;

		// Solve Problem 
		vs[a]++;
		if(mp[a][b] != 1){
			path[0] = a;
			backtrack(a, 1);
			for(int i = 1; i <= n; i++){
				result += check[i];
			}
		}

		// Output
		cout << result << endl;
	}

	// Calc Time
	time_end = clock();
	cout.setf(ios::fixed);
	cout.precision(5);
	cout << "Time: " << double (time_end - time_start) / double (CLOCKS_PER_SEC) << " s." << endl;
	
	return 0;
 }
Leave a Comment