最新华为OD机试真题-生成哈夫曼树(100分)

🍭 大家好这里是清隆学长 ,一枚热爱算法的程序员

✨ 本系列打算持续跟新华为OD-D卷的三语言AC题解

👏 感谢大家的订阅➕ 和 喜欢💗

📎在线评测链接

=> 生成哈夫曼树(100分) <=

华为OD

🌍 评测功能需要 =>订阅专栏<= 后联系清隆解锁~

🍓OJ题目截图

alt

🍪 生成哈夫曼树

问题描述

LYA 是一名计算机专业的学生,最近她学习了哈夫曼编码。为了巩固知识,她决定写一个程序来生成哈夫曼树。

给定一个长度为 的正整数数组,每个数字代表二叉树叶子节点的权值。请你帮助 LYA 生成一棵哈夫曼树,并将其按中序遍历的顺序输出。

为了保证输出的哈夫曼树唯一,需要满足以下条件:

  1. 树中每个非叶子节点的权值等于其左右子节点权值之和。

  2. 对于权值相同的两个节点,左子树的高度应小于等于右子树的高度。

  3. 在满足上述条件的前提下,左子节点的权值应小于等于右子节点的权值。

输入格式

第一行包含一个正整数 ,表示叶子节点的个数。

第二行包含 个正整数,表示每个叶子节点的权值,数值之间用空格分隔。

输出格式

输出一行,包含若干个正整数,表示按中序遍历哈夫曼树得到的节点权值序列,数值之间用空格分隔。

样例输入

5
5 15 40 30 10

样例输出

40 100 30 60 15 30 5 15 10

数据范围

权值

题解

本题考查哈夫曼树的构建。哈夫曼树是一种带权最优二叉树,其特点是带权路径长度最短。

构建哈夫曼树的基本步骤如下:

  1. 将所有节点看成独立的树,并按照权值从小到大排序。

  2. 取出权值最小的两棵树,将它们作为一个新树的左右子树,新树的权值为两棵子树权值之和。

  3. 重复步骤 2,直到只剩下一棵树,即为所求的哈夫曼树。

在实现时,我们可以用优先队列来维护节点,每次取出权值最小的两个节点合并。为了保证输出的哈夫曼树唯一,在优先队列中比较两个节点时,先比较权值,权值相同再比较树高,树高也相同则比较左右子树的权值大小关系。

构建完哈夫曼树后,我们再进行一次中序遍历即可得到输出序列。

参考代码

  • Python
import heapq

class Node:
    def __init__(self, val, left=None, right=None):
        self.val = val
        self.left = left
        self.right = right

    def __lt__(self, other):
        if self.val != other.val:
            return self.val < other.val
        if self.height != other.height:
            return self.height < other.height
        return self.left.val <= other.left.val
    
    @property
    def height(self):
        return max(self.left.height if self.left else 0, 
                   self.right.height if self.right else 0) + 1

def huffman_tree(vals):
    pq = [Node(val) for val in vals]
    heapq.heapify(pq)
    
    while len(pq) > 1:
        left, right = heapq.heappop(pq), heapq.heappop(pq)
        heapq.heappush(pq, Node(left.val + right.val, left, right))
    
    return pq[0]

def inorder_traversal(root):
    if not root:
        return []
    return inorder_traversal(root.left) + [root.val] + inorder_traversal(root.right)

n = int(input())
vals = list(map(int, input().split()))
root = huffman_tree(vals)
print(*inorder_traversal(root))
  • Java
import java.io.*;
import java.util.*;

class Node implements Comparable<Node> {
    int val;
    Node left, right;
    
    Node(int val) {
        this.val = val;
    }
    
    Node(int val, Node left, Node right) {
        this.val = val;
        this.left = left;
        this.right = right;
    }
    
