[关闭]
@boothsun 2017-07-12T01:53:21.000000Z 字数 1697 阅读 1679

线程同步工具 Semaphore类的基础使用

Java多线程


推荐好文:
1. 线程同步工具(一)
2. 线程同步工具(二)控制并发访问多个资源
3. 并发工具类(三)控制并发线程数的Semaphore

简介

Semaphore是基于计数的信号量,可以用来控制同时访问特定资源的线程数量;可以通过设定一个阈值,基于此,多个线程争抢获取许可信号,做完自己的申请后归还,超过阈值后,线程申请许可信号将会被阻塞,直到有其他线程释放许可信号。

Semaphore可以用来构建一些线程池,资源池之类的,比如数据库连接池,我们也可以创建计数为1的Semaphore,将其作为一种类似互斥锁的机制,这也叫二元信号量,表示两种互斥状态。

很多年以来, 我都觉得从字面上很难理解Semaphore所表达的含义,只能把它比作是控制流量的红绿灯,比如XX马路要限制流量,只允许同时有一百辆在这条路上行驶,其他的都必须在路口等待,所以前一百辆会看到绿灯,可以开进这条马路,后面的车会看到红灯,不能驶入XXX马路,但是如果前一百辆中有五辆车已经离开了XX马路,那么后面就允许有5辆车驶入马路,这个例子里说的车就是线程,驶入马路就表示线程在执行,离开马路就表示线程执行完成,看见红灯就表示线程被阻塞,不能执行。

个人理解: Semaphores就是一个控制访问共享资源的计数器。

常用API

在线手册地址

  1. 1. 构造方法:
  2. Semaphore(int permits) 用给定的允许数量和默认的非公平设置创建Semaphore对象。
  3. Semaphore(int permits , boolean fair) 用给定的允许数量和给定的公平设置创建一个Semaphore对象。
  4. 2. 常用方法
  5. 1) void acquire()
  6. 从信号量里获取一个可用的许可,如果没有可用的许可,那么当前线程将被禁用以进行线程调度,并且处于休眠状态。
  7. 2) void release()
  8. 释放一个许可,将其返回给信号量
  9. 3) int availablePermits()
  10. 返回此信号量中当前可用的许可数量。
  11. 4) boolean hasQueuedThreads()
  12. 查询是否有线程正在等待获取。

使用案例

  1. import java.util.concurrent.Semaphore;
  2. public class SemaphoreTest {
  3. public static void main(String[] args) {
  4. Service service = new Service() ;
  5. ThreadRun a = new ThreadRun(service) ;
  6. a.setName("A");
  7. ThreadRun b = new ThreadRun(service);
  8. b.setName("b");
  9. ThreadRun c = new ThreadRun(service) ;
  10. c.setName("C");
  11. a.start();
  12. b.start();
  13. c.start();
  14. }
  15. }
  16. class Service {
  17. private Semaphore semaphore = new Semaphore(2);
  18. public void testMethod() {
  19. try {
  20. semaphore.acquire();
  21. System.out.println(Thread.currentThread().getName() + " begin timer=" + System.currentTimeMillis());
  22. Thread.sleep(5000);
  23. System.out.println(Thread.currentThread().getName() + " end timer=" + System.currentTimeMillis());
  24. semaphore.release();
  25. }catch (Exception e) {
  26. e.printStackTrace();
  27. }
  28. }
  29. }
  30. class ThreadRun extends Thread {
  31. Service service ;
  32. public ThreadRun (Service service) {
  33. this.service = service ;
  34. }
  35. @Override
  36. public void run() {
  37. service.testMethod();
  38. }
  39. }
添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注