从头认识java-18.5 临界区

Posted 李灵晖

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了从头认识java-18.5 临界区相关的知识,希望对你有一定的参考价值。

这一章节我们来讨论一下临界区。

一般来说,我们使用多线程都是直接在方法上面加上synchronized,但是其实这样有些时候对于性能来说,有所欠缺,因此今天来讨论一下临界区的问题。

1.一般做法的例子

class ThreadA implements Runnable {
	private synchronized void test() throws InterruptedException {
		System.out.println("dosomething");
		Thread.sleep(5000);
		System.out.println("dosomething");
	}

	@Override
	public void run() {
		while (true) {
			try {
				test();
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}
	}
}

上面的代码是我们一般的作法,当然,我将上面的时间延长了一些,但是有没有想过,其实在第一个dosomething的地方,我们不需要线程安全(例如,大部分的时候都是查询数据库的功能,不涉及修改),而第二个dosomething的地方才需要线程安全,那么,由于整个方法都要求线程安全,在方法执行的过程中,其他线程不能执行里面的其他方法,如果这个方法需要执行很久,(例如上面的Thread.sleep(5000);),这个时候性能就会出现问题,因此,我们引入临界区这个概念。


2.什么是临界区?

导致竞态条件发生的代码区称作临界区。

也就是上面的第二个dosomething,我们只需要在他这里做线程安全即可


3.怎么使得临界区线程安全?

(1)使用synchronized,因为synchronized不单可以用在方法上面,还可以用在代码块、类上面

class ThreadA implements Runnable {
	private void test() throws InterruptedException {
		System.out.println("dosomething");
		Thread.sleep(5000);
		synchronized (this) {//线程同步的地方
			System.out.println("dosomething");
		}
	}

	@Override
	public void run() {
		while (true) {
			try {
				test();
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}
	}
}

(2)使用ReentrantLock

class ThreadA implements Runnable {

	private ReentrantLock reentrantLock = new ReentrantLock();

	private void test() throws InterruptedException {
		System.out.println("dosomething");
		Thread.sleep(5000);
		reentrantLock.lock();
		try {
			System.out.println("dosomething");
		} finally {
			reentrantLock.unlock();
		}
	}

	@Override
	public void run() {
		while (true) {
			try {
				test();
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}
	}
}

总结:这一章节主要介绍临界区的使用。


这一章节就到这里,谢谢。

-----------------------------------

目录


以上是关于从头认识java-18.5 临界区的主要内容,如果未能解决你的问题,请参考以下文章

同步互斥的实现

临界数据临界区和原子操作

互斥锁自旋锁读写锁和条件变量

OS学习笔记四:同步机制

FreeRTOS临界区

Java并发编程实战—–synchronized