首先解释appen-before,其作用就是保证两个操作的顺序性,特别是多线程中,确保数据的准确性,对于执行顺序会有一定的要求

这里引入volatile手动设置

 

类似的synchronized也可以实现happen-before

测试代码

package com.java.test.happer.before.volatiles;

/**
 * @Description:
 * @Author: Yourheart
 * @Create: 2022/11/3 22:54
 */
public class VolatileTest {


    int x = 0;
    volatile int y = 0;
    int a = 0;
    int b = 0;

    public void set() {
        a = 1;
        x = b;
    }

    public int get() {
        b = 1;
        y = a;
        return y;
    }

    @Override
    public String toString() {
        return "VolatileTest{" +
                "x=" + x +
                ", y=" + y +
                ", a=" + a +
                ", b=" + b +
                '}';
    }
}

  

package com.java.test.happer.before.volatiles;

import lombok.extern.slf4j.Slf4j;

/**
 * @Description:
 * @Author: Yourheart
 * @Create: 2022/11/3 23:24
 */
@Slf4j
public class ThreadDemoOne implements Runnable {

    private VolatileTest volatileTest;

    public ThreadDemoOne(VolatileTest volatileTest) {
        this.volatileTest = volatileTest;
    }

    @Override
    public void run() {
        this.volatileTest.set();
        log.info("【线程一】volatileTest:{}",volatileTest.toString());
    }
}

  

package com.java.test.happer.before.volatiles;

import lombok.extern.slf4j.Slf4j;

/**
 * @Description:
 * @Author: Yourheart
 * @Create: 2022/11/3 23:25
 */
@Slf4j
public class ThreadDemoTwo implements Runnable {


    private VolatileTest volatileTest;

    public ThreadDemoTwo(VolatileTest volatileTest) {
        this.volatileTest = volatileTest;
    }

    @Override
    public void run() {
        int i = this.volatileTest.get();
        log.info("获取到的y的数据:{}",i);
        log.info("【线程二】volatileTest:{}",volatileTest.toString());

    }
}

  

package com.java.test.happer.before.volatiles;

import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;

/**
 * @Description:
 * @Author: Yourheart
 * @Create: 2022/11/3 23:28
 */
public class ThreadPoolExecutorDemo {
    /**
     * 核心线程数
     */
    static int corePoolSize = 3;
    /**
     * 最大线程数
     */
    static int maximumPoolSize = 6;
    /**
     * 超过 corePoolSize 线程数量的线程最大空闲时间
     */
    static long keepAliveTime = 2;
    /**
     * 以秒为时间单位
     */
    static TimeUnit unit = TimeUnit.SECONDS;
    /**
     * 创建工作队列,用于存放提交的等待执行任务
     */
    static BlockingQueue<Runnable> workQueue = new ArrayBlockingQueue<Runnable>(2);

    static ThreadPoolExecutor threadPoolExecutor = null;

    private static void initThreadPool() {
        threadPoolExecutor = new ThreadPoolExecutor(corePoolSize,
                maximumPoolSize,
                keepAliveTime,
                unit,
                workQueue,
                new ThreadPoolExecutor.AbortPolicy());
    }

    public static void executed(Runnable runnable){
        if (threadPoolExecutor==null){
            initThreadPool();
        }
        threadPoolExecutor.execute(runnable);
    }
}

  

package com.java.test.happer.before.volatiles;

import lombok.extern.slf4j.Slf4j;
import org.junit.Test;

/**
 * @Description:
 * @Author: Yourheart
 * @Create: 2022/11/3 23:27
 */
@Slf4j
public class ThreadTest {


    @Test
    public void test(){
       while (true){
           try {
               Thread.sleep(2000);
           } catch (InterruptedException e) {
               e.printStackTrace();
           }
           VolatileTest volatileTest=new VolatileTest();
           ThreadPoolExecutorDemo.executed(new ThreadDemoOne(volatileTest));
           ThreadPoolExecutorDemo.executed(new ThreadDemoTwo(volatileTest));

       }
    }
}

  pom文件

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.java</groupId>
    <artifactId>test-study</artifactId>
    <version>1.0-SNAPSHOT</version>

    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.2.1.RELEASE</version>
        <relativePath/>
    </parent>

    <dependencies>
        <!--tomcat容器-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
        <!--lombok依赖-->
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <version>1.18.16</version>
        </dependency>
        <!--引入junit单元测试依赖-->
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.12</version>
        </dependency>
        <!--判断空的用法  -->
        <dependency>
            <groupId>commons-lang</groupId>
            <artifactId>commons-lang</artifactId>
            <version>2.6</version>
        </dependency>

        <!-- https://mvnrepository.com/artifact/com.oracle.database.jdbc/ojdbc8 -->
        <dependency>
            <groupId>com.oracle.database.jdbc</groupId>
            <artifactId>ojdbc8</artifactId>
            <version>12.2.0.1</version>
        </dependency>
        <!--springboot整合mybatis-->
        <dependency>
            <groupId>org.mybatis.spring.boot</groupId>
            <artifactId>mybatis-spring-boot-starter</artifactId>
            <version>2.1.2</version>
        </dependency>

        <!--添加fastjson依赖-->
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>fastjson</artifactId>
            <version>1.2.70</version>
        </dependency>
        <!-- 热部署模块 -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-devtools</artifactId>
            <optional>true</optional> <!-- 这个需要为 true 热部署才有效 -->
        </dependency>
    </dependencies>


    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
        <finalName>study</finalName>
    </build>


</project>

  在单线程中,执行顺序即使有变化,最终结果不会错乱,但是多线程中就会出现数据错乱的问题

 

原文地址:http://www.cnblogs.com/q202105271618/p/16856219.html

1. 本站所有资源来源于用户上传和网络,如有侵权请邮件联系站长! 2. 分享目的仅供大家学习和交流,请务用于商业用途! 3. 如果你也有好源码或者教程,可以到用户中心发布,分享有积分奖励和额外收入! 4. 本站提供的源码、模板、插件等等其他资源,都不包含技术服务请大家谅解! 5. 如有链接无法下载、失效或广告,请联系管理员处理! 6. 本站资源售价只是赞助,收取费用仅维持本站的日常运营所需! 7. 如遇到加密压缩包,默认解压密码为"gltf",如遇到无法解压的请联系管理员! 8. 因为资源和程序源码均为可复制品,所以不支持任何理由的退款兑现,请斟酌后支付下载 声明:如果标题没有注明"已测试"或者"测试可用"等字样的资源源码均未经过站长测试.特别注意没有标注的源码不保证任何可用性