链表的java实现(单向双向链表,单向链表的反转)

Posted 枸杞仙人

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了链表的java实现(单向双向链表,单向链表的反转)相关的知识,希望对你有一定的参考价值。

链表

链表

虽然顺序表的查询很快,时间复杂度为O(1),但是增删的效率是比较低的,因为每一次增删操作都伴随着大量的数据元素移动。这个问题有没有解决方案呢?有,我们可以使用另外一种存储结构实现线性表,链式存储结构。
链表是一种物理存储单元上非连续、非顺序的存储结构,其物理结构不能只管的表示数据元素的逻辑顺序,数据元素的逻辑顺序是通过链表中的指针链接次序实现的。链表由一系列的结点(链表中的每一个元素称为结点)组成,结点可以在运行时动态生成。

按照面向对象的思想,我们可以设计一个类,来描述结点这个事物,用一个属性描述这个结点存储的元素,用来另外一个属性描述这个结点的下一个结点。

类名Node
构造方法Node(T t,Node next):创建Node对象
成员变量T item:存储数据
Node next:指向下一个结点

结点类实现:

public class Node<T> 
    //存储元素
    public T item;
    //下一个元素的地址
    public Node next;
    
    public Node (T item,Node next)
        this.item = item;
        this.next = next;
    

生成链表:

public static void main(String[] args) throws Exception
        //构建结点
        Node<Integer> first = new Node<>(11,null);
        Node<Integer> second = new Node<>(13,null);
        Node<Integer> third = new Node<>(12,null);
        Node<Integer> forth = new Node<>(7,null);
        Node<Integer> fifth = new Node<>(8,null);
        //生成链表
        first.next = second;
        second.next = third; 
        third.next = forth; 
        forth.next = fifth;
    

单向链表

单向链表是链表的一种,它由多个结点组成,每个结点都由一个数据域和一个指针域组成,数据域用来存储数据,指针域用来指向其后继结点。链表的头结点的数据域不存储数据,指针域指向第一个真正存储数据的结点。

单向链表API设计

类名LinkList
构造方法LinkList():创建LinkList对象
成员方法1.public void clear():空置线性表
2.publicboolean isEmpty():判断线性表是否为空,是返回true,否返回false
3.public int length():获取线性表中元素的个数
4.public T get(int i):读取并返回线性表中的第i个元素的值
5.public void insert(T t):往线性表中添加一个元素;
6.public void insert(int i,T t):在线性表的第i个元素之前插入一个值为t的数据元素。
7.public T remove(int i):删除并返回线性表中第i个数据元素。
8.public int indexOf(T t):返回线性表中首次出现的指定的数据元素的位序号,若不存在,则返回-1。
成员内部类private class Node:结点类
成员变量1.private Node head:记录首结点
2.private int N:记录链表的长度

单向链表代码实现

单向链表类:

import java.util.Iterator;

