Day16_集合与迭代器
Day16-集合
- Day16 集合与迭代器
- 1.1 集合的概念
- 集合继承图
- 1.2 Collection接口
- 1、添加元素
- 2、删除元素
- 3、查询与获取元素
- 不过当我们实际使用都是使用的他的子类Arraylist!!!
- 1.3 API演示
- 1、演示添加
- 2、演示删除
- 3、演示查询与获取元素
- 2 Iterator迭代器
- 2.1 Iterator接口
- 2.2 迭代器的实现原理
- 2.3 Iterable接口
- 1、 Iterable接口依赖Iterator接口
- 2、forEach方法
- 3、新语法糖:foreach循环(增强for循环)
- 2.4 使用Iterator迭代器删除元素
- 2.5 Iterator迭代器的快速失败(fail-fast)机制
- 1、ConcurrentModificationException异常
- 2、modCount变量
- 3 元素要求
- 3.1 有序、可重复
- 1、List接口介绍
- 2、List接口中常用方法
- 3、ListIterator迭代器
- 3.2 无序、不可重复
- 1、Set接口介绍
- 1.1HashSet
- 1.2TreeSet
- 2、元素相等和重复问题:equals和hashCode方法
- 3、大小顺序和重复问题:Comparable和Comparator接口
- 4 CollectionArray
- 4.1数组转换为集合的方法及注意事项
- 5removeif的使用
- 6 Collections 集合工具类
Day16 集合与迭代器
1.1 集合的概念
集合是java中提供的一种容器,可以用来存储多个数据。
集合和数组既然都是容器,它们有啥区别呢?
- 数组的长度是固定的。
- 集合的长度是可变的。
- 数组中可以存储基本数据类型值,也可以存储对象,而集合中只能存储对象
集合主要分为两大系列:Collection和Map,Collection 表示一组对象,Map表示一组映射关系或键值对。
集合继承图
集合框架: 有序无序取决于能否用编号操作他
Collection: 存储的是一个一个的数据-- List: 元素有序且允许重复-- ArrayList: 底层使用数组结构,查询快增删慢-- LinkedList: 底层使用双向链表结构,查询慢增删快-- Set: 元素无序且不允许重复-- HashSet: 底层使用哈希表结构-- TreeSet: 底层使用红黑树结构Map: 存储一对一对的数据-- HashMap:-- TreeMap:-- Hashtable:
1.2 Collection接口
Collection 层次结构中的根接口。Collection 表示一组对象,这些对象也称为 collection 的元素。一些 collection 允许有重复的元素,而另一些则不允许。一些 collection 是有序的,而另一些则是无序的。JDK 不提供此接口的任何直接实现:它提供更具体的子接口(如 Set 和 List、Queue)实现。此接口通常用来传递 collection,并在需要最大普遍性的地方操作这些 collection。
Collection是所有单列集合的父接口,因此在Collection中定义了单列集合(List和Set)通用的一些方法,这些方法可用于操作所有的单列集合。方法如下:
1、添加元素
(1)add(E obj):添加元素对象到当前集合中
(2)addAll(Collection<? extends E> other):添加other集合中的所有元素对象到当前集合中,即this = this ∪ other
2、删除元素
(1) boolean remove(Object obj) :从当前集合中删除第一个找到的与obj对象equals返回true的元素。
(2)boolean removeAll(Collection<?> coll):从当前集合中删除所有与coll集合中相同的元素。即this = this - this ∩ coll
(3)boolean retainAll(Collection<?> coll):从当前集合中删除两个集合中不同的元素,使得当前集合仅保留与c集合中的元素相同的元素,即当前集合中仅保留两个集合的交集,即this = this ∩ coll;
(4)void clear():清空集合
(5)boolean removeIf(Predicate<? super E> filter) :删除满足给定条件的此集合的所有元素。removeIf方法是Java8引入的。
3、查询与获取元素
(1)boolean isEmpty():判断当前集合是否为空集合。
(2)boolean contains(Object obj):判断当前集合中是否存在一个与obj对象equals返回true的元素。
(3)boolean containsAll(Collection<?> c):判断c集合中的元素是否在当前集合中都存在。即c集合是否是当前集合的“子集”。
(4)int size():获取当前集合中实际存储的元素个数
(5)Object[] toArray():返回包含当前集合中所有元素的数组
不过当我们实际使用都是使用的他的子类Arraylist!!!
因为Java中使用ArrayList
作为集合的一个常见选择,主要基于以下几个原因:
-
动态数组:
ArrayList
内部是通过一个动态数组来实现的,这意呀着它可以自动调整其大小以存储更多的元素。这种自动扩容的特性使得ArrayList
在处理未知大小的数据集时非常方便。 -
随机访问:由于
ArrayList
是基于数组的,所以它可以提供快速的随机访问能力。也就是说,你可以通过索引(位置)直接访问或修改元素,这种操作的时间复杂度是O(1)。 -
易用性:
ArrayList
提供了丰富的API,使得添加、删除、查找元素等操作变得简单直观。它实现了List
接口,因此可以使用所有List
接口提供的方法。 -
广泛的使用场景:
ArrayList
适用于大多数需要列表功能的场景,包括存储对象集合、作为栈或队列的简单实现(尽管有专门的Stack
和Queue
接口及其实现),以及作为其他集合(如HashSet
)的基础数据结构等。 -
性能考虑:尽管
ArrayList
在插入和删除元素时可能需要移动其他元素(这会影响性能,特别是在列表的开头或中间位置进行这些操作时),但在许多实际应用中,这种性能开销是可以接受的,特别是当元素添加和删除操作不频繁时。 -
与Java集合框架的兼容性:
ArrayList
是Java集合框架(Java Collections Framework)的一部分,这意味着它可以与其他集合类型(如Set
、Map
等)无缝协作,支持通过Collections
类提供的静态方法(如排序、查找等)进行进一步的操作。
然而,值得注意的是,ArrayList
并不是所有情况下的最佳选择。例如,如果你需要频繁地在列表的开头或中间插入和删除元素,那么LinkedList
可能会是更好的选择,因为它在这些操作上更加高效。同样,如果你需要一个不允许重复元素的列表,那么HashSet
(通过其Set
接口)或LinkedHashSet
(保持元素插入顺序)可能是更合适的选择。选择哪种集合类型取决于你的具体需求和性能考虑。
1.3 API演示
1、演示添加
注意:add和addAll的区别
package com.atguigu.collection;import org.junit.Test;import java.util.ArrayList;
import java.util.Collection;public class TestCollectionAdd {@Testpublic void testAdd(){//ArrayList是Collection的子接口List的实现类之一。Collection<String> coll = new ArrayList<>();coll.add("小李广");coll.add("扫地僧");coll.add("石破天");System.out.println(coll);}@Testpublic void testAddAll(){Collection<String> c1 = new ArrayList<>();c1.add(1);c1.add(2);System.out.println("c1集合元素的个数:" + c1.size());//2System.out.println("c1 = " + c1);Collection<String> c2 = new ArrayList<>();c2.add(1);c2.add(2);System.out.println("c2集合元素的个数:" + c2.size());//2System.out.println("c2 = " + c2);Collection<String> other = new ArrayList<>();other.add(1);other.add(2);other.add(3);System.out.println("other集合元素的个数:" + other.size());//3System.out.println("other = " + other);System.out.println();c1.addAll(other);System.out.println("c1集合元素的个数:" + c1.size());//5System.out.println("c1.addAll(other) = " + c1);c2.add(other);System.out.println("c2集合元素的个数:" + c2.size());System.out.println("c2.add(other) = " + c2);}
}
注意:coll.addAll(other);与coll.add(other);
2、演示删除
注意几种删除方法的区别
package com.atguigu.collection;import org.junit.Test;import java.util.ArrayList;
import java.util.Collection;
import java.util.function.Predicate;public class TestCollectionRemove {@Testpublic void test01(){Collection coll = new ArrayList();coll.add("小李广");coll.add("扫地僧");coll.add("石破天");coll.add("佛地魔");System.out.println("coll = " + coll);coll.remove("小李广");System.out.println("删除元素\"小李广\"之后coll = " + coll);coll.removeIf(new Predicate() {@Overridepublic boolean test(Object o) {String str = (String) o;return str.contains("地");}});System.out.println("删除包含\"地\"字的元素之后coll = " + coll);coll.clear();System.out.println("coll清空之后,coll = " + coll);}@Testpublic void test02() {Collection coll = new ArrayList();coll.add("小李广");coll.add("扫地僧");coll.add("石破天");coll.add("佛地魔");System.out.println("coll = " + coll);Collection other = new ArrayList();other.add("小李广");other.add("扫地僧");other.add("尚硅谷");System.out.println("other = " + other);coll.removeAll(other);System.out.println("coll.removeAll(other)之后,coll = " + coll);System.out.println("coll.removeAll(other)之后,other = " + other);}@Testpublic void test03() {Collection coll = new ArrayList();coll.add("小李广");coll.add("扫地僧");coll.add("石破天");coll.add("佛地魔");System.out.println("coll = " + coll);Collection other = new ArrayList();other.add("小李广");other.add("扫地僧");other.add("尚硅谷");System.out.println("other = " + other);coll.retainAll(other);System.out.println("coll.retainAll(other)之后,coll = " + coll);System.out.println("coll.retainAll(other)之后,other = " + other);}}
3、演示查询与获取元素
package com.atguigu.collection;import org.junit.Test;import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;public class TestCollectionContains {@Testpublic void test01() {Collection coll = new ArrayList();System.out.println("coll在添加元素之前,isEmpty = " + coll.isEmpty());coll.add("小李广");coll.add("扫地僧");coll.add("石破天");coll.add("佛地魔");System.out.println("coll的元素个数" + coll.size());Object[] objects = coll.toArray();System.out.println("用数组返回coll中所有元素:" + Arrays.toString(objects));System.out.println("coll在添加元素之后,isEmpty = " + coll.isEmpty());coll.clear();System.out.println("coll在clear之后,isEmpty = " + coll.isEmpty());}@Testpublic void test02() {Collection coll = new ArrayList();coll.add("小李广");coll.add("扫地僧");coll.add("石破天");coll.add("佛地魔");System.out.println("coll = " + coll);System.out.println("coll是否包含“小李广” = " + coll.contains("小李广"));System.out.println("coll是否包含“宋红康” = " + coll.contains("宋红康"));Collection other = new ArrayList();other.add("小李广");other.add("扫地僧");other.add("尚硅谷");System.out.println("other = " + other);System.out.println("coll.containsAll(other) = " + coll.containsAll(other));}@Testpublic void test03(){Collection c1 = new ArrayList();c1.add(1);c1.add(2);System.out.println("c1集合元素的个数:" + c1.size());//2System.out.println("c1 = " + c1);Collection c2 = new ArrayList();c2.add(1);c2.add(2);System.out.println("c2集合元素的个数:" + c2.size());//2System.out.println("c2 = " + c2);Collection other = new ArrayList();other.add(1);other.add(2);other.add(3);System.out.println("other集合元素的个数:" + other.size());//3System.out.println("other = " + other);System.out.println();c1.addAll(other);System.out.println("c1集合元素的个数:" + c1.size());//5System.out.println("c1.addAll(other) = " + c1);System.out.println("c1.contains(other) = " + c1.contains(other));System.out.println("c1.containsAll(other) = " + c1.containsAll(other));System.out.println();c2.add(other);System.out.println("c2集合元素的个数:" + c2.size());System.out.println("c2.add(other) = " + c2);System.out.println("c2.contains(other) = " + c2.contains(other));System.out.println("c2.containsAll(other) = " + c2.containsAll(other));}}
2 Iterator迭代器
2.1 Iterator接口
在程序开发中,经常需要遍历集合中的所有元素。针对这种需求,JDK专门提供了一个接口java.util.Iterator
。Iterator
接口也是Java集合中的一员,但它与Collection
、Map
接口有所不同,Collection
接口与Map
接口主要用于存储元素,而Iterator
主要用于迭代访问(即遍历)Collection
中的元素,因此Iterator
对象也被称为迭代器。
想要遍历Collection集合,那么就要获取该集合迭代器完成迭代操作,下面介绍一下获取迭代器的方法:
public Iterator iterator()
: iterator(): 获取到集合的迭代器,用来遍历集合里的元素。方法返回的结果是 Iterator类型的对象
下面介绍一下迭代的概念:
- 迭代:即Collection集合元素的通用获取方式。在取元素之前先要判断集合中有没有元素,如果有,就把这个元素取出来,继续在判断,如果还有就再取出出来。直到把集合中的所有元素全部取出。这种取出方式专业术语称为迭代。
Iterator接口的常用方法如下:
public E next()
:返回迭代的下一个元素。public boolean hasNext()
:如果仍有元素可以迭代,则返回 true。
接下来我们通过案例学习如何使用Iterator迭代集合中元素:
public class IteratorDemo {public static void main(String[] args) {// List<String> names = new ArrayList<>();Set<String> names = new HashSet<>();names.add("tom");names.add("jack");names.add("jerry");names.add("rose");// 遍历集合元素的方式一: 使用 for循环(只能用于list)/*for (int i = 0; i < names.size(); i++) {System.out.println(names.get(i));}*/// 遍历集合元素的方式二: 使用迭代器+while循环遍历/*Iterator<String> itr = names.iterator();while (itr.hasNext()) {System.out.println(itr.next());}*/// 遍历集合元素的方式三: 使用迭代器+for循环遍历/*for (Iterator<String> itr = names.iterator(); itr.hasNext(); ) {String name = itr.next();System.out.println(name);}*//*遍历集合元素的方式四: 增强for循环(迭代器的语法糖)for(元素数据类型 元素变量名: 集合) {}*/for (String name : names) {System.out.println(name);}int[] nums = {56, 78};/*for (int i = 0; i < nums.length; i++) {int num = nums[i];System.out.println(num);}*/for (int num : nums) { // 数组没有迭代器,数组的增强for循环是fori循环的语法糖System.out.println(num);}}
}
提示:在进行集合元素取出时,如果集合中已经没有元素了,还继续使用迭代器的next方法,将会发生java.util.NoSuchElementException没有集合元素的错误。
2.2 迭代器的实现原理
我们在之前案例已经完成了Iterator遍历集合的整个过程。当遍历集合时,首先通过调用集合的iterator()方法获得迭代器对象,然后使用hashNext()方法判断集合中是否存在下一个元素,如果存在,则调用next()方法将元素取出,否则说明已到达了集合末尾,停止遍历元素。
Iterator迭代器对象在遍历集合时,内部采用指针的方式来跟踪集合中的元素,接下来通过一个图例来演示Iterator对象迭代元素的过程:
在调用Iterator的next方法之前,迭代器指向第一个元素,当第一次调用迭代器的next方法时,返回第一个元素,然后迭代器的索引会向后移动一位,指向第二个元素,当再次调用next方法时,返回第二个元素,然后迭代器的索引会再向后移动一位,指向第三个元素,依此类推,直到hasNext方法返回false,表示到达了集合的末尾,终止对元素的遍历。
2.3 Iterable接口
1、 Iterable接口依赖Iterator接口
java.lang.Iterable接口的抽象方法:
- public Iterator iterator(): 获取对应的迭代器,用来遍历集合中的元素的。
凡是实现了 Iterable接口的集合,就必须重写 iterator()方法,即还必须为该集合提供一个Iterator接口的实现类,否则就无法完成该方法的重写。
2、forEach方法
java.lang.Iterable接口在Java8还提供了一个forEach默认方法:
- public default void forEach(Consumer<? super T> action):传入Consumer接口的实现类对象,完成集合元素的迭代
java.util.function.Consumer接口的抽象方法:
- void accept(T t):对元素t执行给定的操作
package com.atguigu.iter;import org.junit.Test;import java.util.ArrayList;
import java.util.Collection;
import java.util.function.Consumer;public class TestForEachMethod {@Testpublic void test1(){Collection coll = new ArrayList();coll.add("小李广");coll.add("扫地僧");coll.add("石破天");coll.forEach(new Consumer() {@Overridepublic void accept(Object o) {System.out.println(o);}});}
}
3、新语法糖:foreach循环(增强for循环)
Java5(JDK1.5)中增加了java.lang.Iterable接口,实现这个接口允许对象成为 “foreach” 语句的目标。 Java 5时Collection接口继承了java.lang.Iterable接口,因此Collection系列的集合就可以直接使用foreach循环遍历。
foreach循环的语法格式:
for(元素类型 元素名 : 集合名等){
}
//这里元素名就是一个临时变量,自己命名就可以
对于集合类型来说,foreach循环其实就是使用Iterator迭代器来完成元素的遍历的。
package com.atguigu.iterator;import org.junit.Test;import java.util.ArrayList;
import java.util.Collection;public class TestForeach {@Testpublic void test01(){Collection coll = new ArrayList();coll.add("小李广");coll.add("扫地僧");coll.add("石破天");for (Object o : coll) {System.out.println(o);}}
}
Java中的数组也支持这种语法糖。只不过编译器在处理foreach遍历数组时,是转换为普通for循环的。
代码示例:
package com.atguigu.api;public class TestForeach {public static void main(String[] args) {int[] nums = {1,2,3,4,5};for (int num : nums) {System.out.println(num);}System.out.println("-----------------");String[] names = {"张三","李四","王五"};for (String name : names) {System.out.println(name);}}
}
2.4 使用Iterator迭代器删除元素
java.util.Iterator迭代器中有一个方法:
void remove() ;
那么,既然Collection已经有remove(xx)方法了,为什么Iterator迭代器还要提供删除方法呢?
因为在JDK1.8之前Collection接口没有removeIf方法,即无法根据条件删除。
例如:要删除以下集合元素中的偶数
package com.atguigu.iterator;import org.junit.Test;import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;public class TestIteratorRemove {@Testpublic void test01(){Collection coll = new ArrayList();coll.add(1);coll.add(2);coll.add(3);coll.add(4);// coll.remove(?)//没有removeIf方法无法实现删除“偶数”Iterator iterator = coll.iterator();while(iterator.hasNext()){Integer element = (Integer) iterator.next();if(element%2 == 0){iterator.remove();}}System.out.println(coll);}
}
2.5 Iterator迭代器的快速失败(fail-fast)机制
如果在Iterator、ListIterator迭代器创建后的任意时间从结构上修改了集合(通过迭代器自身的 remove 或 add 方法之外的任何其他方式),则迭代器将抛出 ConcurrentModificationException。因此,面对并发的修改,迭代器很快就完全失败,而不是冒着在将来不确定的时间任意发生不确定行为的风险。
这样设计是因为,迭代器代表集合中某个元素的位置,内部会存储某些能够代表该位置的信息。当集合发生改变时,该信息的含义可能会发生变化,这时操作迭代器就可能会造成不可预料的事情。因此,果断抛异常阻止,是最好的方法。这就是Iterator迭代器的快速失败(fail-fast)机制。
1、ConcurrentModificationException异常
public class IteratorTest {public static void main(String[] args) {List<String> heroes = new ArrayList<>();heroes.add("张飞");heroes.add("诸葛亮");heroes.add("赵云");heroes.add("李白");heroes.add("诸葛亮");heroes.add("诸葛亮");heroes.add("东皇太一");heroes.add("鲁班");heroes.add("诸葛亮");heroes.add("夏侯惇");System.out.println(heroes);/*for (int i = 0; i < heroes.size(); i++) {if ("诸葛亮".equals(heroes.get(i))) {heroes.remove(i);i--;}}*//*for (int i = 0; i < heroes.size(); ) {if ("诸葛亮".equals(heroes.get(i))) {heroes.remove(i);} else {i++;}}*//*for (int i = heroes.size() - 1; i >= 0; i--) {if ("诸葛亮".equals(heroes.get(i))) heroes.remove(i);}*//*迭代器循环里,不能调用集合的remove/add等方法进行增删操作调用next()方法时,第一件事就是比较 modCount 和 expectedModCount如果 modCount != expectedModCount,立马抛出ConcurrentModificationException只要调用了集合的 remove/add等方法对集合进行了增删,modCount会自增,但是迭代器里的 expectedModCount没有同步更新,再次调用next()方法就会抛出异常如果不是调用集合的 remove/add方法对集合增删,而是调用了迭代器的remove()方法删除元素,不会抛出 并发修改异常,迭代器的remove()方法就是调用了集合的remove()方法删除元素(modCount会自增),然后再将 expectedModCount = modCount,同时还调整 cursor 和 lastRet*/Iterator<String> itr = heroes.iterator();for (; itr.hasNext(); ) {String h = itr.next();// if ("诸葛亮".equals(h)) heroes.remove(h);if ("诸葛亮".equals(h)) itr.remove(); // 可以调用迭代器的remove()方法删除元素}System.out.println(heroes);}
2、modCount变量
那么迭代器如何实现快速失败(fail-fast)机制的呢?
- 在ArrayList等集合类中都有一个modCount变量。它用来记录集合的结构被修改的次数。
- 当我们给集合添加和删除操作时,会导致modCount++。
- 然后当我们用Iterator迭代器遍历集合时,创建集合迭代器的对象时,用一个变量记录当前集合的modCount。例如:
int expectedModCount = modCount;
,并且在迭代器每次next()迭代元素时,都要检查expectedModCount != modCount
,如果不相等了,那么说明你调用了Iterator迭代器以外的Collection的add,remove等方法,修改了集合的结构,使得modCount++,值变了,就会抛出ConcurrentModificationException。
下面以AbstractList和ArrayList.Itr迭代器为例进行源码分析:
AbstractList类中声明了modCount变量:
/*** The number of times this list has been <i>structurally modified</i>.* Structural modifications are those that change the size of the* list, or otherwise perturb it in such a fashion that iterations in* progress may yield incorrect results.** <p>This field is used by the iterator and list iterator implementation* returned by the {@code iterator} and {@code listIterator} methods.* If the value of this field changes unexpectedly, the iterator (or list* iterator) will throw a {@code ConcurrentModificationException} in* response to the {@code next}, {@code remove}, {@code previous},* {@code set} or {@code add} operations. This provides* <i>fail-fast</i> behavior, rather than non-deterministic behavior in* the face of concurrent modification during iteration.** <p><b>Use of this field by subclasses is optional.</b> If a subclass* wishes to provide fail-fast iterators (and list iterators), then it* merely has to increment this field in its {@code add(int, E)} and* {@code remove(int)} methods (and any other methods that it overrides* that result in structural modifications to the list). A single call to* {@code add(int, E)} or {@code remove(int)} must add no more than* one to this field, or the iterators (and list iterators) will throw* bogus {@code ConcurrentModificationExceptions}. If an implementation* does not wish to provide fail-fast iterators, this field may be* ignored.*/protected transient int modCount = 0;
翻译解释:modCount是这个list被结构性修改的次数。子类使用这个字段是可选的,如果子类希望提供fail-fast迭代器,它仅仅需要在add(int, E),remove(int)方法(或者它重写的其他任何会结构性修改这个列表的方法)中添加这个字段。调用一次add(int,E)或者remove(int)方法时必须且仅仅给这个字段加1,否则迭代器会抛出伪装的ConcurrentModificationExceptions错误。如果一个实现类不希望提供fail-fast迭代器,则可以忽略这个字段。
Arraylist的Itr迭代器:
private class Itr implements Iterator<E> {int cursor; int lastRet = -1; int expectedModCount = modCount;//在创建迭代器时,expectedModCount初始化为当前集合的modCount的值public boolean hasNext() {return cursor != size;}@SuppressWarnings("unchecked")public E next() {checkForComodification();//校验expectedModCount与modCount是否相等int i = cursor;if (i >= size)throw new NoSuchElementException();Object[] elementData = ArrayList.this.elementData;if (i >= elementData.length)throw new ConcurrentModificationException();cursor = i + 1;return (E) elementData[lastRet = i];}final void checkForComodification() {if (modCount != expectedModCount)//校验expectedModCount与modCount是否相等throw new ConcurrentModificationException();//不相等,抛异常}
}
ArrayList的remove方法:
public boolean remove(Object o) {if (o == null) {for (int index = 0; index < size; index++)if (elementData[index] == null) {fastRemove(index);return true;}} else {for (int index = 0; index < size; index++)if (o.equals(elementData[index])) {fastRemove(index);return true;}}return false;}private void fastRemove(int index) {modCount++;int numMoved = size - index - 1;if (numMoved > 0)System.arraycopy(elementData, index+1, elementData, index,numMoved);elementData[--size] = null; // clear to let GC do its work}
注意,迭代器的快速失败行为不能得到保证,一般来说,存在不同步的并发修改时,不可能作出任何坚决的保证。快速失败迭代器尽最大努力抛出 ConcurrentModificationException
。因此,编写依赖于此异常的程序的方式是错误的,正确做法是:*迭代器的快速失败行为应该仅用于检测 bug。*例如:
package com.atguigu.iterator;import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;public class TestNoConcurrentModificationException {public static void main(String[] args) {Collection coll = new ArrayList();coll.add("hello");coll.add("world");coll.add("java");coll.add("haha");Iterator iterator = coll.iterator();while (iterator.hasNext()) {String str = (String) iterator.next();if (str.contains("a")) {coll.remove(str);//Iterator遍历集合过程中,调用集合的remove方法}}}
}
3 元素要求
3.1 有序、可重复
Collection 层次结构中的根接口。一些 collection 允许有重复的元素,而另一些则不允许。一些 collection 是有序的,而另一些则是无序的。JDK 不提供此接口的任何直接实现:它提供更具体的子接口(如 Set 和 List、Queue)实现。 我们掌握了Collection接口的使用后,再来看看Collection接口中的子接口,他们都具备那些特性呢?
1、List接口介绍
java.util.List
接口继承自Collection
接口,是单列集合的一个重要分支,习惯性地会将实现了List
接口的对象称为List集合。
List的常用实现类有:ArrayList、Vector、LinkedList、Stack等。
List接口特点:
- List集合所有的元素是以一种线性方式进行存储的
- 它是一个元素存取有序的集合。即元素的存入顺序和取出顺序有保证。
- 它是一个带有索引的集合,通过索引就可以精确的操作集合中的元素(与数组的索引是一个道理)。
- 集合中可以有重复的元素
List集合类中元素有序、且可重复。这就像银行门口客服,给每一个来办理业务的客户分配序号:第一个来的是“张三”,客服给他分配的是0;第二个来的是“李四”,客服给他分配的1;以此类推,最后一个序号应该是“总人数-1”。
注意:
List集合关心元素是否有序,而不关心是否重复,请大家记住这个原则。例如“张三”可以领取两个号。
在JavaSE中List名称的类型有两个,一个是java.util.List集合接口,一个是java.awt.List图形界面的组件,别导错包了。
2、List接口中常用方法
List作为Collection集合的子接口,不但继承了Collection接口中的全部方法,而且还增加了一些根据元素索引来操作集合的特有方法,如下:
List除了从Collection集合继承的方法外,List 集合里添加了一些根据索引来操作集合元素的方法。
1、添加元素
- void add(int index, E ele):把元素添加到指定位置
- boolean addAll(int index, Collection<? extends E> eles):把一组元素添加到指定位置
2、删除元素
- E remove(int index):删除指定位置的元素
3、修改元素
- E set(int index, E ele):替换[index]位置的元素
- default void replaceAll(UnaryOperator operator):按指定操作的要求替换元素
4、获取元素
-
E get(int index):返回[index]位置的元素
-
List subList(int fromIndex, int toIndex):返回[fromIndex, toIndex)范围的元素
-
int indexOf(Object obj):查询obj在列表中的位置,如果有重复,返回第1个
-
int lastIndexOf(Object obj):查询obj在列表中的位置,如果有重复,返回最后1个
示例代码:
package com.atguigu.list;import org.junit.Test;import java.util.*;
import java.util.function.UnaryOperator;public class TestListMethod {@Testpublic void test08(){/*演示foreach和Iterator迭代器*/List list = new ArrayList();list.add("hello");list.add("java");list.add("world");list.add("mysql");list.add("java");Iterator iterator = list.iterator();while(iterator.hasNext()){Object next = iterator.next();System.out.println(next);}}@Testpublic void test07(){/*演示foreach和Iterator迭代器*/List list = new ArrayList();list.add("hello");list.add("java");list.add("world");list.add("mysql");list.add("java");for (Object o : list) {System.out.println(o);}}@Testpublic void test6() {/*ArrayList是List接口的实现类.演示查询,截取*/List list = new ArrayList();list.add("hello");list.add("java");list.add("world");list.add("mysql");list.add("java");System.out.println(list.get(1));//javaSystem.out.println(list.indexOf("java"));//1System.out.println(list.lastIndexOf("java"));//4List subList = list.subList(1, 3);//[1,3)下标范围的元素System.out.println("subList = " + subList);//[java, world]}@Testpublic void test5(){List list = new ArrayList();list.add("hello");list.add("java");list.add("world");list.add("mysql");list.add("java");list.replaceAll(new UnaryOperator() {@Overridepublic Object apply(Object o) {return "java".equals(o) ? "atguigu" : o;}});System.out.println(list);}@Testpublic void test4(){/*ArrayList是List接口的实现类。演示修改替换*/List list = new ArrayList();list.add("hello");list.add("java");list.set(0,"hi");System.out.println(list);//[hi, java]}@Testpublic void test03(){/*ArrayList是List接口的实现类。演示删除*/List list =new ArrayList();list.add(10);list.add(20);list.add(30);list.add(40);list.remove(1);//删除[1]位置的元素System.out.println(list);//[10, 30, 40]//list.remove(30);//删除元素值为30的元素list.remove(Integer.valueOf(30));//手动装箱System.out.println(list);//[10, 40]}@Testpublic void test2(){/*ArrayList是List接口的实现类。演示删除*/List<String> list = new ArrayList<>();list.add("hello");list.add("java");list.remove(0);System.out.println(list);}@Testpublic void test01(){/*ArrayList是List接口的实现类。演示添加*/List list = new ArrayList();list.add("hello");list.add("java");list.add(0,"atguigu");list.add(2,"world");System.out.println(list);//[atguigu, hello, world, java]List list2 = Arrays.asList("chai", "lin", "yan");list.addAll(1,list2);System.out.println(list);//[atguigu, chai, lin, yan, hello, world, java]}
}
3、ListIterator迭代器
List 集合额外提供了一个 listIterator() 方法,该方法返回一个 ListIterator 列表迭代器对象, ListIterator 接口继承了 Iterator 接口,提供了专门操作 List 的方法:
- void add():通过迭代器添加元素到对应集合
- void set(Object obj):通过迭代器替换正迭代的元素
- void remove():通过迭代器删除刚迭代的元素
- boolean hasPrevious():如果以逆向遍历列表,往前是否还有元素。
- Object previous():返回列表中的前一个元素。
- int previousIndex():返回列表中的前一个元素的索引
- boolean hasNext()
- Object next()
- int nextIndex()
package com.atguigu.list;import org.junit.Test;import java.util.ArrayList;
import java.util.List;
import java.util.ListIterator;public class TestListIterator {@Testpublic void test7() {/*ArrayList是List接口的实现类。演示ListIterator迭代器*/List list = new ArrayList();list.add("hello");list.add("java");list.add("world");list.add("mysql");list.add("java");ListIterator stringListIterator = list.listIterator();while(stringListIterator.hasNext()){int index = stringListIterator.nextIndex();Object next = stringListIterator.next();if(next.equals("java")){stringListIterator.set("JavaEE");}}System.out.println(list);//[hello, JavaEE, world, mysql, JavaEE]}@Testpublic void test6() {/*ArrayList是List接口的实现类。演示ListIterator迭代器*/List list = new ArrayList();list.add("hello");list.add("java");list.add("world");list.add("mysql");list.add("java");ListIterator stringListIterator = list.listIterator();while(stringListIterator.hasNext()){int index = stringListIterator.nextIndex();Object next = stringListIterator.next();System.out.println("index = " + index +",next = " + next);}System.out.println("---------------");while(stringListIterator.hasPrevious()){int index = stringListIterator.previousIndex();Object previous = stringListIterator.previous();System.out.println("index = " + index +",previous = " + previous);}System.out.println("---------------");stringListIterator = list.listIterator(2);while(stringListIterator.hasNext()){int index = stringListIterator.nextIndex();Object next = stringListIterator.next();System.out.println("index = " + index +",next = " + next);}}
}
3.2 无序、不可重复
1、Set接口介绍
Set接口是Collection的子接口,set接口没有提供额外的方法。但是比Collection
接口更加严格了。
Set 集合不允许包含相同的元素,即元素不可重复。
Set集合支持的遍历方式和Collection集合一样:foreach和Iterator。
Set的常用实现类有:HashSet、TreeSet、LinkedHashSet等。
1.1HashSet
/*
Set在Collection的基础上,没有新增方法
Set集合特点: 存储的元素无序,且不允许重复(判断元素是否重复的依据)HahSet判断元素是否重复的依据:1. 先调用HashCode方法,看哈希值是否相同。如果哈希值不同,直接存入2. 哈希值相同,再调用equals方法。如果equals返回true,就认为是同一个元素,就不再存入
*/
public class HashSetDemo {public static void main(String[] args) {Set<Dog> set = new HashSet<>();set.add(new Dog("2pj", "黄色", 3, "公", 12.5));set.add(new Dog("来福", "黑色", 1, "母", 5.1));set.add(new Dog("大黄", "黄色", 2, "公", 10.5));set.add(new Dog("旺财", "黄色", 3, "公", 12.5));set.add(new Dog("来福", "黑色", 1, "母", 5.1));set.add(new Dog("3Qj", "黄色", 3, "公", 12.5));System.out.println(set);}
}
1.2TreeSet
/*
TreeSet底层使用的红黑树结构:判断元素是否重复的依据: 将元素转换成为Comparable类型,然后再调用元素的 compareTo()方法 //笔记记得记一下这两个接口结果如果是 0就认为是同一个元素,就不再存入TreeSet在存储元素时,要么元素实现 Comparable接口,并实现compareTo()方法
要么在创建TreeSet对象时,调用 TreeSet(Comparator c) 构造方法,传入一个比较器
*/
public class TreeSetDemo {public static void main(String[] args) {Set<String> set = new TreeSet<>();set.add("hello");set.add("good");set.add("ok");set.add("xyz");set.add("yes");set.add("go");set.add("abc");set.add(new String("go"));set.add("hi");System.out.println(set);Set<Integer> nums = new TreeSet<>();nums.add(90);nums.add(73);nums.add(128);nums.add(65);nums.add(99);nums.add(83);nums.add(77);System.out.println(nums);Set<Student> students = new TreeSet<>();students.add(new Student("jack", 18, 90));students.add(new Student("tom", 20, 78));students.add(new Student("rose", 18, 94));System.out.println(students);Set<Dog> dogs = new TreeSet<>(new Comparator<Dog>() {@Overridepublic int compare(Dog o1, Dog o2) {return o1.getAge() - o2.getAge();}});dogs.add(new Dog("大黄", "黄色", 2, "公", 10.8));}
}
2、元素相等和重复问题:equals和hashCode方法
HashSet和LinkedHashSet元素不可重复是依赖于equals和hashCode方法
package com.atguigu.set;public class Circle {private double radius;public Circle(double radius) {this.radius = radius;}public double getRadius() {return radius;}public void setRadius(double radius) {this.radius = radius;}@Overridepublic String toString() {return "Circle{" +"radius=" + radius +'}';}@Overridepublic boolean equals(Object o) {if (this == o) return true;if (o == null || getClass() != o.getClass()) return false;Circle circle = (Circle) o;return Double.compare(circle.radius, radius) == 0;}@Overridepublic int hashCode() {long temp = Double.doubleToLongBits(radius);return (int) (temp ^ (temp >>> 32));}
}
@Testpublic void test08(){HashSet<Circle> set = new HashSet<>();set.add(new Circle(1));set.add(new Circle(1));System.out.println(set);}
3、大小顺序和重复问题:Comparable和Comparator接口
TreeSet元素不可重复是依赖于元素大小关系的。
(1)要么元素类型实现Comparable接口,重写compareTo方法,
(2)要么创建TreeSet时,指定Comparator接口实现类对象,重写compare方法
package com.atguigu.set;public class Rectangle implements Comparable<Rectangle> {private double length;private double width;public Rectangle(double length, double width) {this.length = length;this.width = width;}public double getLength() {return length;}public void setLength(double length) {this.length = length;}public double getWidth() {return width;}public void setWidth(double width) {this.width = width;}public double area(){return length * width;}@Overridepublic String toString() {return "Rectangle{" +"length=" + length +", width=" + width +", area=" + area() +'}';}@Overridepublic int compareTo(Rectangle o) {return Double.compare(this.area(), o.area());
// int result = Double.compare(this.length, o.length);
// return result != 0 ? result : Double.compare(this.width, o.width);}
}
@Testpublic void test06(){TreeSet<Rectangle> set = new TreeSet<>();set.add(new Rectangle(5,3));set.add(new Rectangle(4,2));set.add(new Rectangle(6,1));set.add(new Rectangle(3,2));//进不去,因为Rectangle里面的compareTo方法按照面积比较大小//认为面积相同的就是相同的矩形对象System.out.println(set);/*'[Rectangle{length=6.0, width=1.0, area=6.0}, Rectangle{length=4.0, width=2.0, area=8.0}, Rectangle{length=5.0, width=3.0, area=15.0}]*/}@Testpublic void test05(){TreeSet<Circle> set = new TreeSet<>(new Comparator<Circle>() {@Overridepublic int compare(Circle o1, Circle o2) {return Double.compare(o1.getRadius(),o2.getRadius());}});set.add(new Circle(1.5));set.add(new Circle(1.0));set.add(new Circle(2.3));System.out.println(set);//[Circle{radius=1.0}, Circle{radius=1.5}, Circle{radius=2.3}]}
4 CollectionArray
4.1数组转换为集合的方法及注意事项
/*
数组转换成为集合:调用 Arrays.asList(T...elements) 会得到一个 List集合只能传入一个引用数据类型的数组,如果传入的是一个基本数据类型的数组,会把它当做一个元素int[] arr = {89, 45, 67, 88, 90, 76};List<int[]> list = Arrays.asList(arr); 不会将数组展开集合转换成为数组:调用集合的toArray()方法
*/
public class CollectionArrayDemo {public static void main(String[] args) {List<Integer> list = Arrays.asList(89, 45, 67, 88, 90, 76);System.out.println(list.getClass()); // java.util.Arrays$ArrayList// Arrays.asList方法返回的List集合是一个内部类,不支持 add/remove方法// list.add(81); 尝试添加会报UnsupportedOperationException//它表示尝试执行了某个操作,但该操作在当前上下文中不被支持//但是编译不会报错,因为多态编译看左边,list集合有add方法System.out.println(list.get(0));//alist 不能增/删,只可以查//解决办法:new一个新的集合,再进行操作。List<Integer> x = new ArrayList<>(list);// x.addAll(list);// x.add(83);System.out.println(x);List<Integer> y = new ArrayList<>(Arrays.asList(89, 45, 67, 88, 90, 76));List<String> fruits = new ArrayList<>();fruits.add("apple");fruits.add("orange");fruits.add("peach");fruits.add("pear");// Object[] objects = fruits.toArray();// String[] objects = (String[])fruits.toArray();//类型转换不能数组之间转String[] arr = new String[fruits.size()];fruits.toArray(arr);System.out.println(Arrays.toString(arr));}
}
5removeif的使用
public class RemoveDemo {public static void main(String[] args) {List<String> names = new ArrayList<>();names.add("jack");names.add("tom");names.add("jerry");names.add("rose");names.add("mike");names.add("jordan");/* names.removeIf(s -> s.length() == 4);*/names.removeIf(new Predicate<String>() {@Overridepublic boolean test(String s) {return s.length() == 4;}});//这个和他上面那个一样,上面是一个lambda表达式System.out.println(names);/*remove(names, new Predicate<String>() {@Overridepublic boolean test(String s) {return s.length() == 4;}});System.out.println(names);*/}public static void remove(List<String> list, Predicate<String> predicate) {//Predicate是一个接口,使用removeif 需要实现他//它接受一个输入参数并返回一个布尔值结果Iterator<String> itr = list.iterator();while (itr.hasNext()) {String element = itr.next();// if (element.length() == 4) itr.remove();if (predicate.test(element)) itr.remove();}}
}
6 Collections 集合工具类
有关他的方法详细说明
更多内容可以参考这篇
/*
Collection 是集合,是一个容器,有两个子接口 List 和 Set,可以用来存储数据
Collections 是工具类,用来快速操作集合容器。常见方法:addAll(Collection<T> c,T...elements): 将可变参数的元素都添加进集合c里如果传入的是一个基本数据类型的数组,会将基本数据类型数组当做一个参数
*/
public class CollectionsDemo {public static void main(String[] args) {Set<String> fruits = new HashSet<>();String[] arr = {"apple", "banana", "orange"};Collections.addAll(fruits, arr); // 可以直接传入一个数组// fruits.addAll("peach","pear");System.out.println(fruits);List<Integer> nums = new ArrayList<>();nums.add(45);nums.add(89);Collections.addAll(nums, 88, 90, 78, 56);System.out.println(nums);int[] x = {84, 79};//Collections.addAll(nums,x); //会报错,基本数据类型的数组,在传入到 Object可变参数时,会被当做一个整体System.out.println("nums: " + nums);Integer[] y = {66, 93};Collections.addAll(nums, y); // 引用数据类型的数组会展开int[] arr1 = {92, 86};int[] arr2 = {88, 96};List<int[]> list = new ArrayList<>();Collections.addAll(list, arr1, arr2);List<String> names = new ArrayList<>();/*for (int i = 0; i < 10; i++) {names.set(i,"jack");}*/Collections.fill(names,"jack");System.out.println(names);}
}
相关文章:
Day16_集合与迭代器
Day16-集合 Day16 集合与迭代器1.1 集合的概念 集合继承图1.2 Collection接口1、添加元素2、删除元素3、查询与获取元素不过当我们实际使用都是使用的他的子类Arraylist!!! 1.3 API演示1、演示添加2、演示删除3、演示查询与获取元素 2 Iterat…...
html2canvas + jspdf 纯前端HTML导出PDF的实现与问题
前言 这几天接到一个需求,富文本编辑器的内容不仅要展示出来,还要实现展示的内容导出pdf文件。一开始导出pdf的功能是由后端来做的,然后发现对于宽度太大的图片,导出的pdf文件里部分图片内容被遮盖了,但在前端是正常显…...
【JVM】JVM调优练习-随笔
JVM实战笔记-随笔 前言字节码如何查看字节码文件jclasslibJavapArthasArthurs监控面板Arthus查看字节码信息 内存调优内存溢出的常见场景解决内存溢出发现问题Top命令VisualVMArthas使用案例 Prometheus Grafana案例 堆内存情况对比内存泄漏的原因:代码中的内存泄漏并发请求问…...
如何解决 CentOS 7 官方 yum 仓库无法使用
一、背景介绍 编译基于 CentOS 7.6.1810 镜像的 Dockerfile 过程中,执行 yum install 指令时,遇到了错误:Could not resolve host: mirrorlist.centos.org; Unknown error。 二、原因分析 官方停止维护 CentOS 7。该系统内置的 yum.repo 所使用的域名 mirrorlist.centos.o…...
分布式唯一id的7种方案
背景 为什么需要使用分布式唯一id? 如果我们的系统是单体的,数据库是单库,那无所谓,怎么搞都行。 但是如果系统是多系统,如果id是和业务相关,由各个系统生成的情况下,那每个主机生成的主键id就…...
嵌入式物联网在医疗行业中的应用——案例分析
作者主页: 知孤云出岫 目录 嵌入式物联网在医疗行业中的应用——案例分析引言1. 智能病房监控1.1 实时患者监控系统 2. 智能医疗设备管理2.1 设备使用跟踪与维护 3. 智能药物管理3.1 药物分配与跟踪 4. 智能远程医疗4.1 远程患者监控与诊断 总结 嵌入式物联网在医疗行业中的应…...
C语言 底层逻辑详细阐述指针(一)万字讲解 #指针是什么? #指针和指针类型 #指针的解引用 #野指针 #指针的运算 #指针和数组 #二级指针 #指针数组
文章目录 前言 序1:什么是内存? 序2:地址是怎么产生的? 一、指针是什么 1、指针变量的创建及其意义: 2、指针变量的大小 二、指针的解引用 三、指针类型存在的意义 四、野指针 1、什么是野指针 2、野指针的成因 a、指…...
【人工智能大模型】文心一言介绍以及基本使用指令
目录 一、产品背景与技术基础 二、主要功能与特点 基本用法 指令的使用 注意事项 文心一言(ERNIE Bot)是百度基于其文心大模型技术推出的生成式AI产品。以下是对文心一言的详细介绍: 一、产品背景与技术基础 技术背景:百度…...
AI绘画入门实践|Midjourney 的模型版本
模型分类 Midjourney 的模型主要分为2大类: 默认模型:目前包括:V1, V2, V3, V4, V5.0, V5.1, V5.2, V6 NIJI模型:目前包括:NIJI V4, NIJI V5, NIJI V6 模型切换 你在服务器输入框中输入 /settings: 回车后…...
Web3时代的教育技术革新:智能合约在学习管理中的应用
随着区块链技术的发展和普及,Web3时代正在为教育技术带来前所未有的革新和机遇。智能合约作为区块链技术的核心应用之一,不仅在金融和供应链管理等领域展示了其巨大的潜力,也在教育领域中逐渐探索和应用。本文将探讨智能合约在学习管理中的具…...
云计算实训室的核心功能有哪些?
在当今数字化转型浪潮中,云计算技术作为推动行业变革的关键力量,其重要性不言而喻。唯众,作为教育实训解决方案的领先者,深刻洞察到市场对云计算技能人才的迫切需求,精心打造了云计算实训室。这一实训平台不仅集成了先…...
芯科科技第五届物联网开发者大会走进世界各地,巡回开启注册
中国,北京 – 2024年7月18日 – 致力于以安全、智能无线连接技术,建立更互联世界的全球领导厂商Silicon Labs(亦称“芯科科技”,NASDAQ:SLAB)今日宣布,其2024年Works With开发者大会现正开放注册…...
Python创建Excel表和读取Excel表的基础操作
下载openpyxl第三方库 winr打开命令行输入cmd 这个如果不行可以试试其他方法,在运行Python代码的软件里也有直接下载的地方,可以上网搜索 创建Excel表 示例代码:最后要记得保存,可以加一句提示语句。 import openpyxl lst[100,…...
JVM(day2)经典垃圾收集器
经典垃圾收集器 Serial收集 使用一个处理器或一条收集线程去完成垃圾收集工作,更重要的是强调在它进行垃圾收集时,必须暂停其他所有工作线程,直到它收集结束。 ParNew收集器 ParNew 收集器除了支持多线程并行收集之外,其他与 …...
华为od机试真题 — 分披萨(Python)
题目描述 “吃货”和“馋嘴”两人到披萨店点了一份铁盘(圆形)披萨,并嘱咐店员将披萨按放射状切成大小相同的偶数个小块。 但是粗心服务员将披萨切成了每块大小都完全不同奇数块,且肉眼能分辨出大小。 由于两人都想吃到最多的披萨,他们商量…...
ubuntu22.04 安装boost
下载boost压缩包,我这里上传了一份1_81_0版本tar -xzvf boost_1_81_0.tar.gzcd boost_1_81_0/sudo apt install build-essential g autotools-dev libicu-dev libbz2-dev -ysudo ./bootstrap.sh --prefix/usr/./b2sudo ./b2 install 上述7步完成后,相关…...
基于JAVA+SpringBoot+uniapp的心理小程序(小程序版本)
✌全网粉丝20W,csdn特邀作者、博客专家、CSDN新星计划导师、java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ 技术范围:SpringBoot、Vue、SSM、HLMT、Jsp、SpringCloud、Layui、Echarts图表、Nodejs、爬…...
C语言 ——— 输入两个正整数,求出最小公倍数
目录 何为最小公倍数 题目要求 代码实现 方法一:暴力求解法(不推荐) 方法二:递乘试摸法(推荐) 何为最小公倍数 最小公倍数是指两个或者多个正整数(除了0以外)的最小的公共倍数…...
Langchain 对pdf,word,txt等不同文件的加载解析
项目中遇到各种数据资源想要加载近langchain构建本地知识ai系统,怎么加载对应的文件格式呢,一起研究下 引入Langchain from langchain.document_loaders import UnstructuredWordDocumentLoader,PyPDFium2Loader,DirectoryLoader,PyPDFLoader,TextLoad…...
BL201分布式I/O耦合器连接Profinet网络
钡铼技术的BL201分布式I/O耦合器是一个用于Profinet网络的设备,用于连接远程输入/输出(I/O)设备到控制系统,如可编程逻辑控制器(PLC),能够实现分布式的I/O连接和通信。 它支持标准Profinet IO …...
Pycharm 报错 Environment location directory is not empty 解
删除项目中ven文件夹(已存在的),然后再添加新的ven虚拟环境就可以了...
【Android】Intent基础用法及作用
文章目录 使用Intent在活动中穿梭组成显式Intent隐式Intent显式与隐式区别作用 活动间传递数据向下一个活动传递数据返回数据给上一个活动 使用Intent在活动中穿梭 Intent(意图)是一种重要的消息传递对象,用于在不同组件(如活动&…...
Web开发:ASP.NET CORE的后端小结(基础)
1.后端重定向到指定路由 public IActionResult Index(){return RedirectToAction("Index", "Main");//重定向>Main/Index} 【备注】如果在MainController的Index方法中return View();本质是 return View("Index"),返回和方法同名的…...
侧开知识点合集2
一、try .... catch.. AccessViolationException异常触发后,下列程序的输出结果为 static void Main(string[] args) { try { throw new AccessViolationException(); Console.WriteLine("error1"); } catch (Exception e) { Console.WriteLi…...
ARM/Linux嵌入式面经(十六):蔚来嵌入式一二三面面经
文章目录 static作用,局部static和全局static区别TCP三次握手Linux虚拟内存指针引用区别C++内存分区new/delete和malloc/free区别职业规划为什么选择蔚来介绍一下项目然后问我有没有内核级别开发经验,我说没有什么情况进入内核态一、主动式二、被动式三、其他方式注意事项示例…...
Apache BookKeeper 一致性协议解析
导语 Apache Pulsar 是一个多租户、高性能的服务间消息传输解决方案,支持多租户、低延时、读写分离、跨地域复制(GEO replication)、快速扩容、灵活容错等特性。Pulsar 存储层依托于 BookKeeper 组件,所以本文简单探讨一下 BookK…...
Solana的账户模型
Solana的账户模型与其他区块链平台(如以太坊)有所不同,其设计旨在提高性能和扩展性。以下是Solana账户模型的主要特点和工作原理: Solana账户模型概述 账户类型: 普通账户(User Accounts)&…...
iPython与Matplotlib:数据可视化的秘籍
iPython与Matplotlib:数据可视化的秘籍 前言 欢迎来到"iPython与Matplotlib:数据可视化的秘籍"教程!无论你是数据可视化新手还是希望提升技能的专业人士,这里都是你开始的地方。让我们开始这段数据可视化之旅吧&#…...
做一只勤劳的小蜜蜂
机缘 成为创作者的初心,对我而言,是一个融合了个人兴趣、职业成长以及对知识传播热爱的复杂而纯粹的情感交织。回顾这段旅程的起点,几个核心驱动力始终引领着我前行: 1、记录与反思:在职业生涯的早期,我遇…...
如何处理 PostgreSQL 中死锁的情况?
🍅关注博主🎗️ 带你畅游技术世界,不错过每一次成长机会!📚领书:PostgreSQL 入门到精通.pdf 文章目录 如何处理 PostgreSQL 中死锁的情况?一、认识死锁二、死锁的症状三、死锁的检测四、预防死锁…...
wordpress菜单手机显示下拉/北京网站seo哪家公司好
题目来源:链接 题目描述: 输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字。 示例 1: 输入:matrix [[1,2,3],[4,5,6],[7,8,9]] 输出:[1,2,3,6,9,8,7,4,5] 代码实现: def s…...
好网站建设公司哪家好?/每日舆情信息报送
作者:姚远 Oracle ACE “ 今天是个值得纪念的日子,我成了第一个2022-2023年度login的中国的Oracle ACE。” 因为新app的原因,所有的ACE都要重新登记。我因故没有参加2022年8月18日Oracle ACE新app上线的zoom meeting,感谢赵兄的通知&#x…...
如何用浏览器访问本地的wordpress/小程序怎么开发自己的小程序
链接:https://www.luogu.org/problemnew/show/P3919 分析 很明显我们可以用主席树来维护,所谓主席树就是可持久化线段树,能够查询历史版本而且可以实现修改操作,反正就是复制了一遍。其原理就是动态开点复制前驱版本,在…...
公司的网站推广怎么做/网站优化及推广方案
Android Camera Develop: process preview frames in real time efficiently概述本篇我们暂时不介绍像相机APP增加新功能,而是介绍如何处理相机预览帧数据。想必大多数人都对处理预览帧没有需求,因为相机只需要拿来拍照和录像就好了,实际上本…...
温州市网页制作项文静/关键词优化教程
今天在解决爬虫对加密参数的分析时,需要使用到base64解码。但是过程中出现了TypeError:Incorrect padding的错误提示。以下是解决方法,以便查阅。 其实正常使用base64是不会出现问题的,就比如下面的代码。 1 #!usr/bin/env pytho…...
中小型网站建设与管理设计总结/2023年广州疫情最新消息
如何显示文件夹中的图片,数据库表中,只记录图片名称或是图片路径。写一个用户控件,即轻易把图片显示出来。 另外,如何取得记录的主键,您是用HiddenField,DataKeyNames还是用DataRowView来取得? …...