同步线程包装器是为“原集合对象”的实际操作找一个代理对象,代理在“原集合对象”的一切功能之上又增加了同步功能(只是对这个“代理对象”上的操作同步,“原集合对象”上的操作非同步)。
java同步线程包装器:
public static Collection synchronizedCollection(Collection c);
public static Set synchronizedSet(Set s);
public static List synchronizedList(List list);
public static Map synchronizedMap(Map m);
public static SortedSet synchronizedSortedSet(SortedSet s);
public static SortedMap synchronizedSortedMap(SortedMap m);
java的同步线程包装器是有条件的同步,只有对集合的原子粒度的操作才同步。对于有并发情况的迭代操作,因为迭代操作是通过对对像集的调用间接操作原对像,所以在迭代时要对迭代的对像实现再同步:
Collection c = Collections.synchronizedCollection(myCollection);
...
synchronized(c) {
Iterator i = c.iterator(); // Must be in the synchronized block
while (i.hasNext())
foo(i.next());
}
线程安全的集合包含2个问题
1.多线程并发修改一 个 集合 怎么办?
2.如果迭代的过程中 集合 被修改了怎么办?
a.一个线程在迭代,另一个线程在修改
b.在同一个线程内用同一个迭代器对象进行迭代、修改、迭代、修改. . .
共有有3种解决方案
1.用老的Vector/Hashtable类,上面2个问题都不用担心。
Vector/Hashtable所提供的所有方法都是synchronized的。vector的线程安全是指它的get,add等方法,是通过synchronized修饰的,可以线程安全的修改和读取单个数据。 但vector 的 iterator 方法返回的迭代器是非线性安全的:如果在迭代器创建后的任意时间从结构上修改了vector,则迭代器将抛出concurrentmodificationexception;迭代器的线程安全性与vector的线程安全性无关。 但这种方法效率低下,不建议使用。
2.使用ArrayList/HashMap和同步包装器
可用 同步包装器使容器变成线程安全的
Java代码
List synchArrayList = Collections.synchronizedList(new ArrayList());
Map synchHashMap = Collections.synchronizedMap(new HashMap())
如果要迭代,出于iterator的非线程安全性,需要这样
Java代码
synchronized (synchHashMap){
Iterator iter = synchHashMap.keySet().iterator();
while (iter.hasNext()) . . .;
}
collections.synchronizedlist(new arraylist<student>())返回的包装类,跟vector情况类似,get,add等方法都是线程安全的,但iterator,返回的其实是被包装对象的iterator,因为被包装的arraylist的iterator是快速失败的,所以这种情况也是会抛concurrentmodificationexception异常的。
3.用java5.0新加入的ConcurrentLinkedQueue、ConcurrentHashMap、CopyOnWriteArrayList 和 CopyOnWriteArraySet
对这些集合进行并发修改是安全的。concurrentlinkedqueue的iterator不会抛concurrentmodificationexception,并保证能遍历构造这个iterator时,队列里的元素,但不保证能反映后续的修改。
举例入下:
设置非常简单的一个场景,有一个student集合,对该集合采用迭代器进行遍历之后,又对其添加了一个元素。由于添加元素是改变集合结构的操作,所以集合如果在迭代器构造之后发生改变,就会抛出concurrentmodificationexception异常。
student类代码,很简单:
[code=java]
class student{
public student(string name, int age){
this.name = name;
this.age = age;
}
string name;
int age;
public string tostring(){
return "i am "+name+" , "+age+" years old.";
}
}
1. 创建一个修改collection的线程,实现runnable接口。不采取任何同步措施。
[code=java]
class modifycollectiontask implements runnable{
public modifycollectiontask(collection<student> slist){
this.slist = slist;
}
public void run(){
// 遍历学生列表,
for(student s : slist){
system.out.println(thread.currentthread().getname());
system.out.println(s);
}
// 向学生列表添加元素
slist.add(new student("katie", 30));
}
collection<student> slist;
}
在main函数里启动100个线程,
[code=java]
public class synccollection {
public static void main(string[] args) {
collection<student> slist = new arraylist<student>();
slist.add(new student("aaa",10));
slist.add(new student("bbb",12));
slist.add(new student("ccc",14));
slist.add(new student("ddd",16));
slist.add(new student("eee",18));
for(int i=0;i<100;i++){
new thread(new modifycollectiontask(slist)).start();
}
}
很明显,没有同步控制,那么很快就抛出了concurrentmodificationexception异常
2. 由于vector类是线程安全的动态数组,所以,将集合实现改为vector,在线程run方法中没做任何修改
[code=java]
// 使用vector
list<student> svector = new vector<student>();
svector.add(new student("aaa",10));
svector.add(new student("bbb",12));
svector.add(new student("ccc",14));
svector.add(new student("ddd",16));
svector.add(new student("eee",18));
for(int i=0;i<100;i++){
new thread(new modifycollectiontask(svector)).start();
}
结果还是发生了concurrentmodificationexception异常。这让我有点怀疑vector的同步机制。
3. 使用collections工具类中的同步包装方法,将线程不安全arraylist进行包装,而线程实现方法没有改动
[code=java]
// 使用collections工具类中的同步包装器
list<student> slist2 = collections.synchronizedlist(new arraylist<student>());
slist2.add(new student("aaa",10));
slist2.add(new student("bbb",12));
slist2.add(new student("ccc",14));
slist2.add(new student("ddd",16));
slist2.add(new student("eee",18));
for(int i=0;i<100;i++){
new thread(new modifycollectiontask(slist2)).start();
}
结果还是发生了异常,不明白。。。
4. 下面使用synchronized关键字进行同步控制,对线程实现代码进行了修改
[code=java]
class syncmodifylisttask implements runnable{
public syncmodifylisttask(collection<student> slist){
this.slist = slist;
}
public void run(){
synchronized(slist){
// 遍历学生列表
for(student s : slist){
system.out.println(thread.currentthread().getname());
system.out.println(s);
}
// 向学生列表添加元素
slist.add(new student("katie", 30));
}
}
collection<student> slist;
}
由于有了synchronized关键字对代码片段进行了保护,所以没有出现异常
5. 使用java.util.concurrent包中的高效的同步集合, concurrentlinkedqueue,线程实现代码还是用modifycollectiontask
[code=java]
collection<student> concurrentcollection = new concurrentlinkedqueue<student>();
concurrentcollection.add(new student("aaa",10));
concurrentcollection.add(new student("bbb",12));
concurrentcollection.add(new student("ccc",14));
concurrentcollection.add(new student("ddd",16));
concurrentcollection.add(new student("eee",18));
for(int i=0;i<100;i++){
new thread(new modifycollectiontask(concurrentcollection)).start();
}
结果也没有出现异常,证明高效的同步集合还是很给力的!
参考页面:http://www.haowen.it/question/23060
感谢作者!
分享到:
相关推荐
总结来说,理解Java中的线程安全与不安全集合是非常重要的,这有助于在设计并发程序时选择合适的数据结构和操作方式,以确保程序的正确性和性能。在面对线程安全问题时,我们可以借助如`synchronized`关键字、`...
本项目"JAVA多线程与线程安全实践-基于Http协议的断点续传"探讨了如何在Java中实现多线程以及如何确保线程安全,特别是在处理HTTP协议的断点续传功能时。 断点续传是文件传输的一种优化技术,允许用户在不重新下载...
Java集合框架中有一些线程安全的类,如Vector、HashTable、ConcurrentHashMap等,它们内部实现了同步机制,可以在多线程环境中直接使用,避免了手动同步的复杂性。 十、线程局部变量 ThreadLocal为每个线程都创建了...
二、Java同步 1. **synchronized关键字** - 修饰实例方法:锁定的是对象实例,即整个实例的所有线程都只有一个执行权。 - 修饰静态方法:锁定的是类的Class对象,即所有类实例共享同一把锁。 - 修饰代码块:指定...
6. 同步适配器模式:通过包装已有的线程不安全的类,使其变得线程安全。Java中的Collections.synchronizedXXX()方法和ConcurrentHashMap等就是此类模式的应用。 源代码部分将帮助你更直观地理解这些设计模式的实现...
Java并发工具类 (`java.util.concurrent` 包) 提供了高级线程管理工具,如`ExecutorService`用于管理和控制线程池,`CountDownLatch`用于同步多个线程,`CyclicBarrier`让一组线程等待其他线程到达屏障点,以及`...
以上只是Java多线程面试中可能会遇到的一部分问题,实际的59题集合可能涵盖更多细节和深度,包括线程安全的类、线程通信的高级技术、并发设计模式等。掌握这些知识点,不仅有助于应对面试,更能提高在实际开发中处理...
线程安全的实现通常依赖于同步机制。 10. **并发集合** - **ConcurrentHashMap, CopyOnWriteArrayList, ConcurrentLinkedQueue** 等集合类提供了线程安全的并发操作。 Java线程是Java多线程编程的核心,理解和...
- **线程创建**:Java通过`Thread`类或实现`Runnable`接口来创建线程,理解`start()`方法启动线程与`run()`方法的区别。 - **并发执行**:线程并发执行,提高系统效率,了解`Thread.sleep()`、`yield()`、`join()`...
### Java多线程机制详解 #### 一、Java多线程概述 在计算机科学中,**多线程**是指从软件或硬件上实现多个线程并发执行的技术。它能够提高程序的性能,尤其是在多核处理器的环境下。Java作为一种广泛使用的编程...
线程从新建到运行,可能经历 `start()` 调用、调度器的选择以及可能的同步等待。线程状态的转换受到 `sleep()`, `join()`, `yield()` 等方法的影响。 - **sleep()**:使当前线程进入睡眠状态,指定一段时间后自动...
Java的并发集合如`ConcurrentHashMap`、`ConcurrentLinkedQueue`等提供了线程安全的插入和读取操作,避免了同步问题。 8. **异常处理**:在多线程环境下,异常处理是必须考虑的问题。`Future.get()`方法会抛出一个`...
在实际项目中,我们还需要关注线程安全问题,比如同步机制(synchronized关键字、Lock接口)、线程间通信(wait、notify、Condition)以及并发工具类(ConcurrentHashMap、CountDownLatch、CyclicBarrier、Semaphore...
在Java中,可以通过`synchronized`关键字来控制线程同步,避免数据竞争,确保数据的一致性。 6. **登录机制**:在描述中提到的登录功能,可能涉及到用户身份验证和权限控制。这通常需要存储用户信息,例如用户名和...
Java多线程与锁是Java并发编程中的核心概念,它们对于构建高效、可扩展的并发应用程序至关重要。在Java中,多线程允许程序同时执行多个任务,提高CPU的利用率,而锁则是用来控制多线程间共享资源的访问,确保数据的...
- **实现Callable接口与FutureTask**:Callable接口提供了有返回值的线程,FutureTask用于包装Callable对象并可以获取其计算结果。 2. **线程的状态与生命周期** - **新建(New)**:线程被创建但尚未启动。 - *...
由于long的基本操作(如按位或、按位与、按位异或)在Java中是原子的,因此在单线程环境下,我们可以直接对BitSet进行位操作而不用担心数据不一致的问题。但是,当多个线程同时修改BitSet时,由于这些操作不是原子的...
- `Collections.synchronizedXXX`方法:对普通集合进行同步包装,使其变为线程安全。 - `ConcurrentHashMap`: 并发环境下高效的哈希表。 - `CopyOnWriteArrayList`和`CopyOnWriteArraySet`: 写时复制策略,保证读...
4. **线程安全与同步机制**: - **线程安全问题**:当多个线程访问同一资源时可能会出现数据不一致的问题。 - **同步机制**:通过`synchronized`关键字、`ReentrantLock`等来控制对共享资源的访问,确保线程安全。...
1. **Collections.synchronizedMap()**:可以使用`Collections.synchronizedMap()`静态方法将`HashMap`包装成线程安全的`SynchronizedMap`。但请注意,尽管这个方法可以确保并发修改的安全,但仍然无法避免迭代时的...