iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >怎么在java中实现多线程高并发
  • 691
分享到

怎么在java中实现多线程高并发

2023-06-14 13:06:21 691人浏览 泡泡鱼
摘要

这篇文章将为大家详细讲解有关怎么在java中实现多线程高并发,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。1.JMM数据原子操作read(读取)∶从主内存读取数据load(载入):将主内存读

这篇文章将为大家详细讲解有关怎么在java中实现多线程高并发,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。

1.JMM数据原子操作

  • read(读取)∶从主内存读取数据

  • load(载入):将主内存读取到的数据写入工作内存

  • use(使用):从工作内存读取数据来计算

  • assign(赋值):将计算好的值重新赋值到工作内存中

  • store(存储):将工作内存数据写入主内存

  • write(写入):将store过去的变量值赋值给主内存中的变量

  • lock(定):将主内存变量加锁,标识为线程独占状态

  • unlock(解锁):将主内存变量解锁,解锁后其他线程可以锁定该变量

2.来看volatile关键字

(1)启动两个线程

public class VolatileDemo {     private static boolean flag = false;    public static void main(String[] args) throws InterruptedException {        new Thread(() -> {            while (!flag){            }            System.out.println("跳出while循环了");        }).start();         Thread.sleep(2000);        new Thread(() -> changeFlage()).start();    }     private static void changeFlage() {        System.out.println("开始改变flag值之前");        flag = true;        System.out.println("改变flag值之后");    }}

没加volatile之前,第一个线程的while判断一直满足

怎么在java中实现多线程高并发

(2)给变量flag加了volatile之后

public class VolatileDemo {     private static volatile boolean flag = false;    public static void main(String[] args) throws InterruptedException {        new Thread(() -> {            while (!flag){            }            System.out.println("跳出while循环了");        }).start();         Thread.sleep(2000);        new Thread(() -> changeFlage()).start();    }     private static void changeFlage() {        System.out.println("开始改变flag值之前");        flag = true;        System.out.println("改变flag值之后");    }}

while语句能够满足条件

怎么在java中实现多线程高并发

(3)原理解释:

开启第一个线程时,flag变量通过read从主内存中读出数据,使用load把数据加载进线程一的工作内存,通过use把flag读取到线程中;线程二也是同样的读取操作。线程二通过assign改变了flag的值,线程二工作内存中存储的flag=true,再通过store把flag写入到总线,总线再把flag通过write写入到住内存;由于两个线程读取操作的都是各种工作内存中的值,是主内存的副本,相互不通信,所以线程一一直再循环,线程一的flag为false。

加了volatile后,添加了缓存一致性协议(MESI),CPU通过总线嗅探机制感知到数据的变化而自己缓存里的值失效,此时线程一会把工作内存中存放的flag失效,从主内存中重新读取flag的值,此时满足while条件。

volatile底层通过汇编语言的lock修饰,当变量有修改立马写回主类,避免指令重排序

怎么在java中实现多线程高并发

3.并发编程三大特性

可见性,有序性、原子性

4.双锁判断机制创建单例模式

public class DoubleCheckLockSinglenon {     private static volatile DoubleCheckLockSinglenon doubleCheckLockSingleon = null;     public DoubleCheckLockSinglenon(){}       public static DoubleCheckLockSinglenon getInstance(){        if (null == doubleCheckLockSingleon) {            synchronized(DoubleCheckLockSinglenon.class){                if(null == doubleCheckLockSingleon){                    doubleCheckLockSingleon = new DoubleCheckLockSinglenon();                }            }        }        return doubleCheckLockSingleon;    }      public static void main(String[] args) {        System.out.println(DoubleCheckLockSinglenon.getInstance());     } }

当线程调用getInstance方法创建的时候,先判断是否为空,为空则把对象加上锁,否则多线程的情况会创建重复,再锁里面再次判断是否为空,当new一个对象的时候,先在内存分配空间,再执行对象的init属性赋零操作,再执行初始化赋值操作。

cpu为了优化代码执行效率,会对满足as-if-serial和happens-before原则的代码进行指令重排序,as-if-serial规定线程内的执行代码顺序不影响结果输出,则会进行指令重排;

happens-before规定一些锁的顺序,同一个对象的unlock需要出现下一个lock之前等。

所以为了防止new的时候,指令重排,先进行赋值再执行赋零操作情况,需要加上volatile修饰符,加上volatile修饰后,在new操作时会创建内存屏障,高速cpu不进行指令重排序,底层是lock关键字;内存屏障分为LoadLoad(读读)、storestore(写写)、loadstore(读写)、storeload(写读),底层是c++代码写的,c++代码再调用汇编语言

5.synchronized关键字

(1)没加synchronized之前