public class LinkList<T> implements Iterable<T> 
    //头节点
    private Node head;
    //链表长度
    private int N;

    public LinkList()
        //初始化
        head = new Node(null,null);
        N = 0;
    

    //清空链表
    public void clear()
        head.next = null;
        head.item = null;
        N = 0;
    

    //判断是否为空
    public boolean isEmpty()
        return N == 0;
    

    //获取元素个数
    public int length()
        return N;
    

    //获取第i个元素
    public T get(int i)
        if (i < 0 || i >= N)
            throw new RuntimeException("位置不合法");
        
        Node n = head.next;
        for (int index = 0;index < i;index++)
            n = n.next;
        
        return n.item;
    

    //添加一个元素
    public void insert(T t)
        //找到最后一个节点
        Node n = head;
        while (n.next != null)
            n = n.next;
        
        Node newOne = new Node(t,null);
        n.next = newOne;
        //链表长度++
        N++;
    

    //在i位置添加一个元素
    public void insert(int i,T t)
        if (i<0||i>=N)
            throw new RuntimeException("位置不合法!");
        

        //寻找i位置之前的节点
        Node pre = head;
        for (int j = 0; j <= i-1; j++) 
            pre = pre.next;
        
        //找到位置i的节点
        Node curr = pre.next;
        //构建新节点,让新节点指向i节点
        Node newNode = new Node(t,curr);

        //让i位置之前的节点指向这个节点
        pre.next = newNode;
        //链表长度++
        N++;
    

    //删除并返回第i个位置的元素
    public T delete(int i)
        if (i<0||i>=N)
            throw new RuntimeException("位置不合法!");
        
        //寻找i之前的元素
        Node pre = head;
        for (int index = 0; index <=i-1; index++) 
            pre = pre.next;
        
        //当前i位置的结点
        Node curr = pre.next;

        //前一个结点指向下一个结点,删除当前结点
        pre.next = curr.next;

        //长度-1
        N--;
        return curr.item;
    

    //返回链表中首次出现指定元素的序号,弱不存在返回-1
    public int indexOf(T t)
        Node n = head;
        for (int i = 0;n.next!=null;i++)
            n = n.next;
            if (n.item.equals(t))
                return i;
            
        
        return -1; 

    //节点类
    private class Node
        //存储数据
        T item;
        //下个结点
        Node next;

        public Node(T item,Node next)
            this.item = item;
            this.next = next;
        
    

    @Override
    public Iterator iterator() 
        return new LIterator();
    
    
    private class LIterator implements Iterator<T>
        private Node n;
        
        public LIterator()
            this.n = head;
        

        @Override
        public boolean hasNext() 
            return n.next != null;
        

        @Override
        public T next() 
            n = n.next;
            return n.item;
        
    

测试类:

public class LinkListTest 
    public static void main(String[] args) throws Exception

        LinkList<Integer> list = new LinkList<>();
        System.out.println(list.length());
        list.insert(0,1);
        list.insert(1,2);
        list.insert(2,3);
        list.insert(3,4);
        list.insert(4,5);

        //测试length方法
        for (int i : list) 
            System.out.println(i);
        
        System.out.println(list.length());
        System.out.println("-------------------");

        //测试get方法
        System.out.println(list.get(2));
        System.out.println("------------------------");

        //测试remove方法
        int remove = list.remove(1);
        System.out.println(remove);
        System.out.println(list.length());
        System.out.println("----------------");
        for (int i : list)  System.out.println(i); 
    

0
1
2
3
4
5
5
---------------
3
---------------
2
4
---------------
1
3
4
5

双向链表

双向链表也叫双向表,是链表的一种,它由多个结点组成,每个结点都由一个数据域和两个指针域组成,数据域用来存储数据,其中一个指针域用来指向其后继结点,另一个指针域用来指向前驱结点。链表的头结点的数据域不存储数据,指向前驱结点的指针域值为null,指向后继结点的指针域指向第一个真正存储数据的结点。

结点API设计

类名Node
构造方法Node(T t,Node pre,Node next):创建Node对象
成员变量T item:存储数据
Node next:指向下一个结点
Node pre:指向上一个结点

双向链表API设计

类名TowWayLinkList
构造方法TowWayLinkList():创建TowWayLinkList对象
成员方法1.public void clear():空置线性表
2.publicboolean isEmpty():判断线性表是否为空,是返回true,否返回false
3.public int length():获取线性表中元素的个数
4.public T get(int i):读取并返回线性表中的第i个元素的值
5.public void insert(T t):往线性表中添加一个元素;
6.public void insert(int i,T t):在线性表的第i个元素之前插入一个值为t的数据元素。
7.public T remove(int i):删除并返回线性表中第i个数据元素。
8.public int indexOf(T t):返回线性表中首次出现的指定的数据元素的位序号,若不存在,则返回-1。
9.public T getFirst():获取第一个元素
10.public T getLast():获取最后一个元素
成员内部类private class Node:结点类
成员变量1.private Node first:记录首结点
2.private Node last:记录尾结点
2.private int N:记录链表的长度

双向链表代码实现

双向链表类:

import java.util.Iterator;

