原文: https://www.programiz.com/java-programming/weakhashmap

在本教程中,我们将借助示例学习 Java WeakHashMap及其操作。 我们还将了解WeakHashMapHashMap之间的区别

Java 集合框架的WeakHashMap类提供了哈希表数据结构的功能。

它实现了Map接口

Java `WeakHashMap` - 图1

注意:弱哈希映射的键为WeakReference类型。

如果引用不再在程序中使用,则弱引用类型的对象可以在 Java 中被垃圾回收。

让我们学习首先创建一个弱哈希映射。 然后,我们将学习它与哈希映射的区别。


创建一个WeakHashMap

为了创建一个弱哈希表,我们必须首先导入java.util.WeakHashMap包。 导入包后,可以使用以下方法在 Java 中创建弱哈希表。

  1. //WeakHashMap creation with capacity 8 and load factor 0.6
  2. WeakHashMap<Key, Value> numbers = new WeakHashMap<>(8, 0.6);

在上面的代码中,我们创建了一个名为number的弱哈希映射。

这里,

  • key - 用于关联映射中每个元素(值)的唯一标识符
  • value - 映射中与按键相关联的元素

注意部分new WeakHashMap<>(8, 0.6)。 在此,第一个参数是capcity,第二个参数是loadFactor

  • capcity - 此映射的容量为 8。意味着,它可以存储 8 个条目。
  • loadFactor - 此映射的负载因子为 0.6。 这意味着只要我们的哈希表填充了 60%,条目就会被移到新哈希表中,其大小是原始哈希表的两倍。

默认容量和负载系数

可以在不定义其容量和负载因子的情况下创建弱哈希映射。 例如,

  1. // WeakHashMap with default capacity and load factor
  2. WeakHashMap<Key, Value> numbers1 = new WeakHashMap<>();

默认,

  • 映射的容量将为 16
  • 负载系数将为 0.75

HashMapWeakHashMap之间的差异

让我们看看 Java 中弱哈希表的实现。

  1. import java.util.WeakHashMap;
  2. class Main {
  3. public static void main(String[] args) {
  4. // Creating WeakHashMap of numbers
  5. WeakHashMap<String, Integer> numbers = new WeakHashMap<>();
  6. String two = new String("Two");
  7. Integer twoValue = 2;
  8. String four = new String("Four");
  9. Integer fourValue = 4;
  10. // Inserting elements
  11. numbers.put(two, twoValue);
  12. numbers.put(four, fourValue);
  13. System.out.println("WeakHashMap: " + numbers);
  14. // Make the reference null
  15. two = null;
  16. // Perform garbage collection
  17. System.gc();
  18. System.out.println("WeakHashMap after garbage collection: " + numbers);
  19. }
  20. }

输出

  1. WeakHashMap: {Four=4, Two=2}
  2. WeakHashMap after garbage collection: {Four}

如我们所见,当弱哈希映射的键Two设置为null并执行垃圾回收时,该键将被删除。

这是因为与哈希映射不同,弱哈希映射的键是弱引用类型。 这意味着如果不再使用映射条目,则垃圾收集器将删除该条目。 这对于节省资源很有用。

现在让我们在哈希映射中查看相同的实现。

  1. import java.util.HashMap;
  2. class Main {
  3. public static void main(String[] args) {
  4. // Creating HashMap of even numbers
  5. HashMap<String, Integer> numbers = new HashMap<>();
  6. String two = new String("Two");
  7. Integer twoValue = 2;
  8. String four = new String("Four");
  9. Integer fourValue = 4;
  10. // Inserting elements
  11. numbers.put(two, twoValue);
  12. numbers.put(four, fourValue);
  13. System.out.println("HashMap: " + numbers);
  14. // Make the reference null
  15. two = null;
  16. // Perform garbage collection
  17. System.gc();
  18. System.out.println("HashMap after garbage collection: " + numbers);
  19. }
  20. }

输出

  1. HashMap: {Four=4, Two=2}
  2. HashMap after garbage collection: {Four=4, Two=2}

在此,当哈希映射的键Two被设置为null并执行垃圾收集时,该键不会被删除。

这是因为与弱哈希表不同,哈希表的键是强引用类型。 这意味着垃圾回收器不会删除映射的条目,即使不再使用该条目的键。

注意:哈希映射和弱哈希映射的所有功能都是相似的,只是弱哈希映射的键是弱引用,而哈希映射的键是强引用。


从其他映射创建WeakHashMap

这是我们可以从其他映射创建弱哈希表的方法。

  1. import java.util.HashMap;
  2. import java.util.WeakHashMap;
  3. class Main {
  4. public static void main(String[] args) {
  5. // Creating a hashmap of even numbers
  6. HashMap<String, Integer> evenNumbers = new HashMap<>();
  7. String two = new String("Two");
  8. Integer twoValue = 2;
  9. evenNumbers.put(two, twoValue);
  10. System.out.println("HashMap: " + evenNumbers);
  11. // Creating a weak hash map from other hashmap
  12. WeakHashMap<String, Integer> numbers = new WeakHashMap<>(evenNumbers);
  13. System.out.println("WeakHashMap: " + numbers);
  14. }
  15. }

输出

  1. HashMap: {Two=2}
  2. WeakHashMap: {Two=2}

WeakHashMap的方法

WeakHashMap类提供允许我们在映射上执行各种操作的方法。