    int height() {
        return 1 + Math.max(left != null ? left.height() : 0, 
         

剩余60%内容,订阅专栏后可继续查看/也可单篇购买

最新华为OD机试-E+D卷 文章被收录于专栏

本专栏给大家提供了华为2024最新华为OD-E/D卷的题目汇总和(Java/Cpp/Python)三语言解析 + 部分题目提供OJ在线评测

全部评论
评测功能需要 订阅专栏 后联系清隆解锁~
点赞 回复 分享
发布于 07-01 15:23 浙江

相关推荐

09-13 17:53
已编辑
黑龙江科技大学 iOS开发
题目描述单词接龙的规则是:可用于接龙的单词首字母必须要前一个单词的尾字母相同;当存在多个首字母相同的单词时,取长度最长的单词,如果长度也相等,则取字典序最小的单词;已经参与接龙的单词不能重复使用。现给定一组全部由小写字母组成单词数组,并指定其中的一个单词作为起始单词,进行单词接龙,请输出最长的单词串,单词串是单词拼接而成,中间没有空格。输入描述输入的第一行为一个非负整数,表示起始单词在数组中的索引K,0&nbsp;&lt;=&nbsp;K&nbsp;&lt;&nbsp;N&nbsp;;输入的第二行为一个非负整数,表示单词的个数N;接下来的N行,分别表示单词数组中的单词。输出描述输出一个字符串,表示最终拼接的单词串。补充说明单词个数N的取值范围为[1,&nbsp;20];单个单词的长度的取值范围为[1,&nbsp;30];用例1输入06worddddadcdwordd输出worddwordda说明:先确定起始单词word,再接以d开头的且长度最长的单词dword,剩余以d开头且长度最长的有dd,da,dc,则取字典序最小的da,所以最后输出worddwordda。用例2输入46worddddadcdwordd输出dwordda说明:先确定起始单词word,剩余以d开头且长度最长的有dd,da,dc,则取字典序最小的da,所以最后输出dwordda。解题如下:#include&nbsp;&lt;iostream&gt;#include&nbsp;&lt;vector&gt;#include&nbsp;&lt;string&gt;using&nbsp;namespace&nbsp;std;//&nbsp;判断是否可以接龙bool&nbsp;canChain(const&nbsp;string&amp;amp;&nbsp;a,&nbsp;const&nbsp;string&amp;amp;&nbsp;b)&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;a.back()&nbsp;==&nbsp;b.front();}int&nbsp;main()&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;int&nbsp;k,&nbsp;n;&nbsp;&nbsp;&nbsp;&nbsp;cin&nbsp;&gt;&gt;&nbsp;k&nbsp;&gt;&gt;&nbsp;n;&nbsp;&nbsp;&nbsp;&nbsp;vector&lt;string&gt;&nbsp;words(n);&nbsp;&nbsp;//&nbsp;存储单词列表&nbsp;&nbsp;&nbsp;&nbsp;vector&lt;bool&gt;&nbsp;used(n,&nbsp;false);&nbsp;&nbsp;//&nbsp;标记单词是否已经使用过&nbsp;&nbsp;&nbsp;&nbsp;for&nbsp;(int&nbsp;i&nbsp;=&nbsp;0;&nbsp;i&nbsp;&lt;&nbsp;n;&nbsp;++i)&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;cin&nbsp;&gt;&gt;&nbsp;words[i];&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;&nbsp;&nbsp;&nbsp;string&nbsp;result&nbsp;=&nbsp;words[k];&nbsp;&nbsp;//&nbsp;结果字符串以起始单词开始&nbsp;&nbsp;&nbsp;&nbsp;used[k]&nbsp;=&nbsp;true;&nbsp;&nbsp;//&nbsp;标记起始单词已使用&nbsp;&nbsp;&nbsp;&nbsp;while&nbsp;(true)&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;int&nbsp;nextIndex&nbsp;=&nbsp;-1;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;for&nbsp;(int&nbsp;i&nbsp;=&nbsp;0;&nbsp;i&nbsp;&lt;&nbsp;n;&nbsp;++i)&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;找到未使用的单词并且可以接龙&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(!used[i]&nbsp;&amp;amp;&amp;amp;&nbsp;canChain(result,&nbsp;words[i]))&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;如果没有选择单词或者当前单词比选择的单词更长或者相同长度但字典序更小&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(nextIndex&nbsp;==&nbsp;-1&nbsp;||&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;words[i].length()&nbsp;&gt;&nbsp;words[nextIndex].length()&nbsp;||&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;(words[i].length()&nbsp;==&nbsp;words[nextIndex].length()&nbsp;&amp;amp;&amp;amp;&nbsp;words[i]&nbsp;&lt;&nbsp;words[nextIndex]))&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;nextIndex&nbsp;=&nbsp;i;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;如果没有找到可以接龙的单词,跳出循环&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(nextIndex&nbsp;==&nbsp;-1)&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;break;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;result&nbsp;+=&nbsp;words[nextIndex];&nbsp;&nbsp;//&nbsp;将选择的单词加入结果&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;used[nextIndex]&nbsp;=&nbsp;true;&nbsp;&nbsp;//&nbsp;标记该单词为已使用&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;输出最终拼接的单词串&nbsp;&nbsp;&nbsp;&nbsp;cout&nbsp;&lt;&lt;&nbsp;result&nbsp;&lt;&lt;&nbsp;endl;&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;0;}
查看1道真题和解析 投递华为等公司10个岗位
点赞 评论 收藏
分享
1 3 评论
分享
牛客网
牛客企业服务