java8多线程锁_多线程之8锁问题

Phone 有两个方法:发送邮件和发送短信,每个方法都打印一句话,现在通过不同的方式对方法进行操作,回答出打印的先后顺序(建议先自己看代码认真思考,然后再看答案,文章结尾会对每个问题进行分析)

问题

1、标准访问,两线程中间睡眠 2 毫秒,先打印邮件还是短信?

class Phone {

public synchronized void sendEmail() {

System.out.println("send email");

}

public synchronized void sendSms() {

System.out.println("send sms");

}

}

public class Lock01 {

public static void main(String[] args) throws InterruptedException {

Phone phone = new Phone();

new Thread(() -> phone.sendEmail(), "A").start();

Thread.sleep(200);

new Thread(() -> phone.sendSms(), "B").start();

}

}

查看答案

send email

send sms

2、在 sendEmail() 方法中睡眠 4 秒,先打印邮件还是短信?

class Phone {

public synchronized void sendEmail() {

try {

Thread.sleep(4000);

} catch (InterruptedException e) {

e.printStackTrace();

}

System.out.println("send email");

}

public synchronized void sendSms() {

System.out.println("send sms");

}

}

public class Lock02 {

public static void main(String[] args) throws InterruptedException {

Phone phone = new Phone();

new Thread(() -> phone.sendEmail(), "A").start();

Thread.sleep(200);

new Thread(() -> phone.sendSms(), "B").start();

}

}

查看答案

send email

send sms

3、添加普通的 hello() 方法,先打印邮件还是 hello?

class Phone {

public synchronized void sendEmail() {

try {

Thread.sleep(4000);

} catch (InterruptedException e) {

e.printStackTrace();

}

System.out.println("send email");

}

public void hello() {

System.out.println("hello");

}

}

public class Lock03 {

public static void main(String[] args) throws InterruptedException {

Phone phone = new Phone();

new Thread(() -> phone.sendEmail(), "A").start();

Thread.sleep(200);

new Thread(() -> phone.hello(), "B").start();

}

}

查看答案

hello

send email

4、2 个手机,先打印邮件还是短信?

class Phone {

public synchronized void sendEmail() {

try {

Thread.sleep(4000);

} catch (InterruptedException e) {

e.printStackTrace();

}

System.out.println("send email");

}

public synchronized void sendSms() {

System.out.println("send sms");

}

}

public class Lock04 {

public static void main(String[] args) throws InterruptedException {

Phone phone1 = new Phone();

Phone phone2 = new Phone();

new Thread(() -> phone1.sendEmail(), "A").start();

Thread.sleep(200);

new Thread(() -> phone2.sendSms(), "B").start();

}

}

查看答案

send sms

send email

5、2个静态同步方法,1部手机,先打印邮件还是短信?

class Phone {

public static synchronized void sendEmail() {

try {

Thread.sleep(4000);

} catch (InterruptedException e) {

e.printStackTrace();

}

System.out.println("send email");

}

public static synchronized void sendSms() {

System.out.println("send sms");

}

}

public class Lock05 {

public static void main(String[] args) throws InterruptedException {

Phone phone = new Phone();

new Thread(() -> phone.sendEmail(), "A").start();

Thread.sleep(200);

new Thread(() -> phone.sendSms(), "B").start();

}

}

查看答案

send email

send sms

**6、2个静态同步方法,2部手机,先打印邮件还是短信?**

class Phone {

public static synchronized void sendEmail() {

try {

Thread.sleep(4000);

} catch (InterruptedException e) {

e.printStackTrace();

}

System.out.println("send email");

}

public static synchronized void sendSms() {

System.out.println("send sms");

}

}

public class Lock06 {

public static void main(String[] args) throws InterruptedException {

Phone phone1 = new Phone();

Phone phone2 = new Phone();

new Thread(() -> phone1.sendEmail(), "A").start();

Thread.sleep(200);

new Thread(() -> phone2.sendSms(), "B").start();

}

}

