scala 实现链表

简介: 在学习数据结构时学到了链表,因为日常工作,写java写多了,想换个语言玩下,就试着想用scala实现一个链表。哪到知道碰到很多问题,这里记录下使用scala时,所遇到的那些坑。

在学习数据结构时学到了链表,因为日常工作,写java写多了,想换个语言玩下,就试着想用scala实现一个链表。哪到知道碰到很多问题,
这里记录下使用scala时,所遇到的那些坑。

java链表

首先,我们先使用java来实现一个链表最简单的链表

public static class Node<T> {

    /**
     * 这里为了方便,就不用get和set方法那一套了
     */
    public Node next;

    public T data;

}
public static void main(String[] args) {
    Node<Integer> headNode = new Node<>();
    headNode.data = 1;

    Node<Integer> secondNode = new Node<>();
    secondNode.data = 2;
    headNode.next = secondNode;

    Node<Integer> thirdNode = new Node<>();
    thirdNode.data = 3;
    secondNode.next = thirdNode;

    Node<Integer> fourthNode = new Node<>();
    fourthNode.data = 4;
    thirdNode.next = fourthNode;

    Node<Integer> fifthNode = new Node<>();
    fifthNode.data = 5;
    fourthNode.next = fifthNode;

    //遍历node
    Node<Integer> currentNode = headNode;
    while (currentNode != null) {
        System.out.println(currentNode.data);
        currentNode = currentNode.next;
    }
}

代码很简单,就不解释了,我们使用java实现了如下的数组。

img_4d9d1bfa317fea75f1b245929cd13544.png

同样的代码,我们使用scala实现一遍。

scala链表

case class Node[E](var next: Node[E], var data: E) {
}

def main(args: Array[String]): Unit = {
    val headNode = Node(null, 1)

    val secondNode = Node(null, 2)
    headNode.next = secondNode

    val thirdNode = Node(null, 3)
    secondNode.next = thirdNode

    val fourthNode = Node(null, 4)
    thirdNode.next = fourthNode

    val fifthNode = Node(null, 5)
    fourthNode.next = fifthNode

    var currentNode = headNode
    while (currentNode != null) {
      println(currentNode.data)
      currentNode = currentNode.next
    }
}

使用scala来实现链表也非常简单,除了预发以外基本上和java一模一样。

头部节点

好像从上面看,也没啥问题啊。考虑这样的情况,我们在实现一个链表时,经常会有一个设置一个不表示任何意义的头结点,以便于统一链表的操作(add和remove操作)

img_ebc8a4228fac1791f6456d17830c973d.png

先来看看使用java如何创建这样一个“空节点”

public static class Node<T> {

    /**
     * 这里为了方便,就不用get和set方法那一套了
     */
    public Node next;

    public T data;

}

public static class List<T>{

    private Node<T> headNode = new Node<>();
}

public static void main(String[] args) {
   
    List<Integer> list = new List<>();

}

没啥问题

同样的代码,看看scala


img_654e770a4aae44bf17be2aa6cdf13bcd.png

报了个错,Type mismatch, expected:E actual:Null ,scala的泛型不止是用于引用类型。

img_95d2f34ca81bd00f6db12608831e291d.png

看这个图可以知道,null仅仅是AnyRef的子类,对于AnyVal这样的类型,比如Int,无法使用null赋值。

解决方法一:限制泛型的下界

即,限制泛型必须是Null的父类型

case class Node[E](var next: Node[E], var data: E) {
}

class List[E >: Null] {
var headNode = new Node[E](null, null)
}

def main(args: Array[String]): Unit = {
    val l = new List[Integer] //问题,这里只能用Integer,无法使用Int了
}

这种方式,解决了无法使用null赋值的问题。但是使用这种方式,必须使用new List[Integer] 无法使用基本类型比如 new List[Int]

解决方法二:todo

目录
相关文章
|
存储
链表的实现:无头单向非循环链表的实现
链表的实现:无头单向非循环链表的实现
80 0
链表的实现:无头单向非循环链表的实现
|
Java
两个链表的第一个公共节点(Java实现)
两个链表的第一个公共节点(Java实现)
142 0
两个链表的第一个公共节点(Java实现)
|
消息中间件 存储 缓存
基于数组和链表实现队列
创建大数组实现对象:里面包含的信息公共初始化: 初始化页工厂:索引页工厂、数据页工厂、元数据页工厂,初始化数组索引、初始化数据页索引,通过队列前置索引页工厂获取索引页,获取队列front索引buffer,获取front,也即索引。这个实现和kafka是类似的,也即需要有相关页信息 入队列:进行入队操作是一个追加的操作,首先判断容量是否够,不够,则进行扩容操作。通过缓存拿到映射页实现,然后通过映射页。再通过锁,仅锁定创建页,索引用完后进行移除操作,映射页面实现,使用双向校验,如果为空,则创建页索引对象,通过索引拿到文件名称,然后通过读写通道进行读写操作。使用fileChannal调用映射方法获取
151 0
基于数组和链表实现队列
|
Java
简洁明了!Java实现单向环形链表以解决约瑟夫环Josepfu问题
简洁明了!Java实现单向环形链表以解决约瑟夫环Josepfu问题
181 0
简洁明了!Java实现单向环形链表以解决约瑟夫环Josepfu问题
|
存储 算法 小程序
数据结构与算法——第三节 链表(单向不循环不带头+双向循环带头 C实现+源码剖析+运行+思路分析)
可以看到 ,我如果要吧黑球插入到白球里面,显然,我要把7号位的球移到8号位,5号位的球移到6号位...然后最后才能把2号位的求插进去。如果有N个数据,那么它的算法的时间复杂度达到了O(N)!
122 0
|
存储 JavaScript 开发者
TypeScript实现链表与变相链表
TypeScript实现链表与变相链表
TypeScript实现链表与变相链表
|
存储
【数据结构】认识链表和模拟实现单链表(1)
【数据结构】认识链表和模拟实现单链表
113 0
【数据结构】认识链表和模拟实现单链表(1)
|
存储 C语言 Kotlin
重学数据结构-使用Kotlin实现链表及其他扩展
很简单,链表不像数组那样,不需要我们主动扩容,我们只需要类似递归一样,一层套一层即可,即node1持有node2的引用,node2持有node3…,相应的每次插入我们只需要更改头结点即可,当node-x持有的下一个node引用为null时,我们也可以判定,此时为链表尾节点。
277 0
Data Structures (二) - 链表LinkedList实现(Part B)
Data Structures (二) - 链表LinkedList实现(Part B)
Data Structures (二) - 链表LinkedList实现(Part B)