wait: 线程等待,在等待过程中释放锁,需要其他线程调用notify唤醒。notify: 唤醒一条等待的线程,如果有多条线程等待,随机唤醒一条。notifyAll: 唤醒所有等待线程。lock()获取锁,unlock()释放锁。static ExecutorService newFixedThreadPool(int nThread)submit(Runnable r)、submit(Callable c)run或者call方法的返回值。shutDown()关闭线程池。call()设置线程任务,类似于run方法,但call可以抛出异常,并有返回值。FutureTask实现了Future接口,get()获取call方法的返回值。Collection接口中的方法。ArrayList及LinkedList的使用。在之前的学习中,我们了解了变量和数组用于保存数据,但数组是定长的。当需要添加或删除数据时,数组并不方便,因此我们引入了长度可变的容器——集合。

list.add("张三")。key和value,例如map.put("涛哥","金莲")。Collection是单列集合的顶级接口。
Collection<E> 对象名 = new 实现类对象<E>()。<E>: 泛型,决定了集合中能存储的数据类型,可以统一元素类型。泛型中只能写引用数据类型,如果不写,默认为Object类型,此时可以存储任何类型的数据。boolean add(E e); //将给定的元素添加到当前集合中。
boolean addAll(Collection<? extends E> c); // 将另一个集合的元素添加到当前集合中(集合合并)。
void clear(); // 清除集合中所有的元素。
boolean contains(Object o); // 判断当前集合中是否包含指定的元素。
boolean isEmpty(); // 判断当前集合中是否有元素,即判断集合是否为空。
boolean remove(Object o); // 将指定的元素从集合中删除。
int size(); //返回集合中的元素个数。
Object[] toArray(); // 将集合中的元素存储到数组中。public class Demo01Collection {
public static void main(String[] args) {
Collection<String> collection = new ArrayList<>();
collection.add("萧炎");
collection.add("萧薰儿");
collection.add("彩鳞");
collection.add("小医仙");
collection.add("云韵");
collection.add("涛哥");
System.out.println(collection);
Collection<String> collection1 = new ArrayList<>();
collection1.add("张无忌");
collection1.add("小昭");
collection1.add("赵敏");
collection1.add("周芷若");
collection1.addAll(collection);
System.out.println(collection1);
collection1.clear();
System.out.println(collection1);
boolean result01 = collection.contains("涛哥");
System.out.println("result01 = " + result01);
System.out.println(collection1.isEmpty());
collection.remove("涛哥");
System.out.println(collection);
System.out.println(collection.size());
Object[] arr = collection.toArray();
System.out.println(Arrays.toString(arr));
}
}Iterator<E> iterator(); // 获取迭代器对象。
boolean hasNext(); // 判断集合中是否有下一个元素。
E next(); // 获取下一个元素。public class Demo01Iterator {
public static void main(String[] args) {
ArrayList<String> list = new ArrayList<>();
list.add("楚雨荨");
list.add("慕容云海");
list.add("端木磊");
list.add("上官瑞谦");
list.add("叶烁");
Iterator<String> iterator = list.iterator();
while(iterator.hasNext()){
String element = iterator.next();
System.out.println(element);
}
}
}注意:next方法在获取时不要连续使用多次,否则会抛出NoSuchElementException。
int cursor; // 下一个元素索引位置
int lastRet = -1; // 上一个元素索引位置
Iterator iterator = list.iterator();Iterator是一个接口,等号右边一定是它的实现类对象。
问题:Iterator接收的到底是哪个实现类对象呢?-> ArrayList中的内部类Itr对象。

注意:只有ArrayList使用迭代器时Iterator接口才会指向Itr,其他集合使用迭代器Iterator就指向的不是Itr了。
需求:定义一个集合,存储唐僧、孙悟空、猪八戒、沙僧,遍历集合,如果遍历到猪八戒,往集合中添加一个白龙马。
public class Demo03Iterator {
public static void main(String[] args) {
ArrayList<String> list = new ArrayList<>();
list.add("唐僧");
list.add("孙悟空");
list.add("猪八戒");
list.add("沙僧");
Iterator<String> iterator = list.iterator();
while(iterator.hasNext()){
String element = iterator.next();
if ("猪八戒".equals(element)){
list.add("白龙马");
}
}
System.out.println(list);
}
}
结论:当预期操作次数和实际操作次数不相等时,会出现“并发修改异常”。
我们调用了add方法,而add方法底层只给modCount++,但是再次调用next方法时,并没有给修改后的modCount重新赋值给expectedModCount,导致next方法底层的判断判断出实际操作次数和预期操作次数不相等,所以抛出了“并发修改异常”。
数据结构是一种具有一定逻辑关系,在计算机中应用某种存储结构,并且封装了相应操作的数据元素集合。它包含三方面的内容:逻辑关系、存储关系及操作。

随着应用程序变得越来越复杂和数据越来越丰富,几百万、几十亿甚至几百亿的数据就会出现,而对这么大对数据进行搜索、插入或者排序等的操作就越来越慢,数据结构就是用来解决这些问题的。
数据的逻辑结构指反映数据元素之间的逻辑关系,而与他们在计算机中的存储位置无关:

