题目
{% fold 点击显/隐题目 %}
题解
下意识写了匈牙利算法,不过这道题会爆时间
再仔细看一下题目
可以发现其实求树上父节点和子节点配对能配多少对
可以用动态规划求解
用dp[t][0]
表示编号为t
的节点往下并且不选取节点t
能达到的最多对数
用dp[t][1]
表示编号为t
的节点往下并且选取节点t
能达到的最多对数
很容易就可以发现 dp[t][0] = sum{ max(dp[child][0], dp[child][1])) }
而 dp[t][1]
则根据是否存在没有选取的子节点(上一步有使用dp[child][0]
)有dp[t][1] = dp[t][0]
和 dp[t][1] = dp[t][0] + 1
代码
{% fold 点击显/隐代码 %}```cpp 组队分配 https://github.com/OhYee/sourcecode/tree/master/ACM 代码备份
#include
#include
#include
using namespace std;
const int maxn = 200005;
vector
int n;
int parent[maxn];
int dp[maxn][2];
void init() {
memset(parent, 0, (n + 5) * sizeof(int));
for (int i = 0; i <= n; ++i)
e[i].clear();
}
void addEdge(int u, int v) {
e[u].push_back(v);
e[v].push_back(u);
}
void dfs(int t) {
// printf("dfs(%d)\n", t);
dp[t][0] = dp[t][1] = 0;
bool emptyChild = false;
for (auto child : e[t]) {
if (parent[t] != child) {
parent[child] = t;
dfs(child);
if (dp[child][0] < dp[child][1]) {
dp[t][0] += dp[child][1];
} else {
emptyChild = true;
dp[t][0] += dp[child][0];
}
}
}
dp[t][1] = dp[t][0] + emptyChild;
// printf("dp[%d]={%d,%d}\n", t, dp[t][0], dp[t][1]);
}
int main() {
while (~scanf("%d", &n)) {
init();
for (int i = 1; i < n; ++i) {
int a, b;
scanf("%d%d", &a, &b);
addEdge(a, b);
}
dfs(1);
printf("%d\n", max(dp[1][0], dp[1][1]));
}
return 0;
}
{% endfold %}