多线程之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 emailsend 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 emailsend 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();
}
}
查看答案
hellosend 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 smssend 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 emailsend 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 emailsend 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 smssend 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 smssend 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 锁都不在话下!