Java集合框架-Map、Collections工具类

Map接口

双列数据,储存key-value对的数据 —类似于高中的函数: y =f(x)
HashMap:作为Map的主要实现类;线程不安全的,效率高;存储null的key和value
|----LinkedHashMap:保证在遍历map元素时,可以按照添加的顺序实现遍历。原因:在原有的HashMap底层结构基础上,添加了一对指针,指向前一个和后一个元素。对于频繁的遍历操作,此类执行效率高于HashMap
TreeMap:保证按照添加的key-value对进行排序,实现排序遍历。此时考虑key的自然排序或定制排序,底层使用红黑树
Hashtable:作为古老的实现类;线程安全的,效率低;不能存储nulL的key和vaLue
|----Properties:常用来处理配置文件。key和value都是string类型

HashMap的底层:数组+链表(jdk7及之前),数组+链表+红黑树(jdk 8)

Map结构的理解:
Map中的key:无序的、不可重复的,使用set存储所有的key。key所在的类要重写equals()和hashcode()
Map中的value:无序的、可重复的,使用Collection存储所有的value。value所在的类要重写equals()
一个键值对:key-vaLue构成了一个Entry对象。
Map中的entry:无序的、不可重复的,使用Set存储所有的entry

Hashmap底层实现原理

HashMap的底层实现原理,jdk7为例说明:
HashMap map = new HashMap() 在实例化以后,底层创建了长度是16的一维数组Entry[] table
…可能已经执行过多次put…
map.put( key1,value1) 首先,调用key1所在类的hashCode()计算key1哈希值,此哈希值经过某种算法计算以后,得到在Entry数组中的存放位置。如果此位置上的数据为空,此时的key1-vaLue1添加成功。----情况1
如果此位置上的数据不为空,(意味着此位置上存在一个或多个数据(以链表形式存在)),比较key1和已经存在的一个或多个数据。如果key1的哈希值与已经存在的数据的哈希值都不相同,此时key1-value1添加成功。----情况2
如果key1的哈希值和已经存在的某一个数据(key2-value2)的哈希值相同,继续比较:调用key1所在类的equals(key2)的方法
如果equals()返回false:此时key1-vaLue1添加成功。----情况3
如果equals()返回true:使用value1替换vaLue2

补充:关于情况2和情况3:此时key1-value1和原来的数据以链表的方式存储。

在不断的添加过程中,会涉及到扩容问题,当超出临界值(且要存放的位置非空)时,默认的扩容方式:扩容为原来容量的2倍,并将原有的数据复制过来。

jdk8相较于jdk7在底层实现方面的不同:

  1. new HashMap():底层没有创建一个长度为16的数组
  2. jdk8底层的数组是:Node[].而非Entry[]
  3. 首次调用put()方法时,底层创建长度为16的数组
  4. jdk7底层结构只有:数组+链表。jdk8中底层结构:数组+链表+红黑树。当数组的某一个索引位置上的元素以链表形式存在的数据个数>8且当前数组的长度>64时,此时此索引位置上的所有数据改为使用红黑树存储。

