美文网首页
Java遍历ArrayList,Map和Set四种方法对比

Java遍历ArrayList,Map和Set四种方法对比

作者: 上善若泪 | 来源:发表于2021-11-10 10:23 被阅读0次

1.1 List

1.1.1 遍历ArrayList

package com.test;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
public class ArrayListDemo {
    public static void main(String args[]){
        List<String> list = new ArrayList<String>();
        list.add("test");
        list.add("demo");
        //方法1
        Iterator it1 = list.iterator();
        while(it1.hasNext()){
            System.out.println(it1.next());
        }
        //方法2
        for(Iterator it2 = list.iterator();it2.hasNext();){
             System.out.println(it2.next());
        }
        //方法3
        for(String tmp:list){
            System.out.println(tmp);
        }
        //方法4
        for(int i = 0;i < list.size(); i ++){
            System.out.println(list.get(i));
        }
    }
}

1.1.2 数组与List之间相互转换的方法详解

  1. List转换成为数组(这里的List是实体是ArrayList)
    调用ArrayListtoArray方法。
    public T[] toArray(T[] a)返回一个按照正确的顺序包含此列表中所有元素的数组;返回数组的运行时类型就是指定数组的运行时类型。如果列表能放入指定的数组,则返回放入此列表元素的数组。否则,将根据指定数组的运行时类型和此列表的大小分配一个新的数组。
    如果指定的数组能容纳列表并有剩余空间(即数组的元素比列表的多),那么会将数组中紧跟在集合末尾的元素设置为 null。这对确定列表的长度很有用,但只在调用方知道列表中不包含任何 null 元素时才有用。
    指定者:接口 Collection 中的 toArray
    指定者:接口List 中的 toArray
    覆盖:类 AbstractCollection 中的 toArray
    参数:a - 要存储列表元素的数组,如果它足够大的话;否则,它是一个为存储列表元素而分配的、具有相同运行时类型的新数组。
    返回:包含列表元素的数组。
    抛出:ArrayStoreException - 如果 a 的运行时类型不是此列表中每个元素的运行时类型的超类型
    具体用法:
List list = new ArrayList();
list.add("1");
list.add("2");
final int size = list.size();
String[] arr = (String[])list.toArray(new String[size]);
  1. 数组转换成为List
    调用ArraysasList方法.
    JDK 1.4java.util.Arrays.asList的定义,函数参数是Object[]。所以,在1.4asList()并不支持基本类型的数组作参数。
    JDK 1.5中,java.util.Arrays.asList的定义,函数参数是Varargs, 采用了泛型实现。同时由于autoboxing的支持,使得可以支持对象数组以及基本类型数组。
    不过在使用时,当传入基本数据类型的数组时,会出现小问题,会把传入的数组整个当作返回的List中的第一个元素,例如:
public static void main(String[] args){
    int[] a1 = new int[]{1,2,3};
    String[] a2 = new String[]{"a","b","c"};

    System.out.println(Arrays.asList(a1));
    System.out.println(Arrays.asList(a2));
}


