JDK1.8源码(六)——java.util.LinkedList 类

简介:   上一篇博客我们介绍了List集合的一种典型实现 ArrayList,我们知道 ArrayList 是由数组构成的,本篇博客我们介绍 List 集合的另一种典型实现 LinkedList,这是一个由链表构成的数组,关于链表的介绍,在这篇博客中 我们也详细介绍过,本篇博客我们将介绍 LinkedList 是如何实现的。

  上一篇博客我们介绍了List集合的一种典型实现 ArrayList,我们知道 ArrayList 是由数组构成的,本篇博客我们介绍 List 集合的另一种典型实现 LinkedList,这是一个由链表构成的数组,关于链表的介绍,在这篇博客中 我们也详细介绍过,本篇博客我们将介绍 LinkedList 是如何实现的。

1、LinkedList 定义

  LinkedList 是一个用链表实现的集合,元素有序且可以重复。

1 public class LinkedList<E>
2     extends AbstractSequentialList<E>
3     implements List<E>, Deque<E>, Cloneable, java.io.Serializable

  和 ArrayList 集合一样,LinkedList 集合也实现了Cloneable接口和Serializable接口,分别用来支持克隆以及支持序列化。List 接口也不用多说,定义了一套 List 集合类型的方法规范。

  注意,相对于 ArrayList 集合,LinkedList 集合多实现了一个 Deque 接口,这是一个双向队列接口,双向队列就是两端都可以进行增加和删除操作。

2、字段属性

    //链表元素(节点)的个数
    transient int size = 0;

    /**
     *指向第一个节点的指针
     */
    transient Node<E> first;

    /**
     *指向最后一个节点的指针
     */
    transient Node<E> last;

  注意这里出现了一个 Node 类,这是 LinkedList 类中的一个内部类,其中每一个元素就代表一个 Node 类对象,LinkedList 集合就是由许多个 Node 对象类似于手拉着手构成。

 1     private static class Node<E> {
 2         E item;//实际存储的元素
 3         Node<E> next;//指向上一个节点的引用
 4         Node<E> prev;//指向下一个节点的引用
 5 
 6         //构造函数
 7         Node(Node<E> prev, E element, Node<E> next) {
 8             this.item = element;
 9             this.next = next;
10             this.prev = prev;
11         }
12     }
View Code

  如下图所示:

  

  上图的 LinkedList 是有四个元素,也就是由 4 个 Node 对象组成,size=4,head 指向第一个elementA,tail指向最后一个节点elementD。

3、构造函数 

    public LinkedList() {
    }
    public LinkedList(Collection<? extends E> c) {
        this();
        addAll(c);
    }

  LinkedList 有两个构造函数,第一个是默认的空的构造函数,第二个是将已有元素的集合Collection 的实例添加到 LinkedList 中,调用的是 addAll() 方法,这个方法下面我们会介绍。

  注意:LinkedList 是没有初始化链表大小的构造函数,因为链表不像数组,一个定义好的数组是必须要有确定的大小,然后去分配内存空间,而链表不一样,它没有确定的大小,通过指针的移动来指向下一个内存地址的分配。

4、添加元素

  ①、addFirst(E e)

  将指定元素添加到链表头

  

 

 1     //将指定的元素附加到链表头节点
 2     public void addFirst(E e) {
 3         linkFirst(e);
 4     }
 5     private void linkFirst(E e) {
 6         final Node<E> f = first;//将头节点赋值给 f
 7         final Node<E> newNode = new Node<>(null, e, f);//将指定元素构造成一个新节点,此节点的指向下一个节点的引用为头节点
 8         first = newNode;//将新节点设为头节点,那么原先的头节点 f 变为第二个节点
 9         if (f == null)//如果第二个节点为空,也就是原先链表是空
10             last = newNode;//将这个新节点也设为尾节点(前面已经设为头节点了)
11         else
12             f.prev = newNode;//将原先的头节点的上一个节点指向新节点
13         size++;//节点数加1
14         modCount++;//和ArrayList中一样,iterator和listIterator方法返回的迭代器和列表迭代器实现使用。
15     }
View Code

  ②、addLast(E e)和add(E e)

  将指定元素添加到链表尾

 1     //将元素添加到链表末尾
 2     public void addLast(E e) {
 3         linkLast(e);
 4     }
 5     //将元素添加到链表末尾
 6     public boolean add(E e) {
 7         linkLast(e);
 8         return true;
 9     }
