1. 程式人生 > >2017年第八屆“藍橋杯”國賽B組C/C++ 個人題解

2017年第八屆“藍橋杯”國賽B組C/C++ 個人題解

前言:

我參加了今年第八屆的藍橋杯國賽,只拿了個優秀獎,傷心。官方也沒有公佈試題和答案,在網上搜索了很久都沒有找到藍橋杯國賽的題目。突然有了一個不自量力的想法,趁還有一點記憶,把題目記錄下來,並且附上自己的做法。

第一題:36進位制

題意:

用類似16進位制的表示辦法,A表示10B表示11,……,Y表示25Z表示26,再加上09,就可以表示為36進位制。那麼請問MANY對應的十進位制數是多少?

程式碼:

#include<stdio.h>
int main()
{
	char ch[5] = "MANY";
	int ans = 0;
	for(int i = 0; i < 4; i++)
		ans = ans * 36 + (ch[i]-'A'+10);
	printf("%d\n", ans);
	return 0;
} 

參考答案:1040254

第二題:瓷磚樣式

題意:

有2種不同顏色規格為1*2的瓷磚,用其來鋪設地板,不能重疊和越界。並且,地板中任意2*2的格子不能為同一種顏色。如圖,當地板為2*3時,有10種鋪設方案。問:當地板為3*10時,問有多少種鋪設方案?

 

程式碼:

#include<stdio.h>
#include<string.h>
#define maxn 3
#define maxm 10
int a[maxn][maxm];
int ans;

bool check()
{
	for(int i = 0; i < maxn-1; i++)
		for(int j = 0; j < maxm-1; j++)
		{
			int p = a[i][j];
			if(p == -1) return false; //地板必須全部鋪滿
			if(p == a[i+1][j] && p == a[i][j+1] && p == a[i+1][j+1]) return false; //任意2*2格子不能為同一種顏色
		}
	return true;
}

void dfs(int cur) //準備鋪地板第cur格
{
	if(cur == maxn * maxm)
	{
		if(check())
		{
			ans++;
			/*
			for(int i = 0; i < maxn; i++)
			{
				for(int j = 0; j < maxm; j++) printf("%d", a[i][j]);
				printf("\n");
			}
			printf("-------\n");
			*/
		}
		return;
	}
	int x = (cur-1) / maxm;
	int y = cur - x * maxm -1;
	//格子(x,y)已經鋪有瓷磚
	if(a[x][y] != -1)
		dfs(cur+1);
	//橫著鋪
	if(y+1 < maxm && a[x][y] == -1 && a[x][y+1] == -1)
	{
		a[x][y] = a[x][y+1] = 0;
		dfs(cur+1);
		a[x][y] = a[x][y+1] = 1;
		dfs(cur+1);
		a[x][y] = a[x][y+1] = -1;
	}
	//豎著鋪
	if(x+1 < maxn && a[x][y] == -1 && a[x+1][y] == -1)
	{
		a[x][y] = a[x+1][y] = 0;
		dfs(cur+1);
		a[x][y] = a[x+1][y] = 1;
		dfs(cur+1);
		a[x][y] = a[x+1][y] = -1;
	}
}

int main()
{
	memset(a, -1, sizeof(a));
	ans = 0;
	dfs(1);
	printf("%d\n", ans);
	return 0;
}

參考答案:105760

第三題:希爾伯特曲線

題意:

有一個正方形,可以在上面作出一條希爾伯特曲線。左下角座標為(1,1),右下角座標為(2^n, 2^n)。該曲線在2^(n+1)*2^(n+1)上的正方形上的圖案,可以由2^n * 2^n的情況構造出來。構造方法:左上角和右上角保持不變,左下角順時針90度,右下角逆時針90度,然後各部分按照虛線相連即可。有n<30x,y<2^30。輸入np點座標(x, y),答案輸出曲線上該點覆蓋的格子的序號。注:格子的序號是從(1,1)開始沿著曲線所覆蓋的第幾個格子。

 

程式碼:

#include<stdio.h>
long long f(int n, int x, int y)
{
	if(n == 0) return 1;
	long long m = 1LL << (n-1);
	if(x <= m && y <= m)
		return f(n-1, y, x);
	if(x > m && y <= m)
		return 3LL * m * m + f(n-1,      ,2*m-x+1); //填空
	if(x <= m && y > m)
		return 1LL * m * m + f(n-1, x, y-m);
	if(x > m && y > m)
		return 2LL * m * m + f(n-1, x-m, y-m);
}
int main()
{
	int n, x, y;
	scanf("%d %d %d", &n, &x, &y);
	printf("%lld\n", f(n, x, y));
	return 0;
}

參考答案:m-y+1

