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

黑龙江省建设会计协会网站上海集团平台

黑龙江省建设会计协会网站,上海集团平台,做网站推广复杂吗,湖北省建设安全管理协会网站文章目录一.线程安全的概念1.1 线程安全的概念1.2 线程不安全的原因1.3 解决线程不安全二.synchronized-monitor lock(监视器锁)2.1 synchronized的特性(1)互斥(2)刷新内存(3)可重入2.2 synchronied使用方法1.直接修饰普通方法:2.修饰静态方法:3.修饰代码块:三.死锁3.1死锁的情… 文章目录一.线程安全的概念1.1 线程安全的概念1.2 线程不安全的原因1.3 解决线程不安全二.synchronized-monitor lock(监视器锁)2.1 synchronized的特性(1)互斥(2)刷新内存(3)可重入2.2 synchronied使用方法1.直接修饰普通方法:2.修饰静态方法:3.修饰代码块:三.死锁3.1死锁的情况3.2 死锁的四个必要条件1.互斥使用2.不可抢占3.请求和保持4.循环等待3.3解决死锁的办法四.volatile 关键字五. wait和notify5.1 wait()方法5.2 notify()方法一.线程安全的概念 先来看一段代码 class Counter{public int count 0;public void add(){count;}} public class Thread14 {public static void main(String[] args) throws InterruptedException {Counter counter new Counter();Thread t1 new Thread(()-{for (int i 0; i 50000; i) {counter.add();}});Thread t2 new Thread(() -{for (int i 0; i 50000; i) {counter.add();}});t1.start();t2.start();t1.join();t2.join();System.out.println(count counter.count);} } 可以看到结果是不确定的 1.1 线程安全的概念 先来说一下非线程安全的概念:非线程安全主要是指多个线程对同一个对象中的同一个实例变量进行操作时会出现值被更改、值不同步的情况进而影响程序的执行流程。 则线程安全:如果多线程环境下代码运行的结果是符合我们预期的即在单线程环境应该的结果则说这个程序是线程安全的。 1.2 线程不安全的原因 先解释上述线代码程不安全的原因 如果两个线程并发执行count,此时就相当于两组load add save进行执行,此时不同的线程调度顺序就可能会产生一些结果上的差异 由于线程的抢占执行,导致当前执行到任意一个指令,线程都可能bei调度走,CPU让别的线程来执行 如下图: 导致下面的结果: 线程安全问题的原因: 1.抢占式执行,随机调度(根本原因) 2.代码结构:多个线程同时修改同一个变量 3.原子性(操作是非原子性,容易出现问题) 4.内存可见性问题(如一个线程读,一个线程改) 5.指令重排序 1.3 解决线程不安全 从原子性入手,通过加锁,把非原子的,转成原子的 加了synchronized之后,进入方法就会加锁,出了方法就会解锁,如果两个线程同时尝试加锁,此时一个能获取锁成功,另一个只能阻塞等待(BLOCKED),一直阻塞到刚才的线程解锁,当前线程才能加锁成功 二.synchronized-monitor lock(监视器锁) 2.1 synchronized的特性 (1)互斥 进入sychronized修饰的代码块,相当于加锁退出sychronizde修饰的代码块,相当于解锁 (2)刷新内存 synchronized的工作过程: 1.获得互斥锁 2.从内存拷贝变量的最新副本到工作的内存 3.执行代码 4.将更改后的共享变量的值刷新到主内存 5.释放互斥锁 (3)可重入 synchronized同步块对同一条线程来说是可重入的,不会出现自己把自己锁死的问题(自己可以再次获取自己的内部锁) 理解把自己锁死 一个线程没有释放锁,然后又尝试再次加锁 按照之前对于锁的设定,第二次加锁的时候,就会阻塞等待,而获取不到第一次的锁,就把自己锁死 2.2 synchronied使用方法 1.直接修饰普通方法: 锁的SynchronizedDemo1对象 public class SynchronizedDemo1 {public synchronized void methond() {} }2.修饰静态方法: 锁SynchronizedDemo2对象 public class SynchronizedDemo2 {public synchronized void methond() {} }3.修饰代码块: 明确指定锁哪个对象 public class SychronizedDemo{public void method(){sychronized(this){}} }锁类对象 public class SynchronizedDemo {public void method() {synchronized (SynchronizedDemo.class) {}} }三.死锁 3.1死锁的情况 1.一个线程,连续加锁两次,如果锁是不可重入锁,就会死锁 2.两个线程,两把锁,t1和t2各自先针对锁A和锁B加锁,在获取对方的锁 public class Thread15 {public static void main(String[] args) {Object lock1 new Object();Object lock2 new Object();Thread t1 new Thread(()-{synchronized (lock1){try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}synchronized (lock2){System.out.println(t1把锁1和锁2都获得了);}}});Thread t2 new Thread(()-{synchronized (lock2){try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}synchronized (lock1){System.out.println(t2把锁1和锁2都获得了);}};});t1.start();t2.start();} } 3.多个线程,多把锁(相当于2的一般情况) 3.2 死锁的四个必要条件 1.互斥使用 线程1拿到了锁,线程2就须等着 2.不可抢占 线程1拿到锁A之后,必须是线程1主动释放 3.请求和保持 线程1拿到锁A之后,在尝试获取锁B,A这把锁还是保持的 4.循环等待 线程1尝试获取到锁A和锁B,线程2尝试获取锁B和锁A,线程1在获取B的时候等待线程2释放B,同时线程2 在获取A的时候等待线程1释放A 3.3解决死锁的办法 给锁编号,然后指定一个固定的顺序来加锁,任意线程加把锁,都让线程遵守上述顺序,此时循环等待自然破除 对于synchronied前三个条件都是锁的基本特性,我们只能对四修改 public class Thread15 {public static void main(String[] args) {Object lock1 new Object();Object lock2 new Object();Thread t1 new Thread(()-{synchronized (lock1){try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}synchronized (lock2){System.out.println(t1把锁1和锁2都获得了);}}});Thread t2 new Thread(()-{synchronized (lock1){try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}synchronized (lock2){System.out.println(t2把锁1和锁2都获得了);}};});t1.start();t2.start();} } 四.volatile 关键字 volatile 和内存可见性问题密切相关 一个线程针对一个变量进行读取操作,同时另一个线程针对这个变量进行修改,此时读取到值,不一定是修改之后的值(归根结底是编译器/jvm在多线程下优化时产生了误判) 使用汇编语言解释 1.load,把内存中flag的值,读取到寄存器 2.cmp把寄存器的值和0进行比较,根据比较结果,决定下一不执行. 由于load执行速度太慢(相比于cmp来说),再加上反复load的结果都一样,JVM就不在重复load判定没人改flag值,就只读取一次就好 而给flag加上volatile关键字,告诉编译器变量是易变的,不再进行优化 class MyCounter{volatile public int flag 0; } public class Thread16 {public static void main(String[] args) {MyCounter myCounter new MyCounter();Thread t1 new Thread(() -{while (myCounter.flag 0){//循环体空着}System.out.println(t1循环结束);});Thread t2 new Thread(() -{Scanner scanner new Scanner(System.in);System.out.println(请输入一个整数:);myCounter.flag scanner.nextInt();});t1.start();t2.start();} } 结果: 五. wait和notify wait和notify可以协调线程之间的先后顺序 完成这个协调工作, 主要涉及到三个方法 wait() / wait(long timeout): 让当前线程进入等待状态.notify() / notifyAll():唤醒在当前对象上等待的线程. 注意: wait, notify, notifyAll 都是 Object 类的方法 5.1 wait()方法 wait的操作 1.先释放锁 2.在阻塞等待 3.收到通知之后,重新获取锁,并且在获取锁后,继续往下执行 wait操作需要搭配synchorized来使用 public class Thread17 {public static void main(String[] args) throws InterruptedException {Object object new Object();System.out.println(wait之前);object.wait();System.out.println(wait之后);} }无synchorized的情况 wait无参数版本,就是死等 wait带参数版本,指定了等待的最大时间 5.2 notify()方法 notify()方法是唤醒等待线程 如果有多个线程等待则有线程调度器随机挑选出一个呈 wait 状态的线程。(并没有 “先来后到”) 在notify()方法后当前线程不会马上释放该对象锁要等到执行notify()方法的线程将程序执行完也就是退出同步代码块之后才会释放对象锁。 notfiyAll()方法可以一次唤醒所有的等待线程
http://www.tj-hxxt.cn/news/134534.html

相关文章:

  • 网站显示wordpresswordpress防盗图
  • 中小型企业 公司网站建设啤酒免费代理0元铺货
  • 外贸行销网站网站设计需求模板
  • 做网站需要注意的问题成为直播人的app有哪些
  • 服装公司网站建设开题报告简历模板免费下载word格式
  • 网上做国外兼职网站手机端网站html好看的单页模板
  • 网站页面模板页面布局网站建设注意问题
  • word用来做网站的提高网站关键词排名
  • js做网站统计古镇网站建设公司
  • 做网站 公司有哪些wordpress自己制作主题
  • 太原seo网站排名有没有做兼职的网站吗
  • 建设网站公司哪里好网站如何做地面推广
  • 东莞免费网站建站模板长沙网站大全
  • 高邮建设网站巢湖市网站建设优化
  • 网站现在怎么做排名wordpress 插件官网
  • 什么网站可以学习建设工程法律实践软件开发的工作内容
  • seo外贸网站长沙网页设计哪个公司好
  • 官方网站建设银行在什么网站可以接活做
  • 社保局网站建设意义网站在别人那里已经建好了_公司里要进行修改_怎么做
  • 网站开发需要学什么技能wordpress修改登录地址
  • 做版权保护的网站seo查询seo
  • 中国企业网站西安手机网站建设动力无限
  • 长沙企业建站招聘信息网站搜索引擎收录
  • 公司要做网站网站建设代理网站
  • 资阳的网站建设奉城网站建设
  • 杭州拱墅区网站建设小浣熊做单网站
  • 建设网站用图片需要版权做可视化的网站
  • 徐州网站建设网络推广做电商没几个能赚钱的
  • 做网站语言最好网站图片特效代码
  • 创建网站超市国内vps做网站备案