数据结构篇(2) ts实现单循环链表
JS的class可以通过extends关键字实现类似其他语言的继承效果,比起使用一个extends关键字,在es5中实现继承要复杂一些,可以通过修改原型链的方法实现继承,让一个原型对象等于另一个类型的实例等等,虽然也能够实现,但是不够直观。
constructor()方法中的super()表示调用父类的构造函数,在这里就是调用SIngleList类里面的构造函数。
接下来就可以使用SingleList类中已经实现的函数了,但是由于单向循环链表的某些操作还是不同于单链表的,所以对SingleList类中的一些方法,需要在CirSingleList类中重写。
//继承单链表的属性和方法,有一些方法需要重写
class CirListNode extends ListNode {
constructor() {
super()
}
// 在单循环链表中寻找最后一个节点
/** 使用count进行计数,如果与当前链表的长度相同就返回子节点,这样就可以避免陷入无限循环*/
findLast(): NodeItem {
let count = 0;
let currNode = this.head;
while (currNode.next) {
currNode = currNode.next;
count++;
if (count === this.size) {
return currNode;
}
}
return this.head;
}
// 在单循环链表中寻找数据
/** 如果对应元素不存在会导致无限循环,
所以需要重写搜索函数,如果当前节点等于最后一个子节点就结束循环返回null*/
find(item: any): NodeItem {
let lastNode = this.findLast();
let currNode = this.head;
while (currNode.data != item) {
if (currNode == lastNode) {
currNode = null;
break;
}
currNode = currNode.next;
}
return currNode;
}
// 在数据为item的节点后面插入数据为element元素的节点
/**
* 1. 插入的是头节点位置,如果当前链表为空,则将新节点插入到head节点后面,并指向自己,形成循环
* 2,插入的是头节点位置,当前链表不为空,则将新节点指向头节点的下一个节点,然后再将头节点指向新节点,再将尾节点指向新节点,形成一个新的循环
* 3, 插入的是中间位置,就正常插入即可
*/
insert(item: any, element: any): void {
let newNode = new NodeItem(element);
let itemNode = this.find(item);
let lastNode = this.findLast();
// 插入的位置处于头结点之后,第一个节点之前
if (item === 'head') {
if (this.size === 0) {
this.head.next = newNode;
newNode.next = newNode;
} else {
newNode.next = this.head.next;
this.head.next = newNode;
lastNode.next = newNode;
}
this.size++;
return;
}
// 处于链表中后位置时
newNode.next = itemNode.next;
itemNode.next = newNode;
this.size++;
}
/**
* 1. 当前删除的节点是第一个节点时,如果此时单向循环链表只有一个节点,直接将此单向链表置空即可
* 2. 当前删除的节点是第一个节点时, 且此时单向循环链表不仅只有一个节点时,此时将头节点的next指针指向要删除节点的下一个节点,并将最后一个节点指向要删除节点的下一个节点
* 3. 除了前面的两种情况之外,只要将删除节点的前一个节点next指针指向要删除节点的后一个系欸但即可
*/
remove(item: any): void {
let lastNode = this.findLast();
let itemNode = this.find(item);
let preCurNode = this.head;
while (preCurNode.next !== itemNode) {
preCurNode = preCurNode.next;
}
if (itemNode === this.head.next) {
if (this.size === 1) {
this.head.next = null;
} else {
this.head.next = itemNode.next;
lastNode.next = itemNode.next;
}
} else {
preCurNode.next = itemNode.next;
}
this.size--;
}
/**
* 根据count计数来输出链表内容,防止陷入无限循环
*/
display(): void {
let count = 0;
let currNode = this.head;
let str = '';
while (count !== this.size) {
currNode = currNode.next;
str += currNode.data + '=>';
count++;
}
console.log(str)
}
/**
* 在尾部插入数据
* 用写好的findLast()方法,找到最后一个节点,然后将最后一个节点next指针指向新的节点,再将新的节点指向此链表的第一个节点即可。
*/
append(element: any): void {
let lastNode = this.findLast();
let newNode = new NodeItem(element);
lastNode.next = newNode;
newNode.next = this.head.next;
this.size++;
}
}
// n个人围成一圈,杀死第m个人,直到剩下s个人为止
// 输出存活的人的序号
let myList = new CirListNode();
function killPerson(n: any, m: any, s: any) {
for (let i = 1; i <= n; i++) {
myList.append(i);
}
let currNode = undefined;
let toKill = null;
while(myList.size>s) {
toKill = myList.advance(m, currNode); // 从currNode开始,前进m个节点
currNode = toKill; // 保存要删除的节点作为下一次循环的参数
myList.remove(toKill.data); // 删除第m个节点
}
myList.display();
}
killPerson(41, 3, 2); // head->16->31
// killPerson(5, 4, 1); // head->1
数据结构篇(2) ts实现单循环链表的更多相关文章
- 数据结构篇(2) ts实现单链表
interface NodeItem { prev: NodeItem | null next: NodeItem | null data: any } class NodeItem { prev: ...
- 新秀nginx源代码分析数据结构篇(两) 双链表ngx_queue_t
nginx源代码分析数据结构篇(两) 双链表ngx_queue_t Author:Echo Chen(陈斌) Email:chenb19870707@gmail.com Blog:Blog.csdn. ...
- 菜鸟nginx源代码剖析数据结构篇(八) 缓冲区链表ngx_chain_t
菜鸟nginx源代码剖析数据结构篇(八) 缓冲区链表 ngx_chain_t Author:Echo Chen(陈斌) Email:chenb19870707@gmail.com Blog:Blog. ...
- 菜鸟nginx源码剖析数据结构篇(八) 缓冲区链表ngx_chain_t[转]
菜鸟nginx源码剖析数据结构篇(八) 缓冲区链表 ngx_chain_t Author:Echo Chen(陈斌) Email:chenb19870707@gmail.com Blog:Blog.c ...
- 菜鸟nginx源码剖析数据结构篇(三) 单向链表 ngx_list_t[转]
菜鸟nginx源码剖析数据结构篇(三) 单向链表 ngx_list_t Author:Echo Chen(陈斌) Email:chenb19870707@gmail.com Blog:Blog.csd ...
- 数据结构篇(3)ts 实现双向链表
如今终于到了双向链表了,此前在Node结构中的prev指针终于派上了用场.由于双向链表多了一个前向指针,所以有些操作和单向链表比较起来反而更加的简单. class DbList extends Cir ...
- 数据结构篇(1) ts实现栈的基本操作和解决相关问题
interface Stack { _items: any push(element: any): void pop(): any top(): any size(): any isEmpty(): ...
- 图解Redis之数据结构篇——链表
前言 Redis链表为双向无环链表! 图解Redis之数据结构篇--简单动态字符串SDS提到Redis使用了简单动态字符串,链表,字典(散列表),跳跃表,整数集合,压缩列表这些数据结构 ...
- 图解Redis之数据结构篇——简单动态字符串SDS
图解Redis之数据结构篇--简单动态字符串SDS 前言 相信用过Redis的人都知道,Redis提供了一个逻辑上的对象系统构建了一个键值对数据库以供客户端用户使用.这个对象系统包括字符串对象 ...
随机推荐
- 【仿真】Carla之收集数据快速教程 (附完整代码) [7]
收集过程可视化展示,随后进入正文: 参考与前言 看到仿真群对这类任务下(用carla收集数据然后再做训练等) 需求量大,顺手马上写一个好了,首先收集数据需要考虑清楚: 收集什么数据,需要什么样的数据格 ...
- Blazor 003 : Razor的基础语法
上文,我们通过剖析一个最简单的 Blazor WASM 项目,讲明白了 Razor 文件是什么,以及它被转译成 C#后长什么样子.也介绍了 Razor 中最简单的一个语法:Razor Expressi ...
- kafka 如何不消费重复数据?比如扣款,我们不能重复的扣?
其实还是得结合业务来思考,我这里给几个思路: 比如你拿个数据要写库,你先根据主键查一下,如果这数据都有了,你就别插入 了,update 一下好吧. 比如你是写 Redis,那没问题了,反正每次都是 s ...
- 解释Spring支持的几种bean的作用域?
Spring框架支持以下五种bean的作用域: singleton :bean在每个Spring ioc 容器中只有一个实例. prototype:一个bean的定义可以有多个实例. request: ...
- 重载(Overload)和重写(Override)的区别。重载的 方法能否根据返回类型进行区分?
方法的重载和重写都是实现多态的方式,区别在于前者实现的是编译时的多态性,而后者实现的是运行时的多态性.重载发生在一个类中,同名的方法如果有不同的参数列表(参数类型不同.参数个数不同或者二者都不同)则视 ...
- idea-中的Mark Diretory as的内容
Sources Root:告诉idea这个文件夹及其子文件夹中包含源代码,是需要编译构建的一部分 Test Sources Root:测试源文件夹允许您将与测试相关的代码与生产代码分开.通常,源和测试 ...
- 一文读懂充电宝usb接口电路及制作原理详细
转自:http://www.elecfans.com/dianlutu/dianyuandianlu/20180511675801.html USB充电器套件,又名MP3/MP4充电器,输入AC160 ...
- C#通过LDAP访问目录服务
C#通过LDAP访问目录服务 本文介绍如何编写C#程序通过LDAP协议访问微软目录服务获得用户在目录中的属性信息.在开始部分先简单句介绍LDAP协议,然后是技术比较及实现部分. 目录 什么是LDAP? ...
- css样式权重优先级,css样式优先级
原文:http://www.bkjia.com/Javascri... 样式选择器权重优先级: important > 内嵌样式 > ID > 类 > 标签 | 伪类 | 属性 ...
- 小程序入门系列之 tabBar
本系列为简单入门系列,以一定概括性思路来叙述内容,具体可以查看官网 大部分的电商应用都是底部或顶部多 tab 的模式. 下面我们从配置角度来分析一下: 第一个:position 配置如下: 默认是 b ...