移情别恋c++ ദ്ദി˶ー̀֊ー́ ) ——14.AVL树
1.AVL 树
1.1AVL 树的概念
二叉搜索树虽可以缩短查找的效率,但如果数据有序或接近有序二叉搜索树将退化为单支树,查 找元素相当于在顺序表中搜索元素,效率低下。因此,两位俄罗斯的数学家G.M.Adelson-Velskii 和E.M.Landis在1962年 发明了一种解决上述问题的方法:当向二叉搜索树中插入新结点后,如果能保证每个结点的左右 子树高度之差的绝对值不超过1(需要对树中的结点进行调整),即可降低树的高度,从而减少平均 搜索长度。
一棵AVL树或者是空树,或者是具有以下性质的二叉搜索树:
1.它的左右子树都是AVL树
2.左右子树高度之差(简称平衡因子)的绝对值不超过1(-1/0/1)

如果一棵二叉搜索树是高度平衡的,它就是AVL树。如果它有n个结点,其高度可保持在 O(log_2 n),搜索时间复杂度O(log_2 n)。
2.AVL树节点的定义
template<class K ,class V> struct AVLtreenode {AVLtreenode<K, V>* _left; //右节点AVLtreenode<K, V>* _right; //左节点AVLtreenode<K, V>* _parent; //父节点,三叉链表pair<K, V> kv;int bf;//右子树高度-左子树高度,只有孩子发生变化,bf才有可能发生变化!!!!,若改变父亲,bf不变!!!!!!AVLtreenode(const pair<K, V>& _kv) //初始化列表:_left(nullptr),_right(nullptr),_parent(nullptr),kv(_kv),bf(0){} };
3.AVL树的插入!!!!!!!
3.1初步插入
初步插入与bst的插入一致,不过里面的数据类型是pair<first,second>,并且是根据first进行比较
if (root == nullptr)
{root = new node(_kv);return true;
}
node* parent = nullptr; //比bst多了一个parent
node* cur = root; //while (cur)
{parent = cur;if (cur->kv.first < _kv.first){cur = cur->_right;}else if (cur->kv.first > _kv.first){cur = cur->_left;}else{return false;}
}
cur = new node(_kv);
if (parent->kv.first < _kv.first)
{parent->_right = cur;
}
else
{parent->_left = cur;
}
cur->_parent = parent; //记得连接parent和cur
3.2调整平衡因子


while (cur != root)
{if (cur == parent->_left)parent->bf--;//第一次!!!检查parent左边原来必为空else {parent->bf++;//第一次!!!检查parent右边原来必为空}if (parent->bf == 0) //相当于bf没改变,可直接退出{break;}else if (parent->bf == 1 || parent->bf == -1) //bf改变了继续向上找 {cur = parent;parent = parent->_parent;}else if(parent->bf == -2 || parent->bf == 2){ //-2||2,需要调整(旋转)if (parent->bf == 2 && cur->bf == 1){rotateL(parent);//单左旋,全在右边加}else if (parent->bf == -2 && cur->bf == -1){rotateR(parent);//单右旋,全在左边加}else if (parent->bf == 2 && cur->bf == -1){rotateRL(parent);//先右旋再左旋}else if (parent->bf == -2 && cur->bf == 1){rotateLR(parent);//先左旋再右旋}//旋转让子树变得平衡//旋转降低了子树的高度,恢复到和插入前一样的高度,所以对上一层没有影响break;//1次旋转完成后不需要再调整了}
}
3.3旋转调整!!!!!!
1.新节点插入较高右子树的右侧---右右:左单旋


由上述可知,c必定是x类型的avl树,如果是其他类型的,可能60这个节点的平衡因子就变成-2或2了,(当然,这只是单独举一个例子分析,便于分析代码,不代表所有情况)

void rotateL(node* parent)//左旋,(新节点插入到较高右子树的右侧)// 1.右右
{node* subr = parent->_right;node* subrl = subr->_left;parent->_right = subrl;subr->_left= parent;node* ppnode = parent->_parent;parent->_parent = subr;if (subrl) //subrl可能为空!!!!!!!!!!!!!!!{subrl->_parent = parent;}if (parent == root) // 即如果parent->_parent==nullptr{root = subr;subr->_parent = nullptr;}else{if (ppnode->_left == parent) //需要再查找一下放左边还是右边{ppnode->_left = subr;}else if (ppnode->_right == parent){ppnode->_right = subr;}subr->_parent = ppnode;}parent->bf = subr->bf = 0; //重置平衡因子
}
2.新节点插入较高左子树的左侧---左左:右单旋

和左单旋分析方法一致
void rotateR(node* parent)//右旋,(新节点插入到较高左子树的左侧)// 2.左左
{node* subl = parent->_left;node* sublr = subl->_right;parent->_left = sublr;if (sublr) //sublr可能为空!!!!!!!sublr->_parent = parent;node* ppnode = parent->_parent;subl->_right = parent;parent->_parent=subl;if (root == parent){root = subl;subl->_parent = nullptr;}else{if (ppnode->_left == parent){ppnode->_left = subl;}else if (ppnode->_right == parent){ppnode->_right = subl;}subl->_parent = ppnode;}subl->bf = parent->bf = 0;//重置平衡因子}
3. 新节点插入较高左子树的右侧---左右:先左单旋再右单旋

十分巧妙的是:经过一次左旋(parent->_left)后,就变成左左类型了,这样就能复用右单旋(parent)达成平衡

void rotateLR(node* parent)//左旋一次,再右旋一次,还需要根据不同的_bf更新平衡因子
{node* subl = parent->_left;node* sublr= subl->_right;int _bf = sublr->bf;rotateL(parent->_left);rotateR(parent);if (_bf == 0){//sublr自己就是新增加的节点parent->bf = subl->bf = sublr->bf = 0;}else if (_bf == 1){//sublr的右子树新增parent->bf = 0;subl->bf = -1;sublr->bf = 0;}else if (_bf == -1){//sublr的左子树新增parent->bf = 1;subl->bf = 0;sublr->bf = 0;}
}
4.新节点插入较高右子树的左侧---右左:先右单旋再左单旋
与上方分析方法一致

void rotateRL(node* parent) //右旋一次,再左旋一次,还需要根据不同的_bf更新平衡因子
{node* subr = parent->_right;node* subrl = subr->_left;int _bf = subrl->bf;rotateR(parent->_right);rotateL(parent);if (_bf == 0){//subrl自己就是新增加的节点parent->bf = subr->bf = subrl->bf = 0;}else if (_bf == -1){//subrl的右子树新增parent->bf = 0;subr->bf = 1;subrl->bf = 0;}else if (_bf == 1){ //subrl的左子树新增parent->bf = -1;subr->bf = 0;subrl->bf = 0;}
}
总结:
假如以pParent为根的子树不平衡,即pParent的平衡因子为2或者-2,分以下情况考虑
1. pParent的平衡因子为2,说明pParent的右子树高,设pParent的右子树的根为pSubR
当pSubR的平衡因子为1时,执行左单旋
当pSubR的平衡因子为-1时,执行右左双旋
2. pParent的平衡因子为-2,说明pParent的左子树高,设pParent的左子树的根为pSubL
当pSubL的平衡因子为-1是,执行右单旋
当pSubL的平衡因子为1时,执行左右双旋
旋转完成后,原pParent为根的子树个高度降低,已经平衡,不需要再向上更新。
4.AVL树的判断
AVL树是在二叉搜索树的基础上加入了平衡性的限制,因此要验证AVL树,可以分两步:
1. 验证其为二叉搜索树
:如果中序遍历可得到一个有序的序列,就说明为二叉搜索树
2. 验证其为平衡树
(1)每个节点子树高度差的绝对值不超过1(注意节点中如果没有平衡因子)
(2)节点的平衡因子是否计算正确
void inorder()
{_inorder(root);
}void _inorder(node* root)
{if (root == nullptr)return;_inorder(root->_left);cout << root->kv.first << " ";_inorder(root->_right);
}int _height(node* root)//取得高度
{if (root == nullptr)return 0;int lh = _height(root->_left);int rh = _height(root->_right);return lh > rh ? lh + 1 : rh + 1;//记得+1
}bool isbalance()
{return _isbalance(root);
}bool _isbalance(node* root)
{if (root == nullptr)return true;int lh = _height(root->_left);int rh = _height(root->_right);if ((rh - lh) !=root->bf)//判断是否符合平衡因子计算公式{cout << "异常" << endl;return false;}return (rh - lh) <2 && (rh - lh)>-2 && _isbalance(root->_left) && _isbalance(root->_right);//递归判断左右子树
}
5.完整代码
AVL.h:
template<class K ,class V>
struct AVLtreenode
{AVLtreenode<K, V>* _left;AVLtreenode<K, V>* _right;AVLtreenode<K, V>* _parent;pair<K, V> kv;int bf;//右子树高度-左子树高度,只有孩子发生变化,bf才有可能发生变化!!!!,若改变父亲,bf不变!!!!!!AVLtreenode(const pair<K, V>& _kv):_left(nullptr),_right(nullptr),_parent(nullptr),kv(_kv),bf(0){}
};template<class K, class V>
class AVLtree
{
public:typedef AVLtreenode<K, V> node;bool insert(const pair<K,V>& _kv){if (root == nullptr){root = new node(_kv);return true;}node* parent = nullptr; //比bst多了一个parentnode* cur = root; //while (cur){parent = cur;if (cur->kv.first < _kv.first){cur = cur->_right;}else if (cur->kv.first > _kv.first){cur = cur->_left;}else{return false;}}cur = new node(_kv);if (parent->kv.first < _kv.first){parent->_right = cur;}else{parent->_left = cur;}cur->_parent = parent;//开始调整//调整平衡因子while (cur != root){if (cur == parent->_left)parent->bf--;//第一次!!!检查parent左边原来必为空else {parent->bf++;//第一次!!!检查parent右边原来必为空}if (parent->bf == 0) //相当于bf没改变,可直接退出{break;}else if (parent->bf == 1 || parent->bf == -1) //bf改变了继续向上找 {cur = parent;parent = parent->_parent;}else if(parent->bf == -2 || parent->bf == 2){ //-2||2,需要调整(旋转)if (parent->bf == 2 && cur->bf == 1){rotateL(parent);//单左旋,全在右边加}else if (parent->bf == -2 && cur->bf == -1){rotateR(parent);//单右旋,全在左边加}else if (parent->bf == 2 && cur->bf == -1){rotateRL(parent);//先右旋再左旋}else if (parent->bf == -2 && cur->bf == 1){rotateLR(parent);//先左旋再右旋}//旋转让子树变得平衡//旋转降低了子树的高度,恢复到和插入前一样的高度,所以对上一层没有影响break;//1次旋转完成后不需要再调整了}}return true;}void rotateL(node* parent)//左旋,(新节点插入到较高右子树的右侧)// 1.右右{node* subr = parent->_right;node* subrl = subr->_left;parent->_right = subrl;subr->_left= parent;node* ppnode = parent->_parent;parent->_parent = subr;if (subrl) //subrl可能为空!!!!!!!{subrl->_parent = parent;}if (parent == root) //即如果parent->_parent==nullptr{root = subr;subr->_parent = nullptr;}else{if (ppnode->_left == parent){ppnode->_left = subr;}else if (ppnode->_right == parent){ppnode->_right = subr;}subr->_parent = ppnode;}parent->bf = subr->bf = 0; //重置平衡因子}void rotateR(node* parent)//右旋,(新节点插入到较高左子树的左侧)// 2.左左{node* subl = parent->_left;node* sublr = subl->_right;parent->_left = sublr;if (sublr) //sublr可能为空!!!!!!!sublr->_parent = parent;node* ppnode = parent->_parent;subl->_right = parent;parent->_parent=subl;if (root == parent){root = subl;subl->_parent = nullptr;}else{if (ppnode->_left == parent){ppnode->_left = subl;}else if (ppnode->_right == parent){ppnode->_right = subl;}subl->_parent = ppnode;}subl->bf = parent->bf = 0;}void rotateRL(node* parent) //右旋一次,再左旋一次,还需要根据不同的_bf更新平衡因子{node* subr = parent->_right;node* subrl = subr->_left;int _bf = subrl->bf;rotateR(parent->_right);rotateL(parent);if (_bf == 0){//subrl自己就是新增加的节点parent->bf = subr->bf = subrl->bf = 0;}else if (_bf == -1){//subrl的右子树新增parent->bf = 0;subr->bf = 1;subrl->bf = 0;}else if (_bf == 1){ //subrl的左子树新增parent->bf = -1;subr->bf = 0;subrl->bf = 0;}}void rotateLR(node* parent)//左旋一次,再右旋一次,还需要根据不同的_bf更新平衡因子{node* subl = parent->_left;node* sublr= subl->_right;int _bf = sublr->bf;rotateL(parent->_left);rotateR(parent);if (_bf == 0){//sublr自己就是新增加的节点parent->bf = subl->bf = sublr->bf = 0;}else if (_bf == 1){//sublr的右子树新增parent->bf = 0;subl->bf = -1;sublr->bf = 0;}else if (_bf == -1){//sublr的左子树新增parent->bf = 1;subl->bf = 0;sublr->bf = 0;}}void inorder(){_inorder(root);}void _inorder(node* root){if (root == nullptr)return;_inorder(root->_left);cout << root->kv.first << " ";_inorder(root->_right);}int _height(node* root){if (root == nullptr)return 0;int lh = _height(root->_left);int rh = _height(root->_right);return lh > rh ? lh + 1 : rh + 1;}bool isbalance(){return _isbalance(root);}bool _isbalance(node* root){if (root == nullptr)return true;int lh = _height(root->_left);int rh = _height(root->_right);if ((rh - lh) !=root->bf){cout << "异常" << endl;return false;}return (rh - lh) <2 && (rh - lh)>-2 && _isbalance(root->_left) && _isbalance(root->_right);}private:node* root = nullptr;
};
test.c:
#define _CRT_SECURE_NO_WARNINGS
#include<iostream>
#include<map>using namespace std;#include"AVL.h"int main()
{int arr[] = { 4, 2, 6, 1, 3, 5, 15, 7, 16, 14 };//int arr[] = { 16, 3, 7, 11, 9, 26, 18, 14, 15 };AVLtree<int, int> it;for (auto i : arr){it.insert(make_pair(i,i));}it.inorder();cout << endl<<it.isbalance() << endl;return 0;
}

6.AVL树的性能
AVL树是一棵绝对平衡的二叉搜索树,其要求每个节点的左右子树高度差的绝对值都不超过1,这 样可以保证查询时高效的时间复杂度,即$og_2 (N)。但是如果要对AVL树做一些结构修改的操 作,性能非常低下,比如:插入时要维护其绝对平衡,旋转的次数比较多,更差的是在删除时, 有可能一直要让旋转持续到根的位置。因此:如果需要一种查询高效且有序的数据结构,而且数 据的个数为静态的(即不会改变),可以考虑AVL树,但一个结构经常修改,就不太适合。
相关文章:
移情别恋c++ ദ്ദി˶ー̀֊ー́ ) ——14.AVL树
1.AVL 树 1.1AVL 树的概念 二叉搜索树虽可以缩短查找的效率,但如果数据有序或接近有序二叉搜索树将退化为单支树,查 找元素相当于在顺序表中搜索元素,效率低下。因此,两位俄罗斯的数学家G.M.Adelson-Velskii 和E.M.Landis在1962…...
Python 的数据类型与操作
一、常用内置类型(Built - in Types) Python 拥有多种内置数据类型,这些类型满足了各种编程需求,从简单的数据存储到复杂的数据结构表示。 1. 数值类型(Numeric Types) 整数(int)&a…...
Python燃烧废气排放推断算法模型
🎯要点 宏观能耗场景模型参数化输入数据,分析可视化输出结果,使用场景时间序列数据模型及定量和定性指标使用线图和箱线图、饼图、散点图、堆积条形图、桑基图等可视化模型输出结果根据气体排放过程得出其时间序列关系,使用推断模…...
Qt中多语言的操作(以QtCreator为例)
1、首先,我们在代码中与文本相关的且需要支持多语言的地方,用tr来包含多语言key(多语言key是我们自己定义的),如下 //举例 QPushButton* btnnew QPushButton(this); btn->move(20,20); btn->resize(100,50); //…...
计算机毕业设计 社区医疗服务系统的设计与实现 Java实战项目 附源码+文档+视频讲解
博主介绍:✌从事软件开发10年之余,专注于Java技术领域、Python人工智能及数据挖掘、小程序项目开发和Android项目开发等。CSDN、掘金、华为云、InfoQ、阿里云等平台优质作者✌ 🍅文末获取源码联系🍅 👇🏻 精…...
html+css学习
html 元素 html元素是HTML的根元素,一个文档只能有一个,其他所有元素都是其后代元素 html有一个属性为lang,其作用是: 帮助语言合成工具确定要使用的发音帮助翻译工具确定要使用的翻译规则 当属性lang“en”则表示告诉其浏览器…...
2.gitlab ce 细粒度的权限控制
需求: 在提交merge reqeust时,必须指定审核人,并且要选审核人清单里的 有个code owners应该可以做到(gitlab ce应该也可以用) 下面是参考的文档 细粒度的代码权限怎么做?极狐GitLab 代码所有者来帮忙 -…...
G - Merchant Takahashi / F - Useless for LIS
G - Merchant Takahashi 首先考虑暴力 DP。 设最后一步走到编号 ii 的城镇的方案的最大收益为 fifi,则每次集市相当于是 fTi←fj−C∣Ti−j∣Pi(1≤j≤n)。 这样每次可以通过枚举 j 来转移,这样总时间复杂度是 O(nm) 的&…...
自然语言处理实例
引子:基于聊天机器人项目的自然语言处理(NLP)学习路线 自然语言处理(Natural Language Processing,简称 NLP)是人工智能的重要分支,旨在帮助计算机理解、生成和处理人类语言。NLP 技术广泛应用于搜索引擎、机器翻译、语音识别、文本摘要、情感分析、对话系统等领域。为…...
『功能项目』主角属性值显示【75】
本章项目成果展示 我们打开上一篇74穿戴装备的项目, 本章要做的事情是制作主角属性界面,实现在面板上显示主角的攻击力等数值 制作一个简易的主角界面(创建Image与Text显示即可) 创建一个空物体 重命名为PlayerInfo 在其子级下创…...
单片机嵌入式编程中常用技术点
Open CV,QT,Linux,多线程,网络编程,文件编程在单片机嵌入式编程中,这些技术在单片机嵌入式编程中的作用: 一、OpenCV 在单片机嵌入式编程中,虽然单片机的计算能力相对有限…...
【毕业论文+源码】基于ASP+NET的人事管理系统
引言 人事管理系统是针对企业内部人事管理设计,分角色实现对公司部门及各部门员工的增、删、改、查以及对员工考勤的管理。 编写目的: 在系统需求分析的基础上,对需求分析中产生的功能模块进行过程描述,设计功能模块的内部细节&…...
计算机毕业设计 校园志愿者管理系统的设计与实现 Java实战项目 附源码+文档+视频讲解
博主介绍:✌从事软件开发10年之余,专注于Java技术领域、Python人工智能及数据挖掘、小程序项目开发和Android项目开发等。CSDN、掘金、华为云、InfoQ、阿里云等平台优质作者✌ 🍅文末获取源码联系🍅 👇🏻 精…...
速通LLaMA2:《Llama 2: Open Foundation and Fine-Tuned Chat Models》全文解读
文章目录 概览LLaMA和LLaMA2的区别AbstractIntroductionPretrainingFine-tuning1. 概括2、Supervised Fine-Tuning(SFT)3、⭐Reinforcement Learning with Human Feedback(RLHF)🔺总览Training Objectives:…...
如何使用VM中win10搭建Hfish蜜罐(危险感知平台)。从下载到部署详细教程
得而不惜就该死。 -----古月方源 引言:最近跟一个老师做东西,叫我搞清楚蜜罐的搭建和一些底层逻辑,所以记录一下。 一、实验准备 (一)win10虚拟机 (若有需要可以后台私信) (二&…...
Rust: AES 加密算法库
在Rust中,进行AES加密通常会用到一些现有的库,因为Rust标准库中并不直接提供AES加密的API。一个非常流行的库是crypto-box或者更广泛使用的ring库,但ring库由于依赖问题有时可能难以编译,另一个常用的库是cryptography的Rust绑定&…...
计算机网络34——Windows内存管理
1、计算机体系结构 2、内存管理 分为连续分配管理和非连续分配管理 在块内存在的未使用空间叫内部碎片,在块外存在的未使用空间叫外部碎片 固定分区分配可能出现内部碎片,动态分区分配可能出现外部碎片 3、逻辑地址和实际地址的互相转换 4、缺页中断 …...
Redisson 总结
1. 基础使用 1.1 引入依赖 <dependencies><dependency><groupId>org.redisson</groupId><artifactId>redisson-spring-boot-starter</artifactId></dependency> </dependencies>包含的依赖如下 1.2 配置文件 其实默认主机就…...
EfficientFormer实战:使用EfficientFormerV2实现图像分类任务(一)
摘要 EfficientFormerV2是一种通过重新思考ViT设计选择和引入细粒度联合搜索策略而开发出的新型移动视觉骨干网络。它结合了卷积和变换器的优势,通过一系列高效的设计改进和搜索方法,实现了在移动设备上既轻又快且保持高性能的目标。这一成果为在资源受…...
文心智能体搭建步骤
通过使用文心智能体平台来创建智能体的过程。这种方法可以让没有编程经验的人也能快速构建智能体,降低了技 术门槛。以下是一些建议和心得: 1.选择合适的平台:文心智能体平台是一个优秀的选择,它提供了零代码和低代码的开发环境,极大地降低了…...
观成科技:隐蔽隧道工具Ligolo-ng加密流量分析
1.工具介绍 Ligolo-ng是一款由go编写的高效隧道工具,该工具基于TUN接口实现其功能,利用反向TCP/TLS连接建立一条隐蔽的通信信道,支持使用Let’s Encrypt自动生成证书。Ligolo-ng的通信隐蔽性体现在其支持多种连接方式,适应复杂网…...
【CSS position 属性】static、relative、fixed、absolute 、sticky详细介绍,多层嵌套定位示例
文章目录 ★ position 的五种类型及基本用法 ★ 一、position 属性概述 二、position 的五种类型详解(初学者版) 1. static(默认值) 2. relative(相对定位) 3. absolute(绝对定位) 4. fixed(固定定位) 5. sticky(粘性定位) 三、定位元素的层级关系(z-i…...
2023赣州旅游投资集团
单选题 1.“不登高山,不知天之高也;不临深溪,不知地之厚也。”这句话说明_____。 A、人的意识具有创造性 B、人的认识是独立于实践之外的 C、实践在认识过程中具有决定作用 D、人的一切知识都是从直接经验中获得的 参考答案: C 本题解…...
html css js网页制作成品——HTML+CSS榴莲商城网页设计(4页)附源码
目录 一、👨🎓网站题目 二、✍️网站描述 三、📚网站介绍 四、🌐网站效果 五、🪓 代码实现 🧱HTML 六、🥇 如何让学习不再盲目 七、🎁更多干货 一、👨…...
Caliper 负载(Workload)详细解析
Caliper 负载(Workload)详细解析 负载(Workload)是 Caliper 性能测试的核心部分,它定义了测试期间要执行的具体合约调用行为和交易模式。下面我将全面深入地讲解负载的各个方面。 一、负载模块基本结构 一个典型的负载模块(如 workload.js)包含以下基本结构: use strict;/…...
在 Spring Boot 项目里,MYSQL中json类型字段使用
前言: 因为程序特殊需求导致,需要mysql数据库存储json类型数据,因此记录一下使用流程 1.java实体中新增字段 private List<User> users 2.增加mybatis-plus注解 TableField(typeHandler FastjsonTypeHandler.class) private Lis…...
上位机开发过程中的设计模式体会(1):工厂方法模式、单例模式和生成器模式
简介 在我的 QT/C 开发工作中,合理运用设计模式极大地提高了代码的可维护性和可扩展性。本文将分享我在实际项目中应用的三种创造型模式:工厂方法模式、单例模式和生成器模式。 1. 工厂模式 (Factory Pattern) 应用场景 在我的 QT 项目中曾经有一个需…...
【堆垛策略】设计方法
堆垛策略的设计是积木堆叠系统的核心,直接影响堆叠的稳定性、效率和容错能力。以下是分层次的堆垛策略设计方法,涵盖基础规则、优化算法和容错机制: 1. 基础堆垛规则 (1) 物理稳定性优先 重心原则: 大尺寸/重量积木在下…...
FFmpeg avformat_open_input函数分析
函数内部的总体流程如下: avformat_open_input 精简后的代码如下: int avformat_open_input(AVFormatContext **ps, const char *filename,ff_const59 AVInputFormat *fmt, AVDictionary **options) {AVFormatContext *s *ps;int i, ret 0;AVDictio…...
HTML版英语学习系统
HTML版英语学习系统 这是一个完全免费、无需安装、功能完整的英语学习工具,使用HTML CSS JavaScript实现。 功能 文本朗读练习 - 输入英文文章,系统朗读帮助练习听力和发音,适合跟读练习,模仿学习;实时词典查询 - 双…...