查看答案

send email

send sms

**7、1个静态同步方法,1个普通同步方法,1部手机,先打印邮件还是短信?**

class Phone {

public static synchronized void sendEmail() {

try {

Thread.sleep(4000);

} catch (InterruptedException e) {

e.printStackTrace();

}

System.out.println("send email");

}

public synchronized void sendSms() {

System.out.println("send sms");

}

}

public class Lock07 {

public static void main(String[] args) throws InterruptedException {

Phone phone = new Phone();

new Thread(() -> phone.sendEmail(), "A").start();

Thread.sleep(200);

new Thread(() -> phone.sendSms(), "B").start();

}

}

查看答案

send sms

send email

8、1个静态同步方法,1个普通同步方法,2部手机,先打印邮件还是短信?

class Phone {

public static synchronized void sendEmail() {

try {

Thread.sleep(4000);

} catch (InterruptedException e) {

e.printStackTrace();

}

System.out.println("send email");

}

public synchronized void sendSms() {

System.out.println("send sms");

}

}

public class Lock08 {

public static void main(String[] args) throws InterruptedException {

Phone phone1 = new Phone();

Phone phone2 = new Phone();

new Thread(() -> phone1.sendEmail(), "A").start();

Thread.sleep(200);

new Thread(() -> phone2.sendSms(), "B").start();

}

}

查看答案

send sms

send email

分析

🎨 问题一

当一个对象里有多个同步(synchronized)方法,有一个线程访问了其中一个同步方法,其它线程只能等待其访问完成后才能访问,因为此时锁的是当前对象 this,其它的线程都不能进入到当前对象的其它的同步方法。

如果没有添加 Thread.sleep(200); 则打印的顺序是不一定的,因为线程的调度和操作系统有关。 添加 Thread.sleep(200); 则保证了线程 A 比 B 先执行。

🎨 问题二

由于线程 A 先执行,会先调用 sendEmail() 方法,Phone 实例就会被锁住,线程 B 只能等待 A 执行完在执行。

🎨 问题三

hello() 方法并不是同步方法,因此不受锁的影响。

🎨 问题四

现在有两个实例,前面我们说过,synchronized 锁的是 this,所以会产生两把锁,它们之间互不干扰,谁先执行完谁就先打印。

🎨 问题五、问题六

synchronized 实现同步的基础:Java 中的每一个对象都可以作为锁,具体表现为以下三种形式:

对于普通同步方法,锁的是当前实例对象 this

对于静态同步方法,锁的是当前类的 Class 对象

对于同步方法块,锁是Synchonized括号里配置的对象

所以,无论是 1 个对象还是 2 个对象,静态同步方法锁的都是 Class,只能按照线程执行的顺序打印。

🎨 问题七、问题八

这两种情况都是 1 个静态同步方法,1 个非静态同步方法,它们的锁都不是同一个对象,因此相互不受影响

总结

1、当一个线程试图访问同步代码块时,它首先必须得到锁,退出或抛出异常时必须释放锁。

2、Java 中的每一个对象都可以作为锁;普通同步方法锁 this,静态同步方法锁 Class,同步方法块锁括号;

3、只要锁的对象不是同一个,就直接按照线程执行的快慢来决定;锁的对象是同一个,就按照线程进入的先后顺序决定。

只要掌握了锁的对象是什么,无论是 8 锁还是 100 锁都不在话下!

b739ec46bb5c46d9c0aa4ce35ba1ea56.png

关于找一找教程网

本站文章仅代表作者观点,不代表本站立场,所有文章非营利性免费分享。

本站提供了软件编程、网站开发技术、服务器运维、人工智能等等IT技术文章,希望广大程序员努力学习,让我们用科技改变世界。

[多线程之8锁问题]http://www.zyiz.net/tech/detail-132234.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值