Java线程:线程安全类和Callable与Future(有返回值的线程)

简介: 一、线程安全类   当一个类已经很好的同步以保护它的数据时,这个类就称为线程安全的。当一个集合是安全的,有两个线程在操作同一个集合对象,当第一个线程查询集合非空后,删除集合中所有元素的时候,第二个线程也来执行与第一个线程相同的操作,也许第一个线程查询后,第二个也查出非空,但是此时明显是不对的。

一、线程安全类

  当一个类已经很好的同步以保护它的数据时,这个类就称为线程安全的。当一个集合是安全的,有两个线程在操作同一个集合对象,当第一个线程查询集合非空后,删除集合中所有元素的时候,第二个线程也来执行与第一个线程相同的操作,也许第一个线程查询后,第二个也查出非空,但是此时明显是不对的。如:

 1 public class NameList { 
 2     private List nameList = Collections.synchronizedList(new LinkedList()); 
 3 
 4     public void add(String name) { 
 5         nameList.add(name); 
 6     } 
 7 
 8     public String removeFirst() { 
 9         if (nameList.size() > 0) { 
10             return (String) nameList.remove(0); 
11         } else { 
12             return null; 
13         } 
14     } 
15 }
 1 public class Test { 
 2     public static void main(String[] args) { 
 3         final NameList nl = new NameList(); 
 4         nl.add("aaa"); 
 5         class NameDropper extends Thread{ 
 6             public void run(){ 
 7                 String name = nl.removeFirst(); 
 8                 System.out.println(name); 
 9             } 
10         } 
11 
12         Thread t1 = new NameDropper(); 
13         Thread t2 = new NameDropper(); 
14         t1.start(); 
15         t2.start(); 
16     } 
17 }

  虽然集合对象private List nameList=Collections.synchronizedList(new LinkedList())是同步的,但是程序并不是线程安全的。

  原因是:一个线程操作列表的过程无法阻止另一个线程对列表的其他操作。解决办法是:在操作集合对象的NameList上面做一个同步。改写后的代码为:

 1 public class NameList { 
 2     private List nameList = Collections.synchronizedList(new LinkedList()); 
 3 
 4     public synchronized void add(String name) { 
 5         nameList.add(name); 
 6     } 
 7 
 8     public synchronized String removeFirst() { 
 9         if (nameList.size() > 0) { 
10             return (String) nameList.remove(0); 
11         } else { 
12             return null; 
13         } 
14     } 
15 }

  此时,一个线程访问其中一个方法时,其他线程等待。

二、Callable与Future

  通过实现Callable接口实现有返回值的任务,与Runnable接口处理无返回值的任务类似。

执行了Callable任务后,可以获得一个Future对象,在该对象上调用get就可以获得Callable任务返回的Object了。如:

 1 package Thread;
 2 
 3 import java.util.concurrent.Callable;
 4 import java.util.concurrent.ExecutionException;
 5 import java.util.concurrent.ExecutorService;
 6 import java.util.concurrent.Executors;
 7 import java.util.concurrent.Future;
 8 
 9 public class CallableTest {
10     public static void main(String[] args)throws ExecutionException,InterruptedException{
11         ExecutorService pool=Executors.newFixedThreadPool(2);
12         Callable c1=new MyCallable("A");
13         Callable c2=new MyCallable("B");
14         Future f1=pool.submit(c1);
15         Future f2=pool.submit(c2);
16         System.out.println(">>>"+f1.get().toString());
17         System.out.println(">>>"+f2.get().toString());
18         pool.shutdown();
19     }
20 }
21 class MyCallable implements Callable{
22     private String oid;
23     MyCallable(String oid){
24         this.oid=oid;
25     }
26     public Object call()throws Exception{
27         return oid+"任务返回的内容";
28     }
29     
30 }
View Code
1 >>>A任务返回的内容
2 >>>B任务返回的内容

 

当神已无能为力,那便是魔渡众生
目录
相关文章
|
5月前
|
并行计算 Java 大数据
Callable和Future
Callable和Future
|
21天前
|
Java
【JavaEE】——多线程常用类
Callable的call方法,FutureTask类,ReentrantLock可重入锁和对比,Semaphore信号量(PV操作)CountDownLatch锁存器,
|
21天前
|
Java 程序员 调度
【JavaEE】线程创建和终止,Thread类方法,变量捕获(7000字长文)
创建线程的五种方式,Thread常见方法(守护进程.setDaemon() ,isAlive),start和run方法的区别,如何提前终止一个线程,标志位,isinterrupted,变量捕获
|
21天前
|
安全 Java API
【JavaEE】多线程编程引入——认识Thread类
Thread类,Thread中的run方法,在编程中怎么调度多线程
|
2月前
|
安全 Java
Java多线程集合类
本文介绍了Java中线程安全的问题及解决方案。通过示例代码展示了使用`CopyOnWriteArrayList`、`CopyOnWriteArraySet`和`ConcurrentHashMap`来解决多线程环境下集合操作的线程安全问题。这些类通过不同的机制确保了线程安全,提高了并发性能。
|
3月前
lua面向对象(类)和lua协同线程与协同函数、Lua文件I/O
Lua的面向对象编程、协同线程与协同函数的概念和使用,以及Lua文件I/O操作的基本方法。
46 4
lua面向对象(类)和lua协同线程与协同函数、Lua文件I/O
|
3月前
|
Java 开发者
在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口
【10月更文挑战第20天】在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口。本文揭示了这两种方式的微妙差异和潜在陷阱,帮助你更好地理解和选择适合项目需求的线程创建方式。
47 3
|
3月前
|
Java
在Java多线程编程中,实现Runnable接口通常优于继承Thread类
【10月更文挑战第20天】在Java多线程编程中,实现Runnable接口通常优于继承Thread类。原因包括:1) Java只支持单继承,实现接口不受此限制;2) Runnable接口便于代码复用和线程池管理;3) 分离任务与线程,提高灵活性。因此,实现Runnable接口是更佳选择。
74 2
|
3月前
|
Java
Java中多线程编程的基本概念和创建线程的两种主要方式:继承Thread类和实现Runnable接口
【10月更文挑战第20天】《JAVA多线程深度解析:线程的创建之路》介绍了Java中多线程编程的基本概念和创建线程的两种主要方式:继承Thread类和实现Runnable接口。文章详细讲解了每种方式的实现方法、优缺点及适用场景,帮助读者更好地理解和掌握多线程编程技术,为复杂任务的高效处理奠定基础。
50 2
|
3月前
|
Java 开发者
Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点
【10月更文挑战第20天】Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点,重点解析为何实现Runnable接口更具灵活性、资源共享及易于管理的优势。
56 1

热门文章

最新文章