10     void linkLast(E e) {
11         final Node<E> l = last;//将l设为尾节点
12         final Node<E> newNode = new Node<>(l, e, null);//构造一个新节点,节点上一个节点引用指向尾节点l
13         last = newNode;//将尾节点设为创建的新节点
14         if (l == null)//如果尾节点为空,表示原先链表为空
15             first = newNode;//将头节点设为新创建的节点(尾节点也是新创建的节点)
16         else
17             l.next = newNode;//将原来尾节点下一个节点的引用指向新节点
18         size++;//节点数加1
19         modCount++;//和ArrayList中一样,iterator和listIterator方法返回的迭代器和列表迭代器实现使用。
20     }
View Code

  ③、add(int index, E element)

  将指定的元素插入此列表中的指定位置

 

    //将指定的元素插入此列表中的指定位置
    public void add(int index, E element) {
        //判断索引 index >= 0 && index <= size中时抛出IndexOutOfBoundsException异常
        checkPositionIndex(index);

        if (index == size)//如果索引值等于链表大小
            linkLast(element);//将节点插入到尾节点
        else
            linkBefore(element, node(index));
    }
    void linkLast(E e) {
        final Node<E> l = last;//将l设为尾节点
        final Node<E> newNode = new Node<>(l, e, null);//构造一个新节点,节点上一个节点引用指向尾节点l
        last = newNode;//将尾节点设为创建的新节点
        if (l == null)//如果尾节点为空,表示原先链表为空
            first = newNode;//将头节点设为新创建的节点(尾节点也是新创建的节点)
        else
            l.next = newNode;//将原来尾节点下一个节点的引用指向新节点
        size++;//节点数加1
        modCount++;//和ArrayList中一样,iterator和listIterator方法返回的迭代器和列表迭代器实现使用。
    }
    Node<E> node(int index) {
        if (index < (size >> 1)) {//如果插入的索引在前半部分
            Node<E> x = first;//设x为头节点
            for (int i = 0; i < index; i++)//从开始节点到插入节点索引之间的所有节点向后移动一位
                x = x.next;
            return x;
        } else {//如果插入节点位置在后半部分
            Node<E> x = last;//将x设为最后一个节点
            for (int i = size - 1; i > index; i--)//从最后节点到插入节点的索引位置之间的所有节点向前移动一位
                x = x.prev;
            return x;
        }
    }
    void linkBefore(E e, Node<E> succ) {
        final Node<E> pred = succ.prev;//将pred设为插入节点的上一个节点
        final Node<E> newNode = new Node<>(pred, e, succ);//将新节点的上引用设为pred,下引用设为succ
        succ.prev = newNode;//succ的上一个节点的引用设为新节点
        if (pred == null)//如果插入节点的上一个节点引用为空
            first = newNode;//新节点就是头节点
        else
            pred.next = newNode;//插入节点的下一个节点引用设为新节点
        size++;
        modCount++;
    }