HashMap源诮中的重要常量
DEFAULT_INITIAL_CAPACITY : HashMap的默认容量,16
MAXIMUM_CAPACITY:HashMap的最大支持容量,2^30
DEFAULT_LOAD_FACTOR: HashMap的默认加载因子
TREEIFY_THRESHOLD:Bucket中链表长度大于该默认值,转化为红黑树
UNTREEIFY_THRESHOLD:Bucket中红黑树存储的Node小于该默认值,转化为链表
MIN_TREEIFY_CAPACITY:桶中的Node被树化时最小的hash表容量。(当桶中Node的数量大到需要变红黑树时,锁定若hash表容量小于MIN_TREEIFY_CAPACITY时,此时应执行resize扩容操作这个MIN_TREEIFY_CAPACITY的值至少是TREEIFY_THRESHOLD的4倍
table:存储元素的数组,总是2的n次幂
entrySet:存储具体元素的集
size:HashMap中存储的键值对的数量
modCount:HashMap扩容和结构改变的次数。
threshold:扩容的临界值,=容量 * 填充因子
loadFactor:填充因子

LinkedHashMap底层实现原理

在这里插入图片描述
LinkedHashMap记录了前后的数据引用位置,所以得以按照添加顺序输出

map中定义的方法

添加、删除、修改操作:
object put(Object key,0bject value):将指定key-value添加到(或修改)当前map对象中
void putAll(Map m):将m中的所有key-value对存放到当前map中
Object remove(object key):移除指定key的key-vaLue对,并返回vaLue
void clear():清空当前map中的所有数据

public class test{
    public static void main(String[] args) {
        Map map = new HashMap();
        map.put("a",1);
        map.put("test",3);
        map.put("c",2);
        map.put("test",4);
        System.out.println(map);
        Map map1 = new HashMap();
        map1.put("aa",11);
        map1.put("cc",22);
        map.putAll(map1);
        System.out.println(map);
        Object obj = map.remove("aa");
        System.out.println(map);
        System.out.println(obj);
        map.clear();
        System.out.println(map);
    }
}

元素查询的操作:
object get(object key庄获取指定key对应的value
boolean containsKey(object key):是否包含指定的ey
boolean containsVaLlue(object value):是否包含指定的value
int size():返回map中key-value对的个数
boolean isEmpty():判断当前map是否为空
boolean equals(object obj):判断当前map和参数对象obj是否相等

public class test{
    public static void main(String[] args) {
        Map map = new HashMap();
        map.put("a",1);
        map.put("c",2);
        map.put("test",4);
        System.out.println(map.get("test"));
        System.out.println(map.containsKey("test"));
        System.out.println(map.containsValue(4));
        System.out.println(map.size());
        System.out.println(map.isEmpty());
        Map map1 = new HashMap();
        map1.put("a",1);
        map1.put("c",2);
        map1.put("test",4);
        System.out.println(map.equals(map1));

    }
}

元视图操作的方法:
set keyset():返回所有key构成的Set集合
Collection values():返回所有vaLue构成的Collection集合
set entrySet():返回所有key-value对构成的Set集合

Treemap添加方式

public class test{
    public static void main(String[] args) {
        TreeMap treeMap = new TreeMap();
        treeMap.put(new Person("Tom",13),1);
        treeMap.put(new Person("Jack",16),2);
        treeMap.put(new Person("Lily",19),3);
        Set set = treeMap.entrySet();
        Iterator iterator = set.iterator();
        while (iterator.hasNext()){
            System.out.println(iterator.next());
        }
    }
}

class Person implements Comparable{
    String name;
    int age;
    Person(String name,int age){
        this.name = name;
        this.age = age;
    }

    @Override
    public String toString() {
        return this.name + this.age;
    }

    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        Person person = (Person) o;
        return Objects.equals(name, person.name);
    }

    @Override
    public int compareTo(Object o) {
        if (o instanceof Person){
            Person person = (Person)o;
            if (this.age > person.age){
                return 1;
            }else if(this.age < person.age){
                return -1;
            }else {
                return this.name.compareTo(person.name);
            }
        }else {
            throw new RuntimeException("类型不正确");
        }
    }
}
public class test{
    public static void main(String[] args) {
        Comparator comparator = new Comparator() {
            @Override
            public int compare(Object o1, Object o2) {
                if (o1 instanceof Person && o2 instanceof Person){
                    Person person1 = (Person)o1;
                    Person person2 = (Person)o2;
                    if (person1.age > person2.age){
                        return 1;
                    }else if(person1.age < person2.age){
                        return -1;
                    }else{
                        return person1.name.compareTo(person2.name);
                    }
                }else{
                    throw new RuntimeException("类型不正确");
                }
            }
        };
        TreeMap treeMap = new TreeMap(comparator);
        treeMap.put(new Person("Tom",13),1);
        treeMap.put(new Person("Jack",16),2);
        treeMap.put(new Person("Lily",19),3);
        Set set = treeMap.entrySet();
        Iterator iterator = set.iterator();
        while (iterator.hasNext()){
            System.out.println(iterator.next());
        }
    }
}

class Person{
    String name;
    int age;
    Person(String name,int age){
        this.name = name;
        this.age = age;
    }

    @Override
    public String toString() {
        return this.name + this.age;
    }
}

Propertise

public class test{
    public static void main(String[] args) throws IOException {
        Properties properties = new Properties();
        FileInputStream file = new FileInputStream("setting.properties");
        properties.load(file);
        String name = properties.getProperty("name");
        String password = properties.getProperty("password");
        System.out.println(name);
        System.out.println(password);
    }
}

Collections工具类

Collections是一个操作Set、List和l Map等集合的工具类
Collections中提供了一系列静态的方法对集合元素进行排序、查询和修改等操作,还提供了对集合对象设置不可变、对集合对象实现同步控制等方法
排序操作:(均为static方法)
reverse(List):反转List中元素的顺序
shuffle(List):对List集合元素进行随机排序
sort(List):根据元素的自然顺序对指定List集合元素按升序排序
swap(List,int,int):将指定list集合中的i处元素和j处元素进行交换
sort(List,Comparator):根据指定的Comparator产生的顺序对List集合元素进行排序

object max(Collection):根据元素的自然顺序,返回给定集合中的最大元素
object max(Collection,Comparator):根据Comparator指定的顺序,返回给定集合中的最大值
object min( collection)
object min(Collection,Comparator)
int frequency(Collection,object):返回指定集合中指定元素的出现次数
void copy(List dest,List src):将src中的内容复制到dest中
boolean replaceAll(List list,object oldVal,0bject newVal):使用新值替换List对V

public class test{
    public static void main(String[] args) throws IOException {
        List list = new ArrayList();
        list.add(1);
        list.add(2);
        list.add(3);
        List list1 = Arrays.asList(new Object[list.size()]);
        Collections.copy(list1,list);
        System.out.println(list1);
    }
}

Collections常用方法:同步控制
Collections类中提供了多个synchronizedXxx()方法,该方法可使将指定集合包装成线程同步的集合,从而可以解决多线程并发访问集合时的线程安全问题

List list2 = Collections.synchronizedList(list1);
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值