dfs加上各種減支:
- 如果所有stick的長度和(sum)不能被4整除直接輸出no
- 如果最長stick的長度已經大於(sum/4)直接輸出no
- dfs的時候已經排完三個邊直接return true,因為第四個邊一定排得出來
- 先將stick由大排到小,每次dfs的時候stick不要從第一個開始挑選,而是要從上一層dfs選到的stick的下一個開始選。
基本上第四點是關鍵,有跟沒有就是0.0x秒和TLE的差別。
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#include <cstdio> | |
#include <algorithm> | |
int N, M; | |
int length[25]; | |
bool dfs(int, int, int, int, bool []); | |
int main() | |
{ | |
scanf("%d", &N); | |
while (N--) { | |
scanf("%d", &M); | |
int sum = 0, Max = 0; | |
for (int i = 0; i < M; ++i) { | |
scanf("%d", &length[i]); | |
Max = std::max(Max, length[i]); | |
sum += length[i]; | |
} | |
if (sum%4 != 0 || Max > sum/4) {puts("no"); continue;} | |
std::sort(length, length+M, [](const int &a, const int &b){return a>b;}); | |
bool used[25] = {0}; | |
if (dfs(0, 0, 0, sum / 4, used)) puts("yes"); | |
else puts("no"); | |
} | |
} | |
// num_of_edge: 目前已經組好了幾個邊 | |
// sum: 現在在排的這個邊已經排了多長 | |
// start: 從第start個stick開始挑選 | |
// edge_length: 每邊的長度必須要排成edge_length | |
// used[]: used[i]==true 表示第i的stick已經被挑選 | |
bool dfs(int num_of_edge, int sum, int start, const int edge_length, bool used[]) | |
{ | |
if (num_of_edge == 3) return true; | |
if (sum == edge_length) | |
if (dfs(num_of_edge + 1, 0, 0, edge_length, used)) return true; | |
for (int i = start; i < M; ++i) { | |
if (!used[i] && sum + length[i] <= edge_length) { | |
used[i] = true; | |
if (dfs(num_of_edge, sum + length[i], i+1, edge_length, used)) | |
return true; | |
used[i] = false; | |
} | |
} | |
return false; | |
} |
沒有留言:
張貼留言