Executors创建多线程demo

直接上代码,直接复制可用。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 list = t.batchGetObject();
		System.out.println(list.size()+"==最后返回的对象数量");
	}
	//说明:循环的次数要跟计数器的一样,如果循环的次数大于计数器,返回的结果会有偏差,如果计数器中的值
	//大于了循环的值,这个时候就会出现阻塞,因为执行完了任务,计数器还没有释放,主线程已经开发跑了。
	// Executors创建线程有几种方式,这里只是简单的一种,关于线程。建议是在启动服务器的时候就初始化好线程池的数量,不要每次都关
	// 到String窗口关闭的时候再去销毁线程,这样对系统更好。线程可以重复的调用。一般在系统中要看执行任务的数量,不建议开太多线程,
	// 这样太消耗性能,一般计算一下,一秒能执行多少个线程,你执行的循环多少次,大概开多少个线程可以跑完。然后其他的代码有没有会用到这个线程池
	// 一般都是写一个线程工具类,整个块使用。
	public List batchGetObject(){
		List 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;
	}
}

你可能感兴趣的:(Executors创建多线程demo)