【解釋】給出的程式碼是使用分治法,通過分析給出的4if語句可以判斷出依次是處理左下、右下、左上、右上,需要我們填空的部分是處理右下角的。分治法是將大問題變為相同性質的小問題,因此是要將右下角的圖形逆時針90度。

假設正方形是m*m,那麼順時針90度,就是將第1行變為第1列,第2行變為第2列,……第m行變為第m列。而逆時針90度,就是將第1行變為第m列,第2行變為第m-1列,……,第m行變為第1列。右下角的部分需要縱座標平移m個單位後,再逆時針90度。

順時針90度:(x, y) ==> (y, x)

逆時針90度:(x, y) ==> (m-y+1, m-x+1)

第四題:找環

題意:

編號為1nn個點,以及n-1條邊構成一棵樹。現在在樹上加上一條邊,這樣就構成了一個含環的圖了。請你找出該環上的結點,從小到大輸出這些結點編號。

測試資料:

30%資料:n<1000

100%資料:n<100000

輸入樣例:

5

1 2

2 5

4 2

1 3

5 3
輸出樣例:

1 2 3 5

【題解】

計算出每個點的度數。明顯度數為1的點不可能在環上,那麼與該點唯一相連的邊也就不在環上了,可以刪去。刪去這條邊後,這邊的另一個端點度數減去1,如果因此而導致它的度數變為1了,同樣說明該點不在環上,可以去掉,不斷重複以上操作,直到沒有邊可以刪去。

參考程式碼中用一個佇列來維護邊。由於點數n很大,無法使用鄰接矩陣,只能使用鄰接表來儲存圖,程式碼中使用vector容器來實現。

參考程式碼:

#include<stdio.h>
#include<string.h>
#include<vector>
#include<queue>
using namespace std;
#define maxn 100010
int d[maxn];
vector<int> e[maxn];
typedef struct EDGE
{
	int u, v;
	EDGE(int a=0, int b=0):u(a),v(b){}
}Edge;
queue<Edge> q;
int n;
void init()
{
	memset(d, 0, sizeof(d));
    scanf("%d", &n);
	int a, b;
	for(int i = 1; i <= n; i++)
	{
		scanf("%d %d", &a, &b);
		d[a]++;
		d[b]++;
		e[a].push_back(b);
		e[b].push_back(a);
		q.push(Edge(a,b));
	}
}
void out()
{
	bool flag = false;
	for(int i = 1; i <= n; i++)
		if(d[i] > 0)
		{
			if(!flag){ printf("%d", i); flag = true; }
			else
				printf(" %d", i);
		}
		putchar('\n');
}

void solve()
{
	while(!q.empty())
	{
		Edge tmp = q.front();  q.pop();
		if(d[tmp.u] == 1 || d[tmp.v] == 1) //其中一個端點的度為1時,刪去該邊
		{
			d[tmp.u]--;
			d[tmp.v]--;
			int p;
			if(d[tmp.u] == 1) 
			{
				p = tmp.u;
				int size = e[p].size();
				for(int i = 0; i < size; i++)
				{
					int t = e[p][i];
					if(d[t] > 0) q.push(Edge(p, t));
				}
			}
			if(d[tmp.v] == 1) 
			{
				p = tmp.u;
				int size = e[p].size();
				for(int i = 0; i < size; i++)
				{
					int t = e[p][i];
					if(d[t] > 0) q.push(Edge(p, t));
				}
			}
		}
	}
}

int main()
{
	init();
	solve();
	out();
	return 0;
}

第五題:線上匹配

題意:

有n個人在線上玩遊戲,每人的積分分別為Ai。如果線上的某兩個人的積分恰好相差為k時,他們就會被進行匹配。問線上最多會有多少人,他們任意兩人均沒法進行匹配?

測試資料:

30%資料:n<10

100%資料:n<100000Ai<100000k<100000

輸入樣例1

10 0

1 4 2 3 6 7 1 4 2 3

輸出樣例1

6

輸入樣例2

10 3

1 1 1 1 4 4 1 1 1 1

輸出樣例2

8

【題解】

方法一:

30%資料n<10,所以可以用深度搜索,每個數只有選和不選,得到一組待定的數列後,再檢查一下是否兩兩不能匹配,是則用當前數列的個數去嘗試更新答案。時間複雜度是O(n*2^n)

方法二:

將每個數值看作點,把可以匹配的兩個數連一條邊,那麼問題轉化為:刪去最少的點,使得剩下的點中不存在相連的邊。

對於每條邊,明顯兩個端點只能選一個,按照貪心思想,易知應該優先去掉度數多的那個點。

演算法步驟:

(1)構圖,將積分恰好相差k的兩個點相連;

(2)計算出所有點的度數;

(3)找出度數最大的那個點;

(4)把與該點相連的所有點的度數減去1,把該點的度數置為-1表示已經刪去該點;

