二叉检索树的实现——增删改查、读取命令文件、将结果写入新文件

news2024/10/6 14:32:30

看这篇文章前的知识储备

链接: 二叉树的性质和分类
链接: 二叉检索树的概念 、insert方法的图解、实现、时间代价分析
链接: 二叉检索树的search、remove方法的图解、实现、时间代价分析

1、中序遍历及中序遍历写进文件的区别

两者思路一致,将二叉树分为三部分:左子树,根结点,右子树。对左子树,右子树递归遍历。

(1)写入数组:

在这里插入图片描述

(2)写入文件

需要借助两个函数,目的:在传参时不需要手动重复传入二叉树根结点

第一个函数只需要传入输出文件名称即可(打开关闭文件,写入文件)
第二个函数具体实现写入文件

在这里插入图片描述
在这里插入图片描述

2、处理命令

为方便用户进行增删改查
将增删改查的命令符号分别定义为 + - = ?
具体实现如下
在这里插入图片描述

3、utf-8 和 gbk 编码的区别

UTF-8和GBK是两种不同的字符编码方式,它们之间存在一些显著的差异:

  • 字符集范围
    GBK编码主要用于支持中文字符和日韩字符,而UTF-8编码则支持全球范围内的字符。这意味着,如果需要处理多种语言字符,UTF-8将是一个更好的选择。
  • 编码方式
    GBK编码采用双字节编码,即不论中、英文字符均使用双字节来表示。而UTF-8编码则采用变长编码,一个字符的编码长度可为1~4个字节,甚至更长。这种变长编码方式使得UTF-8编码能够根据字符的实际长度来分配存储空间,因此在存储中占用的空间相对较小。
  • 兼容性
    GBK编码在国内应用广泛,但在国际化应用上可能受到限制。而UTF-8编码则具有更好的国际化兼容性,可以在各国各种支持UTF-8字符集的浏览器上显示,无需额外下载语言支持包。
  • 存储空间大小
    由于GBK编码每个字符固定占用2个字节,因此在存储中占用的空间相对较大。而UTF-8编码采用变长编码,可以根据需要分配存储空间,因此在存储中占用的空间相对较小。

综上所述,UTF-8和GBK在字符集范围、编码方式、兼容性和存储空间大小等方面存在显著差异。选择哪种编码方式取决于具体的应用场景和需求。在处理多种语言字符或需要国际化应用的场合,UTF-8编码通常更为合适;而在主要处理中文字符的场合,GBK编码可能更为常用。

(1)注意

在读取写入文件时,最好表明编码方式,防止出现UnicodeDecodeError错误
在这里插入图片描述

(2)判断文件的编码类型

在这里插入图片描述

import chardet

# 打开文件并读取一部分内容
with open('name of your file', 'rb') as f:
    rawdata = f.read(100)  # 读取前100个字节

# 使用chardet检测编码
result = chardet.detect(rawdata)
print(result)
encoding = result['encoding']

print(f"Detected encoding: {encoding}")

4、全部代码理解

(1)前期准备——Elem()、BinNode()

在这里插入图片描述

(2)查——search

在这里插入图片描述

(3)改——update

在这里插入图片描述

(4)增——insert

在这里插入图片描述

(5)删——remove

在这里插入图片描述

(6)其他方法

在这里插入图片描述

(7)读取命令

在这里插入图片描述

(8)具体命令处理

在这里插入图片描述

(8)实例

在这里插入图片描述

(9)结果

命令文件(部分)

在这里插入图片描述

运行结果(部分)

在这里插入图片描述

5、全部代码

class Elem():
    def __init__(self,val):
        if val is not None:#把要插入的元素提取出来
            self._key = val[1:-1].strip().split(',')[0]
            self._value= val[1:-1].strip().split(',')[1]
        else:
            self._key=None

    def key(self):#获取key()
        return self._key

    def value(self):
        return self._value

    def set_key(self, key):
        self._key = key

    def set_value(self, value):
        self._value = value

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

