2025.05.17淘天机考笔试真题第三题

📌 点击直达笔试专栏 👉《大厂笔试突围》

💻 春秋招笔试突围在线OJ 👉 笔试突围OJ

03. 奇偶平衡树分割问题

问题描述

K小姐是一位园林设计师,她设计了一个由多个花坛组成的树形公园。每个花坛中种植了不同数量的花,数量由整数表示。K小姐发现,当公园被分割成多个独立区域时,如果每个区域中的花朵总数都是偶数,游客会感到更加和谐。

现在,K小姐想要通过关闭若干条连接花坛的小路(即删除树的边),将公园分割成若干个独立区域(连通块),使得每个区域内的花朵总数都是偶数。

请你求出,对于每个 k ( 1 ≤ k ≤ n − 1 ) k (1 \leq k \leq n-1) k(1kn1),关闭 k k k 条小路后得到的 k + 1 k+1 k+1 个独立区域满足条件的方案数。如果不存在满足条件的方案,对应的答案记为 0 0 0

注意:两种关闭小路的方案若关闭的路径集合不同,则视为不同的方案。

输入格式

第一行给出一个整数 n n n,表示花坛的数量。

第二行给出 n n n 个整数 W 1 , W 2 , . . . , W n W_1, W_2, ..., W_n W1,W2,...,Wn,其中 W i W_i Wi 表示第 i i i 个花坛中的花朵数量。

接下来 n − 1 n-1 n1 行,每行包含两个整数 u u u v ( 1 ≤ u , v ≤ n , u ≠ v ) v (1 \leq u, v \leq n, u \neq v) v(1u,vn,u=v),表示花坛 u u u 与花坛 v v v 之间有一条小路,保证给定的图构成一棵树。

输出格式

输出 n − 1 n-1 n1 个数,第 i i i 个数表示关闭 i i i 条小路后满足条件的方案数。由于答案可能非常大,请对答案取模 1 0 9 + 7 10^9+7 109+7 后输出。

样例输入

5
1 2 3 4 4
1 2
1 3
2 4
2 5

样例输出

3 3 1 0
样例解释说明
样例1 k = 1 k=1 k=1 时,关闭方案有 {(1,2)}, {(2,4)}, {(2,5)},共 3 种。
k = 2 k=2 k=2 时,关闭方案有 {(1,2), (2,4)}, {(1,2), (2,5)}, {(2,5), (2,4)},共 3 种。
k = 3 k=3 k=3 时,关闭方案有 {(1,2), (2,4), (2,5)},共 1 种。
k = 4 k=4 k=4 时,没有满足条件的方案。

数据范围

  • 2 ≤ n ≤ 1 0 5 2 \leq n \leq 10^5 2n105
  • 1 ≤ W i ≤ 1 0 9 1 \leq W_i \leq 10^9 1Wi109
  • 1 ≤ u , v ≤ n 1 \leq u, v \leq n 1u,vn

题解

这道题乍看复杂,但仔细分析后会发现其中的数学规律和解决方案。

首先,我们需要理解什么情况下一个连通块的花朵数能够为偶数。显然,如果一个连通块内奇数花坛的个数为偶数,那么总花朵数一定是偶数(偶数+偶数=偶数,奇数+奇数=偶数)。

我们的目标是通过删除边,将树分成多个连通块,使得每个连通块内的奇数花坛数量都是偶数。那么问题转化为:找出那些删除后能让两侧奇数花坛数量都是偶数的边。

关键观察:一个连通块内奇数花坛数量为奇数时,无论如何分割,都无法使所有子连通块内的奇数花坛数量都为偶数。因为奇数个奇数,无论如何分组,总有一组含有奇数个奇数。

因此,如果整棵树中奇数花坛的总数为奇数,则无解。

接下来,我们定义一个边是"好边",如果删除这条边后,两个连通块内的奇数花坛数量都是偶数。一条边是好边当且仅当这条边连接的子树内奇数花坛数量为偶数。

如果好边数量为g,那么要删除k条边使所有连通块花朵数为偶数,就相当于从g条好边中选择k条,即组合数C(g,k)。