1
2 [[I@dc8569]
[a, b, c]

下面说说Arrays.asList()的返回值:
JDK文档是这么说的:

public static <T> List<T> asList(T... a)返回一个受指定数组支持的固定大小的列表。(对返回列表的更改会直接写到数组。)此方法同 Collection.toArray()一起,充当了基于数组的API与基于 collectionAPI之间的桥梁。返回的列表是可序列化的,并且实现了 RandomAccess。此方法还提供了一个创建固定长度的列表的便捷方法,该列表被初始化为包含多个元素: List<String> stooges = Arrays.asList("Larry", "Moe", "Curly");

/**
 * @serial include
 */
 private static class ArrayList<E>extends AbstractList<E>implements RandomAccess, java.io.Serializable {
     private static final long serialVersionUID = -2764017481108945198L;
     private final E[] a;

     ArrayList(E[] array) {
         if (array == null)
         throw new NullPointerException();
         a = array;
     }

     public int size() {
        return a.length;
     }

     public Object[] toArray() {
        return a.clone();
     }

     public <T> T[] toArray(T[] a) {
         int size = size();
         if (a.length < size)
            return Arrays.copyOf(this.a, size, (Class<? extends T[]>) a.getClass());
         System.arraycopy(this.a,0, a, 0, size);
         if (a.length > size)
            a[size] = null;
         return a;
     }

     public E get(int index) {
        return a[index];
     }

     public E set(int index, E element) {
         E oldValue = a[index];
         a[index] = element;
         return oldValue;
     }

     public int indexOf(Object o) {
         if (o == null) {
             for (int i = 0; i < a.length; i++)
             if (a[i] == null)
             return i;
         }else {
             for (int i = 0; i < a.length; i++)
             if (o.equals(a[i]))
             return i;
         }
         return -1;
     }
     public boolean contains(Object o) {
        return indexOf(o) != -1;
     }
}

我们都知道,List的一个典型的特性就是其长度是可变的,我们可以很方便地对它进行插入和删除元素的操作,这是它与数组所存在的一个很大的区别,后者的长度是固定的,而且我们不能从数组中删除元素,只能修改元素的值。利用Arrays.asList(array)将返回一个List,然而这个返回的List并不支持addremove的操作。这是什么原因呢?
Arrays.asList源码:

public static <T> List<T> asList(T... a) {
   return new ArrayList<T>(a);
}

这里的ArrayList并不是java.util.ArrayList,而是Arrays的内部类:
我们可以看到该内部类继承的是AbstractList,下面是AbstractListaddremove方法源码:

public boolean add(E e) {
    add(size(), e);
    return true;
}

public void add(int index, E element) {
    throw new UnsupportedOperationException();
}

public E remove(int index) {
    throw new UnsupportedOperationException();
}

所以,当我们对Arrays.asList返回的List进行添加或删除时将会报 java.lang.UnsupportedOperationException 异常。

1.2 遍历Map

1.2.1 四种遍历Map方法

  1. 利用Map. keyset方法。 在map中方法定义如下:
Set<K> keySet();   

例如:

for (String stuKey : stuMap.keySet()) {  
          stuStr = stuKey;   
          stu = stuMap.get(stuKey);  
     }  
  1. 只能遍历values 用到方法是Map.values
for (Student student : stuMap.values()) {  
             stu = student;  
        }  
  1. 利用Map.Keyset().iterator()方法。
Iterator<Map.Entry<String,Student>> it = stuMap.entrySet().iterator();   
ile (it.hasNext()) {  
     Map.Entry<String,Student> entry = it.next();  
     stuStr = entry.getKey();   
     stu = entry.getValue();  
 }  
  1. 结合 Map.Entry<String, Student> entryentrySet 方法:
for (Map.Entry<String, Student> entry : stuMap.entrySet()) {  
         stuStr = entry.getKey();   
         stu = entry.getValue();  
     }  

1.2.2 分析四种方法性能

首先创建了一个测试内为: student,如下:

public class Student {    
    private int age;  
    private String name;  
  
    public Student(String name, int age) {  
        this.age = age;  
        this.name = name;  
    }  
}  

主类:

import java.util.HashMap;  
import java.util.Iterator;  
import java.util.Map;  
  
public class MapListor {  
  
    private static Map<String, Student> stuMap = new HashMap<String, Student> ();  
      
    public static void main(String[] args) {  
          
        intMap(200);  
          
        //method 1: Map.Keyset()  
         long endTime = 0;  
         String stuStr = ""; // key used to be set when listing map.  
         Student stu = null; // value used to be set when listing map.  
         long startTime = System.currentTimeMillis();  
         for (String stuKey : stuMap.keySet()) {  
              stuStr = stuKey;   
              stu = stuMap.get(stuKey);  
         }  
         endTime = System.currentTimeMillis();  
         System.out.println("Using the first method to list Map("+ stuMap.size()+") costs: " + (endTime - startTime) );  
           
         //method 2: Map.Keyset().iterator()  
         startTime = System.currentTimeMillis();  
         Iterator<Map.Entry<String,Student>> it = stuMap.entrySet().iterator();   
         while (it.hasNext()) {  
             Map.Entry<String,Student> entry = it.next();  
             stuStr = entry.getKey();   
             stu = entry.getValue();  
         }  
         endTime = System.currentTimeMillis();  
         System.out.println("Using the second method to list Map("+ stuMap.size()+") costs: " + (endTime - startTime) );  
           
         //method 3: Map.Entry<String, Student> entry and entrySet method.  
         startTime = System.currentTimeMillis();  
         for (Map.Entry<String, Student> entry : stuMap.entrySet()) {  
             stuStr = entry.getKey();   
             stu = entry.getValue();  
         }  
         endTime = System.currentTimeMillis();  
         System.out.println("Using the third method to list Map("+ stuMap.size()+") costs: " + (endTime - startTime) );   
           
         //method 4: only list values.  
         for (Student student : stuMap.values()) {  
             stu = student;  
        }  
    }  
  
    /** 
     * init the stuMap according to the input number of students. 
     * @param stuNumber 
     */  
    private static void intMap(int stuNumber) {  
        Student stu = null;  
        for (int i = 0; i < stuNumber; i++) {  
            stu = new Student("student" + i, (int)Math.random() * 100);  
            stuMap.put(String.valueOf(i), stu);  
        }  
    }  
}

1.3 Set

1.3.1 HashSet

javaSet集合是一个不包含重复元素的Collection,首先我们先看看遍历方法

package com.sort;  
  
import java.util.HashSet;  
import java.util.Iterator;  
import java.util.Set;  
  
/** 
 * 一个不包含重复元素的 collection。更确切地讲,set 不包含满足 e1.equals(e2) 的元素对 e1 和 e2, 
 * @author Owner 
 * 
 */  
public class SetTest2 {  
  
    public static void main(String[] args) {  
        Set<String> set = new HashSet<String>();  
          
        set.add("a");  
        set.add("b");  
        set.add("c");  
        set.add("d");  
        set.add("e");  
          
        set.add("e");//不能放入重复数据  
          
        /** 
         * 遍历方法一,迭代遍历 
         */  
        for(Iterator<String> iterator = set.iterator();iterator.hasNext();){  
            System.out.print(iterator.next()+" ");  
        }  
          
        System.out.println();  
        System.out.println("********************");  
          
        /** 
         * for增强循环遍历 
         */  
        for(String value : set){  
            System.out.print(value+" ");  
        }  
    }  
}  

注意:这里Set集合中放入的是String类型,假如我们放入一个自己定义的类实例的时候,比如Person类实例,这时候我们要自己重新hashcodeequal方法,用自己的关键字段来重写,因为当使用HashSet时,hashCode()方法就会得到调用,判断已经存储在集合中的对象的hash code值是否与增加的对象的hash code值一致;如果不一致,直接加进去;如果一致,再进行equals方法的比较,equals方法如果返回true,表示对象已经加进去了,就不会再增加新的对象,否则加进去。

1.3.1 TreeSet

TreeSet使用元素的自然顺序对元素进行排序,或者根据创建 set 时提供的 Comparator 进行排序,具体取决于使用的构造方法。
通俗一点讲,就是可以按照排序后的列表显示,也可以按照指定的规则排序

    Set<String> set = new TreeSet<String>();  
          
    set.add("f");  
        set.add("a");  
        set.add("b");  
        set.add("c");  
        set.add("d");  
        set.add("e");  
          
        System.out.println(set);  

输出:[a, b, c, d, e, f]

那么如果我们想让他倒序输出呢?当然方法很多。这里我采用指定一个规则让他倒序输出

package com.sort;  
  
import java.util.Comparator;  
import java.util.Iterator;  
import java.util.Set;  
import java.util.TreeSet;  
  
public class TreeSetTest3 {  
  
    public static void main(String[] args) {  
        Set<String> set = new TreeSet<String>(new MyComparator());  
          
        set.add("a");  
        set.add("b");  
        set.add("c");  
        set.add("d");  
        set.add("e");  
        set.add("A");  
          
        for(Iterator<String> iterator = set.iterator();iterator.hasNext();){  
            System.out.print(iterator.next()+" ");  
        }  
    }  
}  
  
class MyComparator implements Comparator<String>{    
    @Override  
    public int compare(String o1, String o2) {            
        return o2.compareTo(o1);//降序排列  
    }  
}  

输出:e d c b a A 

如果Set集合中放入的是我们自己定义的一个类类型呢?
注意:一定要定义一个排序规则类实现Comparator接口,与上面的方法类似

package com.sort;  
  
import java.util.Comparator;  
import java.util.Iterator;  
import java.util.Set;  
import java.util.TreeSet;  
  
public class TreeSetTest2 {  
  
    public static void main(String[] args) {  
        Set<Person> set = new TreeSet<Person>(new PersonComparator());  
          
        Person p1 =  new Person(10);  
        Person p2 =  new Person(20);  
        Person p3 =  new Person(30);  
        Person p4 =  new Person(40);  
          
        set.add(p1);  
        set.add(p2);  
        set.add(p3);  
        set.add(p4);  
          
        for(Iterator<Person> iterator = set.iterator();iterator.hasNext();){  
            System.out.print(iterator.next().score+" ");  
        }  
    }  
}  
  
class Person{  
    int score;  
      
    public Person(int score){  
        this.score = score;  
    }  
      
    public String toString(){  
        return String.valueOf(this.score);  
    }  
}  
  
class PersonComparator implements Comparator<Person>{  
  
    @Override  
    public int compare(Person o1, Person o2) {  
          
        return o1.score - o2.score;  
    }  
      
}  

输出:10 20 30 40 

如果按照一个人的分数的倒序排列,只需要更改compare方法中的o2.score-o1.score

相关文章

网友评论

      本文标题:Java遍历ArrayList,Map和Set四种方法对比

      本文链接:https://www.haomeiwen.com/subject/khpbzltx.html