class BST():
    def __init__(self):
        self.root=None

    def search(self,key):
        return self.search_help(self.root, key)

    def search_help(self,root,key):
        if root is None:
            return None
        it=root.val
        if it.key()>key:
            return self.search_help(root.left,key)
        elif it.key()<key:
            return self.search_help(root.right,key)
        return it.value()

    def search_node(self, root, key):
        if root is None:
            return None
        it = root.val
        if it.key() > key:
            return self.search_node(root.left, key)
        elif it.key() < key:
            return self.search_node(root.right, key)
        return root.val  # 返回节点对象

    def update(self, key, value):
        node = self.search_node(self.root, key)
        if node is not None:
            node.set_value(value)
            return True
        else:
            return False

    def insert(self,elem):
        self.root=self._insert(self.root,elem)
    # _代表私有函数 辅助insert函数实现插入操作
    def _insert(self,root,elem):
        if root is None:
            return BinNode(elem)
        it=root.val#rooot.val是一个Elem()对象
        if it.key()>elem.key():
            root.left=self._insert(root.left,elem)
        elif it.key()<elem.key():
            root.right=self._insert(root.right,elem)
        if it.key()==elem.key():#如果key值相同就替换
            self.update(elem.key(),elem.value())
            # root.val=elem
        return root#返回当前结点,在递归的时候只改变最后的结点

    def getmin(self,root):
        if root.left is None:
            return root.val
        else:
            return self.getmin(root.left)
    def del_min(self,root):
        if root.left is None:
            root=root.right
        else:
            root.left=self.del_min(root.left)
        return root
    def remove(self,key):
        self.remove_help(self.root,key)
    def remove_help(self,root,key):
        if root is None:
            return None
        it=root.val
        #先找到要删除的元素,思想类似于del_min(),路径中结点不会改变子结点指针值
        if it.key()>key:
            root.left=self.remove_help(root.left,key)
        elif it.key()<key:#要规范代码写法
            root.right=self.remove_help(root.right,key)
        #找到该元素,判断该结点类型(叶子、一分支、(叶子的判断可以包含在一分支判断中)二分支)
        else:
            if root.left is None:
                root=root.right
            elif root.right is None:
                root=root.left
            else:#两个分支
                temp=self.getmin(root)#获取最小元素
                root.val=temp#替换
                self.del_min(root.right)#删除最小元素(del_min返回root)
        return root

    def clear(self):
        self.root=None

    def is_Empty(self):
        return self.root is None

    def size(self, root):#递归得出二叉树大小
        if root is None:
            return 0
        else:
            return 1 + self.size(root.left) + self.size(root.right)#根结点+左子树+右子树
    def height(self, root):#递归得出二叉树高度
        if root is None:
            return 0
        else:
            return 1 + max(self.height(root.left), self.height(root.right))
    def showStructure(self, outputFile):
        with open(outputFile, 'a+',encoding='utf-8') as f:
            f.write('---------------------------\n')
            f.write(f'There are {self.size(self.root)} nodes in this BST.\n')
            f.write(f'The height of this BST is {self.height(self.root)}\n')
            f.write('---------------------------\n')
    def printInorder(self, outputFile):
        with open(outputFile, 'w',encoding='utf-8') as f:
            self._printInorder(self.root, f)
    def _printInorder(self, root, outputFile):
        if root is not None:
            self._printInorder(root.left, outputFile)
            outputFile.write(f'[{root.val.key()} - < {root.val.value()} >]\n')
            self._printInorder(root.right, outputFile)


bst=BST()
def readInputFile(bst,inputFile):#把命令文件读进来并将相应命令结果输出到文件中
    with open(inputFile, 'r',encoding='utf-8') as f:
        for line in f:
            if line.startswith('+( ') or line.startswith('=( ') or line.startswith('-( ') or line.startswith('?( ')  :
                handleOperation(bst,line.strip())
            if  line.startswith('#'):
                bst.showStructure('r.txt')
def handleOperation(bst,operation):#处理命令,将命令符号,key,value分离出来
    operation_type = operation[0]
    key_value = operation[3:-2].split(' , ')#包含3 不包含-2
    if len(key_value)>1:#命令中既有key 也有value——增改命令
        key = key_value[0]
        value = key_value[1][1:-1]#去掉引号[1:-1]
        if operation_type == '+':#判断命令类型
            bst.insert(Elem(f'({key}, "{value}")'))
        elif operation_type == '=':
            if bst.update(key, value) == True:
                with open('r.txt','a+',encoding='utf-8')as f:
                    f.write(f'update success ---{key} {value}\n')
            else:
                with open('r.txt', 'a+',encoding='utf-8') as f:
                    f.write(f'update unsuccessful ---{key}\n')
    else:#如果仅有Key 没有value——删查命令
        key=key_value[0]
        if operation_type == '-':
            root=bst.remove(key)
            if root is not None:
                with open('r.txt','a+',encoding='utf-8')as f:
                    f.write(f"remove success ---{root.val.value()}\n")
            else:
                with open('r.txt','a+',encoding='utf-8')as f:
                    f.write(f"remove unsuccessful ---{key}\n")
        elif operation_type == '?':
            result = bst.search(key)
            if result is not None:
                with open('r.txt','a+',encoding='utf-8')as f:
                    f.write(f"search success ---{key} {result}\n")
            else:
                with open('r.txt','a+',encoding='utf-8')as f:
                    f.write(f"search unsuccessful ---{key}\n")