具体算法如下:

  1. 统计整棵树中奇数花坛的总数,如果为奇数,则无解。
  2. 通过DFS计算每个子树中奇数花坛的数量。
  3. 检查每条边,如果删除后两侧连通块中奇数花坛数量都是偶数,则该边为好边。
  4. 计算从g条好边中选k条的组合数C(g,k),即为答案。

时间复杂度分析:

  • DFS遍历树的复杂度为O(n)
  • 检查每条边的复杂度为O(n)
  • 计算组合数的复杂度为O(n)
    总体时间复杂度为O(n),空间复杂度为O(n)。

参考代码

  • Python
import sys
input = lambda: sys.stdin.readline().strip()def solve():# 读取输入n = int(input())w = [0] + list(map(int, input().split()))adj = [[] for _ in range(n+1)]for _ in range(n-1):u, v = map(int, input().split())adj[u].append(v)adj[v].append(u)# 计算奇数花坛的总数odd_count = sum(1 for val in w[1:] if val % 2 == 1)# 如果奇数花坛总数为奇数,则无解if odd_count % 2 == 1:print(" ".join(["0"] * (n-1)))return# 计算子树中奇数花坛的数量t = [0] * (n+1)good_edges = 0# DFS计算子树信息def dfs(node, parent):nonlocal good_edgesis_odd = w[node] % 2  # 当前节点是否为奇数for child in adj[node]:if child != parent:# 递归计算子树dfs(child, node)# 更新当前节点的奇数计数is_odd ^= t[child]t[node] = is_odd# 检查是否为好边if parent != 0 and t[node] == 0:good_edges += 1# 从根节点开始DFSdfs(1, 0)# 计算组合数MOD = 10**9 + 7# 预计算阶乘和逆元fact = [1] * (n+1)inv_fact = [1] * (n+1)for i in range(1, n+1):fact[i] = (fact[i-1] * i) % MOD# 计算逆元(使用费马小定理)def pow_mod(x, p):res = 1while p > 0:if p & 1:res = (res * x) % MODx = (x * x) % MODp >>= 1return resinv_fact[n] = pow_mod(fact[n], MOD - 2)for i in range(n, 0, -1):inv_fact[i-1] = (inv_fact[i] * i) % MOD# 计算组合数C(n,k)def comb(n, k):if k < 0 or k > n:return 0return (fact[n] * inv_fact[k] % MOD * inv_fact[n-k] % MOD)# 计算并输出结果result = []for k in range(1, n):ans = comb(good_edges, k)result.append(str(ans))print(" ".join(result))if __name__ == "__main__":solve()
  • Cpp
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int MOD = 1e9 + 7;// 快速幂计算a^b mod MOD
ll pow_mod(ll a, ll b) {ll res = 1;while (b > 0) {if (b & 1) res = (res * a) % MOD;a = (a * a) % MOD;b >>= 1;}return res;
}int main() {ios::sync_with_stdio(false);cin.tie(nullptr);int n;cin >> n;// 读取花坛权值vector<int> w(n+1);for (int i = 1; i <= n; i++) cin >> w[i];// 构建树vector<vector<int>> adj(n+1);for (int i = 0; i < n-1; i++) {int u, v;cin >> u >> v;adj[u].push_back(v);adj[v].push_back(u);}// 计算奇数花坛总数int tot_odd = 0;for (int i = 1; i <= n; i++)tot_odd += (w[i] & 1);// 如果奇数总数为奇数,无解if (tot_odd & 1) {for (int k = 1; k <= n-1; k++)cout << "0" << (k == n-1 ? '\n' : ' ');return 0;}// 存储子树奇数数量vector<int> t(n+1, 0);int good = 0; // 好边数量// DFS计算子树信息vector<bool> vis(n+1, false);function<void(int, int)> dfs = [&](int u, int p) {int odd = w[u] & 1; // 当前节点是否为奇数for (int v : adj[u]) {if (v != p) {dfs(v, u);odd ^= t[v]; // 更新奇数计数}}t[u] = odd;// 检查是否为好边if (p != 0 && t[u] == 0)good++;};dfs(1, 0);// 预计算阶乘和逆元vector<ll> fact(n+1, 1), inv_fact(n+1, 1);for (int i = 1; i <= n; i++)fact[i] = (fact[i-1] * i) % MOD;inv_fact[n] = pow_mod(fact[n], MOD-2); // 费马小定理for (int i = n; i >= 1; i--)inv_fact[i-1] = (inv_fact[i] * i) % MOD;// 组合数计算函数auto comb = [&](int n, int k) -> ll {if (k < 0 || k > n) return 0;return (fact[n] * inv_fact[k] % MOD * inv_fact[n-k] % MOD);};// 输出结果for (int k = 1; k <= n-1; k++) {ll res = comb(good, k);cout << res << (k == n-1 ? '\n' : ' ');}return 0;
}
  • Java