View Code

  ④、addAll(Collection<? extends E> c)

  按照指定集合的​​迭代器返回的顺序,将指定集合中的所有元素追加到此列表的末尾

  此方法还有一个 addAll(int index, Collection<? extends E> c),将集合 c 中所有元素插入到指定索引的位置。其实 

    addAll(Collection<? extends E> c) ==  addAll(size, Collection<? extends E> c)

  源码如下:

 1     //按照指定集合的​​迭代器返回的顺序,将指定集合中的所有元素追加到此列表的末尾。
 2     public boolean addAll(Collection<? extends E> c) {
 3         return addAll(size, c);
 4     }
 5     //将集合 c 中所有元素插入到指定索引的位置。
 6     public boolean addAll(int index, Collection<? extends E> c) {
 7         //判断索引 index >= 0 && index <= size中时抛出IndexOutOfBoundsException异常
 8         checkPositionIndex(index);
 9 
10         Object[] a = c.toArray();//将集合转换成一个 Object 类型的数组
11         int numNew = a.length;
12         if (numNew == 0)//如果添加的集合为空,直接返回false
13             return false;
14 
15         Node<E> pred, succ;
16         if (index == size) {//如果插入的位置等于链表的长度,就是将原集合元素附加到链表的末尾
17             succ = null;
18             pred = last;
19         } else {
20             succ = node(index);
21             pred = succ.prev;
22         }
23 
24         for (Object o : a) {//遍历要插入的元素
25             @SuppressWarnings("unchecked") E e = (E) o;
26             Node<E> newNode = new Node<>(pred, e, null);
27             if (pred == null)
28                 first = newNode;
29             else
30                 pred.next = newNode;
31             pred = newNode;
32         }
33 
34         if (succ == null) {
35             last = pred;
36         } else {
37             pred.next = succ;
38             succ.prev = pred;
39         }
40 
41         size += numNew;
42         modCount++;
43         return true;
44     }
View Code

  看到上面向 LinkedList 集合中添加元素的各种方式,我们发现LinkedList 每次添加元素只是改变元素的上一个指针引用和下一个指针引用,而且没有扩容。,对比于 ArrayList ,需要扩容,而且在中间插入元素时,后面的所有元素都要移动一位,两者插入元素时的效率差异很大,下一篇博客会对这两者的效率,以及何种情况选择何种集合进行分析。

  还有,每次进行添加操作,都有modCount++ 的操作,