readInputFile(bst,'testcases.txt')
bst.printInorder('bstInorder.txt')


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

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

相关文章

Linux信号(产生)

个人主页&#xff1a;Lei宝啊 愿所有美好如期而遇 目录 信号是什么&#xff1f; 为什么要有信号&#xff1f; 信号是如何产生的&#xff1f; kill命令 键盘产生信号 系统调用 kill系统调用 raise函数 abort函数 自制kill命令 ​编辑 软件条件 举例一&#xff1…

C++ :设计模式实现

文章目录 原则单一职责原则开闭原则依赖倒置原则接口隔离原则里氏替换原则 设计模式单例模式观察者模式策略模式代理模式 原则 单一职责原则 定义&#xff1a; 即一个类只负责一项职责 问题&#xff1a; 类 T 负责两个不同的职责&#xff1a;职责 P1&#xff0c;职责 P2。当…

大数据第六天

这里写目录标题 问题解决问题查询插入(时间慢)练习sql数据清理 问题 FAILED: ParseException line 1:16 mismatched input ‘input’ expecting INPATH near ‘local’ in load statement MismatchedTokenException(24!155) 加载数据的时候出现了这个错误&#xff0c;我们解释…

【六十】【算法分析与设计】用一道题目解决dfs深度优先遍历,dfs中节点信息,dfs递归函数模板进入前维护出去前回溯,唯一解的剪枝飞升返回值true

路径之谜 题目描述 小明冒充X星球的骑士,进入了一个奇怪的城堡。 城堡里边什么都没有,只有方形石头铺成的地面。 假设城堡地面是nn个方格。如下图所示。 按习俗,骑士要从西北角走到东南角。可以横向或纵向移动,但不能斜着音走,也不能跳跃。每走到一个新方格,就要向正北 方和正西…

短信视频提取批量工具,免COOKIE,博主视频下载抓取,爬虫

痛点&#xff1a;关于看了好多市面的软件&#xff0c;必须要先登录自己的Dy号才能 然后找到自己的COOKIE 放入软件才可以继续搜索&#xff0c;并且无法避免长时间使用 会导致无法正常显示页面的问题。 有没有一种方法 直接可以使用软件&#xff0c;不用设置的COOKIE的方法呢 …

对于地理空间数据,PostGIS扩展如何在PostgreSQL中存储和查询地理信息?

文章目录 一、PostGIS扩展简介二、PostGIS存储地理空间数据1. 创建空间数据表2. 插入空间数据 三、PostGIS查询地理空间数据1. 查询指定范围内的地理空间数据2. 计算地理空间数据之间的距离3. 对地理空间数据进行缓冲区分析 四、总结 地理空间数据是指描述地球表面物体位置、形…

开源社区与开发者的故事

开源社区与开发者的故事 什么是开源社区你参加开源社区的主要目的你是否在开源社区中贡献&#xff0c;或者开源自己的项目&#xff1f;你认为个人开发者是否应该从开源中获利&#xff1f;如果是&#xff0c;该如何获利&#xff1f; 今天要谈及的主题是开源社区&#xff0c;那么…

2024年新算法-牛顿-拉夫逊优化算法(NRBO)优化BP神经网络回归预测

亮点&#xff1a; 输出多个评价指标&#xff1a;R2&#xff0c;RMSE&#xff0c;MSE&#xff0c;MAPE和MAE 满足需求&#xff0c;分开运行和对比的都有对应的主函数&#xff1a;main_BP, main_NRBO, main_BPvsBP_NRBO&#xff0c;并且详细中文注释 方便快捷&#xff1a;替换…

打破企业差旅管理困局,让金融CEO眼前一亮的出行方案

在国内券商投行部工作是怎样一种体验&#xff1f; “长期出差&#xff0c;而且出长差&#xff0c;时常让人有漂泊的孤独感。”这是某问答平台上的高赞回答的第一条。 对金融人来说&#xff0c;说走就走的旅行可能根本没有什么吸引力&#xff0c;时刻准备着说走就走的出差才是生…

MVCC的执行原理

