解析Java编程之Synchronized锁住的对象
图片上传 密码修改为 synchronized是java中用于同步的关键字,一般我们通过Synchronized锁住一个对象,来进行线程同步。我们需要了解在程序执行过程中,synchronized锁住的到底是哪个对象,否则我们在多线程的程序就有可能出现问题。
看下面的代码,我们定义了一个静态变量n,在run方法中,我们使n增加10,然后在main方法中,我们开辟了100个线程,来执行n增加的操作,如果线程没有并发执行,那么n最后的值应该为1000,显然下面的程序执行完结果不是1000,因为我们没有进行线程同步。
import java.util.concurrent.TimeUnit; public class SynchronizedTest1 extends Thread { public static int n = 0; public void run() { try { //使n自加10次 for (int i = 0; i < 10; i++) { n = n + 1; TimeUnit.MILLISECONDS.sleep(10); } } catch (InterruptedException e) { e.printStackTrace(); } } public static void main(String[] args) throws InterruptedException { Thread[] threads = new Thread[100]; for (int i = 0; i < threads.length; i++) { threads[i] = new SynchronizedTest1(); threads[i].start(); } //使所有其他线程执行完,再继续执行main线程,这样得出的n是最终的结果 for (Thread thread : threads) { thread.join(); } System.out.println(n); } }
为了实现同步,我们修改上面的代码,增加一个increase方法,如下。但是当我们执行下面的代码时,会发现n仍然不是1000.
import java.util.concurrent.TimeUnit; public class SynchronizedTest2 extends Thread { public static int n = 0; public synchronized void increase() { n++; } public void run() { try { //使n自加10次 for (int i = 0; i < 10; i++) { increase(); TimeUnit.MILLISECONDS.sleep(10); } } catch (InterruptedException e) { e.printStackTrace(); } } public static void main(String[] args) throws InterruptedException { Thread[] threads = new Thread[100]; for (int i = 0; i < threads.length; i++) { threads[i] = new SynchronizedTest2(); threads[i].start(); } //使所有其他线程执行完,再继续执行main线程,这样得出的n是最终的结果 for (Thread thread : threads) { thread.join(); } System.out.println(n); } }
其实原因很简单,上面的多个线程在执行时根本就没有竞争同一个对象锁。当我们执行用synchronized修饰的非静态方法时,线程会首先获得调用这个方法的对象的锁,然后才能继续执行代码。那么调用这个方法的到底是哪个对象,是this对象。在上面的例子中,thread[i]所代表的线程获取的锁对象是thread[i]对象,也就是该线程对象本身。因此上面所开辟的100个线程只要获得自身对象就可以执行,这样就使同步失去了作用。
我们再次修改代码:即将increase方法改为i静态的,此时程序执行完后n的值为1000。
import java.util.concurrent.TimeUnit; public class SynchronizedTest3 extends Thread { public static int n = 0; public synchronized static void increase() { n++; } public void run() { try { //使n自加10次 for (int i = 0; i < 10; i++) { increase(); TimeUnit.MILLISECONDS.sleep(10); } } catch (InterruptedException e) { e.printStackTrace(); } } public static void main(String[] args) throws InterruptedException { Thread[] threads = new Thread[100]; for (int i = 0; i < threads.length; i++) { threads[i] = new SynchronizedTest3(); threads[i].start(); } //使所有其他线程执行完,再继续执行main线程,这样得出的n是最终的结果 for (Thread thread : threads) { thread.join(); } System.out.println(n); } }
当synchronized 修饰static方法,它锁住的是该类的Class对象,而不是某一个具体对象。在上面的例子中,它锁住的就是SynchronizedTest3.class对象。在程序执行过程中,类的Class对象只有一份,所以上面线程竞争的是同一个对象锁。
下面是对synchronized锁住对象的总结:
(1)对于同步方法,锁当前对象(this)
(2)对于静态同步方法,锁当前类的Class对象
(3)对于同步代码块,锁住的是synchronized括号中的对象
总结
以上就是本文关于解析Java编程之Synchronized锁住的对象的全部内容,希望对大家有所帮助。感兴趣的朋友可以继续参阅本站:Java编程redisson实现分布式锁代码示例、Java并发编程之重入锁与读写锁等,有什么问题可以直接留言,小编会及时回复大家的。下面推荐本站基本Java编程相关的书籍,免费下载,供朋友们学习参考。
Java初级开发工程师面试题汇总.PDF
//www.jb51.net/books/576989.html
Java经典实例(第三版) 完整版 ([美]达尔文) 中文pdf扫描版
//www.jb51.net/books/577859.html
希望大家能够喜欢。
栏 目:Java编程
下一篇:Java编程获取文件列表及子文件目录的方法(非递归)
本文标题:解析Java编程之Synchronized锁住的对象
本文地址:https://www.xiuzhanwang.com/a1/Javabiancheng/8443.html
您可能感兴趣的文章
- 01-10Java咖啡馆(1)——叹咖啡
- 01-10Java Socket编程(三) 服务器Sockets
- 01-10Java进阶:Struts多模块的技巧
- 01-10Java Socket编程(一) Socket传输模式
- 01-10Java Socket编程(二) Java面向连接的类
- 01-10Java运行时多态性的实现
- 01-10Java经验点滴:处理没有被捕获的异常
- 01-10Java Socket编程(四) 重复和并发服务器
- 01-10Java中的浮点数分析
- 01-10面向对象编程:Java中的抽象数据类型
阅读排行
本栏相关
- 01-10Java咖啡馆(1)——叹咖啡
- 01-10JVM的垃圾回收机制详解和调优
- 01-10Java Socket编程(三) 服务器Sockets
- 01-10Java进阶:Struts多模块的技巧
- 01-10J2SE 1.5版本的新特性一览
- 01-10Java Socket编程(一) Socket传输模式
- 01-10Java运行时多态性的实现
- 01-10Java Socket编程(二) Java面向连接的类
- 01-10Java Socket编程(四) 重复和并发服务
- 01-10Java经验点滴:处理没有被捕获的异常
随机阅读
- 01-10C#中split用法实例总结
- 08-05DEDE织梦data目录下的sessions文件夹有什
- 04-02jquery与jsp,用jquery
- 01-10使用C语言求解扑克牌的顺子及n个骰子
- 08-05织梦dedecms什么时候用栏目交叉功能?
- 01-11Mac OSX 打开原生自带读写NTFS功能(图文
- 01-10delphi制作wav文件的方法
- 08-05dedecms(织梦)副栏目数量限制代码修改
- 01-10SublimeText编译C开发环境设置
- 01-11ajax实现页面的局部加载