(5)重複(4)(5),直到沒有邊可以刪去;

(6)這時剩餘的點的度數均為0,統計出其個數,就是最大不能兩兩匹配的個數。

時間複雜度為O(n^2)

n<100000,該方法仍然無法通過所有測試資料,希望有大神能夠提供更優秀的方法)

參考程式碼:

#include<stdio.h>
#include<string.h>
#include<vector>
using namespace std;
#define maxn 100010
vector<int> e[maxn];
int a[maxn]; //a[i]表示第i個人的積分
int d[maxn];  //d[i]表示第i個點的度數
bool exist[maxn]; //exist[i]=false表示第i個點刪去
int n, k;

void init()
{
	scanf("%d %d", &n, &k);
	for(int i = 1; i <= n; i++)
	{
		scanf("%d", &a[i]);
	}
}

void out()
{
	int ans = 0;
	for(int i = 1; i <= n; i++)
		if(exist[i]) ans++;
	printf("%d\n", ans);
}
void solve()
{
	memset(d, 0, sizeof(d));
	int i, j;
	for(i = 1; i < n; i++)
		for(j = i+1; j <= n; j++)
		{
			if(a[i] - a[j] == k || a[j] - a[i] == k)
			{
				e[i].push_back(j);
				e[j].push_back(i);
				d[i]++;
				d[j]++;
			}
		}
	
	memset(exist, true, sizeof(exist));
	do{
		int maxv = 0, sign = -1;
		int i;
		for(i = 1; i <= n; i++)
			if(exist[i])
			{
				if(d[i] > maxv) maxv = d[sign = i];
			}
		if(maxv == 0) 
			break;
		else{
			int size = e[sign].size();
			for(i = 0; i < size; i++)
			{
				int p = e[sign][i];
				if(exist[p]) d[p]--;
			}
			exist[sign] = false;
			d[sign] = -1;
		}
	}while(1);
}

int main()
{
	init();
	solve();
	out();
	return 0;
}

第六題:觀光鐵路

       跳蚤國正在大力發展旅遊業,每個城市都被打造成了旅遊景點。
       許多跳蚤想去其他城市旅遊,但是由於跳得比較慢,它們的願望難以實現。這時,小C聽說有一種叫做火車的交通工具,在鐵路上跑得很快,便抓住了商機,創立了一家鐵路公司,向跳蚤國王請示在每兩個城市之間都修建鐵路。
      然而,由於小C不會扳道岔,火車到一個城市以後只能保證不原路返回,而會隨機等概率地駛向與這個城市有鐵路連線的另外一個城市。
      跳蚤國王向廣大居民徵求意見,結果跳蚤們不太滿意,因為這樣修建鐵路以後有可能只遊覽了3個城市(含出發的城市)以後就回來了,它們希望能多遊覽幾個城市。於是跳蚤國王要求小C提供一個方案,使得每隻跳蚤坐上火車後能多遊覽幾個城市才回來。
      小C提供了一種方案給跳蚤國王。跳蚤國王想知道這個方案中每個城市的居民旅遊的期望時間(設火車經過每段鐵路的時間都為1),請你來幫跳蚤國王。
【輸入格式】
輸入的第一行包含兩個正整數n、m,其中n表示城市的數量,m表示方案中的鐵路條數。
接下來m行,每行包含兩個正整數u、v,表示方案中城市u和城市v之間有一條鐵路。
保證方案中無重邊無自環,每兩個城市之間都能經過鐵路直接或間接到達,且火車由任意一條鐵路到任意一個城市以後一定有路可走。
【輸出格式】
輸出n行,第i行包含一個實數ti,表示方案中城市i的居民旅遊的期望時間。你應當輸出足夠多的小數位數,以保證輸出的值和真實值之間的絕對或相對誤差不超過1e-9。
【樣例輸入】
4 5
1 2
2 3
3 4
4 1
1 3
【樣例輸出】
3.333333333333
5.000000000000
3.333333333333
5.000000000000
【樣例輸入】
10 15
1 2
1 9
1 5
2 3
2 7
3 4
3 10
4 5
4 8
5 6
6 7
6 10
7 8
8 9
9 10
【樣例輸出】
10.000000000000
10.000000000000
10.000000000000
10.000000000000
10.000000000000
10.000000000000
10.000000000000
10.000000000000
10.000000000000
10.000000000000
【資料規模與約定】
對於10%的測試點,n <= 10;
對於20%的測試點,n <= 12;
對於50%的測試點,n <= 16;
對於70%的測試點,n <= 19;
對於100%的測試點,4 <= k <= n <= 21,1 <= u, v <= n。資料有梯度。
資源約定:
峰值記憶體消耗 < 256M
CPU消耗  < 2000ms

【題解】

這題我也沒有什麼思路,還沒想到什麼解決方法。