5、删除元素

  删除元素和添加元素一样,也是通过更改指向上一个节点和指向下一个节点的引用即可,这里就不作图形展示了。

  ①、remove()和removeFirst()

  从此列表中移除并返回第一个元素

 1     //从此列表中移除并返回第一个元素
 2     public E remove() {
 3         return removeFirst();
 4     }
 5     //从此列表中移除并返回第一个元素
 6     public E removeFirst() {
 7         final Node<E> f = first;//f设为头结点
 8         if (f == null)
 9             throw new NoSuchElementException();//如果头结点为空,则抛出异常
10         return unlinkFirst(f);
11     }
12     private E unlinkFirst(Node<E> f) {
13         // assert f == first && f != null;
14         final E element = f.item;
15         final Node<E> next = f.next;//next 为头结点的下一个节点
16         f.item = null;
17         f.next = null; // 将节点的元素以及引用都设为 null,便于垃圾回收
18         first = next; //修改头结点为第二个节点
19         if (next == null)//如果第二个节点为空(当前链表只存在第一个元素)
20             last = null;//那么尾节点也置为 null
21         else
22             next.prev = null;//如果第二个节点不为空,那么将第二个节点的上一个引用置为 null
23         size--;
24         modCount++;
25         return element;
26     }
View Code

  ②、removeLast()

  从该列表中删除并返回最后一个元素

 1     //从该列表中删除并返回最后一个元素
 2     public E removeLast() {
 3         final Node<E> l = last;
 4         if (l == null)//如果尾节点为空,表示当前集合为空,抛出异常
 5             throw new NoSuchElementException();
 6         return unlinkLast(l);
 7     }
 8     
 9     private E unlinkLast(Node<E> l) {
10         // assert l == last && l != null;
11         final E element = l.item;
12         final Node<E> prev = l.prev;
13         l.item = null;
14         l.prev = null; //将节点的元素以及引用都设为 null,便于垃圾回收
15         last = prev;//尾节点为倒数第二个节点
16         if (prev == null)//如果倒数第二个节点为null
17             first = null;//那么将节点也置为 null
18         else
19             prev.next = null;//如果倒数第二个节点不为空,那么将倒数第二个节点的下一个引用置为 null
20         size--;
21         modCount++;
22         return element;
23     }
View Code

  ③、remove(int index)

  删除此列表中指定位置的元素

 1     //删除此列表中指定位置的元素
 2     public E remove(int index) {
 3         //判断索引 index >= 0 && index <= size中时抛出IndexOutOfBoundsException异常
 4         checkElementIndex(index);
 5         return unlink(node(index));
 6     }
 7     E unlink(Node<E> x) {
 8         // assert x != null;
 9         final E element = x.item;
10         final Node<E> next = x.next;
11         final Node<E> prev = x.prev;
12 
13         if (prev == null) {//如果删除节点位置的上一个节点引用为null(表示删除第一个元素)
14             first = next;//将头结点置为第一个元素的下一个节点
15         } else {//如果删除节点位置的上一个节点引用不为null
16             prev.next = next;//将删除节点的上一个节点的下一个节点引用指向删除节点的下一个节点(去掉删除节点)
17             x.prev = null;//删除节点的上一个节点引用置为null
18         }
19 
20         if (next == null) {//如果删除节点的下一个节点引用为null(表示删除最后一个节点)
21             last = prev;//将尾节点置为删除节点的上一个节点
22         } else {//不是删除尾节点
23             next.prev = prev;//将删除节点的下一个节点的上一个节点的引用指向删除节点的上一个节点
24             x.next = null;//将删除节点的下一个节点引用置为null
25         }
26 
27         x.item = null;//删除节点内容置为null,便于垃圾回收
28         size--;
29         modCount++;
30         return element;
31     }
View Code

  ④、remove(Object o)

  如果存在,则从该列表中删除指定元素的第一次出现

  此方法本质上和 remove(int index) 没多大区别,通过循环判断元素进行删除,需要注意的是,是删除第一次出现的元素,不是所有的。

 1     public boolean remove(Object o) {
 2         if (o == null) {
 3             for (Node<E> x = first; x != null; x = x.next) {
 4                 if (x.item == null) {
 5                     unlink(x);
 6                     return true;
 7                 }
 8             }
 9         } else {
10             for (Node<E> x = first; x != null; x = x.next) {
11                 if (o.equals(x.item)) {
12                     unlink(x);
13                     return true;
14                 }
15             }
16         }
17         return false;
18     }
View Code

 6、修改元素

  通过调用 set(int index, E element) 方法,用指定的元素替换此列表中指定位置的元素。

    public E set(int index, E element) {
        //判断索引 index >= 0 && index <= size中时抛出IndexOutOfBoundsException异常
        checkElementIndex(index);
        Node<E> x = node(index);//获取指定索引处的元素
        E oldVal = x.item;
        x.item = element;//将指定位置的元素替换成要修改的元素
        return oldVal;//返回指定索引位置原来的元素
    }

  这里主要是通过 node(index) 方法获取指定索引位置的节点,然后修改此节点位置的元素即可。

7、查找元素

  ①、getFirst()

  返回此列表中的第一个元素

1     public E getFirst() {
2         final Node<E> f = first;
3         if (f == null)
4             throw new NoSuchElementException();
5         return f.item;
6     }
View Code

  ②、getLast()

  返回此列表中的最后一个元素

1     public E getLast() {
2         final Node<E> l = last;
3         if (l == null)
4             throw new NoSuchElementException();
5         return l.item;
6     }
View Code

  ③、get(int index)

  返回指定索引处的元素

1     public E get(int index) {
2         checkElementIndex(index);
3         return node(index).item;
4     }
View Code

  ④、indexOf(Object o)

  返回此列表中指定元素第一次出现的索引,如果此列表不包含元素,则返回-1。

 1     //返回此列表中指定元素第一次出现的索引,如果此列表不包含元素,则返回-1。
 2     public int indexOf(Object o) {
 3         int index = 0;
 4         if (o == null) {//如果查找的元素为null(LinkedList可以允许null值)
 5             for (Node<E> x = first; x != null; x = x.next) {//从头结点开始不断向下一个节点进行遍历
 6                 if (x.item == null)
 7                     return index;
 8                 index++;
 9             }
10         } else {//如果查找的元素不为null
11             for (Node<E> x = first; x != null; x = x.next) {
12                 if (o.equals(x.item))
13                     return index;
14                 index++;
15             }
16         }
17         return -1;//找不到返回-1
18     }
View Code

8、遍历集合

  ①、普通 for 循环

