直接上代码,直接复制可用。java多线程安全
package com.a;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class ThreadTest {
public static void main(String[] args) {
ThreadTest t = new ThreadTest();
List<Student> list = t.batchGetObject();
System.out.println(list.size()+"==最后返回的对象数量");
}
//说明:循环的次数要跟计数器的一样,如果循环的次数大于计数器,返回的结果会有偏差,如果计数器中的值
//大于了循环的值,这个时候就会出现阻塞,因为执行完了任务,计数器还没有释放,主线程已经开发跑了。
// Executors创建线程有几种方式,这里只是简单的一种,关于线程。建议是在启动服务器的时候就初始化好线程池的数量,不要每次都关
// 到String窗口关闭的时候再去销毁线程,这样对系统更好。线程可以重复的调用。一般在系统中要看执行任务的数量,不建议开太多线程,
// 这样太消耗性能,一般计算一下,一秒能执行多少个线程,你执行的循环多少次,大概开多少个线程可以跑完。然后其他的代码有没有会用到这个线程池
// 一般都是写一个线程工具类,整个块使用。
public List<Student> batchGetObject(){
List<Student> list = new ArrayList<>();
ExecutorService service = Executors.newCachedThreadPool();
final CountDownLatch cdOrder = new CountDownLatch(1);
final CountDownLatch cdAnswer = new CountDownLatch(3);
for (int i = 0; i < 3; i++) {
Runnable runnable = new Runnable() {
public void run() {
try {
cdOrder.await();
Student student = createObject();//要执行的任务
list.add(student);
cdAnswer.countDown();
} catch (Exception e) {
e.printStackTrace();
}
}
};
service.execute(runnable);// 为线程池添加任务
}
try {
cdOrder.countDown();
cdAnswer.await();
} catch (Exception e) {
e.printStackTrace();
}
service.shutdown(); // 任务结束,停止线程池的所有线程
return list;
}
public Student createObject() {
Student stu = new Student();
System.out.println("开始创建对象");
return stu;
}
}