求二叉树的带权路径长度(深搜或广搜)

考研昨天结束了,专业课我考得不好,是408,算法题13分。今年是求二叉树的带权路径长度,我就写了算法思想,代码空白,因为时间来不及了,慌了,是心态的问题吧,做到最后有种天要塌下来的感觉,回来后很沮丧,因为我觉得自己是可以写出程序的,不服啊! 

下面说说这道题目。树的带权路径长度(Weighted Path Length)定义:树中所有叶子的带权路径长度之和。比如下面这棵树,WPL就是3*2+7*1 = 13。

                         2
                        / \
                       /   \
                      4     7
                       \  
                        \
                         3

1、深搜(前序遍历)的算法可以用递归,程序简洁,先判断当前的根是不是叶子,若是则加上该叶子的带权路径长度,叶子的深度可以作为参数传递得到,求深度是个必须解决的问题。然后对左子树、右子树递归进行。因为是递归,所以我用了全局变量来记录部分和,不知道有没有不用全局变量的方法。

2、广搜(层次遍历)中的难题是求每层的深度,我参考了王道书上的算法,用了一个 last 变量来记录每层的最右结点,我觉得这个方法很巧,有点难想。

两个算法的实现如下(测试数据建的是上面那棵树,假设树根的深度为0)

/********************************************************************
created:    2014/01/06
created:    14:42
file base:  main
file ext:   cpp
author:     Justme0 (http://blog.csdn.net/Justme0)

purpose:    求二叉树的带权路径长度
*********************************************************************/

#define _CRT_SECURE_NO_WARNINGS
#include <iostream>
#include <queue>
#include <cassert>
using namespace std;

/*
** 二叉链表的结点定义
*/
struct Node {
    Node *left;
    Node *right;
    int weight;

    Node() : left(NULL), right(NULL), weight(0) {}

    bool has_left() const {
        return NULL != left;
    }

    bool has_right() const {
        return NULL != right;
    }

    bool is_leaf() const {
        return !has_left() && !has_right();
    }
};

#pragma region 深搜

/*
** 前序遍历求树的带权路径长度(weighted path length)
*/
int wpl_dfs = 0;    // wpl_dfs 为加权的部分和
void DFS(Node *root, int depth) {
    if (root == NULL) { // root 为空树
        return ;
    }

    if (root->is_leaf()) {
        wpl_dfs += root->weight * depth;
    } else {
        DFS(root->left, depth + 1);
        DFS(root->right, depth + 1);
    }
}

int get_WPL_dfs(Node *root) {
    DFS(root, 0);
    return wpl_dfs;
}

#pragma endregion 深搜


#pragma region 广搜

/*
** 层次遍历求树的带权路径长度(weighted path length)
*/
int get_WPL_bfs(Node *root) {
    if (root == NULL) { // root 是空树
        return 0;
    }

    int wpl_bfs = 0;
    int depth = 0;
    Node *last = root;  // last 为每层最右结点的地址,用于确定高度
    queue<Node *> Q;
    Q.push(root);
    do {
        Node *p = Q.front();
        Q.pop();

        if (p->is_leaf()) {
            wpl_bfs += p->weight * depth;
        } else {
            if (p->has_left()) {
                Q.push(p->left);
            }
            if (p->has_right()) {
                Q.push(p->right);
            }
        }

        if (p == last && !Q.empty()) {  // 此时处理到该层的最右结点
            last = Q.back();    // 队尾恰好是下一层的最右结点的地址
            ++depth;
        }
    } while (!Q.empty());

    return wpl_bfs;     
}

#pragma endregion 广搜

/*
** 输入前序序列动态生成树
*/
void creat_tree(Node *&root) {
    int info;
    cin >> info;
    if (info == -1) {
        root = NULL;
    } else {
        root = new Node;
        root->weight = info;
        creat_tree(root->left);
        creat_tree(root->right);
    }
}

/*
** 后序遍历释放树
*/
void free_tree(Node *&root) {
    if (root == NULL) {
        return ;
    }

    free_tree(root->left);
    free_tree(root->right);
    delete root;
    root = NULL;
}

int main(int argc, char **argv) {
    freopen("cin.txt", "r", stdin);

    Node *tree = NULL;
    creat_tree(tree);

    int dfs_wpl = get_WPL_dfs(tree);
    int bfs_wpl = get_WPL_bfs(tree);
    assert(dfs_wpl == bfs_wpl);
    cout << "深搜:WPL=" << dfs_wpl << endl;    // 11
    cout << "广搜:WPL=" << bfs_wpl << endl;    // 11

    free_tree(tree);

    return 0;
}

/*cin.txt
1 2 -1 4 -1 -1 3 -1 -1
*/