LinkedList<String> linkedList = new LinkedList<>();
linkedList.add("A");
linkedList.add("B");
linkedList.add("C");
linkedList.add("D");
for(int i = 0 ; i < linkedList.size() ; i++){
    System.out.print(linkedList.get(i)+" ");//A B C D
}
View Code

  代码很简单,我们就利用 LinkedList 的 get(int index) 方法,遍历出所有的元素。

  但是需要注意的是, get(int index) 方法每次都要遍历该索引之前的所有元素,这句话这么理解:

  比如上面的一个 LinkedList 集合,我放入了 A,B,C,D是个元素。总共需要四次遍历:

  第一次遍历打印 A:只需遍历一次。

  第二次遍历打印 B:需要先找到 A,然后再找到 B 打印。

  第三次遍历打印 C:需要先找到 A,然后找到 B,最后找到 C 打印。

  第四次遍历打印 D:需要先找到 A,然后找到 B,然后找到 C,最后找到 D。

  这样如果集合元素很多,越查找到后面(当然此处的get方法进行了优化,查找前半部分从前面开始遍历,查找后半部分从后面开始遍历,但是需要的时间还是很多)花费的时间越多。那么如何改进呢?

  ②、迭代器

 1 LinkedList<String> linkedList = new LinkedList<>();
 2 linkedList.add("A");
 3 linkedList.add("B");
 4 linkedList.add("C");
 5 linkedList.add("D");
 6 
 7 
 8 Iterator<String> listIt = linkedList.listIterator();
 9 while(listIt.hasNext()){
10     System.out.print(listIt.next()+" ");//A B C D
11 }
12 
13 //通过适配器模式实现的接口,作用是倒叙打印链表
14 Iterator<String> it = linkedList.descendingIterator();
15 while(it.hasNext()){
16     System.out.print(it.next()+" ");//D C B A
17 }
View Code

  在 LinkedList 集合中也有一个内部类 ListItr,方法实现大体上也差不多,通过移动游标指向每一次要遍历的元素,不用在遍历某个元素之前都要从头开始。其方法实现也比较简单:

  1     public ListIterator<E> listIterator(int index) {
  2         checkPositionIndex(index);
  3         return new ListItr(index);
  4     }
  5 
  6     private class ListItr implements ListIterator<E> {
  7         private Node<E> lastReturned;
  8         private Node<E> next;
  9         private int nextIndex;
 10         private int expectedModCount = modCount;
 11 
 12         ListItr(int index) {
 13             // assert isPositionIndex(index);
 14             next = (index == size) ? null : node(index);
 15             nextIndex = index;
 16         }
 17 
 18         public boolean hasNext() {
 19             return nextIndex < size;
 20         }
 21 
 22         public E next() {
 23             checkForComodification();
 24             if (!hasNext())
 25                 throw new NoSuchElementException();
 26 
 27             lastReturned = next;
 28             next = next.next;
 29             nextIndex++;
 30             return lastReturned.item;
 31         }
 32 
 33         public boolean hasPrevious() {
 34             return nextIndex > 0;
 35         }
 36 
 37         public E previous() {
 38             checkForComodification();
 39             if (!hasPrevious())
 40                 throw new NoSuchElementException();
 41 
 42             lastReturned = next = (next == null) ? last : next.prev;
 43             nextIndex--;
 44             return lastReturned.item;
 45         }
 46 
 47         public int nextIndex() {
 48             return nextIndex;
 49         }
 50 
 51         public int previousIndex() {
 52             return nextIndex - 1;
 53         }
 54 
 55         public void remove() {
 56             checkForComodification();
 57             if (lastReturned == null)
 58                 throw new IllegalStateException();
 59 
 60             Node<E> lastNext = lastReturned.next;
 61             unlink(lastReturned);
 62             if (next == lastReturned)
 63                 next = lastNext;
 64             else
 65                 nextIndex--;
 66             lastReturned = null;
 67             expectedModCount++;
 68         }
 69 
 70         public void set(E e) {
 71             if (lastReturned == null)
 72                 throw new IllegalStateException();
 73             checkForComodification();
 74             lastReturned.item = e;
 75         }
 76 
 77         public void add(E e) {
 78             checkForComodification();
 79             lastReturned = null;
 80             if (next == null)
 81                 linkLast(e);
 82             else
 83                 linkBefore(e, next);
 84             nextIndex++;
 85             expectedModCount++;
 86         }
 87 
 88         public void forEachRemaining(Consumer<? super E> action) {
 89             Objects.requireNonNull(action);
 90             while (modCount == expectedModCount && nextIndex < size) {
 91                 action.accept(next.item);
 92                 lastReturned = next;
 93                 next = next.next;
 94                 nextIndex++;
 95             }
 96             checkForComodification();
 97         }
 98 
 99         final void checkForComodification() {
100             if (modCount != expectedModCount)
101                 throw new ConcurrentModificationException();
102         }
103     }
View Code

  这里需要重点注意的是 modCount 字段,前面我们在增加和删除元素的时候,都会进行自增操作 modCount,这是因为如果想一边迭代,一边用集合自带的方法进行删除或者新增操作,都会抛出异常。(使用迭代器的增删方法不会抛异常)

