Semaphore -信号量、实现限流

信号量模型

一个计数器、一个等待队列、三个方法。计数器和等待队列对外是透明的
三个方法:
Init(); 初始化计数器
down(); 计数器的值-1;如果此时计数器的值小于0,则当前线程将被阻塞,否则当前线程继续执行; – 在Semphore中是acquire()方法
up(); 计数器的值+1;如果此时计数器的值大于或等于0,则唤醒等待队列中的一个线程,并将其从等待队列中移除; – 在Semphore中是release()方法
在 Java SDK 里面,信号量模型是由 java.util.concurrent.Semaphore 实现的,Semaphore 这个类能够保证这三个方法都是原子操作。
在这里插入图片描述

简单使用信号量

static int count;
//初始化信号量
static final Semaphore s = new Semaphore(1);
//用信号量保证互斥    
static void addOne() {
  s.acquire();
  try {
    count+=1;
  } finally {
    s.release();
  }
}

上面简单实现的信号量
定义多个线程同时去调用addOne方法,该方法首先调用了Semaphore.acquire方法,因为Semaphore初始化设置了1,表示同时只允许一个线程可以通过,当线程1和线程2同时去调用addOne方法时,因为acquire方法时原子的,同时只有一个线程可以通过,假设线程1通过了,那么线程1可以继续执行,线程2则进入了Semaphore的等待队列中被阻塞。等到线程1 执行完成后会调用Semaphore.release方法,还原了Semaphore中的计数器,同时由于其计数器》=0所以,会随机释放等待队列中的线程2 ,线程就可以继续执行了。
这种方式也是一个简单的互斥锁的功能
Semaphore 可以允许多个线程访问一个临界区

限流器简单实现

/**
 * 简单限流器
 *
 * @author lhb
 * @date 2019-12-5
 */
public class SampleLimiting<T, R> {

    final List<T> pool;
    /**
     * 用信号量实现限流器
     */
    final Semaphore sem;
    // 构造函数
    SampleLimiting(int size, T t) {
        pool = new Vector<T>() {
        };
        for (int i = 0; i < size; i++) {
            pool.add(t);
        }
        sem = new Semaphore(size);
    }

    void exec2() {
        T t = null;
        try {
            sem.acquire();
            t = pool.remove(0);
            TestSemaphore.run((TestSemaphore) t);
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            pool.add(t);
            sem.release();
        }
    }

这里生成了一个限流器,同时只允许5个线程进入,
初始化的时候会往对象池中放入5个指定的对象,每次调用exce都会从对象池中获取需要的对象,并且通过Semaphore实现限流方式,等处理结束后把对象放回对象池中,同时调用Semaphore释放等待线程

public static void main(String[] args) throws InterruptedException {
    SampleLimiting<TestSemaphore, String> stringSampleLimiting = new SampleLimiting<>(5,
        new TestSemaphore());

    List<Thread> threads = new ArrayList<>(100);

    for (int i = 0; i < 100; i++) {
        Thread thread = new Thread(new Runnable() {
            @Override
            public void run() {
                stringSampleLimiting.exec2();
            }
        });
        threads.add(thread);
    }
    for (Thread thread : threads) {
        thread.start();
    }
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
下面是一个使用Semaphore实现两线程同步的C++代码: ```c++ #include <iostream> #include <thread> #include <mutex> #include <condition_variable> #include <semaphore.h> using namespace std; sem_t sem; void thread1() { cout << "Thread 1 is running." << endl; sem_post(&sem); // 发送信号量 } void thread2() { sem_wait(&sem); // 等待信号量 cout << "Thread 2 is running." << endl; } int main() { sem_init(&sem, 0, 0); // 初始化信号量 thread t1(thread1); thread t2(thread2); t1.join(); t2.join(); sem_destroy(&sem); // 销毁信号量 return 0; } ``` 这个例子中,我们使用了`sem_t`和`sem_init()`、`sem_wait()`、`sem_post()`、`sem_destroy()`四个函数来实现Semaphore。`sem_t`是一个结构体类型,用于存储Semaphore的信息。`sem_init()`函数用于初始化Semaphore,第一个参数是Semaphore结构体指针,第二个参数是0表示这个Semaphore是用于线程间同步的,第三个参数是Semaphore的初值,这里设为0。`sem_wait()`函数用于等待Semaphore的信号,如果Semaphore的值为0,则该函数阻塞线程,直到Semaphore的值大于0;如果Semaphore的值大于0,则该函数将Semaphore的值减1,然后线程继续执行。`sem_post()`函数用于发送Semaphore的信号,将Semaphore的值加1,表示Semaphore可以被下一个等待的线程使用。`sem_destroy()`函数用于销毁Semaphore。 在这个例子中,我们创建了两个线程,线程1通过调用`sem_post()`函数发送Semaphore的信号,线程2通过调用`sem_wait()`函数等待Semaphore的信号。由于Semaphore的初值为0,线程2会在等待Semaphore的信号时被阻塞,直到线程1发送了Semaphore的信号。这样就实现了两个线程的同步。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值