zoukankan      html  css  js  c++  java
  • 单向链表

    【链表】
    是一种线性表,但是并不会按线性的顺序存储数据,而是在每一个节点里存到下一个节点的指针(Pointer)。
    由于不必须按顺序存储,链表在插入的时候可以达到O(1)的复杂度,比另一种线性表顺序表快得多,
    但是查找一个节点或者访问特定编号的节点则需要O(n)的时间,
    而顺序表相应的时间复杂度分别是O(㏒ n)和O(1)。 
    【单向链表】 
    是链表的一种,其特点是链表的链接方向是单向的,对链表的访问要通过顺序读取从头部开始
      一个节点包含两个域,一个信息域和一个指针域,指针域指向下一个节点,最后一个节点指向一个空值
    单向链表有一个头节点,可以不存储任何信息,包含一个指向第一个节点的指针
    特点:
    1.访问节点速度慢,需要从头节点一次遍历链表
    2.更新节点快,节点不需要移动,更改相应指针即可

    单向链表节点类定义如下:

    public class Node<AnyType>{
    
        AnyType element;
        
        Node<AnyType> next;    
    
    }

    要实现单向列表的相关操作,可定义类SingleLinkedList如下:

    public class SingleLinkedList<AnyType>{
    
        private static class Node<AnyType>{
    
            AnyType element;
            
            Node<AnyType> next;    
            
            public Node(AnyType element){
                this.element = element;
            }
    
        }
        
        private Node<AnyType> header = new Node<AnyType>(null);
        
        private int size = 0;
        
        public boolean isEmpty(){
            return size==0;
        }
        
        public void makeEmpty(){
            header = null;
        }
        
        public int size(){
            return size;
        } 
        
        /**
         * Get element by index
         * */
        public AnyType get(int index){        
            return getNode(index).element;
        }
        
        /**
         * Add the element to the end of this list
         * */
        public void add(AnyType element){
            add(new Node<AnyType>(element));
        }
        
        /**
         * Inserts the specified element at the specified position in this list
         * 插入逻辑:
         * 1.创建一个新节点
         * 2.将原index节点的前一个节点的指针指向新节点
         * 3.将新节点的指针指向index节点
         * 4.插入后,新节点的位置为index
         * */
        public void add(int index,AnyType element){
            Node<AnyType> newNode = new Node<AnyType>(element);
            Node<AnyType> previous = getNode(index-1);
            //index节点
            Node<AnyType> node = previous.next;
            //将原index节点的前一个节点的指针指向newNode
            previous.next = newNode;
            //将newNode的指针指向index节点
            newNode.next = node;
            size++;
        }
        
        /**
         * Inserts the specified element at the specified position in this list
         * 删除逻辑:
         * 1.获得index节点的前一个节点previousNode
         * 2.获得index节点的后一个节点nextNode
         * 3.将previousNode的指针指向nextNode
         * */
        public void remove(int index){
            Node<AnyType> previous = getNode(index-1);
            Node<AnyType> next = previous.next.next;
            previous.next = next;
            size--;
        }
    
        
        /**
         * 定义此方法是为了便于测试
         * */
        public List<Integer> getElements(){
            if(isEmpty()){
                return null;
            }else{
                List<Integer> elements = new ArrayList<Integer>();
                Node<AnyType> node = (Node<AnyType>) header;
                while(node.next!=null){
                    node = node.next;
                    elements.add(((Element)node.element).getValue());
                }
                return elements;
            }
        }
        
    
        //private methods
        private Node<AnyType> getNode(int index){
            if(isEmpty()){
                throw new RuntimeException("List is empty");
            }
            int i = 0;
            Node<AnyType> node = header;
            while(i<=index){
                node = node.next;
                i++;
            }
            return node;
        }
        
        private void add(Node<AnyType> newNode){
            Node<AnyType> node = header;
            while(node.next!=null){
                node = node.next;
            }
            node.next = newNode;
            size++;
        }
            
    }

    如何实现链表反向:

    /**
         * 链表反向
         * */
        public void reverse(){
            if(!isEmpty()){
                reverse(header.next,header.next.next);
            }        
        }
    private void reverse(Node<AnyType> node,Node<AnyType> nextNode){
            if(nextNode.next!=null){
                reverse(nextNode,nextNode.next);            
            }else{
                //如果该节点是表尾,那么用头节点指向此节点
                header.next = nextNode;
            }    
            //该节点的指针指向前一个节点P,并将节点P的指针设置为空
            nextNode.next = node;
            node.next = null;
        }
  • 相关阅读:
    angularJS处理table中checkbox的选中状态
    大公司常见表格样式
    github:当你想要使用VSCODE开心提交代码时,出现Git:git@github.com:Permission denied(publickey)解决方案
    angularjs 使用angular-sortable-view实现拖拽效果(包括拖动完成后的方法使用)
    sortable结合angularjs实现拖动排序
    promise的一个简单易懂实例
    angular-ui-select (系列二)远程搜索,页面方框显示的值跟传给后台的值不一样解决方案
    leetcode刷题笔记二十九 两数相除
    scala学习 包和引入
    leetcode刷题笔记二十六 、 二十七、 二十八
  • 原文地址:https://www.cnblogs.com/tangyanbo/p/4282369.html
Copyright © 2011-2022 走看看