final void checkForComodification() {
            if (modCount != expectedModCount)
                throw new ConcurrentModificationException();
        }

  比如:

 1 LinkedList<String> linkedList = new LinkedList<>();
 2 linkedList.add("A");
 3 linkedList.add("B");
 4 linkedList.add("C");
 5 linkedList.add("D");
 6 
 7 
 8 Iterator<String> listIt = linkedList.listIterator();
 9 while(listIt.hasNext()){
10     System.out.print(listIt.next()+" ");//A B C D
11     //linkedList.remove();//此处会抛出异常
12     listIt.remove();//这样可以进行删除操作
13 }
View Code

  迭代器的另一种形式就是使用 foreach 循环,底层实现也是使用的迭代器,这里我们就不做介绍了。

LinkedList<String> linkedList = new LinkedList<>();
linkedList.add("A");
linkedList.add("B");
linkedList.add("C");
linkedList.add("D");
for(String str : linkedList){
    System.out.print(str + "");
}
View Code

9、迭代器和for循环效率差异

 1 LinkedList<Integer> linkedList = new LinkedList<>();
 2 for(int i = 0 ; i < 10000 ; i++){//向链表中添加一万个元素
 3     linkedList.add(i);
 4 }
 5 long beginTimeFor = System.currentTimeMillis();
 6 for(int i = 0 ; i < 10000 ; i++){
 7     System.out.print(linkedList.get(i));
 8 }
 9 long endTimeFor = System.currentTimeMillis();
10 System.out.println("使用普通for循环遍历10000个元素需要的时间:"+ (endTimeFor - beginTimeFor));
11 
12 
13 long beginTimeIte = System.currentTimeMillis();
14 Iterator<Integer> it = linkedList.listIterator();
15 while(it.hasNext()){
16     System.out.print(it.next()+" ");
17 }
18 long endTimeIte = System.currentTimeMillis();
19 System.out.println("使用迭代器遍历10000个元素需要的时间:"+ (endTimeIte - beginTimeIte));

  打印结果为:

  

  一万个元素两者之间都相差一倍多的时间,如果是十万,百万个元素,那么两者之间相差的速度会越来越大。下面通过图形来解释:

  普通for循环:每次遍历一个索引的元素之前,都要访问之间所有的索引。

  

  迭代器:每次访问一个元素后,都会用游标记录当前访问元素的位置,遍历一个元素,记录一个位置。

  

 

参考文档:https://docs.oracle.com/javase/8/docs/api/java/util/LinkedList.html#

