个人博客网:https://wushaopei.github.io/    (你想要这里多有)

一、案例环境初始化

1、环境搭建与准备

Spring Boot 项目,https://start.spring.io/

Git 管理代码,https://github.com/wushaopei/concurrency

码云:https://gitee.com/wushaopei

点击Generate -Ctrl + 将项目下载到本地,并解压。

使用git bash 将码云仓库下载到本地:

git clone https://gitee.com/wushaopei/concurrency.git

将concurrency.zip压缩包解压后的src、mvnw.cmd、mvnw、pom.xml文件复制到git仓库目录concurrency下,

配置依赖驱动,加载项目

二、案例准备工作

1、自定义注解ThreadSafe.java类,用于标识线程安全的类:

/**
* @ClassName ThreadSafe
* @Description TODO
* @Author wushaopei
* @Date 2019/10/30 14:39
* @Version 1.0
* 用来标记线程安全的类或写法
*/
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.SOURCE)
public @interface ThreadSafe { String value() default "";
}

2、自定义注解NoThreadSafe.java类,用于标识线程不安全的类:

/**
* @ClassName NoThreadSafe
* @Description TODO
* @Author wushaopei
* @Date 2019/10/30 14:44
* @Version 1.0
* 用来标记线程不安全的类或写法
*
*/
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.SOURCE)
public @interface NoThreadSafe { String value() default "";
}

3、用来标记 不推荐 的类或者写法

/**
* 用来标记 不推荐 的类或者写法
* */
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.SOURCE)
public @interface NoRecommend { String value() default "";
}

4、Recommend.java 用来标记 推荐 的类或者写法

/**
* 用来标记 推荐 的类或者写法
* */
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.SOURCE)
public @interface Recommend { String value() default "";
}

三、并发模拟 - 工具

常用工具:

Postman : Http请求模拟工具

Apache Bench AB) : Apache附带的工具,测试网站性能

JMeter : Apache 组织开发的压力测试工具

代码: Semaphore 、CountDownLatch等

1、并发模拟——POSTMAN

1) 创建测试接口:

/**
* @ClassName TestController
* @Description TODO
* @Author wushaopei
* @Date 2019/10/30 15:48
* @Version 1.0
*/
@Controller
@Slf4j
public class TestController { @RequestMapping("/test")
@ResponseBody public String test(){
return "test";
}
}

使用POSTMAN模拟并发:

2) 新建测试用用户组:

3) 执行Run Concurrency按钮:

2、使用专业并发模拟测试工具——Apache Bench(AB)

使用ab命令指定每秒的请求数和间隔时间以及请求接口地址

3、压测工具JMeter:

jmeter文件目录下文件概览:

启动方式:

linux环境使用jmeter.sh启动脚本测试

Windows环境使用jmeter.bat启动脚本测试

使用界面:

Ramp-Up Period 代表虚拟用户增长时长,可理解为一段时间内多个用户登录的总时间范围,如8点15到9点这段时间是打卡上班的时间,也就是45分钟*60秒=2700秒

添加HTTP请求:

添加监听窗口-图形结果和察看结果树

点击绿色三角按钮启动HTTP请求

请求响应结果查看:

四、并发模拟-代码

1、 CountDownLatch (计数器向下减的闭锁)

案例解析该类的使用: 假设计数器值 cnt = 3, 线程A在调用了await()方法之后,当前线程就进入了等待状态awaiting;之后在其他进程中每次执行countDown()方法(如T1)之后计数器 cnt 就会减一,然后当前线程继续执行;之后的T2、T3一次执行完成,当计数器 cnt 变成 0之后,线程A才会继续执行。

说明: CountDownLatch这个类可以阻塞线程,并在满足某种特定条件下去执行。

2、Semaphore

假如一条公路上只有两条车道,那么同时只有两辆车可以通过同一个点;当两辆车中的其中一辆车让开以后,其他的等待的车就可以继续通过了。

说明: Semaphore可以阻塞进程,并且可以控制同一时间请求的并发量

CountDownLatch 和 Semaphore 通常会和线程池一起使用

3、并发模拟

package com.mmall.concurrency;

import lombok.extern.slf4j.Slf4j;

import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Semaphore; /**
* @ClassName ConcurrencyTest
* @Description TODO
* @Author wushaopei
* @Date 2019/10/30 16:50
* @Version 1.0
*/
@Slf4j
@NoThreadSafe //该标识表示当前线程及线程池的并发处理使用方式不正确,建议不要这么写
public class ConcurrencyTest { //请求总数
public static int clientTotal = 5000; // 同时并发执行的线程数
public static int threadTotal = 200; public static int count = 0; public static void main(String[] args) throws InterruptedException {
ExecutorService executorService = Executors.newCachedThreadPool();
final Semaphore semaphore = new Semaphore(threadTotal);
final CountDownLatch countDownLatch = new CountDownLatch(clientTotal);
for (int i = 0 ; i < clientTotal ; i++){
executorService.execute(()->{
try {
semaphore.acquire();
add();
semaphore.release();
}catch (Exception e){
log.error("exception",e);
}
countDownLatch.countDown();
});
}
countDownLatch.await();
executorService.shutdown();
log.info("count:{}",count);
} private static void add(){
count ++;
}
}

执行结果:

第一次

16:58:27.063 [main] INFO com.mmall.concurrency.ConcurrencyTest - count:4944

Process finished with exit code 0

第二次:

16:59:48.185 [main] INFO com.mmall.concurrency.ConcurrencyTest - count:4938

Process finished with exit code 0

由结果可知:程序执行结果数量少于5000,并且每次都不同。所以这是不确定的结果,存在线程安全的问题。

