不同企业的网络营销网站,社群营销与运营,网站无后台可以上框架,参考网是正规网站吗一、什么是synchronized 在Java当中synchronized通常是用来标记一个方法或者代码块。在Java当中被synchronized标记的代码或者方法在同一个时刻只能够有一个线程执行被synchronized修饰的方法或者代码块。因此被synchronized修饰的方法或者代码块不会出现数据竞争的情况#x…一、什么是synchronized 在Java当中synchronized通常是用来标记一个方法或者代码块。在Java当中被synchronized标记的代码或者方法在同一个时刻只能够有一个线程执行被synchronized修饰的方法或者代码块。因此被synchronized修饰的方法或者代码块不会出现数据竞争的情况也就是说被synchronized修饰的代码块是并发安全的。synchronized是java内置关键字,是内置锁,JVM中内置了,颗粒度比较大 二、synchronized的四种用法
2.1、修饰一个代码块
被修饰的代码块称为同步语句块其作用的范围是大括号{} 括起来的代码作用的对象是调用这个代码块的对象;
2.2、修饰一个方法 被修饰的方法称为同步方法其作用的范围是整个方法作用的对象是调用这个方法的对象;
非静态方法使用 synchronized 修饰的写法修饰实例方法时锁定的是当前实例对象
2.3、修饰一个静态的方法
其作用的范围是整个静态方法作用的对象是这个类的所有对象;
2.4、修饰一个类
其作用的范围是synchronized后面括号括起来的部分作用对象是这个类的所有对象。
三、使用案例分析
3.1、修饰一个方法
class SyncDemo {private int count;public void add() {count;}public static void main(String[] args) throws InterruptedException {SyncDemo syncDemo new SyncDemo();Thread t1 new Thread(()-{for (int i 0; i 1000 ; i) {syncDemo.add();}});Thread t2 new Thread(()-{for (int i 0; i 1000 ; i) {syncDemo.add();}});t1.start();t2.start();t1.join(); //线程的阻塞方法,线程t1执行完毕,再执行main主线程打印语句t2.join(); //线程的阻塞方法,线程t2执行完毕,再执行main主线程打印语句System.out.println(syncDemo.count);}
} 由于add方法没有使用synchronized修饰线程t1和线程t2可能同时去执行add方法那么就会导致最终count的结果小于20000因为count操作不具备原子性。 将上述add方法被synchronized修饰 public synchronized void add() {count;} 由于add方法被synchronized修饰,因此每一个时刻只能有一个线程执行add方法因此上面打印的结果是20000 总结: synchronized修饰的add方法一个时刻只能有一个线程执行的意思是对于一个SyncDemo类的对象来说一个时刻只能有一个线程进入。比如现在有两个SyncDemo的对象s1和s2一个时刻只能有一个线程进行s1的add方法一个时刻只能有一个线程进入s2的add方法但是同一个时刻可以有两个不同的线程执行s1和s2的add方法也就说s1的add方法和s2的add是没有关系的一个线程进入s1的add方法并不会阻止另外的线程进入s2的add方法也就是说synchronized在修饰一个非静态方法的时候“锁”住的只是一个实例对象并不会“锁”住其它的对象。其实这也很容易理解一个实例对象是一个独立的个体别的对象不会影响他他也不会影响别的对象。 3.2、修饰一个静态的方法
class SyncDemo {private static int count; //静态变量public static synchronized void add() { //静态方法count;}public static void main(String[] args) throws InterruptedException {SyncDemo syncDemo new SyncDemo();Thread t1 new Thread(()-{for (int i 0; i 1000 ; i) {syncDemo.add();}});Thread t2 new Thread(()-{for (int i 0; i 1000 ; i) {syncDemo.add();}});t1.start();t2.start();t1.join(); //线程的阻塞方法,线程t1执行完毕,再执行main主线程打印语句t2.join(); //线程的阻塞方法,线程t2执行完毕,再执行main主线程打印语句System.out.println(syncDemo.count); //输出结果为2000}
} 上面的代码最终输出的结果也是20000但是与前一个程序不同的是。这里的add方法用static修饰的在这种情况下真正只能有一个线程进入到add方法因为用static修饰的add方法是静态方法,静态方法所有对象公共的方法因此和前面的那种情况不同不存在两个不同的线程同一时刻执行不同实例对象的add方法。 你仔细想想如果能够让两个不同的线程执行add代码块那么count的执行就不是原子的了。那为什么没有用static修饰的代码为什么可以呢因为当没有用static修饰时每一个对象的count都是不同的内存地址不一样因此在这种情况下count这个操作仍然是原子的 3.3、修饰一个代码块
class SyncDemo {private int count; //静态变量public void add() {System.out.println(进入了add方法);synchronized (this){count;}}public void minus() {System.out.println(进入了minus方法);synchronized (this){count--;}}public static void main(String[] args) throws InterruptedException {SyncDemo syncDemo new SyncDemo();Thread t1 new Thread(()-{for (int i 0; i 10 ; i) {syncDemo.add();}});Thread t2 new Thread(()-{for (int i 0; i 10 ; i) {syncDemo.minus();}});t1.start();t2.start();t1.join(); //线程的阻塞方法,线程t1执行完毕,再执行main主线程打印语句t2.join(); //线程的阻塞方法,线程t2执行完毕,再执行main主线程打印语句System.out.println(syncDemo.count); //输出结果为0}
} 有时候我们并不需要用synchronized去修饰一个方法因为这样并发度就比较低了一个方法一个时刻只能有一个线程在执行。因此我们可以选择用synchronized去修饰代码块只让某个代码块一个时刻只能有一个线程执行除了这个代码块之外的代码还是可以并行的。 比如上面的代码当中add和minus方法没有使用synchronized进行修饰因此一个时刻可以有多个线程执行这个两个方法。在上面的synchronized代码块当中我们使用了this对象作为锁对象只有拿到这个锁对象的线程才能够进入代码块执行而在同一个时刻只能有一个线程能够获得锁对象。也就是说add函数和minus函数用synchronized修饰的两个代码块同一个时刻只能有一个代码块的代码能够被一个线程执行因此上面的结果同样是0。 这里说的锁对象是this也就SyncDemo类的一个实例对象因为它锁住的是一个实例对象因此当实例对象不一样的时候他们之间是没有关系的也就是说不同实例用synchronized修饰的代码块是没有关系的他们之间是可以并发的。 3.4、修饰一个类
class SyncDemo {private int count; //静态变量public void add() {System.out.println(进入了add方法);synchronized (SyncDemo.class){count;}}public void minus() {System.out.println(进入了minus方法);synchronized (SyncDemo.class){count--;}}public static void main(String[] args) throws InterruptedException {SyncDemo syncDemo new SyncDemo();Thread t1 new Thread(()-{for (int i 0; i 10 ; i) {syncDemo.add();}});Thread t2 new Thread(()-{for (int i 0; i 10 ; i) {syncDemo.minus();}});t1.start();t2.start();t1.join(); //线程的阻塞方法,线程t1执行完毕,再执行main主线程打印语句t2.join(); //线程的阻塞方法,线程t2执行完毕,再执行main主线程打印语句System.out.println(syncDemo.count); //输出结果为0}
} 上面的代码是使用synchronized修饰类锁对象是SyncDemo.class,这个时候他不再是锁住一个对象了而是一个类了这个时候的并发度就变小了上一份代码当锁对象是SyncDemo的实例对象时并发度更大一些因为当锁对象是实例对象的时候只有实例对象内部是不能够并发的实例之间是可以并发的。但是当锁对象是SyncDemo.class的时候实例对象之间时不能够并发的因为这个时候的锁对象是一个类。 四、Synchronized与可见性和重排序
4.1互斥性/排他性(非公平锁)
synchronized非公平锁会起到互斥效果某个线程执⾏到某个对象的 synchronized 中时其他线程如果也执⾏到同⼀个对象 synchronized 就会阻塞等待。
进⼊ synchronized 修饰的代码块, 相当于加锁。退出 synchronized 修饰的代码块, 相当于解锁。 PS公平锁 VS 非公平锁 公平锁按资排辈先到先得。需要“唤醒”这一步操作会牺牲一定的性能。(线程发现锁占用尝试获取锁一段时间后进入休眠状态进入排队队列中等待。上一个线程释放锁后会唤醒排队等候的其他线程中最前面的线程从阻塞状态又切换至运行状态 非公平锁来得早不如来得巧不需要“唤醒”性能高。线程1发现锁占用尝试获取锁一段时间后进入休眠状态。此时线程2来了处于运行状态尝试获取锁此时刚好上一个线程释放了锁那么线程2直接得到了锁并去运行它的任务了。排队等待的其他线程获取锁的顺序不是按照访问的顺序先来先到的而是由线程自己竞争随机获取到锁Java里所有的锁默认是非公平锁。 4.2可见性 当一个线程进入到synchronized同步代码块的时候将会刷新所有对该线程的可见的变量也就是说如果其他线程修改了某个变量而且线程需要在Synchronized代码块当中使用那就会重新刷新这个变量到内存当中保证这个变量对于执行同步代码块的线程是可见的。 当一个线程从同步代码块退出的时候也会将线程的工作内存同步到内存当中保证在同步代码块当中修改的变量对其他线程可见。
4.3可重入性(可重入性锁) synchronized 同步块对同⼀条线程来说是可重⼊的不会出现⾃⼰把⾃⼰锁死的问题。 /*** synchronized 可重入性测试*/
public class ThreadSynchronized4 {public static void main(String[] args) {synchronized (ThreadSynchronized4.class) {System.out.println(当前主线程已经得到了锁); //当执行到此行代码时表示已经获得锁synchronized (ThreadSynchronized4.class) { //同一个线程获取锁两次System.out.println(当前主线程再次得到了锁); //若两行代码都能打印说明具备可重入性}}}
} 五、synchronized实现原理 面试必问synchronized是如何实现的 ①在Java代码层面 synchronized加锁的对象里有一个的隐藏的对象头这个对象头可看作一个类里有很多属性其中比较关注的两个属性是是否加锁的标识和拥有当前锁的线程id。 每次进⼊ synchronized 修饰的代码块时会去对象头中先判断加锁的标识再判断拥有当前锁的线程id从而决定当前线程能否往下继续执行。 判断加锁标识为false-对象头未加锁当前线程可以进入synchronized 修饰的代码块并设置加锁标识为true设置拥有当前锁的线程id为此线程id。 判断加锁标识为true-对象头已加锁需进一步判断拥有当前锁的线程id是否为此线程id若是则继续往下执行否则不能往下执行需要等待锁资源释放后重新竞争再获取锁。 ②在JVM层面和操作系统层面 synchronized同步锁是通过JVM内置的Monitor监视器实现的而监视器又是依赖操作系统的互斥锁Mutex实现的。↓ ———————————————— 原文链接https://blog.csdn.net/WWXDwrn/article/details/129115774 六、synchronized历史发展进程
在JDK1.6之前多使用Locksynchronized使用很少那时synchronized默认使用重量级锁实现所以性能较差。在JDK1.6时synchronized做了优化。锁升级流程如下 1,无锁没有线程访问时默认是无锁状态加了synchronized也是无锁状态。有线程访问时才加锁。更大程度上减少锁带来的程序上的开销。 2,偏向锁当有一个线程访问时会升级为偏向锁。(在对象头里存了这样一把锁后面再来线程时会判断如果线程id和拥有锁的线程id相同会让它进去只偏向某一个线程其他线程来了之后要继续等) 3,轻量级锁当有多个线程访问时会升级为轻量级锁。 4,重量级锁当大量线程访问同时竞争锁资源的时候会升级为重量级锁。 原文链接https://baijiahao.baidu.com/s?id1740505389877266267wfrspiderforpc 文章转载自: http://www.morning.wtbzt.cn.gov.cn.wtbzt.cn http://www.morning.rdfq.cn.gov.cn.rdfq.cn http://www.morning.daidudu.com.gov.cn.daidudu.com http://www.morning.prkdl.cn.gov.cn.prkdl.cn http://www.morning.lxmks.cn.gov.cn.lxmks.cn http://www.morning.nsfxt.cn.gov.cn.nsfxt.cn http://www.morning.jcffp.cn.gov.cn.jcffp.cn http://www.morning.dpppx.cn.gov.cn.dpppx.cn http://www.morning.yrycb.cn.gov.cn.yrycb.cn http://www.morning.tdnbw.cn.gov.cn.tdnbw.cn http://www.morning.jmdpp.cn.gov.cn.jmdpp.cn http://www.morning.lphtm.cn.gov.cn.lphtm.cn http://www.morning.yrddl.cn.gov.cn.yrddl.cn http://www.morning.mxftp.com.gov.cn.mxftp.com http://www.morning.rfldz.cn.gov.cn.rfldz.cn http://www.morning.sbrrf.cn.gov.cn.sbrrf.cn http://www.morning.lxhrq.cn.gov.cn.lxhrq.cn http://www.morning.zmwzg.cn.gov.cn.zmwzg.cn http://www.morning.rltw.cn.gov.cn.rltw.cn http://www.morning.hytqt.cn.gov.cn.hytqt.cn http://www.morning.mpgfk.cn.gov.cn.mpgfk.cn http://www.morning.gsksm.cn.gov.cn.gsksm.cn http://www.morning.wjfzp.cn.gov.cn.wjfzp.cn http://www.morning.qnlbb.cn.gov.cn.qnlbb.cn http://www.morning.qmzwl.cn.gov.cn.qmzwl.cn http://www.morning.qhjkz.cn.gov.cn.qhjkz.cn http://www.morning.bztzm.cn.gov.cn.bztzm.cn http://www.morning.cxnyg.cn.gov.cn.cxnyg.cn http://www.morning.xnkb.cn.gov.cn.xnkb.cn http://www.morning.pjyrl.cn.gov.cn.pjyrl.cn http://www.morning.syxmx.cn.gov.cn.syxmx.cn http://www.morning.rdzlh.cn.gov.cn.rdzlh.cn http://www.morning.yesidu.com.gov.cn.yesidu.com http://www.morning.skql.cn.gov.cn.skql.cn http://www.morning.jpbpc.cn.gov.cn.jpbpc.cn http://www.morning.bgqqr.cn.gov.cn.bgqqr.cn http://www.morning.qhtlq.cn.gov.cn.qhtlq.cn http://www.morning.bpmfg.cn.gov.cn.bpmfg.cn http://www.morning.xnlj.cn.gov.cn.xnlj.cn http://www.morning.cdlewan.com.gov.cn.cdlewan.com http://www.morning.tsdjj.cn.gov.cn.tsdjj.cn http://www.morning.tkryt.cn.gov.cn.tkryt.cn http://www.morning.tyklz.cn.gov.cn.tyklz.cn http://www.morning.yjfmj.cn.gov.cn.yjfmj.cn http://www.morning.ygth.cn.gov.cn.ygth.cn http://www.morning.jpkk.cn.gov.cn.jpkk.cn http://www.morning.lfdzr.cn.gov.cn.lfdzr.cn http://www.morning.nxrgl.cn.gov.cn.nxrgl.cn http://www.morning.lsmnn.cn.gov.cn.lsmnn.cn http://www.morning.zbnts.cn.gov.cn.zbnts.cn http://www.morning.lwnwl.cn.gov.cn.lwnwl.cn http://www.morning.lpzyq.cn.gov.cn.lpzyq.cn http://www.morning.a3e2r.com.gov.cn.a3e2r.com http://www.morning.dsxgc.cn.gov.cn.dsxgc.cn http://www.morning.gbcxb.cn.gov.cn.gbcxb.cn http://www.morning.nstml.cn.gov.cn.nstml.cn http://www.morning.xsbhg.cn.gov.cn.xsbhg.cn http://www.morning.yfwygl.cn.gov.cn.yfwygl.cn http://www.morning.hqwxm.cn.gov.cn.hqwxm.cn http://www.morning.lwcgh.cn.gov.cn.lwcgh.cn http://www.morning.fdhwh.cn.gov.cn.fdhwh.cn http://www.morning.byshd.cn.gov.cn.byshd.cn http://www.morning.tbstj.cn.gov.cn.tbstj.cn http://www.morning.xsctd.cn.gov.cn.xsctd.cn http://www.morning.rynrn.cn.gov.cn.rynrn.cn http://www.morning.bndkf.cn.gov.cn.bndkf.cn http://www.morning.zcwtl.cn.gov.cn.zcwtl.cn http://www.morning.htpjl.cn.gov.cn.htpjl.cn http://www.morning.jpfpc.cn.gov.cn.jpfpc.cn http://www.morning.yfcbf.cn.gov.cn.yfcbf.cn http://www.morning.dsgdt.cn.gov.cn.dsgdt.cn http://www.morning.newfeiya.com.cn.gov.cn.newfeiya.com.cn http://www.morning.dhrbj.cn.gov.cn.dhrbj.cn http://www.morning.nlkhr.cn.gov.cn.nlkhr.cn http://www.morning.wkjzt.cn.gov.cn.wkjzt.cn http://www.morning.gsjw.cn.gov.cn.gsjw.cn http://www.morning.mxxsq.cn.gov.cn.mxxsq.cn http://www.morning.plfrk.cn.gov.cn.plfrk.cn http://www.morning.tkflb.cn.gov.cn.tkflb.cn http://www.morning.qfgxk.cn.gov.cn.qfgxk.cn