java實現單項鍊表(複製,本身總結的在下一篇)
- 1、單向鏈表的結構。
-
- (1)、首先節點的結構,其中包含本節點內容,以及須要指向下一個節點。
-
-
- Java代碼
- private static class Entry<E>{
- E e;
- Entry<E> nextEntry;
-
- public Entry(E e,Entry<E> nextEntry){
- this.e=e;
- this.nextEntry=nextEntry;
- }
- }
-
- private static class Entry<E>{
- E e;
- Entry<E> nextEntry;
-
- public Entry(E e,Entry<E> nextEntry){
- this.e=e;
- this.nextEntry=nextEntry;
- }
- }Java代碼
-
-
-
- Java代碼
-
-
- 其中e則指向本節點的對象,而nextEntry則指向下一個節點。
-
-
-
- (2)、任什麼時候候都須要知道表頭在哪裏。畢竟是單向鏈表嘛,由於只有一個方向,找到了表頭就能找到所有。
-
- Java代碼
- private Entry<E> head;
-
- private Entry<E> head;
-
- (3)、還能夠記錄鏈表中總共有多少個節點。它是在某些判斷時爲了提升效率而存在的。不是絕對須要的。畢竟知道表頭後,一個一個按順序去尋找就行了。
-
-
- Java代碼
- private int size;
-
- public int size(){
- return this.size;
- }
-
- private int size;
-
- public int size(){
- return this.size;
- }
- Java代碼
-
-
- 好了有這三樣,就足夠了。就看咱們如何用他們了。
-
-
-
- 2、內部實現。
-
- (1)、第一次向鏈表中插入。此時鏈表中節點爲null,第一次插入,無非就是把節點頭插入而已。
-
- 能夠看出就是把鏈表頭初始化了,而且鏈表大小漲1。其中modCount記錄整個鏈表修改的次數,鏈表的增長和刪除它都會增長。畢竟第一次插入相對外調用是透明的,因此應該是私有的咯。(透明就是不可見,這裏只得是外部不必知道它的存在)
-
- Java代碼
- private void addFirst(E e){
- head=new Entry<E>(e,null);
- size++;
- modCount++;
- }
-
- private void addFirst(E e){
- head=new Entry<E>(e,null);
- size++;
- modCount++;
- }
-
- (2)、表頭插入。在鏈表的頭前插入一個元素,新增的元素變成新的表頭。這個插入是效率最高的,畢竟你時刻知道鏈表的頭在哪裏。
-
- Java代碼
- public void addHead(E e){
- if(head==null){
- this.addFirst(e);
- }else{
- Entry<E> newEntry=new Entry<E>(e,head);
- head=newEntry;
- size++;
- modCount++;
- }
- }
-
- public void addHead(E e){
- if(head==null){
- this.addFirst(e);
- }else{
- Entry<E> newEntry=new Entry<E>(e,head);
- head=newEntry;
- size++;
- modCount++;
- }
- }能夠看出頭爲null的時候,則代表鏈表中沒值,只需調用第一次插入。不然對給定的元素創新增一個節點,新增節點的下一個指向頭節點,固然此時本身已經變成頭結點了,索引要更新頭節點的引用。(能夠看出想要清空鏈表,只須要將頭置爲null就行了)
-
-
-
- (3)、指定節點插入(插隊)。在鏈表的指定節點插入一個元素,效率很是低。因爲規則上你只能從隊伍第一個開始日後找,找到你要插隊位置的前一個,並將你插入其中,你先要告訴你身前人你在他身後,而且你本身要清楚你身後是誰。反正夠麻煩的。
-
-
- Java代碼
- public void addSpecifyIndex(E e,int index){
- if(index<0||index>size||size==0){
- throw new NoSuchElementException();
- }
- if(index==0){
- this.addHead(e);
- return;
- }
- int count=0;
- for (Entry<E> p=head; p!=null;p=p.nextEntry) {
- if(count+1==index){
- Entry<E> newEntry=new Entry<E>(e,p.nextEntry);
- p.nextEntry=newEntry;
- size++;
- modCount++;
- return;
- }
- count++;
- }
- }
-
- public void addSpecifyIndex(E e,int index){
- if(index<0||index>size||size==0){
- throw new NoSuchElementException();
- }
- if(index==0){
- this.addHead(e);
- return;
- }
- int count=0;
- for (Entry<E> p=head; p!=null;p=p.nextEntry) {
- if(count+1==index){
- Entry<E> newEntry=new Entry<E>(e,p.nextEntry);
- p.nextEntry=newEntry;
- size++;
- modCount++;
- return;
- }
- count++;
- }
- }
- Java代碼
-
-
- 先進行判斷index是否正確,規定不能插入null鏈表。並且不能跳着走,畢竟鏈表要連起來。因爲要找到前一個,可是表頭的前一個是沒有的,因此index==0時要單獨判斷。後面則用count進行計數,找到其index-1節點,而後進行插隊處理。
-
-
-
- (4)、尾插入。其實也是插隊了,只是老是須要插到最後一個以後。
-
- Java代碼
- public void add(E e){
- if(head==null){
- this.addFirst(e);
- }else{
- this.addSpecifyIndex(e, size);
- }
- }
-
- public void add(E e){
- if(head==null){
- this.addFirst(e);
- }else{
- this.addSpecifyIndex(e, size);
- }
- }
-
- (5)、指定節點獲取元素。效率低,一樣從頭開始找到指定的節點把其中元素取出
-
-
- Java代碼
- public E get(int index){
- if(index<0||index>=size){
- throw new NoSuchElementException();
- }
- E result=null;
- int count=0;
- for (Entry<E> p=head;p!=null;p=p.nextEntry) {
- if(count==index){
- result=p.e;
- }
- count++;
- }
- return result;
- }
-
- public E get(int index){
- if(index<0||index>=size){
- throw new NoSuchElementException();
- }
- E result=null;
- int count=0;
- for (Entry<E> p=head;p!=null;p=p.nextEntry) {
- if(count==index){
- result=p.e;
- }
- count++;
- }
- return result;
- }
- Java代碼
-
-
-
-
- (6)、指定節點刪除。效率低,一樣須要找到指定節點前一節點,直接把指定節點跳過就行了。
-
- Java代碼
- public void remove(int index){
- if(index<0||index>=size){
- throw new NoSuchElementException();
- }
- if(index==0){
- head=head.nextEntry;
- size--;
- modCount++;
- return;
- }
- int count=0;
- for (Entry<E> p=head;p.nextEntry!=null;p=p.nextEntry) {
- if(count+1==index){
- p.nextEntry=p.nextEntry.nextEntry;
- size--;
- modCount++;
- break;
- }
- count++;
- }
- }
-
- public void remove(int index){
- if(index<0||index>=size){
- throw new NoSuchElementException();
- }
- if(index==0){
- head=head.nextEntry;
- size--;
- modCount++;
- return;
- }
- int count=0;
- for (Entry<E> p=head;p.nextEntry!=null;p=p.nextEntry) {
- if(count+1==index){
- p.nextEntry=p.nextEntry.nextEntry;
- size--;
- modCount++;
- break;
- }
- count++;
- }
- }
-
- (7)、循環。爲了好進行遍歷演示,下面的就是循環遍歷所用的了,你們隨意看一下就行了。
-
-
-
- Java代碼
- private transient Entry<E> current;
-
- public void setCursor(int index){
- if(index<0||index>=size){
- throw new NoSuchElementException();
- }
- int count=0;
- for (Entry<E> p=head;p!=null;p=p.nextEntry) {
- if(count==index){
- current=p;
- break;
- }
- count++;
- }
- }
-
- public boolean hasNext(){
- return current!=null;
- }
-
- public E next(){
- E result=current.e;
- current=current.nextEntry;
- return result;
- }
-
- private transient Entry<E> current;
-
- public void setCursor(int index){
- if(index<0||index>=size){
- throw new NoSuchElementException();
- }
- int count=0;
- for (Entry<E> p=head;p!=null;p=p.nextEntry) {
- if(count==index){
- current=p;
- break;
- }
- count++;
- }
- }
-
- public boolean hasNext(){
- return current!=null;
- }
-
- public E next(){
- E result=current.e;
- current=current.nextEntry;
- return result;
- }
-
- 3、測試。。一個main方法,測試一下。
-
- Java代碼
- public static void main(String[] args) {
- SingleChain<String> singleChain=new SingleChain<String>();
- for (int i = 0; i < 4; i++) {
- singleChain.add(i+"");
- }
- //頭插入
- // singleChain.addHead("head");
- //尾插入
- // singleChain.add("tail");
- //指定節點插入
- // singleChain.addSpecifyIndex("Specify", 1);
- //指定節點刪除
- // singleChain.remove(3);
- //設置循環的初始節點
- singleChain.setCursor(0);
- int count=0;
- System.out.println("######SIZE"+singleChain.size()+"#######");
- while(singleChain.hasNext()){
- System.out.println("index:"+count+",entry:"+singleChain.next());
- count++;
- }
-
- System.out.println(singleChain.get(singleChain.size()-1));
- }
-
- public static void main(String[] args) {
- SingleChain<String> singleChain=new SingleChain<String>();
- for (int i = 0; i < 4; i++) {
- singleChain.add(i+"");
- }
- //頭插入
- // singleChain.addHead("head");
- //尾插入
- // singleChain.add("tail");
- //指定節點插入
- // singleChain.addSpecifyIndex("Specify", 1);
- //指定節點刪除
- // singleChain.remove(3);
- //設置循環的初始節點
- singleChain.setCursor(0);
- int count=0;
- System.out.println("######SIZE"+singleChain.size()+"#######");
- while(singleChain.hasNext()){
- System.out.println("index:"+count+",entry:"+singleChain.next());
- count++;
- }
-
- System.out.println(singleChain.get(singleChain.size()-1));
- }
-
-
-
- 4、所有代碼
-
-
-
- Java代碼
- package paladin.chain;
-
- import java.util.NoSuchElementException;
-
- public class SingleChain<E> implements Chain<E>{
-
- private Entry<E> head;
-
- private transient Entry<E> current;
-
- private int size;
-
- private int modCount;
-
-
- private void addFirst(E e){
- head=new Entry<E>(e,null);
- size++;
- modCount++;
- }
-
- public void addHead(E e){
- if(head==null){
- this.addFirst(e);
- }else{
- Entry<E> newEntry=new Entry<E>(e,head);
- head=newEntry;
- size++;
- modCount++;
- }
- }
-
- public void addSpecifyIndex(E e,int index){
- if(index<0||index>size||size==0){
- throw new NoSuchElementException();
- }
- if(index==0){
- this.addHead(e);
- return;
- }
- int count=0;
- for (Entry<E> p=head; p!=null;p=p.nextEntry) {
- if(count+1==index){
- Entry<E> newEntry=new Entry<E>(e,p.nextEntry);
- p.nextEntry=newEntry;
- size++;
- modCount++;
- return;
- }
- count++;
- }
- }
-
- public void add(E e){
- if(head==null){
- this.addFirst(e);
- }else{
- this.addSpecifyIndex(e, size);
- }
- }
-
- public E get(int index){
- if(index<0||index>=size){
- throw new NoSuchElementException();
- }
- E result=null;
- int count=0;
- for (Entry<E> p=head;p!=null;p=p.nextEntry) {
- if(count==index){
- result=p.e;
- }
- count++;
- }
- return result;
- }
-
- public void remove(int index){
- if(index<0||index>=size){
- throw new NoSuchElementException();
- }
- if(index==0){
- head=head.nextEntry;
- size--;
- modCount++;
- return;
- }
- int count=0;
- for (Entry<E> p=head;p.nextEntry!=null;p=p.nextEntry) {
- if(count+1==index){
- p.nextEntry=p.nextEntry.nextEntry;
- size--;
- modCount++;
- break;
- }
- count++;
- }
- }
-
- public void setCursor(int index){
- if(index<0||index>=size){
- throw new NoSuchElementException();
- }
- int count=0;
- for (Entry<E> p=head;p!=null;p=p.nextEntry) {
- if(count==index){
- current=p;
- break;
- }
- count++;
- }
- }
-
- public boolean hasNext(){
- return current!=null;
- }
-
- public E next(){
- E result=current.e;
- current=current.nextEntry;
- return result;
- }
-
- public int size(){
- return this.size;
- }
-
- public static void main(String[] args) {
- SingleChain<String> singleChain=new SingleChain<String>();
- for (int i = 0; i < 4; i++) {
- singleChain.add(i+"");
- }
- //頭插入
- // singleChain.addHead("head");
- //尾插入
- // singleChain.add("tail");
- //指定節點插入
- // singleChain.addSpecifyIndex("Specify", 1);
- //指定節點刪除
- // singleChain.remove(3);
- //設置循環的初始節點
- singleChain.setCursor(0);
- int count=0;
- System.out.println("######SIZE"+singleChain.size()+"#######");
- while(singleChain.hasNext()){
- System.out.println("index:"+count+",entry:"+singleChain.next());
- count++;
- }
-
- System.out.println(singleChain.get(singleChain.size()-1));
- }
-
- private static class Entry<E>{
- E e;
- Entry<E> nextEntry;
-
- public Entry(E e,Entry<E> nextEntry){
- this.e=e;
- this.nextEntry=nextEntry;
- }
- }
- }
歡迎關注本站公眾號,獲取更多信息