将元素插入WeakHashMap

  • put() - 将指定的键/值映射插入到映射中
  • putAll() - 将指定映射中的所有条目插入此映射中
  • putIfAbsent() - 如果映射中不存在指定的键,则将指定的键/值映射插入到映射中

例如,

  1. import java.util.WeakHashMap;
  2. class Main {
  3. public static void main(String[] args) {
  4. // Creating WeakHashMap of even numbers
  5. WeakHashMap<String, Integer> evenNumbers = new WeakHashMap<>();
  6. String two = new String("Two");
  7. Integer twoValue = 2;
  8. // Using put()
  9. evenNumbers.put(two, twoValue);
  10. String four = new String("Four");
  11. Integer fourValue = 4;
  12. // Using putIfAbsent()
  13. evenNumbers.putIfAbsent(four, fourValue);
  14. System.out.println("WeakHashMap of even numbers: " + evenNumbers);
  15. //Creating WeakHashMap of numbers
  16. WeakHashMap<String, Integer> numbers = new WeakHashMap<>();
  17. String one = new String("One");
  18. Integer oneValue = 1;
  19. numbers.put(one, oneValue);
  20. // Using putAll()
  21. numbers.putAll(evenNumbers);
  22. System.out.println("WeakHashMap of numbers: " + numbers);
  23. }
  24. }

输出

  1. WeakHashMap of even numbers: {Four=4, Two=2}
  2. WeakHashMap of numbers: {Two=2, Four=4, One=1}

访问WeakHashMap元素

1.使用entrySet()keySet()values()

  • entrySet() - 返回一组所有键/值映射的映射
  • keySet() - 返回映射的所有键的集合
  • values() - 返回一组所有映射值

例如:

  1. import java.util.WeakHashMap;
  2. class Main {
  3. public static void main(String[] args) {
  4. // Creating WeakHashMap of even numbers
  5. WeakHashMap<String, Integer> numbers = new WeakHashMap<>();
  6. String one = new String("One");
  7. Integer oneValue = 1;
  8. numbers.put(one, oneValue);
  9. String two = new String("Two");
  10. Integer twoValue = 2;
  11. numbers.put(two, twoValue);
  12. System.out.println("WeakHashMap: " + numbers);
  13. // Using entrySet()
  14. System.out.println("Key/Value mappings: " + numbers.entrySet());
  15. // Using keySet()
  16. System.out.println("Keys: " + numbers.keySet());
  17. // Using values()
  18. System.out.println("Values: " + numbers.values());
  19. }
  20. }

输出

  1. WeakHashMap: {Two=2, One=1}
  2. Key/Value mappings: [Two=2, One=1]
  3. Keys: [Two, One]
  4. Values: [1, 2]

2.使用get()getOrDefault()

  • get() - 返回与指定键关联的值。 如果找不到键,则返回null
  • getOrDefault() - 返回与指定键关联的值。 如果找不到键,则返回指定的默认值。

例如:

  1. import java.util.WeakHashMap;
  2. class Main {
  3. public static void main(String[] args) {
  4. // Creating WeakHashMap of even numbers
  5. WeakHashMap<String, Integer> numbers = new WeakHashMap<>();
  6. String one = new String("One");
  7. Integer oneValue = 1;
  8. numbers.put(one, oneValue);
  9. String two = new String("Two");
  10. Integer twoValue = 2;
  11. numbers.put(two, twoValue);
  12. System.out.println("WeakHashMap: " + numbers);
  13. // Using get()
  14. int value1 = numbers.get("Two");
  15. System.out.println("Using get(): " + value1);
  16. // Using getOrDefault()
  17. int value2 = numbers.getOrDefault("Four", 4);
  18. System.out.println("Using getOrDefault(): " + value2);
  19. }
  20. }

输出

  1. WeakHashMap: {Two=2, One=1}
  2. Using get(): 2
  3. Using getOrDefault(): 4

删除WeakHashMap元素

  • remove(key) - 返回并从映射中删除与指定键关联的条目
  • remove(key, value) - 仅当指定键映射到指定值并返回布尔值时,才从映射中删除条目

例如:

  1. import java.util.WeakHashMap;
  2. class Main {
  3. public static void main(String[] args) {
  4. // Creating WeakHashMap of even numbers
  5. WeakHashMap<String, Integer> numbers = new WeakHashMap<>();
  6. String one = new String("One");
  7. Integer oneValue = 1;
  8. numbers.put(one, oneValue);
  9. String two = new String("Two");
  10. Integer twoValue = 2;
  11. numbers.put(two, twoValue);
  12. System.out.println("WeakHashMap: " + numbers);
  13. // Using remove() with single parameter
  14. int value = numbers.remove("Two");
  15. System.out.println("Removed value: " + value);
  16. // Using remove() with 2 parameters
  17. boolean result = numbers.remove("One", 3);
  18. System.out.println("Is the entry {One=3} removed? " + result);
  19. System.out.println("Updated WeakHashMap: " + numbers);
  20. }
  21. }

输出

  1. WeakHashMap: {Two=2, One=1}
  2. Removed value: 2
  3. Is the entry {One=3} removed? False
  4. Updated WeakHashMap: {One=1}

WeakHashMap的其他方法

方法 描述
clear() 从映射中删除所有条目
containsKey() 检查映射是否包含指定的键并返回布尔值
containsValue() 检查映射是否包含指定的值并返回布尔值
size() 返回映射的大小
isEmpty() 检查映射是否为空并返回布尔值

要了解更多信息,请访问 Java WeakHashMap(官方 Java 文档)