MVCC的执行原理 MVCC简介事务的隔离级别MVCC作用当前读和快照读MVCC实现原理Undo LogUndo Log 版本链Read View判断方法判断规则 小结 MVCC简介 MVCC&#xff08;Multi-Version Concurrency Control&#xff09;是一种并发控制机制&#xff0c;用于解决数据库并发访问中&#…

pyqt 动态更换表头和数据

目录 pyqt 动态更换表头和数据代码 效果图&#xff1a; pyqt 动态更换表头和数据代码 from PyQt5.QtGui import QColor, QBrush from PyQt5.QtWidgets import QApplication, QTableWidget, QVBoxLayout, QWidget, QPushButton, QTableWidgetItemclass Example(QWidget):def _…

如何诊断并解决PostgreSQL中的磁盘空间不足问题?

文章目录 诊断磁盘空间不足问题1. 检查服务器磁盘空间2. 检查PostgreSQL数据目录大小3. 检查PostgreSQL中的大表和大对象 解决磁盘空间不足问题1. 清理不必要的文件和日志2. 清理或压缩大表和大对象3. 扩展磁盘容量4. 优化数据库配置和查询 在使用PostgreSQL数据库时&#xff0…

华为云实验 -- 对云硬盘数据盘进行备份

文章目录 备份Linux系统备份1.购买Linux操作系统的ESC(云服务器)2.挂载数据盘--初始化--分区--格式化2.1.点击"远程登录"a.查看/dev/vdb数据盘b.新建主分区/dev/vdb1 2.2.查看新建分区大小,分区格式信息a.确定之前的分区操作是否正确b.确认完成后&#xff0c;将分区结…

【MATLAB源码-第32期】基于matlab的通信及雷达中常用伪随机码m序列的仿真。

操作环境&#xff1a; MATLAB 2022a 1、算法描述 M序列&#xff0c;也称为最大长度序列或者伪随机序列&#xff0c;是一种特殊的二进制序列。它的特点是在有限的长度内&#xff0c;尽管它是伪随机的&#xff0c;但它会在特定的周期内不重复地循环。 在数学上&#xff0c;M序…

利用fft算法重写公式并理解频率和像素变化率的关系(完美解决问题)

算法我就不贴了。算法就是算法导论的内容。 我直接写推导过程。 假设变化率为f(n1)-f(n) 首先计算二进制数&#xff0c;这里我假设为3位二进制。 例如:f(5)-f(4)&#xff0c; 5和4的二进制为101,100。所以逆序数为101&#xff0c;001 101对应的频率为5, 001对应的频率为1…

力扣HOT100 - 236. 二叉树的最近公共祖先

解题思路&#xff1a; dfs 节点p,q异侧时&#xff0c;节点root为它们的公共祖先。 class Solution {public TreeNode lowestCommonAncestor(TreeNode root, TreeNode p, TreeNode q) {if (root null || p root || q root) return root;TreeNode left lowestCommonAncest…

max各种相机导出到ue4匹配镜头的工具集

总览 rollout export_UE4Cam_v2 "导出UE4Cam_v2:半自动" width:200 height:120(HyperLink explain "在打开的max文件中使用" pos:[25,12] width:200 height:15 color:(color 255 155 0) GroupBox grp1 "要导出的相机名" pos:[5,28] width:179 …

NeRF in the Wild: Neural Radiance Fields for Unconstrained Photo Collections

NeRF in the Wild: Neural Radiance Fields for Unconstrained Photo Collections(野外的 NERF: 用于无约束照片采集的神经辐射场&#xff09; Abstract 我们提出了一种基于学习的方法来合成新的视图的复杂场景使用只有非结构化的收集野生照片。我们建立在神经辐射场(neRF)的…

深度学习算法简介(一)

目录 ⛳️推荐 前言 1、深度神经网络&#xff08;DNN&#xff09; 2、卷积神经网络&#xff08;CNN&#xff09; 3、残差网络&#xff08;ResNet&#xff09; 4、LSTM&#xff08;长短时记忆网络&#xff09; 5、Word2Vec 6、Transformer 7、生成对抗网络&#xff08;…

MySQL常见的约束

什么是约束&#xff1f; 限制&#xff0c;限制我们表中的数据&#xff0c;保证添加到数据表中的数据准确和可靠性&#xff01;凡是不符合约束的数据&#xff0c;插入时就会失败&#xff0c;插入不进去的&#xff01; 比如&#xff1a;学生信息表中&#xff0c;学号就会约束不…