public class TwoWayLinkList<T> implements Iterable<T> 
    //结点类
    private class Node
        public Node(T item, Node pre, Node next)
            this.item = item;
            this.pre = pre;
            this.next = next;
        

        //存储数据
        private T item;
        //存储上一个结点
        private Node pre;
        //存储下一个节点
        private Node next;
    

    //首结点
    private Node head;
    //尾结点
    private Node last;
    //链表长度
    private int N;

    //构造方法
    public TwoWayLinkList()
        last = null;
        head = new Node(null,null,null);
        N = 0;
    

    //清空链表
    public void clear()
        N = 0;
        last = null;
        head.next = last;
        head.pre = null;
        head.item = null;
    

    //判断是否为空
    public boolean isEmpty()
        return N==0;
    

    //获取元素个数
    public int length()
        return N;
    

    //获取位置i的元素值
    public T get(int i)
        if (i<0 || i>= N)
            throw new RuntimeException("位置不合法");
        
        Node curr = head.next;
        for (int j = 0; j < i; j++) 
            curr = curr.next;
        
        return curr.item;
    

    //添加一个元素在末尾
    public void insert(T t)
        if(last == null)
            last = new Node(t,head,null);
            head.next = last;
        else
            Node oldlast = last;
            Node node = new Node(t,oldlast,null);
            oldlast.next = node;
            last = node;
        
        N++;
    

    //在i位置添加一个元素
    public void insert(int i, T t)
        if (i<0 || i>=N)
            throw new RuntimeException("位置不合法");
        

        //找到i位置的前一个结点
        Node pre = head;
        for (int index=0; index < i;index++)
            pre = pre.next;
        

        //设置当前结点
        Node curr = pre.next;
        //构建这个结点
        Node newNode = new Node(t,pre,curr);
        curr.pre = newNode;
        pre.next = newNode;
        //长度++
        N++;
    

    //删除并返回第i个元素
    public T remove(int i)
        if (i<0 || i>=N)
            throw new RuntimeException("位置不合法");
        
        //找到前一个结点
        Node pre = head;
        for (int index=0; index < i;index++)
            pre = pre.next;
        
        //设置当前结点
        Node curr = pre.next;
        //找到后一个结点
        Node curr_next = curr.next;

        pre.next = curr_next;
        curr_next.pre = pre;
        N--;
        return curr.item;
    

    //返回表中某元素首次出现的index,如果不存在就返回-1
    public int indexOf(T t)
        Node n = head;
        for (int i = 0; n.next != null ; i++) 
            n = n.next;
            if (n.next.equals(t))
                return i;
            
        
        return -1;
    

    //获取第一个元素
    public T getFirst()
        if (isEmpty())
            return null;
        
        return head.next.item;
    

    //获取最后一个元素
    public T getLast()
        if (isEmpty())
            return null;
        
        return last.item;
    

    @Override
    public Iterator<T> iterator() 
        return new TIterator();
    

    private class TIterator implements Iterator
        private Node n =head;

        @Override
        public boolean hasNext() 
            return n.next != null;
        

        @Override
        public Object next() 
            n = n.next;
            return n.item;
        
    

Test类:

public class TwoWayLinkListTest 
    public static void main(String[] args) throws Exception

        TwoWayLinkList<Integer> list = new TwoWayLinkList<>();
        System.out.println(list.length());
        list.insert(0,1);
        list.insert(1,2);
        list.insert(2,3);
        list.insert(3,4);
        list.insert(4,5);

        //测试length方法
        for (int i : list) 
            System.out.println(i);
        
        System.out.println(list.length());
        System.out.println("-------------------");

        以上是关于链表的java实现(单向双向链表,单向链表的反转)的主要内容,如果未能解决你的问题,请参考以下文章

用Java语言实现单向链表

[算法]反转单向链表和双向链表

算法总结之 反转单向和双向链表

[单向链表与双向链表的实现]

双向链表的实现(双向链表与单向链表的简单区别联系和实现)

双向链表的实现(双向链表与单向链表的简单区别联系和实现)