package com.qingyun; public class SynchronizedDemo {     public static void main(String[] args) throws InterruptedException {         Num num = new Num();       Thread t1 = new Thread(() -> {            for (int i = 0;i < 100000;i++) {                num.incrent();            }        });        t1.start();         for (int i = 0;i < 100000;i++) {            num.incrent();        }        t1.join();        System.out.println(num.getNum());    }}
package com.qingyun; public class Num {     public int num = 0;     public void incrent() {        num++;    }     public int getNum(){        return num;    }}

输出结果不是我们想要的,由于线程和for循环同时去调加的方法,导致最后输出的结果不是我们想要的

怎么在java中实现多线程高并发

(2)加上synchronized之后

public synchronized void incrent() {        num++;    } //或者   public  void incrent() {        synchronized(this){            num++;        }    }

怎么在java中实现多线程高并发

输出的结果是我们想要的,synchronized关键字底层使用的lock,是重量级锁,互斥锁、悲观锁,jdk1.6之前的锁,线程会放到一个队列里面等待着执行

6.AtomicIntger原子操作

(1)给原子加1的操作,可以使用AtomicInteger实现,与synchronized相比,性能大大提升

public class Num {    // public int num = 0;    AtomicInteger atomicInteger = new AtomicInteger();     public  void incrent() {       atomicInteger.incrementAndGet(); //原子加1    }     public int getNum(){        return atomicInteger.get();    }}

AtomicInteger源码有一个value字段,使用volatile修饰,volatile底层使用lock修饰,保证多线程并发结果的正确

private volatile int value;

(2)atomicInteger.incrementAndGet()方法做的事情:先获取到value的值,给值加1,再使用旧的值和atomicInteger进行比较,相等了把newValue设置进去,由于使用多线程可能值会不相等的情况,所以使用while进行循环比对,相等了执行完才推出

while(true) {    int oldValue = atomicInteger.get();    int newValue = oldValue+1;    if(atomicInteger.compareAndSet(oldValue,newValue)){       break;    }}

(3)atomicInteger.compareAndSet比对完值后才设置新值的方式即为CAS:无锁、乐观锁、轻量级锁,synchroznied存在线程阻塞、上行文切换、操作系统调度比较费时;CAS一直循环比对执行,效率要高

(4)compareAndSetInt底层使用native修饰,底层是c++代码,实现了原子性问题,在汇编语言使用代码lock cmpxchqq保证了原子性,是缓存行锁

(5)ABA问题:线程一那到一个变量,线程二执行比较快,也拿到这个变量,把变量的值进行修改,再快速修改回原来的值,这样变量的值有过一次变化,线程一再去执行compareAndSet的时候,虽然值还是之前的没变,但是已经发生过变化了,出现ABA问题

(6)解决ABA问题就是给变量加版本,每次操作变量版本加1,JDK带版本的锁有AtomicStampedReference,这样就算变量被其它线程修改过再回复原值,版本号也是不一致的。

7.锁优化

(1)重量级锁会把等待的线程放到队列中,重量级锁锁定的是monitor,存在上下问切换的资源占用;轻量级锁若是线程太多,会存在自旋,耗费cpu

(2)jdk1.6之后,锁升级为无状态-》偏向锁(锁id指定)-》轻量级锁(自旋膨胀)-》重量级锁(队列存储)

(3)创建一个对象,此时对象为无状态,当启动了一个线程时,再创建一个对象时,启用偏向锁,偏向锁执行完之后不会释放锁;当再启用一个线程时,有两个线程来挣抢对象时,立马又偏向锁升级为轻量级锁;当再创建一个线程的来挣抢对象锁时,由轻量级锁升级为重量级锁

(4)分段CAS,底层有一个base记录变量值,当有多个线程类访问此变量是,base的值会分为多个cell,组成数组,每个cell对应一到多个线程的cas处理,避免了线程的自旋空转,这样还是轻量级锁,返回数据的时候,底层调用的是所有cell数组和base的加和

public class Num {     LongAdder longAdder = new LongAdder();     public  void incrent() {         longAdder.increment();    }     public long getNum(){       return longAdder.longValue();    }}
public long longValue() {        return sum();    }
public long sum() {        Cell[] as = cells; Cell a;        long sum = base;        if (as != null) {            for (int i = 0; i < as.length; ++i) {                if ((a = as[i]) != null)                    sum += a.value;            }        }        return sum;    }

关于怎么在java中实现多线程高并发就分享到这里了,希望以上内容可以对大家有一定的帮助,可以学到更多知识。如果觉得文章不错,可以把它分享出去让更多的人看到。

--结束END--

本文标题: 怎么在java中实现多线程高并发

本文链接: https://www.lsjlt.com/news/271242.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 怎么在java中实现多线程高并发
    这篇文章将为大家详细讲解有关怎么在java中实现多线程高并发,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。1.JMM数据原子操作read(读取)∶从主内存读取数据load(载入):将主内存读...
    99+
    2023-06-14
  • java怎么实现多线程并发执行
    Java实现多线程并发执行的方式有两种:继承Thread类和实现Runnable接口。 继承Thread类: 定义一个类,继承...
    99+
    2023-10-25
    java
  • java多线程并发执行怎么实现
    在Java中实现多线程的并发执行有多种方式,以下是其中的几种常见方法:1. 继承Thread类:创建一个继承自Thread类的子类,...
    99+
    2023-09-27
    java
  • java多线程中如何实现线程并发库
    本篇文章给大家分享的是有关java多线程中如何实现线程并发库,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。多线程之线程并发库原子性操作类java.util.concurrent...
    99+
    2023-06-19
  • 多线程并发编程如何在Java项目中实现
    本篇文章为大家展示了多线程并发编程如何在Java项目中实现 ,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。一、多线程操作系统有两个容易混淆的概念,进程和线程。进程:一个计算机程序的运行实例,包含了需...
    99+
    2023-05-31
    java 多线程 并发编程
  • DelayQueue怎么在Java多线程并发开发中使用
    这篇文章给大家介绍DelayQueue怎么在Java多线程并发开发中使用,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。Delayed,一种混合风格的接口,用来标记那些应该在给定延迟时间之后执行的对象。此接口的实现必须定...
    99+
    2023-05-31
    java delayqueue 多线程并发
  • 在Java项目中实现多线程并发编程的方法
    在Java项目中实现多线程并发编程的方法?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。Java 中的锁通常分为两种:通过关键字 synchronized 获取的...
    99+
    2023-05-31
    并发 java并发 多线程
  • java的多线程高并发详解
    目录1.JMM数据原子操作2.来看volatile关键字3.并发编程三大特性4.双锁判断机制创建单例模式5.synchronized关键字6.AtomicIntger原子操作7.锁优...
    99+
    2024-04-02
  • Java httpClient连接池支持多线程高并发的实现
    当采用HttpClient httpClient = HttpClients.createDefault() 实例化的时候。会导致Address already in use的异常。...
    99+
    2024-04-02
  • python怎么实现多线程并发抓取
    这篇文章主要为大家展示了“python怎么实现多线程并发抓取”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“python怎么实现多线程并发抓取”这篇文章吧。多线程并发抓取单线程太慢的话,就需要多线...
    99+
    2023-06-26
  • 怎么在Java中实现多线程排序
    这期内容当中小编将会给大家带来有关怎么在Java中实现多线程排序,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。java基本数据类型有哪些Java的基本数据类型分为:1、整数类型,用来表示整数的数据类型。2...
    99+
    2023-06-14
  • java怎么实现高并发
    Java可以通过以下几种方法来实现高并发:1. 线程池:使用线程池来管理线程,避免频繁创建和销毁线程,提高线程的重用性和效率。2. ...
    99+
    2023-08-12
    java
  • Java 多线程并发ReentrantLock
    目录背景ReentrantLock可重入特性公平锁设置参数源码分析Lock 接口加锁操作内部类SynctryLockinitialTryLocklocklockInterruptib...
    99+
    2024-04-02
  • Python多线程与高并发
    主要讲解了关于Python多线程的一些例子和高并发的一些应用场景# -*- coding: utf-8 -*- # @Author: Clarence...
    99+
    2023-01-31
    多线程 Python
  • 《PHP容器并发编程实践:如何实现高效的多线程并发?》
    PHP容器并发编程实践:如何实现高效的多线程并发? 在如今的互联网时代,高并发已经成为了一个家喻户晓的话题。而对于PHP开发者来说,如何实现高效的多线程并发则成为了一个必须掌握的技能。本文将介绍PHP容器并发编程的实践方法,帮助开发者实现高...
    99+
    2023-10-02
    并发 教程 容器
  • Java中多线程怎么实现
    这篇文章主要讲解了“Java中多线程怎么实现”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Java中多线程怎么实现”吧!线程是一些可以并行的,独立的执行的代码.之前我编的程序都只能做一件事情...
    99+
    2023-06-03
  • Java 多线程并发LockSupport
    目录概览源码分析静态方法BlockerunparkUnsafe 的 unpark 方法park不带 blocker 参数的分组需要 blocker 参数的分组park/unpark ...
    99+
    2024-04-02
  • Java多线程并发AbstractQueuedSynchronizer怎么使用
    这篇文章主要介绍“Java多线程并发AbstractQueuedSynchronizer怎么使用”,在日常操作中,相信很多人在Java多线程并发AbstractQueuedSynchronizer怎么使用问题上存在疑惑,小编查阅了各式资料,...
    99+
    2023-07-02
  • Java多线程并发ReentrantLock怎么使用
    这篇文章主要介绍“Java多线程并发ReentrantLock怎么使用”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Java多线程并发ReentrantLock怎么使用”文章能帮助大家解决问题。背景...
    99+
    2023-07-02
  • 怎么在Java中使用ReentrantReadWriteLock实现多线程
    本篇文章给大家分享的是有关怎么在Java中使用ReentrantReadWriteLock实现多线程,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。Java有哪些集合类Java中...
    99+
    2023-06-14
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作