import java.io.*;
import java.util.*;public class Main {static final int MOD = (int)1e9 + 7;public static void main(String[] args) throws IOException {BufferedReader br = new BufferedReader(new InputStreamReader(System.in));// 读取花坛数量int n = Integer.parseInt(br.readLine().trim());// 读取每个花坛的花朵数String[] vals = br.readLine().trim().split(" ");int[] w = new int[n+1];for (int i = 1; i <= n; i++) {w[i] = Integer.parseInt(vals[i-1]);}// 构建树List<Integer>[] adj = new ArrayList[n+1];for (int i = 0; i <= n; i++) {adj[i] = new ArrayList<>();}for (int i = 0; i < n-1; i++) {String[] edge = br.readLine().trim().split(" ");int u = Integer.parseInt(edge[0]);int v = Integer.parseInt(edge[1]);adj[u].add(v);adj[v].add(u);}// 计算奇数花坛总数int oddCount = 0;for (int i = 1; i <= n; i++) {if (w[i] % 2 == 1) {oddCount++;}}// 如果奇数总数为奇数,无解if (oddCount % 2 == 1) {StringBuilder sb = new StringBuilder();for (int k = 1; k <= n-1; k++) {sb.append("0");if (k < n-1) sb.append(" ");}System.out.println(sb.toString());return;}// 存储子树奇数数量和好边数量int[] t = new int[n+1];int[] good = new int[1]; // 用数组便于在DFS中修改// DFS计算子树信息dfs(1, 0, w, adj, t, good);// 预计算阶乘和逆元long[] fact = new long[n+1];long[] invFact = new long[n+1];fact[0] = 1;for (int i = 1; i <= n; i++) {fact[i] = (fact[i-1] * i) % MOD;}invFact[n] = powMod(fact[n], MOD-2);for (int i = n; i > 0; i--) {invFact[i-1] = (invFact[i] * i) % MOD;}// 输出结果StringBuilder result = new StringBuilder();for (int k = 1; k <= n-1; k++) {long ans = combination(good[0], k, fact, invFact);result.append(ans);if (k < n-1) result.append(" ");}System.out.println(result.toString());}// DFS计算子树信息static void dfs(int node, int parent, int[] w, List<Integer>[] adj, int[] t, int[] good) {int isOdd = w[node] % 2; // 当前节点是否为奇数for (int child : adj[node]) {if (child != parent) {dfs(child, node, w, adj, t, good);isOdd ^= t[child]; // 更新奇数计数}}t[node] = isOdd;// 检查是否为好边if (parent != 0 && t[node] == 0) {good[0]++;}}// 快速幂计算static long powMod(long a, int b) {long res = 1;while (b > 0) {if ((b & 1) == 1) res = (res * a) % MOD;a = (a * a) % MOD;b >>= 1;}return res;}// 计算组合数C(n,k)static long combination(int n, int k, long[] fact, long[] invFact) {if (k < 0 || k > n) return 0;return (fact[n] * invFact[k] % MOD * invFact[n-k] % MOD);}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/80313.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

第三十五节:特征检测与描述-ORB 特征

1. 引言:为什么需要ORB? 在计算机视觉领域,特征检测与描述是许多任务(如图像匹配、目标跟踪、三维重建等)的核心基础。传统的算法如SIFT(尺度不变特征变换)和SURF(加速稳健特征)因其优异的性能被广泛应用,但它们存在两个显著问题: 专利限制:SIFT和SURF受专利保护,…

深入解读WPDRRC信息安全模型:构建中国特色的信息安全防护体系

目录 前言1 WPDRRC模型概述2 模型结构详解2.1 预警&#xff08;Warning&#xff09;2.2 保护&#xff08;Protect&#xff09;2.3 检测&#xff08;Detect&#xff09;2.4 响应&#xff08;React&#xff09;2.5 恢复&#xff08;Restore&#xff09;2.6 反击&#xff08;Count…

《算法导论(第4版)》阅读笔记:p82-p82

《算法导论(第4版)》学习第 17 天&#xff0c;p82-p82 总结&#xff0c;总计 1 页。 一、技术总结 1. Matrix Matrices(矩阵) (1)教材 因为第 4 章涉及到矩阵&#xff0c;矩阵属于线性代数(linear algebra)范畴&#xff0c;如果不熟悉&#xff0c;可以看一下作者推荐的两本…

基于Spring Boot和Vue的在线考试系统架构设计与实现(源码+论文+部署讲解等)

源码项目获取联系 请文末卡片dd我获取更详细的演示视频 系统介绍 基于Spring Boot和Vue的在线考试系统。为学生和教师/管理员提供一个高效、便捷的在线学习、考试及管理平台。系统采用前后端分离的架构&#xff0c;后端基于成熟稳定的Spring Boot框架&#xff0c;负责数据处理…

Codeforces Round 1024 (Div.2)

比赛链接&#xff1a;CF1024 A. Dinner Time 只有当 n n n 是 p p p 的倍数而且 n ⋅ q p ̸ m \frac{n \cdot q}{p} \not m pn⋅q​m 时输出 NO&#xff0c;其余情况均满足条件。 时间复杂度&#xff1a; O ( 1 ) O(1) O(1)。 #include <bits/stdc.h> using na…

【LeetCode 热题 100】二叉树的最大深度 / 翻转二叉树 / 二叉树的直径 / 验证二叉搜索树

⭐️个人主页&#xff1a;小羊 ⭐️所属专栏&#xff1a;LeetCode 热题 100 很荣幸您能阅读我的文章&#xff0c;诚请评论指点&#xff0c;欢迎欢迎 ~ 目录 二叉树的中序遍历二叉树的最大深度翻转二叉树对称二叉树二叉树的直径二叉树的层序遍历将有序数组转换为二叉搜索树验…

Tomcat发布websocket

一、tomcal的lib放入文件 tomcat-websocket.jar websocket-api.jar 二、代码示例 package com.test.ws;import com.test.core.json.Jmode;import javax.websocket.*; import javax.websocket.server.ServerEndpoint; import java.util.concurrent.CopyOnWriteArraySet; imp…

LLM笔记(二)LLM数据基础-分词算法(2)

文章目录 1. 分词算法概述1.1 基于词典的&#xff08;或基于规则的&#xff09;分词算法1.2 基于统计的&#xff08;或基于机器学习的&#xff09;分词算法1.3 基于深度学习的分词算法1.4 子词&#xff08;Subword&#xff09;分词算法1.5 混合分词算法1.6 针对不同语言的特点 …

Uniapp开发鸿蒙应用时如何运行和调试项目

经过前几天的分享&#xff0c;大家应该应该对uniapp开发鸿蒙应用的开发语法有了一定的了解&#xff0c;可以进行一些简单的应用开发&#xff0c;今天分享一下在使用uniapp开发鸿蒙应用时怎么运行到鸿蒙设备&#xff0c;并且在开发中怎么调试程序。 运行 Uniapp项目支持运行到…

数据湖与数据仓库融合:Hudi、Iceberg、Delta Lake 实践对比

在实时与离线一体化的今天,数据湖与数据仓库边界不断融合,越来越多企业选用如 Hudi、Iceberg、Delta Lake 等开源方案实现统一的数据存储、计算、分析平台。本篇将围绕以下关键点,展开实战对比与解决方案分享: ✅ 实时写入能力 ✅ ACID 保证 ✅ 增量数据处理能力 ✅ 流批一…

Python爬虫(29)Python爬虫高阶:动态页面处理与云原生部署全链路实践(Selenium、Scrapy、K8s)

目录 引言&#xff1a;动态爬虫的技术挑战与云原生机遇一、动态页面处理&#xff1a;Selenium与Scrapy的协同作战1.1 Selenium的核心价值与局限1.2 Scrapy-Selenium中间件开发1.3 动态分页处理实战&#xff1a;京东商品爬虫 二、云原生部署&#xff1a;Kubernetes架构设计与优化…

数据结构(十)——排序

一、选择排序 1.简单选择排序 基本思想&#xff1a;假设排序表为[1,…,n]&#xff0c;第i趟排序即从[i,…,n]中选择关键字最小的元素与L[i]交换 eg&#xff1a;给定关键字序列{87&#xff0c;45&#xff0c;78&#xff0c;32&#xff0c;17&#xff0c;65&#xff0c;53&…

小结:jvm 类加载过程

类加载过程 是Java虚拟机&#xff08;JVM&#xff09;将字节码文件&#xff08;.class文件&#xff09;加载到内存中&#xff0c;并转换为运行时数据结构的过程。这个过程可以分为多个步骤&#xff0c;每个步骤都有其特定的任务和目的。根据你提供的信息&#xff0c;以下是类加…

2024 山东省ccpc省赛

目录 I&#xff08;签到&#xff09; 题目简述&#xff1a; 思路&#xff1a; 代码&#xff1a; A&#xff08;二分答案&#xff09; 题目简述&#xff1a; 思路&#xff1a; 代码&#xff1a; K&#xff08;构造&#xff09; 题目&#xff1a; 思路&#xff1a; 代…

turn.js与 PHP 结合使用来实现 PDF 文件的页面切换效果

将 Turn.js 与 PHP 结合使用来实现 PDF 文件的页面切换效果&#xff0c;你需要一个中间步骤将 PDF 转换为 Turn.js 可以处理的格式&#xff08;如 HTML 页面或图片&#xff09;。以下是实现这一功能的步骤和示例代码&#xff1a; 步骤 1: 安装必要的库 首先&#xff0c;你需要…

Python实现NOA星雀优化算法优化卷积神经网络CNN回归模型项目实战

说明&#xff1a;这是一个机器学习实战项目&#xff08;附带数据代码文档视频讲解&#xff09;&#xff0c;如需数据代码文档视频讲解可以直接到文章最后关注获取。 1.项目背景 在当今数据驱动的时代&#xff0c;卷积神经网络&#xff08;CNN&#xff09;不仅在图像分类任务中…

(面试)View相关知识

1、View绘制流程 onMeasure() 确定View的测量宽高。onLayout() 确定View的最终宽高和四个顶点的位置。onDraw() 将View 绘制到屏幕上。 2、MeasureSpec有三种测量模式&#xff1a; 2.1. EXACTLY&#xff08;精确模式&#xff09; 含义&#xff1a;父容器明确指定了子View的精…

数组名既可作为指针也可作为变量名

在C语言中&#xff0c;数组名在不同的上下文中既可以作为指向数组首个元素的指针&#xff0c;也可以代表整个数组&#xff0c;这是由C语言的设计和语法规则决定的&#xff0c;下面我来详细解释一下。 1. 数组名作为指向首元素的指针 在大多数情况下&#xff0c;当数组名出现在…

Java异常、泛型与集合框架实战:从基础到应用

在Java编程的世界里&#xff0c;异常处理、泛型和集合框架是构建高效、健壮应用的关键技术。通过掌握这些技术&#xff0c;我们可以更好地管理程序运行时的错误&#xff0c;提高代码的复用性和类型安全性。今天&#xff0c;我将通过一系列实验&#xff0c;分享如何在Java中使用…

Spring源码之解决循环依赖 三级缓存

目录 三级缓存核心原理 循环依赖的解决过程 1. Bean A创建过程中提前曝光工厂 2. Bean B创建时发现依赖A&#xff0c;从缓存获取 3. Bean A继续完成初始化 三级缓存的作用总结 二级缓存为何不够解决缓存依赖&#xff1f; 三级缓存如何解决&#xff1f; 为什么不直接在…