当前位置: 首页 > news >正文

校园网站建设方案网站分辨率做96是否会更好

校园网站建设方案,网站分辨率做96是否会更好,注册个人公司流程及费用,网站 设计理念在Java早期版本中#xff0c;synchronized属于重量级锁#xff0c;效率低下#xff0c;因为监视器锁#xff08;monitor#xff09;是依赖于底层的操作系统的Mutex Lock来实现的#xff0c;挂起线程和恢复线程都需要转入内核态去完成#xff0c;阻塞或唤醒一个Java线程需…在Java早期版本中synchronized属于重量级锁效率低下因为监视器锁monitor是依赖于底层的操作系统的Mutex Lock来实现的挂起线程和恢复线程都需要转入内核态去完成阻塞或唤醒一个Java线程需要操作系统切换CPU状态来完成(用户态和内核态之间的切换)这种状态切换需要耗费处理器时间如果同步代码块中内容过于简单这种切换的时间可能比用户代码执行的时间还长”时间成本相对较高这也是为什么早期的synchronized效率低的原因 Java 6之后为了减少获得锁和释放锁所带来的性能消耗引入了锁升级 1 为什么每个Java对象都可以上锁 1.1 C底层源码 Java的基类Object是由C底层源码实现的,每一个Object对象都有一个ObjectMonitor实例 Monitor可以理解为一种同步工具也可理解为一种同步机制常常被描述为一个Java对象。Java对象是天生的Monitor每一个Java对象都有成为Monitor的潜质因为在Java的设计中 每一个Java对象自打娘胎里出来就带了一把看不见的锁它叫做内部锁或者Monitor锁 Monitor的本质是依赖于底层操作系统的Mutex Lock实现操作系统实现线程之间的切换需要从用户态到内核态的转换成本非常高。 1.2 Monitor(监视器锁) Monitor是在jvm底层实现的底层代码是c。本质是依赖于底层操作系统的Mutex Lock实现操作系统实现线程之间的切换需要从用户态到内核态的转换状态转换需要耗费很多的处理器时间成本非常高。所以synchronized是Java语言中的一个重量级操作。 1.3 Monitor与java对象以及线程是如何关联 如果一个java对象被某个线程锁住则该java对象的Mark Word字段中LockWord指向monitor的起始地址Monitor的Owner字段会存放拥有相关联对象锁的线程id 1.4 java6以后开始优化Synchronized 我们先来回顾一下锁升级对 synchronized 性能的影响所谓的锁升级是指 synchronized 从无锁升级到偏向锁再到轻量级锁最后到重量级锁的过程它叫锁升级也叫锁升级。 JDK 1.6 之前synchronized 是重量级锁也就是说 synchronized 在释放和获取锁时都会从用户态转换成内核态而转换的效率是比较低的。但有了锁升级机制之后synchronized 的状态就多了无锁、偏向锁以及轻量级锁了这时候在进行并发操作时大部分的场景都不需要用户态到内核态的转换了这样就大幅的提升了 synchronized 的性能。 2. synchronized锁种类及升级步骤 synchronized用的锁是存在Java对象头里的Mark Word中.锁升级功能主要依赖MarkWord中锁标志位和释放偏向锁标志位 2.1 无锁 锁消除 锁消除是在JIT编译器级别的事情。 在即时编译器时如果发现不可能被共享的对象则可以消除这些对象的锁操作。 也许你会觉得奇怪既然有些对象不可能被多线程访问那为什么要加锁呢写代码时直接不加锁不就好了。 但是有时这些锁并不是程序员所写的有的是JDK实现中就有锁的比如Vector和StringBuffer这样的类它们中的很多方法都是有锁的。当我们在一些不会有线程安全的情况下使用这些类的方法时达到某些条件时编译器会将锁消除来提高性能 public class TestLockEliminate {public static String getString(String s1, String s2) {StringBuffer sb new StringBuffer();sb.append(s1);sb.append(s2);return sb.toString();}public static void main(String[] args) {long tsStart System.currentTimeMillis();for (int i 0; i 1000000; i) {getString(TestLockEliminate , Suffix);}System.out.println(一共耗费 (System.currentTimeMillis() - tsStart) ms);}上述代码中的StringBuffer.append是一个同步操作但是StringBuffer却是一个局部变量并且方法也并没有把StringBuffer返回所以不可能会有多线程去访问它。那么此时StringBuffer中的同步操作就是没有意义的。 开启锁消除是在JVM参数上设置的当然需要在server模式下 并且要开启逃逸分析。 逃逸分析的作用呢就是看看变量是否有可能逃出作用域的范围 当JVM参数为 -server -XX:DoEscapeAnalysis -XX:EliminateLocks DEMO 2 /*** 锁消除* 从JIT角度看相当于无视它synchronized (o)不存在了,这个锁对象并没有被共用扩散到其它线程使用* 极端的说就是根本没有加这个锁对象的底层机器码消除了锁的使用*/ public class LockClearUPDemo {static Object objectLock new Object();//正常的public void m1(){//锁消除,JIT会无视它synchronized(对象锁)不存在了。不正常的Object o new Object();synchronized (o){System.out.println(-----hello LockClearUPDemo\to.hashCode()\tobjectLock.hashCode());}}public static void main(String[] args){LockClearUPDemo demo new LockClearUPDemo();for (int i 1; i 10; i) {new Thread(() - {demo.m1();},String.valueOf(i)).start();}} }锁粗化 锁粗化是指将多个连续的加锁、解锁操作连接在一起扩展成一个范围更大的锁。 我只听说锁“细化”可以提高程序的执行效率也就是将锁的范围尽可能缩小这样在锁竞争时等待获取锁的线程才能更早的获取锁从而提高程序的运行效率但锁粗化是如何提高性能的呢 没错锁细化的观点在大多数情况下都是成立了但是一系列连续加锁和解锁的操作也会导致不必要的性能开销从而影响程序的执行效率比如这段代码 public String method() {StringBuilder sb new StringBuilder();for (int i 0; i 10; i) {// 伪代码加锁操作sb.append( i);// 伪代码解锁操作}return sb.toString(); }这里我们不考虑编译器优化的情况如果在 for 循环中定义锁那么锁的范围很小但每次 for 循环都需要进行加锁和释放锁的操作性能是很低的但如果我们直接在 for 循环的外层加一把锁那么对于同一个对象操作这段代码的性能就会提高很多如下伪代码所示 public String method() {StringBuilder sb new StringBuilder();// 伪代码加锁操作for (int i 0; i 10; i) {sb.append( i);}// 伪代码解锁操作return sb.toString(); }锁粗化的作用如果检测到同一个对象执行了连续的加锁和解锁的操作则会将这一系列操作合并成一个更大的锁从而提升程序的执行效率。 DEMO 2: /*** 锁粗化* 假如方法中首尾相接前后相邻的都是同一个锁对象那JIT编译器就会把这几个synchronized块合并成一个大块* 加粗加大范围一次申请锁使用即可避免次次的申请和释放锁提升了性能*/ public class LockBigDemo {static Object objectLock new Object();public static void main(String[] args){new Thread(() - {synchronized (objectLock) {System.out.println(11111);}synchronized (objectLock) {System.out.println(22222);}synchronized (objectLock) {System.out.println(33333);}},a).start();new Thread(() - {synchronized (objectLock) {System.out.println(44444);}synchronized (objectLock) {System.out.println(55555);}synchronized (objectLock) {System.out.println(66666);}},b).start();} }2.2 偏向锁 当一段同步代码一直被同一个线程多次访问由于只有一个线程那么该线程在后续访问时便会自动获得锁。会修改偏向锁位置为1,并且前面的54位放入当前线程的指针JavaThread*,标识目前的偏向锁被哪一个线程所持有。 理论落地 在实际应用运行过程中发现“锁总是同一个线程持有很少发生竞争”也就是说锁总是被第一个占用他的线程拥有这个线程就是锁的偏向线程。 那么只需要在锁第一次被拥有的时候记录下偏向线程ID。这样偏向线程就一直持有着锁(后续这个线程进入和退出这段加了同步锁的代码块时不需要再次加锁和释放锁。而是直接比较对象头里面是否存储了指向当前线程的偏向锁)。 如果相等表示偏向锁是偏向于当前线程的就不需要再尝试获得锁了直到竞争发生才释放锁。以后每次同步检查锁的偏向线程ID与当前线程ID是否一致如果一致直接进入同步。无需每次加锁解锁都去CAS更新对象头。如果自始至终使用锁的线程只有一个很明显偏向锁几乎没有额外开销性能极高。 假如不一致意味着发生了竞争锁已经不是总是偏向于同一个线程了这时候可能需要升级变为轻量级锁才能保证线程间公平竞争锁。偏向锁只有遇到其他线程尝试竞争偏向锁时持有偏向锁的线程才会释放锁线程是不会主动释放偏向锁的。 技术实现 一个synchronized方法被一个线程抢到了锁时那这个方法所在的对象就会在其所在的Mark Word中将偏向锁修改状态位同时还 会有占用前54位来存储线程指针作为标识。若该线程再次访问同一个synchronized方法时该线程只需去对象头的Mark Word 中去判断一下是否有偏向锁指向本身的ID无需再进入 Monitor 去竞争对象了。 细化案例Account对象举例说明 偏向锁的操作不用直接捅到操作系统不涉及用户到内核转换不必要直接升级为最高级我们以一个account对象的“对象头”为例 假如有一个线程执行到synchronized代码块的时候JVM使用CAS操作把线程指针ID记录到Mark Word当中并修改标偏向标示标示当前线程就获得该锁。锁对象变成偏向锁通过CAS修改对象头里的锁标志位字面意思是“偏向于第一个获得它的线程”的锁。执行完同步代码块后线程并不会主动释放偏向锁。 这时线程获得了锁可以执行同步代码块。当该线程第二次到达同步代码块时会判断此时持有锁的线程是否还是自己持有锁的线程ID也在对象头里JVM通过account对象的Mark Word判断当前线程ID还在说明还持有着这个对象的锁就可以继续进入临界区工作。由于之前没有释放锁这里也就不需要重新加锁。 如果自始至终使用锁的线程只有一个很明显偏向锁几乎没有额外开销性能极高。 结论JVM不用和操作系统协商设置Mutex(争取内核)它只需要记录下线程ID就标示自己获得了当前锁不用操作系统接入。 上述就是偏向锁在没有其他线程竞争的时候一直偏向偏心当前线程当前线程可以一直执行。 JVM参数说明 实际上偏向锁在JDK1.6之后是默认开启的但是启动时间有延迟所以需要添加参数**-XX:BiasedLockingStartupDelay0让其在程序启动时立刻启动只是实验用平时默认设置就好 开启偏向锁 -XX:UseBiasedLocking -XX:BiasedLockingStartupDelay0 关闭偏向锁 -XX:-UseBiasedLocking 关闭之后程序默认会直接进入轻量级锁状态。** 偏向锁的撤销 偏向锁使用一种等到竞争出现才释放锁的机制只有当其他线程竞争锁时持有偏向锁的原来线程才会被撤销要升级为轻量级锁。 撤销需要等待全局安全点(该时间点上没有字节码正在执行)同时检查持有偏向锁的线程是否还在执行 ① 第一个线程正在执行synchronized方法(处于同步块)它还没有执行完其它线程来抢夺该偏向锁会被取消掉并出现锁升级。 此时轻量级锁由原持有偏向锁的线程持有继续执行其同步代码而正在竞争的线程会进入自旋等待获得该轻量级锁。 ② 第一个线程执行完成synchronized方法(退出同步块)则将对象头设置成无锁状态并撤销偏向锁重新偏向 。 总体步骤流程图示 2.3 轻量级锁本质就是CAS自旋 轻量级锁是为了在线程近乎交替执行同步块时提高性能。 主要目的 在没有多线程竞争的前提下通过CAS减少重量级锁使用操作系统互斥量产生的性能消耗本质就是先进行CAS自旋实在不行了再去升级为重量级锁。说白了先自旋再阻塞。 升级时机 当关闭偏向锁功能或多线程竞争偏向锁会导致偏向锁升级为轻量级锁 假如线程A已经拿到锁这时线程B又来抢该对象的锁由于该对象的锁已经被线程A拿到当前该锁已是偏向锁了。 而线程B在争抢时发现对象头Mark Word中的线程ID不是线程B自己的线程ID(而是线程A)那线程B就会进行CAS操作希望能获得锁。 此时线程B操作中有两种情况 如果锁获取成功直接替换Mark Word中的线程ID为B自己的ID(A → B)重新偏向于其他线程(即将偏向锁交给其他线程相当于当前线程被释放了锁)该锁会保持偏向锁状态A线程OverB线程上位 如果锁获取失败则偏向锁升级为轻量级锁此时轻量级锁由原持有偏向锁的线程持有继续执行其同步代码而正在竞争的线程B会进入自旋等待获得该轻量级锁。 自旋达到一定次数和程度 java6之前 默认启用默认情况下自旋的次数是 10 次。使用JVM 参数 -XX:PreBlockSpin10来修改或者自旋线程数超过cpu核数一半 java6之后(自适应) 自适应意味着自旋的次数不是固定不变的,而是根据同一个锁上一次自旋的时间, 拥有锁线程的状态来决定。 轻量锁与偏向锁的区别和不同 争夺轻量级锁失败时自旋尝试抢占锁。轻量级锁每次退出同步块都需要释放锁而偏向锁是在竞争发生时才释放锁 3. 各种锁优缺点、synchronized锁升级和实现原理 synchronized锁升级过程总结一句话就是先自旋不行再阻塞。实际上是把之前的悲观锁(重量级锁)变成在一定条件下使用偏向锁以及使用轻量级(自旋锁CAS)的形式 synchronized在修饰方法和代码块在字节码上实现方式有很大差异但是内部实现还是基于对象头的MarkWord来实现的。 JDK1.6之前synchronized使用的是重量级锁JDK1.6之后进行了优化拥有了无锁-偏向锁-轻量级锁-重量级锁的升级过程而不是无论什么情况都使用重量级锁。 偏向锁: 适用于单线程适用的情况在不存在锁竞争的时候进入同步方法/代码块则使用偏向锁。 **轻量级锁**适用于竞争较不激烈的情况(这和乐观锁的使用范围类似) 存在竞争时升级为轻量级锁轻量级锁采用的是自旋锁如果同步方法/代码块执行时间很短的话采用轻量级锁虽然会占用cpu资源但是相对比使用重量级锁还是更高效。 **重量级锁**适用于竞争激烈的情况如果同步方法/代码块执行时间很长那么使用轻量级锁自旋带来的性能消耗就比使用重量级锁更严重这时候就需要升级为重量级锁。 4. 锁升级流程图 5. 64位的MarkWord标记图 参考 源码级别的讲解JAVA 中的CAS ava对象内存布局及对象头详解
http://www.tj-hxxt.cn/news/136110.html