小结:

Postman : Http请求模拟工具

Apache Bench(AB):Apache附带的工具,测试网站性能

JMeter : Apache组织开发的压力测试工具

代码: Semaphore、CountDownLatch等

Java并发编程 (三) 项目准备的更多相关文章

  1. Java并发编程三个性质:原子性、可见性、有序性

      并发编程 并发程序要正确地执行,必须要保证其具备原子性.可见性以及有序性:只要有一个没有被保证,就有可能会导致程序运行不正确  线程不安全在编译.测试甚至上线使用时,并不一定能发现,因为受到当时的 ...

  2. 【Java并发编程三】闭锁

    1.什么是闭锁? 闭锁(latch)是一种Synchronizer(Synchronizer:是一个对象,它根据本身的状态调节线程的控制流.常见类型的Synchronizer包括信号量.关卡和闭锁). ...

  3. Java 并发编程(三):如何保证共享变量的可见性?

    上一篇,我们谈了谈如何通过同步来保证共享变量的原子性(一个操作或者多个操作要么全部执行并且执行的过程不会被任何因素打断,要么就都不执行),本篇我们来谈一谈如何保证共享变量的可见性(多个线程访问同一个变 ...

  4. Java并发编程(三):ReentrantLock

    ReentrantLock是可以用来代替synchronized的.ReentrantLock比synchronized更加灵活,功能上面更加丰富,性能方面自synchronized优化后两者性能没有 ...

  5. 【Java并发编程实战】----- AQS(三):阻塞、唤醒:LockSupport

    在上篇博客([Java并发编程实战]----- AQS(二):获取锁.释放锁)中提到,当一个线程加入到CLH队列中时,如果不是头节点是需要判断该节点是否需要挂起:在释放锁后,需要唤醒该线程的继任节点 ...

  6. 《Java并发编程实战》第三章 对象的共享 读书笔记

    一.可见性 什么是可见性? Java线程安全须要防止某个线程正在使用对象状态而还有一个线程在同一时候改动该状态,并且须要确保当一个线程改动了对象的状态后,其它线程能够看到发生的状态变化. 后者就是可见 ...

  7. [Java并发编程(三)] Java volatile 关键字介绍

    [Java并发编程(三)] Java volatile 关键字介绍 摘要 Java volatile 关键字是用来标记 Java 变量,并表示变量 "存储于主内存中" .更准确的说 ...

  8. Java并发编程原理与实战三十一:Future&FutureTask 浅析

    一.Futrue模式有什么用?------>正所谓技术来源与生活,这里举个栗子.在家里,我们都有煮菜的经验.(如果没有的话,你们还怎样来泡女朋友呢?你懂得).现在女票要你煮四菜一汤,这汤是鸡汤, ...

  9. Java并发编程(三)volatile域

    相关文章 Java并发编程(一)线程定义.状态和属性 Java并发编程(二)同步 Android多线程(一)线程池 Android多线程(二)AsyncTask源代码分析 前言 有时仅仅为了读写一个或 ...

随机推荐

  1. hue搭建

    1.安装依赖: sudo yum -y install gcc-c++ asciidoc cyrus-sasl-devel cyrus-sasl-gssapi krb5-devel libxml2-d ...

  2. 王颖奇 20171010129《面向对象程序设计(java)》第十三周学习总结

      实验十三  图形界面事件处理技术 实验时间 2018-11-22 1.实验目的与要求 (1) 掌握事件处理的基本原理,理解其用途: (2) 掌握AWT事件模型的工作机制: (3) 掌握事件处理的基 ...

  3. redis crackit入侵事件总结

    今天发现服务器有异常进程/opt/yam/yam,上网搜了搜,是由于redis未授权引起的入侵,查了些资料,这里做下总结. 1. 现象 有以下其一现象就要注意是否被入侵 crontab -l 可以看到 ...

  4. CF#633 D. Edge Weight Assignment

    D. Edge Weight Assignment 题意 给出一个n个节点的树,现在要为边赋权值,使得任意两个叶子节点之间的路径权值异或和为0,问最多,最少有多少个不同的权值. 题解 最大值: 两个叶 ...

  5. springBoot第二种配置文件yaml书写方式及读取数据、整合myBatis和整合junit

    一.yaml文件格式:key-value形式:可以表示对象 集合 1.语法:key:value 冒号后面必须跟一个空格再写value值 key1: key2: key3:value 2.属性取值:a. ...

  6. css3 常用。

    CSS3 边框 通过 CSS3,您能够创建圆角边框,向矩形添加阴影,使用图片来绘制边框 - 并且不需使用设计软件,比如 PhotoShop. 在本章中,您将学到以下边框属性: border-radiu ...

  7. MySQL zip解压 安装过程和配置

    MYSQL官网下载地址:https://dev.mysql.com/downloads/mysql/   1.下载mysql-5.7.19-winx64.zip,解压到指定的文件夹, 例如:E:\so ...

  8. storm-jdbc详解

    今天来说说Storm集成Jdbc是如何完成的,代码如下: 写入数据: 先来讲讲官方API: Map hikariConfigMap = Maps.newHashMap(); hikariConfigM ...

  9. Spring Bean 定义

    Bean 定义 被称作 bean 的对象是构成应用程序的支柱.也是由 Spring IoC 容器管理的. bean 是一个被实例化,组装,并通过 Spring IoC 容器所管理的对象. 这些 bea ...

  10. Hyperledger Fabric——balance transfer(二)注册用户

    详细分析blance transfer示例的用户注册(register)与登录(enroll)功能. 源码分析 1.首先分析项目根目录的app.js文件中关于用户注册和登录的路由函数.注意这里的tok ...