我考试时写的算法思想是广搜,然后一想到程序中得用队列,还得考虑深度,我就蒙了,只写了个函数头,希望能施舍一点分。。。我觉得这题是个典型的题目,不偏,出得挺好的。

20140110 
1、下面这种方法基于的原理是 WPL 为所有分支结点的权的和(这里的分支结点的权定义为它的左儿子和右儿子的权的和,同 Huffman 树)。

/*
** 后序遍历将分支结点的左儿子和右儿子的权相加作为该分支结点的权(覆盖原来无用的权)
** 再将所有的分支结点的权相加即 WPL
*/
int wpl_postorder = 0;
int get_WPL_postorder(Node *root) {
    if (root == NULL) {
        return 0;
    }

    get_WPL_postorder(root->left);
    get_WPL_postorder(root->right);
    if (!root->is_leaf()) {
        root->weight = 0;
        if (root->has_left()) {
            root->weight += root->left->weight;
        }
        if (root->has_right()) {
            root->weight += root->right->weight;
        }
        wpl_postorder += root->weight;
    }
    return wpl_postorder;
}
2、下面的方法有点偷懒,在结点定义时加了一个深度字段,用以解决求深度的问题。这种方法可以任何一种方式遍历树,当用前序时原理同开始的深搜算法,只不过那里的深搜将深度作为参数求出,这个干脆为它增加一个字段。

/*
** 在结点中加深度字段
*/
int wpl_depth = 0;

void visit(Node *root) {
    if (NULL == root) {
        return ;
    }

    if (root->is_leaf()) {
        wpl_depth += root->weight * root->depth;
    } else {
        if (root->has_left()) {
            root->left->depth = root->depth + 1;
        }
        if (root->has_right()) {
            root->right->depth = root->depth + 1;
        }
        visit(root->left);
        visit(root->right);
    }
}

