91名师指路-头部
91名师指路

hashSet 线程不安全的演示和解决方案

由于某些原因,现在不支持支付宝支付,如需要购买源码请加博主微信进行购买,微信号:13248254750

概述:hashset线程不安全,底层是hashMap。


(1)如何证明hashSet线程不安全。
(2)如何解决hashSet线程不安全。
(3)底层是hashMap,但是hashMap的add方法是添加k和v,为什么hashSet可以只添加k就可以了。
带着这三个疑问,我们下面来进行解答。

一:如何证明hashset线程不安全。

代码如下:

package com.mszl.thread;

import java.util.HashSet;
import java.util.Set;
import java.util.UUID;

/**
* 功能:证明HashSet线程不安全
* 备注:更多资料请访问 http://www.91mszl.com
* @author bobo teacher
*/
public class HashSetDemo {

public static void main(String[] args) {
Set<String> st=new HashSet<String>();

for(int i=1; i<=50; i++){
Thread t=new Thread(new Runnable() {
@Override
public void run() {
st.add(UUID.randomUUID().toString().substring(0, 8));
System.out.println(Thread.currentThread().getName() + "\t" + st);
}
});
t.start();
}
}


}


执行结果:

Exception in thread "Thread-20" java.util.ConcurrentModificationException

由此证明了hashSet 线程不安全。


二:如何解决hashSet线程不安全。

(1)使用辅助类包装一层。Collections.synchronizedSet(new HashSet<String>())

(2)使用CopyOnWriteArraySet


(1)使用辅助类包装一层,代码如下

package com.mszl.thread;

import java.util.Collections;
import java.util.HashSet;
import java.util.Set;
import java.util.UUID;

/**
* 功能:证明HashSet线程不安全和解决方案
* 备注:更多资料请访问 http://www.91mszl.com
* @author bobo teacher
*/
public class HashSetDemo {

public static void main(String[] args) {
// Set<String> st=new HashSet<String>();
Set<String> st=Collections.synchronizedSet(new HashSet<String>());

for(int i=1; i<=50; i++){
Thread t=new Thread(new Runnable() {
@Override
public void run() {
st.add(UUID.randomUUID().toString().substring(0, 8));
System.out.println(Thread.currentThread().getName() + "\t" + st);
}
});
t.start();
}
}


}


(2)使用CopyOnWriteArraySet,代码如下。

package com.mszl.thread;

import java.util.Collections;
import java.util.HashSet;
import java.util.Set;
import java.util.UUID;
import java.util.concurrent.CopyOnWriteArraySet;

/**
* 功能:证明HashSet线程不安全和解决方案
* 备注:更多资料请访问 http://www.91mszl.com
* @author bobo teacher
*/
public class HashSetDemo {

public static void main(String[] args) {
// Set<String> st=new HashSet<String>();
// Set<String> st=Collections.synchronizedSet(new HashSet<String>());
Set<String> st=new CopyOnWriteArraySet<String>();

for(int i=1; i<=50; i++){
Thread t=new Thread(new Runnable() {
@Override
public void run() {
st.add(UUID.randomUUID().toString().substring(0, 8));
System.out.println(Thread.currentThread().getName() + "\t" + st);
}
});
t.start();
}
}


}


三:hashSet 底层是hashMap,但是hashMap的add方法是添加k和v,为什么hashSet可以只添加k就可以了。

我们看hashSet的add方法源码,如下图所示。



可以知道value是一个Object PRESENT的固定值。所以我们不用去关心value是什么,只用管key。


2019-11-26 13:05:51     阅读(1178)

名师出品,必属精品    https://www.91mszl.com

联系博主    
用户登录遮罩层
x

账号登录

91名师指路-底部