一、情景一
1、代码
class Phone {
public synchronized void sendSMS() throws Exception {
System.out.println("------sendSMS");
}
public synchronized void sendEmail() throws Exception {
System.out.println("------sendEmail");
}
}
public class Lock_8 {
public static void main(String[] args) throws InterruptedException {
Phone phone = new Phone();
new Thread(() -> {
try {
phone.sendSMS();
} catch (Exception e) {
e.printStackTrace();
}
}, "AA").start();
Thread.sleep(100);
new Thread(() -> {
try {
phone.sendEmail();
} catch (Exception e) {
e.printStackTrace();
}
}, "BB").start();
}
}
2、问题:标准访问,先打印短信还是邮件?
3、运行结果
1 2 | ------sendSMS ------sendEmail |
4、分析
synchronized 修饰普通方法,锁住的是当前对象即 this(调用的 phone对象),执行到第一个 synchronized 方法时,会把 this 上锁,也就不能调用 this 类中其他的 synchronized 方法了,只能阻塞等待,等到第一个 synchronized 方法执行完毕,其他线程才能继续执行。
二、情景二
1、代码
class Phone {
public synchronized void sendSMS() throws Exception {
//停留4秒
TimeUnit.SECONDS.sleep(4);
System.out.println("------sendSMS");
}
public synchronized void sendEmail() throws Exception {
System.out.println("------sendEmail");
}
}
public class Lock_8 {
public static void main(String[] args) throws InterruptedException {
Phone phone = new Phone();
new Thread(() -> {
try {
phone.sendSMS();
} catch (Exception e) {
e.printStackTrace();
}
}, "AA").start();
Thread.sleep(100);
new Thread(() -> {
try {
phone.sendEmail();
} catch (Exception e) {
e.printStackTrace();
}
}, "BB").start();
}
}
2、问题:停4秒在短信方法内,先打印短信还是邮件?
3、运行结果
1 2 | ------sendSMS ------sendEmail |
4、分析
synchronized 修饰普通方法,锁住的是当前对象即 this(调用的 phone对象),执行到第一个 synchronized 方法时,会把 this 上锁,也就不能调用 this 类中其他的 synchronized 方法了,只能阻塞等待,等到第一个 synchronized 方法执行完毕,在这里让执行 sendSMS 方法休眠4秒,其他线程只能处于等待情况,只有该线程执行完毕,其他线程才能继续执行。
第一、二场景:
线程 A、B不能同时进入到资源类中,只能有一个线程进入到资源类中。
总结:一个对象里面如果有多个 synchronized 方法,某一个时刻内,只要有一个线程去调用其中的一个 synchronized 方法了,其他的线程都只能等待。
换句话说,某一个时刻内,只能有唯一一个线程去访问这些 synchronized 方法锁的当前对象是 this,被锁定后,其他的线程都不能进入到当前对象的其他 synchronized方法中锁的对象是当前的实例对象 —— this。
三、情景三
1、代码
class Phone {
public synchronized void sendSMS() throws Exception {
//停留4秒
TimeUnit.SECONDS.sleep(4);
System.out.println("------sendSMS");
}
public synchronized void sendEmail() throws Exception {
System.out.println("------sendEmail");
}
public void getHello() {
System.out.println("------getHello");
}
}
public class Lock_8 {
public static void main(String[] args) throws InterruptedException {
Phone phone = new Phone();
new Thread(() -> {
try {
phone.sendSMS();
} catch (Exception e) {
e.printStackTrace();
}
}, "AA").start();
Thread.sleep(100);
new Thread(() -> {
try {
phone.getHello();
} catch (Exception e) {
e.printStackTrace();
}
}, "BB").start();
}
}
2、问题:新增普通的hello方法,是先打短信还是hello?
3、运行结果
1 2 | ------getHello ------sendSMS |
4、分析
新增了普通方法,没有加 synchronized锁,其他线程执行时,不用获取锁,所以普通方法先执行。
四、情景四
1、代码
class Phone {
public synchronized void sendSMS() throws Exception {
//停留4秒
TimeUnit.SECONDS.sleep(4);
System.out.println("------sendSMS");
}
public synchronized void sendEmail() throws Exception {
System.out.println("------sendEmail");
}
public void getHello() {
System.out.println("------getHello");
}
}
public class Lock_8 {
public static void main(String[] args) throws InterruptedException {
Phone phone = new Phone();
Phone phone2 = new Phone();
new Thread(() -> {
try {
phone.sendSMS();
} catch (Exception e) {
e.printStackTrace();
}
}, "AA").start();
Thread.sleep(100);
new Thread(() -> {
try {
phone2.sendEmail();
} catch (Exception e) {
e.printStackTrace();
}
}, "BB").start();
}
}
2、问题:现在有两部手机,先打印短信还是邮件?
3、运行结果
1 2 | ------sendEmail ------sendSMS |
4、分析
synchronized 修改普通方法,锁住的是当前对象(this),也就是调用者 phone 和 phone2 对象,第一个线程的锁对象是 phone,第二个线程的锁对象是 phone2,这两个线程锁的对象不一样(不是同一把锁),所以互不影响,各自执行。
五、情景五
1、代码
class Phone {
public static synchronized void sendSMS() throws Exception {
//停留4秒
TimeUnit.SECONDS.sleep(4);
System.out.println("------sendSMS");
}
public static synchronized void sendEmail() throws Exception {
System.out.println("------sendEmail");
}
public void getHello() {
System.out.println("------getHello");
}
}
public class Lock_8 {
public static void main(String[] args) throws InterruptedException {
Phone phone = new Phone();
new Thread(() -> {
try {
phone.sendSMS();
} catch (Exception e) {
e.printStackTrace();
}
}, "AA").start();
Thread.sleep(100);
new Thread(() -> {
try {
phone.sendEmail();
} catch (Exception e) {
e.printStackTrace();
}
}, "BB").start();
}
}
2、问题:两个静态同步方法,1部手机,先打印短信还是邮件?
3、运行结果
1 2 | ------sendSMS ------sendEmail |
4、分析
synchronized 修饰静态方法,锁对象是当前类的字节码文件(Phone.class),这两个线程用的是同一把锁,即 Phone.class,所以第一个线程获取到锁执行时,其他线程只能阻塞等待。
六、情景六
1、代码
class Phone {
public static synchronized void sendSMS() throws Exception {
//停留4秒
TimeUnit.SECONDS.sleep(4);
System.out.println("------sendSMS");
}
public static synchronized void sendEmail() throws Exception {
System.out.println("------sendEmail");
}
public void getHello() {
System.out.println("------getHello");
}
}
public class Lock_8 {
public static void main(String[] args) throws InterruptedException {
Phone phone = new Phone();
Phone phone2 = new Phone();
new Thread(() -> {
try {
phone.sendSMS();
} catch (Exception e) {
e.printStackTrace();
}
}, "AA").start();
Thread.sleep(100);
new Thread(() -> {
try {
phone2.sendEmail();
} catch (Exception e) {
e.printStackTrace();
}
}, "BB").start();
}
}
2、问题:两个静态同步方法,2部手机,先打印短信还是邮件?
3、运行结果
1 2 | ------sendSMS ------sendEmail |
4、分析
synchronized 修饰静态方法,锁对象是当前类的字节码文件(Phone.class),所以这两个线程都是用的同一把锁,即 Phone.class,所以第一个线程获取到锁执行时,其他线程只能阻塞等待。
七、情景七
1、代码
class Phone {
public static synchronized void sendSMS() throws Exception {
//停留4秒
TimeUnit.SECONDS.sleep(4);
System.out.println("------sendSMS");
}
public synchronized void sendEmail() throws Exception {
System.out.println("------sendEmail");
}
public void getHello() {
System.out.println("------getHello");
}
}
public class Lock_8 {
public static void main(String[] args) throws InterruptedException {
Phone phone = new Phone();
new Thread(() -> {
try {
phone.sendSMS();
} catch (Exception e) {
e.printStackTrace();
}
}, "AA").start();
Thread.sleep(100);
new Thread(() -> {
try {
phone.sendEmail();
} catch (Exception e) {
e.printStackTrace();
}
}, "BB").start();
}
}
2、问题:1个静态同步方法,1个普通同步方法,1部手机,先打印短信还是邮件?
3、运行结果
1 2 | ------sendEmail ------sendSMS |
4、分析
synchronized 修改静态方法,锁对手是字节码Class对象,即 Phone 的 Class 对象。
synchronized 修改普通方法,锁对象是当前对象(this),即调用者 phone 对象。
第一个线程使用的锁是 Phone.class 对象,第二个线程使用的锁是 phone 对象,这不是同一把锁,锁的范围也不一样,所以两个线程互不影响。
八、情景八
1、代码
class Phone {
public static synchronized void sendSMS() throws Exception {
//停留4秒
TimeUnit.SECONDS.sleep(4);
System.out.println("------sendSMS");
}
public synchronized void sendEmail() throws Exception {
System.out.println("------sendEmail");
}
public void getHello() {
System.out.println("------getHello");
}
}
public class Lock_8 {
public static void main(String[] args) throws InterruptedException {
Phone phone = new Phone();
Phone phone2 = new Phone();
new Thread(() -> {
try {
phone.sendSMS();
} catch (Exception e) {
e.printStackTrace();
}
}, "AA").start();
Thread.sleep(100);
new Thread(() -> {
try {
phone2.sendEmail();
} catch (Exception e) {
e.printStackTrace();
}
}, "BB").start();
}
}
2、问题:1个静态同步方法,1个普通同步方法,2部手机,先打印短信还是邮件?
3、运行结果
1 2 | ------sendEmail ------sendSMS |
4、分析
synchronized 修改静态方法,锁对象是字节码Class对象,即 Phone 的 Class 对象。
synchronized 修改普通方法,锁对象是当前对象(this),即调用者 phone2 对象。
第一个线程使用的锁是 Phone.class 对象,第二个线程使用的锁是 phone2 对象,这不是同一把锁,锁的范围也不一样,所以两个线程互不影响。
九、总结
1、一个对象里面如果有多个 synchronized 方法,某一个时刻内,只要一个线程去调用其中的一个 synchronized 方法了,其它的线程都只能等待。
换句话说,某一个时刻内,只能有唯一一个线程去访问这些synchronized 方法。
锁的是当前对象 this,被锁定后,其它的线程都不能进入到当前对象的其它的synchronized 方法。
2、加个普通方法后发现和同步锁无关。
3、换成两个对象后,不是同一把锁了,情况立刻变化。锁的是各自的实例对象。
4、synchronized 实现同步的基础: Java 中的每一个对象都可以作为锁。具体表现为以下 3 种形式:
1 2 3 | 对于普通同步方法,锁是当前实例对象。 对于静态同步方法,锁是当前类的 Class 对象。 对于同步方法块,锁是 Synchonized 括号里配置的对象 |
6、当一个线程试图访问同步代码块时,它首先必须得到锁,退出或抛出异常时必须释放锁。
也就是说如果一个实例对象的非静态同步方法获取锁后,该实例对象的其他非静态同步方法必须等待获取锁的方法释放锁后才能获取锁,
可是别的实例对象的非静态同步方法因为跟该实例对象的非静态同步方法用的是不同的锁,
所以毋须等待该实例对象已获取锁的非静态同步方法释放锁就可以获取他们自己的锁。
7、所有的静态同步方法用的也是同一把锁——类对象本身,这两把锁是两个不同的对象。
这两把锁(this/Class)是两个不同的对象,所以静态同步方法与非静态同步方法之间是不会有竞态条件的。
但是一旦一个静态同步方法获取锁后,其他的静态同步方法都必须等待该方法释放锁后才能获取锁,而不管是同一个实例对象的静态同步方法之间,还是不同的实例对象的静态同
步方法之间,只要它们同一个类的实例对象!
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】博客园社区专享云产品让利特惠,阿里云新客6.5折上折
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 微软正式发布.NET 10 Preview 1:开启下一代开发框架新篇章
· C# 集成 DeepSeek 模型实现 AI 私有化(本地部署与 API 调用教程)
· DeepSeek R1 简明指南:架构、训练、本地部署及硬件要求
· 没有源码,如何修改代码逻辑?
· NetPad:一个.NET开源、跨平台的C#编辑器