int get_WPL_depth(Node *root) {
    root->depth = 0;
    visit(root);
    return wpl_depth;
}


  • 7
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
5.1 数的逻辑结构 5.1.1 1、树的定义 在树中常常将数据元素称为结点 (1)有且仅有一个特定的称为根的结点; (2)当n>1时,除根结点之外的其余结点被分成m(m>0)个互不相交的有限集合T1,T2,•••Tm,其中每个集合又是一棵树,并称为这个节点的子树。 2、树的基本术语: 结点的度、树的度 叶子节点、分支结点 孩子节点、分支结点、兄弟节点 路径路径长度 祖先、子孙 结点的层数、树的深度(高度) 层序编号 有序树、无序树 森林 5.1.2 树的抽象数据类型定义 5.1.3树的遍历操作 1、前序遍历 树的前序遍历操作定义为: 若树为空,则空操作返回;否则 (1)访问根结点 (2)按照从左向右的顺序前序遍历根结点的每一棵子树。 2、中序遍历 树的中序遍历操作定义为: 若树为空,则空操作返回;否则 (1)按照从左向右的顺序后序遍历根结点的每一棵子树; (2)访问根结点。 3、层序遍历 树的层序遍历也称作树的广泛遍历,其操作定义为树的第一层开始,自上而下逐层遍历,在同一层中,按从左向右的顺序对结点逐个访问。 5.2树的存储结构 5.2.1 双亲表示法 由树的定义可知,树中每个结点都有且仅有一个双亲结点。所以利用这一特性,可以用一维数组来存储各个结点,数组中一个元素对应一个结点,数组元素包括树中结点的数据信息以及该结点的双亲在数组中的下标。 其中: Data为数据域,存储树中结点的数据信息; Parent为指针即游标,存储该结点的双亲在数组中的小标。 5.2.2孩子表示法 1、多重链表表示法 (1)指针域的个数等于该结点的度。 (2)指针域的个数等于树的度。 2、孩子链表表示法 把孩子看成一个线性表,且以单链表存储,称为该结点的孩子链表。则n个结点有n个孩子链表。 孩子节点有两类:孩子节点、表头结点。 5.2.3 双亲孩子表示法 即将双亲表示法和孩子链表表示法相结合的存储方法。仍将各结点的孩子分别组成单链表,同时用一维数组顺序存储树中的各结点,数组元素除了包括结点的数据信息和该结点的孩子链表的头指针之外,还增设一个域存储该结点的双亲在数组的下标。 5.2.4孩子兄弟表示法 又称二链表表示法,其方法是链表中每个结点除数据域外,还设置了两个指针分别指向该结点的第一个孩子和右兄弟链表的结构: Firstchild data rightsib 指针域,存储第一个孩子结点的存储地址 数据域,存储该结点的数据信息 指针域,存储该结点右兄弟结点的存储地址 5.3二叉树的逻辑结构 最简单的树结构,特别适合计算机处理,而且任何数都可以简单的转换为二叉树。(重点内容) 5.3.1二叉树的定义 二叉树是n(n>=0)个结点的有限集合,该集合或者为空集,或者有一个根节点和两棵互不相交的、分别称为根节点的左子树和右子树的二叉树组成。 二叉树具有五种基本形态: 1、空二叉树; 2、只有一个根结点; 3、根结点只有左子树; 4、根结点只有右子树; 5、根结点既有左子树又有右子树 特殊二叉树: 1、斜树; 2、满二叉树; 3、完全二叉树; 5.3.2二叉树的基本性质 性质5-1 二叉树的第i层上最多有2^(i-1)个结点(i>=1)。 性质5-2 在一棵深度为k的二叉树中,最多有2^k-1个结点,最少有k个结点。 性质5-3 在一棵二叉树中,如果叶子结点的个数为n0,度为2的结点个数为n2,则n0=n2+1. 性质5-4 具有n个结点的完全二叉树的深度为【log2^n】+1。 性质5-5 对一棵具有n个结点的完全二叉树中的结点从一开始按层序编号,则对于任意的编号为i(1<=i<=n)的结点,有: (1)如果i>1,则结点i的双亲的编号为【i/2】;否则结点i是根结点,无双亲。 (2)如果2i<=n,则 结点i的左孩子的编号为2i;否则结点i无左孩子。 (3)如果2i+1<=n,则结点i的右孩子的编号为2i+1,否则结点i无右孩子。 5.3.3 二叉树的抽象数据类型定义 同树类似,在不同的应用中,二叉树的基本操作不尽相同。 5.3.4 二叉树的遍历操作 二叉树的遍历是指从根节点出发,按照某种次序访问二叉树是所有结点,使得每个结点被访问一次且仅被访问一次。由于二叉树中每个结点都可能有两个子树,因此需要寻找一条合适的路径。 1、前序遍历 前序遍历二叉树操作定义为: 若树为空,则空操作返回;否则 (1)访问根结点 (2)前序遍历根结点的左子树 (3)前序遍历根结点的右子树 2、中序遍历 中序遍历二叉树操作定义为: 若树为空,则空操作返回;否则 (1)中序遍历根结点的左子树 (2)访问根结点 (3)中序遍历根结点的右子树 3、后序遍历 后序遍历根结点的左子树 后序遍历根结点的右子树 访问根结点 4、层序遍历 二叉树的层序遍历是指从二叉树的第一层开始,从上之下逐层遍历,在同一层中,按从左到右的顺序对结点逐个访问。 5.4 二叉树存储结构及实现 5.4.1 顺序存储结构 具体步骤: (1)将二叉树按完全二叉树编号。 (2)将二叉树中的结点一编号顺序存储到一维数组中。 5.4.2 二叉链表 基本思想: 令二叉树的每个结点对应一个链表结点,链表结点除了存放于二叉树结点有关的数据信息外,还要设置指示左右孩子的指针。 5.4.3 三叉链表 在二叉链表存储方式下,从某个结点出发可以直接访问它的孩子结点,但要找到它的双亲结点,则需要从根节点开始索,最坏的情况下,需要遍历整个二叉链表。此时采用三叉树链表储存二叉树。 其中,data,lchild,rchild三个域的含义同二叉树,parent域为指向该结点的双亲结点指针。 5.4.4 线索链表 按照某种遍历次序对二叉树进行遍历,可以把二叉树中所有结点排成一个线性序列。在集体应用中,有时需要访问二叉树中的结点在某种遍历序列中前驱和后继,此时,在存储结构中应该保存结点在某种遍历序列中的前驱和后继信息。 前驱和后继结点的指针称为线索,加上线索的二叉树称为线索二叉树,加上线索的二叉链表称为线索链表。 5.5 二叉树遍历的非递归算法 5.5.1 前序遍历非递归算法 关键:在前序遍历过某个左子树后,如何找到该结点的右子树的根指针。 一般的前序遍历执行过程中,设要遍历二叉树的根指针为bt,可能出现两种情况: (1)若bt!=NULL,则表明当前二叉树不为空,此时,应输入根结点bt的值并将bt保存到栈中,准备继续遍历bt的左子树。 (2)若bt=NULL,则表明以bt为根指针的二叉树遍历完毕,并且bt是栈顶指针所指结点的左子树,若栈不空,则应根据栈顶指针所指结点找到待遍历右子树的根指针并赋予bt,以继续遍历下去;若栈空,则表明整个二叉树遍历完毕。 5.5.2 中序遍历非递归算法 此算法只是需要将前序遍历的非递归算法中输出的语句cout<<bt->data移到bt=s[top--]之后即可。 5.5.3 后序遍历非递归算法 后序遍历的不同在于:结点要出入两次栈,出两次栈,这种情况的含义和处理方法为: (1)第一次出栈:只遍历晚左子树,右子树尚未遍历,则该结点不出栈,利用栈顶结点找到它的右子树,准备遍历它的右子树。 (2)第二次出栈:遍历完右子树,该结点出栈,并访问它。 设根指针为bt,则可能有以下两种情况: (1)若bt!=NULL,则bt及标志flag入栈,遍历其左子树。 (2)若bt=NULL,此时栈空,则整个遍历结束;若栈不空,则表明栈顶结点的左子树或右子树已遍历结束。若栈顶点的标志flag=1,则表明栈结点的左子树已遍历完毕,将flag修改为2,修改为2,并遍历栈定点的右子树;若栈顶结点的标志flag=2,则表明栈结点的右子树也遍历完毕,输出栈顶结点。 5.6 树、森林与二叉树的转换 1.树转换为二叉树 将一棵树转换为二叉树的方法为: (1)加线——树中所有相邻的兄弟结点之间加一条线; (2)去线——对树中的每个节点,只保留它与第一个孩子结点之间的连线,删去它与其他孩子结点之间的连线。 (3)层次调节——以根结点为轴心,将树顺时针转动一定角度,使之层次分明。 2.森林转换成二叉树 (1)将森林中的每一棵二叉树转化成二叉树; (2)从第二课二叉树开始,依次把后一棵二叉树的根结点作为一棵二叉树根节点的右孩子,当所有二叉树连起来后,此时所得到的二叉树就是由森林转换得到的二叉树。 3、二叉树转换为树或森林 (1)加线——若某个结点x是其双亲y的左孩子,则把结点x的右孩子、右孩子的右孩子、……,都与结点y用线连起来; (2)去线——删去原二叉树中所有的双亲结点与右孩子结点的连线; (3)层次调整——整理由(1)、(2)两步所得到的树或森林,使之层次分明。 (4)森林的遍历 两种遍历方法;前序遍历后续遍历。 5.7 应用举例 5.7.1 二叉树的应用举例——哈夫曼及哈夫曼编码 1、哈夫曼树也称最优二叉树,在实际中有着广泛的应用。 叶子节点的权值 是对叶子结点赋予的一个有意义的数值量。 二叉树带权路径长度二叉树具有n个带权值的叶子节点,从根节点到叶子节点的路径长度与相应的叶子节点权值的乘积之和叫做二叉树带权路径长度,记为: WPL=EWkLk 哈夫曼树 给定一组具有确定权值的叶子结点,可以构造出不同的二叉树,将其中带权路径长度最小的二叉树称为哈夫曼树。 哈夫曼算法基本思想: (1)初始化:由给定的n个权值构造n棵只有一个根结点的二叉树,从而得到一个二叉树集合。 (2)选取与合并:在F中选取根结点的权值最小的两棵二叉树分别作为左、右子树构造一棵新的二叉树,这棵新的二叉树的根结点的权值为其左右子树根结点的权值之和。 (3)删除与加入:在F中删除作为左、右子树的两棵二叉树,并将新建的二叉树加入到F中。 (4)重复(2)(3)两步的操作,当集合F只剩下一棵二叉树时这棵二叉树便是哈夫曼树。 2、哈夫曼编码 在进行程序设计时,通常给每一个字符记一个单独的代码来表示一组字符,我们称之为编码。
二叉树带权路径长度(WPL)是指二叉树中所有叶结点的带权路径长度之和。\[1\]带权路径长度是从树的根结点到任意结点的路径长度(经过的边数)与该结点上权值的乘积。\[2\]要设计一个算法来解给定二叉树T的WPL。 一个可行的思路是使用递归来遍历二叉树的所有根结点,并将每个根结点的带权路径长度与一个静态局部变量相加,最终得到二叉树的WPL。\[3\]具体实现时,可以定义一个递归函数,该函数的参数为当前结点和该结点的深度。通过判断当前结点是否有左右子结点,来递归遍历所有的根结点。在遍历过程中,将每个根结点的带权路径长度与静态局部变量相加,最终得到二叉树的WPL。 以下是一个示例的二叉树结点数据类型的定义: typedef struct BitNode { int weight; BitNode *left, *right; } BitNode, *BitTree; 通过实现上述算法,可以计算出给定二叉树带权路径长度(WPL)。 #### 引用[.reference_title] - *1* [【数据结构】二叉树带权路径长度WPL是二叉树中所有叶结点的带权路径长度之和。给定一颗二叉树T,采用二叉...](https://blog.csdn.net/weixin_51280571/article/details/126234359)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* *3* [二叉树带权路径长度(WPL)是二叉树中所有叶结点的带权路径长度之和。给定一棵二叉树T,采用二叉链表存储,...](https://blog.csdn.net/weixin_48239221/article/details/120337506)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值