快速鸟瞰并发编程, 呕心沥血整理的架构技术【3】

做者:享学课堂James老师java

转载请声明出处!程序员

接着第1, 2篇后,咱们继续来跟进一下并发编程的其它内容,以下:编程

第9节 java.util.concurrent包

线程池

线程池的核心接口是 ExecutorServicejava.util.concurrent还提供了一个静态工厂类 Executors,其中包含用于建立配置线程池的工厂方法。安全

其实 静态工厂方法以下bash

注意:数据结构

调整线程池大小时,大小是根据你的计算机中的逻辑核心数而定的。这个大小能够经过调用Runtime.getRuntime().availableProcessors()方法得到该值。多线程

任务随着 ExecutorService#submitExecutorService#invokeAll或者提交, ExecutorService#invokeAny对于不一样类型的任务具备多个重载。并发

其实 功能接口以下ide

Future

Future是对于具体的Runnable任务或Callable任务的执行结果进行取消、查询是否完成、获取结果。必要时能够经过get方法获取执行结果,该方法会阻塞直到任务返回结果。高并发

ExecutorService使用 Future做为返回类型。

ExecutorService executorService = Executors.newSingleThreadExecutor();
Future<String> future = executorService.submit(() -> "结果");
try {
	String result = future.get(1L, TimeUnit.SECONDS);
	System.out.println("结果为 '" + result + "'.");
}
catch (InterruptedException e) {
	Thread.currentThread().interrupt();
	throw  new  RuntimeException(e);
}
catch (ExecutionException e) {
	throw  new  RuntimeException(e.getCause());
}
catch (TimeoutException e) {
	throw  new  RuntimeException(e);
}
assert future.isDone();
复制代码

ReentrantLock锁

java.util.concurrent.locks软件包括了常常使用到的 Lock接口。ReentrantLock类其实也实现了 synchronized关键字的功能,还提供了其它功能,例如获取有关锁的状态,非阻塞 tryLock()和可中断锁的信息。使用显式 ReentrantLock的示例以下:

class  JamesCounter {
	private  final  Lock lock = new  ReentrantLock();
	private  int value;
	int increment() {
		lock.lock();
		try {
			return ++value;
		}
		finally {
			lock.unlock();
		}
	}
}
复制代码

ReadWriteLock读写锁

java.util.concurrent.locks还包含一个 ReadWriteLock接口( ReentrantReadWriteLock实现),读写锁,一般容许多个并发读取,但只容许一个写入。

class  JamesStatistic {
	private  final  ReadWriteLock lock = new  ReentrantReadWriteLock();
	private  int value;
	void increment() {
		lock.writeLock().lock();
		try {
			value++;
		}
		finally {
			lock.writeLock().unlock();
		}
	}
	int current() {
		lock.readLock().lock();
		try {
			return value;
		}
		finally {
			lock.readLock().unlock();
		}
	}
}
复制代码

CountDownLatch工具

CountDownLatch主要用过计数,好比开项目大会,项目经理在会议室门口,有5个程序员A B C D E(至关于5个线程)分别来会议室开会,项目经理手写拿了一份会议人员名单,程序员A进入了会议室后,项目经理把A名单打个勾表示来了(至关于建立了线程A),B进会议室后,在名单上把B也打勾(至关于建立了线程B),但请注意,人没到齐, A,B程序员只能在座位上等待(线程全在等待阻塞中),还不能开会,等5个程序员都到齐了,才开会(5个线程同时被唤醒,开始工做)。

@SpringBootTest(classes = TripApplication.class)
@RunWith(SpringJUnit4ClassRunner.class)
public  class  JamesTestInvokeRemote {
	private  static  final  int THREADS = 200;//200线程模拟用户提交并发
	RestTemplate rest = new  RestTemplate();
	private  final  String url = "http://127.0.0.1:8090/buyTicket?idcard=123456";
	private  static  CountDownLatch cdl = new  CountDownLatch(THREADS);//200
	@Test
	public  void  TestInvoke() throws  InterruptedException {
		for (int i = 0; i < THREADS; i++){
			new  Thread(new  TicketRequest()).start();
			//模拟5个程序员陆陆续续进门
		}
	}
	public  class  JamesTicketRequest  implements  Runnable{
		@Override
		public  void run() {
			cdl.countDown();
			//项目经理的名单上勾掉一个,其实就是减1
			try {
				cdl.await();
				//全部程序员末到位前,都在椅子上等待(全部线程等待),直到                                         //cdl.countDown()减为0时唤醒
			}
			catch (InterruptedException e) {
				e.printStackTrace();
			}
			//5个线程同时(并发请求)执行业务逻辑`
			String str = rest.getForEntity(url, String.class).getBody();
			//并发同时请求
			System.out.println(str);
		}
	}
}
复制代码

并发知识汇总

集合线程安全的最简单方法就是使用 Collections#synchronized锁定方法。因为此解决方案在高并发场景下表现不佳,所以 java.util.concurrent提供了针对并发使用进行了优化的各类数据结构。

List集合

Maps集合

Set集合

Queue队列

队列实际上是充当了“生产者”和“消费者”之间的管道。换句话来讲就是个“先进先出”(FIFO)顺序而已。BlockingQueue接口扩展 Queue,提供永久阻塞或按指定的时间段进行阻塞的方法,它的等待条件会因另外一个线程的操做而发生改变。

花了几天时间总算写完了,但愿对你们有帮助。看完这篇,你们至少要对多线程的一些经常使用工具类要有所了解。

关注我,还有更多技术干货分享~

相关文章
相关标签/搜索