作者: YSOcean
本文版权归作者所有,欢迎转载,但未经作者同意不能转载,否则保留追究法律责任的权利。
目录
相关文章
|
21天前
|
XML Java 编译器
Java注解的底层源码剖析与技术认识
Java注解(Annotation)是Java 5引入的一种新特性,它提供了一种在代码中添加元数据(Metadata)的方式。注解本身并不是代码的一部分,它们不会直接影响代码的执行,但可以在编译、类加载和运行时被读取和处理。注解为开发者提供了一种以非侵入性的方式为代码提供额外信息的手段,这些信息可以用于生成文档、编译时检查、运行时处理等。
59 7
|
1月前
|
数据采集 人工智能 Java
Java产科专科电子病历系统源码
产科专科电子病历系统,全结构化设计,实现产科专科电子病历与院内HIS、LIS、PACS信息系统、区域妇幼信息平台的三级互联互通,系统由门诊系统、住院系统、数据统计模块三部分组成,它管理了孕妇从怀孕开始到生产结束42天一系列医院保健服务信息。
32 4
|
1月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
77 2
|
1月前
|
Java 开发者
在 Java 中,一个类可以实现多个接口吗?
这是 Java 面向对象编程的一个重要特性,它提供了极大的灵活性和扩展性。
148 57
|
4天前
|
JSON Java Apache
Java基础-常用API-Object类
继承是面向对象编程的重要特性,允许从已有类派生新类。Java采用单继承机制,默认所有类继承自Object类。Object类提供了多个常用方法,如`clone()`用于复制对象,`equals()`判断对象是否相等,`hashCode()`计算哈希码,`toString()`返回对象的字符串表示,`wait()`、`notify()`和`notifyAll()`用于线程同步,`finalize()`在对象被垃圾回收时调用。掌握这些方法有助于更好地理解和使用Java中的对象行为。
|
13天前
|
存储 JavaScript 前端开发
基于 SpringBoot 和 Vue 开发校园点餐订餐外卖跑腿Java源码
一个非常实用的校园外卖系统,基于 SpringBoot 和 Vue 的开发。这一系统源于黑马的外卖案例项目 经过站长的进一步改进和优化,提供了更丰富的功能和更高的可用性。 这个项目的架构设计非常有趣。虽然它采用了SpringBoot和Vue的组合,但并不是一个完全分离的项目。 前端视图通过JS的方式引入了Vue和Element UI,既能利用Vue的快速开发优势,
77 13
|
27天前
|
缓存 监控 Java
Java线程池提交任务流程底层源码与源码解析
【11月更文挑战第30天】嘿,各位技术爱好者们,今天咱们来聊聊Java线程池提交任务的底层源码与源码解析。作为一个资深的Java开发者,我相信你一定对线程池并不陌生。线程池作为并发编程中的一大利器,其重要性不言而喻。今天,我将以对话的方式,带你一步步深入线程池的奥秘,从概述到功能点,再到背景和业务点,最后到底层原理和示例,让你对线程池有一个全新的认识。
54 12
|
21天前
|
JavaScript 安全 Java
java版药品不良反应智能监测系统源码,采用SpringBoot、Vue、MySQL技术开发
基于B/S架构,采用Java、SpringBoot、Vue、MySQL等技术自主研发的ADR智能监测系统,适用于三甲医院,支持二次开发。该系统能自动监测全院患者药物不良反应,通过移动端和PC端实时反馈,提升用药安全。系统涵盖规则管理、监测报告、系统管理三大模块,确保精准、高效地处理ADR事件。
|
1月前
|
存储 缓存 安全
java 中操作字符串都有哪些类,它们之间有什么区别
Java中操作字符串的类主要有String、StringBuilder和StringBuffer。String是不可变的,每次操作都会生成新对象;StringBuilder和StringBuffer都是可变的,但StringBuilder是非线程安全的,而StringBuffer是线程安全的,因此性能略低。
53 8
|
1月前
|
人工智能 监控 数据可视化
Java智慧工地信息管理平台源码 智慧工地信息化解决方案SaaS源码 支持二次开发
智慧工地系统是依托物联网、互联网、AI、可视化建立的大数据管理平台,是一种全新的管理模式,能够实现劳务管理、安全施工、绿色施工的智能化和互联网化。围绕施工现场管理的人、机、料、法、环五大维度,以及施工过程管理的进度、质量、安全三大体系为基础应用,实现全面高效的工程管理需求,满足工地多角色、多视角的有效监管,实现工程建设管理的降本增效,为监管平台提供数据支撑。
45 3