相关文章:

  • 有口碑的做网站随州市网站建设
  • 东莞在线网站制作平台公司怎么搭建自己网站
  • 有源码如何做网站网站名称注册保护
  • 互联网服务平台登录安阳网站制作优化
  • 中小企业网站建设效果企业网站的主要内容
  • 平面设计网站源码公司网站建设图片素材怎么找
  • 十大家居家装网站哪个好网站建设脑图
  • 怎么咨询网络服务商深圳网站建设制作优化
  • 国内电子商务网站有哪些建设一个网站需要哪些
  • 企业网站主要功能网络推广方法有几种
  • 网站建设制作设计公司公司logo墙设计图片
  • 网站设计的逻辑河北高端网站设计
  • 网站建设网站模版广告网站模板下载 迅雷下载安装
  • 淘宝网站可以做轮播吗个人网站 创意
  • 室内设计网站配色app广州的服装网站建设
  • 杭州企业网站建站模板网站运营单位是什么意思
  • 网站升级停止访问如何做中华会计网校
  • 做自己的网站能赚钱吗网站基础建设英文
  • WordPress多语言多站点详情页设计详细教程
  • 35互联做网站怎么样360广告推广平台
  • 西安商城类网站制作首选大型网站建站公司
  • 可以做笔记的网站网站建设原因
  • 天津建设注册执业中心网站企业宣传片策划团队
  • 网站源码下载 用户注册天津建站服务
  • dw网站模版wordpress如何把文章
  • 北京住房城乡建设厅网站南京网站开发南京乐识赞
  • 网站设计与网页制作岗位招聘信息种植园网站模板
  • 已有的网站如何做排名优化山东网站建设都有那些
  • 建站网站模板做网站用什么面板好
  • 企业官方网站建设推广软件哪个好