数据的物理结构/存储结构:是描述数据具体在内存中的存储(如:顺序结构、链式结构、索引结构、哈希结构)等,一种数据逻辑结构可表示成一种或多种物理存储结构。
数据结构是一门完整并且复杂的课程,那么我们今天只是简单的讨论常见的几种数据结构,让我们对数据结构与算法有一个初步的了解。



a.节点:一个节点分为两部分。 第一部分:数据域(存数据)。 第二部分:指针域(保存下一个节点地址)。 b.特点:前面节点记录后面节点的地址,但是后面节点地址不记录前面节点地址。
a.节点:一个节点分为三部分。 第一部分:指针域(保存上一个节点地址)。 第二部分:数据域(保存的数据)。 第三部分:指针域(保存下一个节点地址)。 b.特点: 前面节点记录后面节点地址,后面节点也记录前面节点地址。

List是Collection接口的子接口。
ArrayListLinkedListVector1.概述:ArrayList是List接口的实现类。
2.特点:
a.元素有序。
b.元素可重复。
c.有索引,可以利用索引去操作元素。
d.线程不安全。
3.数据结构:数组。
4.常用方法:
boolean add(E e) : 将元素添加到集合中。
void add(int index, E element) : 在指定索引位置上添加元素。
boolean remove(Object o) : 删除指定的元素。
E remove(int index) : 删除指定索引位置上的元素,返回的是被删除的那个元素。
E set(int index, E element) : 将指定索引位置上的元素修改成后面的element元素。
E get(int index) : 根据索引获取元素。
int size() : 获取集合元素个数。public class Demo01ArrayList {
public static void main(String[] args) {
ArrayList<String> list = new ArrayList<>();
list.add("铁胆火车侠");
list.add("喜洋洋");
list.add("火影忍者");
list.add("灌篮高手");
list.add("网球王子");
System.out.println(list);
list.add(2,"涛哥");
System.out.println(list);
list.remove("涛哥");
System.out.println(list);
String element = list.remove(0);
System.out.println(element);
System.out.println(list);
String element2 = list.set(0, "金莲");
System.out.println(element2);
System.out.println(list);
System.out.println(list.get(0));
System.out.println(list.size());
}
}public class Demo02ArrayList {
public static void main(String[] args) {
ArrayList<String> list = new ArrayList<>();
list.add("铁胆火车侠");
list.add("喜洋洋");
list.add("火影忍者");
list.add("灌篮高手");
list.add("网球王子");
Iterator<String> iterator = list.iterator();
while(iterator.hasNext()){
System.out.println(iterator.next());
}
System.out.println("=====================");
for (int i = 0;i<list.size();i++){
System.out.println(list.get(i));
}
System.out.println("=====================");
for (int i = 0; i < list.size(); i++) {
System.out.println(list.get(i));
}
}
}public class Demo03ArrayList {
public static void main(String[] args) {
ArrayList<Integer> list = new ArrayList<>();
list.add(2);
System.out.println(list);
/*
需求:删除2
remove(Object o) : 直接删除指定元素。
remove(int index) : 删除指定索引位置上的元素。
如果remove中直接传递整数,默认调用按照指定索引删除元素的remove。
但是此时list中没有2索引,所以越界。
解决:我们可以将2包装成包装类,变成包装类之后,其父类就是Object了。
*/
list.remove(Integer.valueOf(2));
System.out.println(list);
}
}ArrayList() 构造一个初始容量为十的空列表
=========================================
private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};
Object[] elementData; ->ArrayList底层的那个数组
public ArrayList() {
this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;
}
=========================================
list.add("a");
public boolean add(E e) {
modCount++;
add(e, elementData, size);// e->要存的元素 elementData->集合数组,长度开始为0,size->0
return true;
}
private void add(E e->元素, Object[] elementData->集合数组, int s->0) {
if (s == elementData.length)
elementData = grow();
elementData[s] = e;
size = s + 1;
}
private Object[] grow() {
return grow(size + 1);
}
private Object[] grow(int minCapacity->1) {
int oldCapacity = elementData.length;//0
if (oldCapacity > 0 || elementData != DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
int newCapacity = ArraysSupport.newLength(oldCapacity,
minCapacity - oldCapacity, /* minimum growth */
oldCapacity >> 1 /* preferred growth */);
return elementData = Arrays.copyOf(elementData, newCapacity);
} else {
return elementData = new Object[Math.max(DEFAULT_CAPACITY->10, minCapacity->1)];
}
}
==========================================
// 假设ArrayList中存了第11个元素,会自动扩容-> Arrays.copyOf
private Object[] grow(int minCapacity) {//11
int oldCapacity = elementData.length;//10
if (oldCapacity > 0 || elementData != DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
int newCapacity(15) = ArraysSupport.newLength(oldCapacity->10,
minCapacity - oldCapacity->1, /* minimum growth */
oldCapacity >> 1 ->5 /* preferred growth */);
return elementData = Arrays.copyOf(elementData, newCapacity);
} else {
return elementData = new Object[Math.max(DEFAULT_CAPACITY, minCapacity)];
}
}
public static int newLength(int oldLength->10, int minGrowth->1, int prefGrowth->5) {
// preconditions not checked because of inlining
// assert oldLength >= 0
// assert minGrowth > 0
int prefLength = oldLength + Math.max(minGrowth, prefGrowth); // 15
if (0 < prefLength && prefLength <= SOFT_MAX_ARRAY_LENGTH) {
return prefLength;
} else {
// put code cold in a separate method
return hugeLength(oldLength, minGrowth);
}
} ArrayList(int initialCapacity); //构造具有指定初始容量的空列表 ArrayList<String> list = new ArrayList<>(5);
// ==============================================
public ArrayList(int initialCapacity->5) {
if (initialCapacity > 0) {
this.elementData = new Object[initialCapacity];//直接创建长度为5的数组
} else if (initialCapacity == 0) {
this.elementData = EMPTY_ELEMENTDATA;
} else {
throw new IllegalArgumentException("Illegal Capacity: "+
initialCapacity);
}
} public void addFirst(E e); //将指定元素插入此列表的开头。public void addLast(E e);将指定元素添加到此列表的结尾。public E getFirst();返回此列表的第一个元素。public E getLast();返回此列表的最后一个元素。public E removeFirst();移除并返回此列表的第一个元素。public E removeLast();移除并返回此列表的最后一个元素。public E pop();从此列表所表示的堆栈处弹出一个元素。public void push(E e);将元素推入此列表所表示的堆栈。public boolean isEmpty();如果列表没有元素,则返回true。public class Demo05LinkedList {
public static void main(String[] args) {
LinkedList<String> linkedList = new LinkedList<>();
linkedList.add("吕布");
linkedList.add("刘备");
linkedList.add("关羽");
linkedList.add("张飞");
linkedList.add("貂蝉");
System.out.println(linkedList);
linkedList.addFirst("孙尚香");
System.out.println(linkedList);
linkedList.addLast("董卓");
System.out.println(linkedList);
System.out.println(linkedList.getFirst());
System.out.println(linkedList.getLast());
linkedList.removeFirst();
System.out.println(linkedList);
linkedList.removeLast();
System.out.println(linkedList);
System.out.println("======================");
Iterator<String> iterator = linkedList.iterator();
while(iterator.hasNext()){
System.out.println(iterator.next());
}
System.out.println("=======================");
for (int i = 0; i < linkedList.size(); i++) {
System.out.println(linkedList.get(i));
}
}
}public E pop();从此列表所表示的堆栈处弹出一个元素。
public void push(E e);将元素推入此列表所表示的堆栈。
public class Demo06LinkedList {
public static void main(String[] args) {
LinkedList<String> linkedList = new LinkedList<>();
linkedList.add("吕布");
linkedList.add("刘备");
linkedList.add("关羽");
linkedList.add("张飞");
linkedList.add("貂蝉");
linkedList.pop();
System.out.println(linkedList);
linkedList.push("涛哥");
System.out.println(linkedList);
}
} transient int size = 0; // 元素个数
transient Node<E> first; //第一个节点对象
transient Node<E> last; // 最后一个节点对象 private static class Node<E> {
E item;//节点上的元素
Node<E> next;//记录着下一个节点地址
Node<E> prev;//记录着上一个节点地址
Node(Node<E> prev, E element, Node<E> next) {
this.item = element;
this.next = next;
this.prev = prev;
}
}LinkedList<String> list = new LinkedList<>();
list.add("a");
list.add("b");
void linkLast(E e) {
final Node<E> l = last;
final Node<E> newNode = new Node<>(l, e, null);
last = newNode;
if (l == null)
first = newNode;
else
l.next = newNode;
size++;
modCount++;
}public E get(int index) {
checkElementIndex(index);
return node(index).item;
}
Node<E> node(int index) {
// assert isElementIndex(index);
if (index < (size >> 1)) {
Node<E> x = first;
for (int i = 0; i < index; i++)
x = x.next;
return x;
} else {
Node<E> x = last;
for (int i = size - 1; i > index; i--)
x = x.prev;
return x;
}
}index < (size >> 1)采用二分思想,先将index与长度size的一半比较,如果index<size/2,就只从位置0往后遍历到位置index处,而如果index>size/2,就只从位置size往前遍历到位置index处。这样可以减少一部分不必要的遍历。
1.作用: 遍历集合或者数组。 2.格式:
for(元素类型 变量名:要遍历的集合名或者数组名){
变量名就是代表的每一个元素
}3.快捷键:集合名或者数组名.for
public class Demo01ForEach {
public static void main(String[] args) {
ArrayList<String> list = new ArrayList<>();
list.add("张三");
list.add("李四");
list.add("王五");
list.add("赵六");
for (String s : list) {
System.out.println(s);
}
System.out.println("=====================");
int[] arr = {1,2,3,4,5};
for (int i : arr) {
System.out.println(i);
}
}
}
所以不管是用迭代器还是使用增强for,在遍历集合的过程中都不要随意修改集合长度,否则会出现并发修改异常。