下载安卓APP箭头
箭头给我发消息

客服QQ:3315713922

Java代码有哪些重要作用?

作者:课课家教育     来源: http://www.kokojia.com点击数:1110发布时间: 2019-08-02 10:19:47

标签: 编程语言JavaJavascript

大神带你学编程,欢迎选课

java是一门面向对象编程语言,不仅吸收了C++语言的各种优点,还摒弃了C++里难以理解的多继承、指针等概念,因此Java语言具有功能强大和简单易用两个特征。Java语言作为静态面向对象编程语言的代表,极好地实现了面向对象理论,允许程序员以优雅的思维方式进行复杂的编程 。

Java通过代码模拟高并发可以以最快的方式发现我们系统中潜在的线程安全性问题,此处使用Semaphore(信号量)和 CountDownLatch(闭锁)搭配ExecutorService(线程池)来进行模拟。

Java代码有哪些重要作用_编程语言_Java_Javascript_课课家
 
 

Java通过代码模拟高并发可以以最快的方式发现我们系统中潜在的线程安全性问题,此处使用Semaphore(信号量)和 CountDownLatch(闭锁)搭配ExecutorService(线程池)来进行模拟,主要介绍如下:

1、Semaphore

JDk 1.5之后会提供这个类

Semaphore是一种基于计数的信号量。它可以设定一个阈值,基于此,多个线程竞争获取许可信号,做完自己的申请后归还,超过阈值后,线程申请许可信号将会被阻塞。Semaphore可以用来构建一些对象池,资源池之类的,比如数据库连接池,我们也可以创建计数为1的Semaphore,将其作为一种类似互斥锁的机制,这也叫二元信号量,表示两种互斥状态。

2、CountDownLatch

JDK 1.5之后会提供这个类,

CountDownLatch这个类能够使一个线程等待其他线程完成各自的工作后再执行。例如,应用程序的主线程希望在负责启动框架服务的线程已经启动所有的框架服务之后再执行。

CountDownLatch是通过一个计数器来实现的,计数器的初始值为线程的数量。每当一个线程完成了自己的任务后,计数器的值就会减1。当计数器值到达0时,它表示所有的线程已经完成了任务,然后在闭锁上等待的线程就可以恢复执行任务。

如下图:

 

以上两个类可以搭配使用,达到模拟高并发的效果,以下使用代码的形式进行举例:

  1. package modules; 
  2. import java.util.concurrent.CountDownLatch; 
  3. import java.util.concurrent.ExecutorService; 
  4. import java.util.concurrent.Executors; 
  5. import java.util.concurrent.Semaphore; 
  6. public class CountExample { 
  7.  // 请求总数 
  8.  public static int clientTotal = 5000; 
  9.  // 同时并发执行的线程数 
  10.  public static int threadTotal = 200; 
  11.  public static int count = 0; 
  12.  public static void main(String[] args) throws Exception { 
  13.  ExecutorService executorService = Executors.newCachedThreadPool(); 
  14.  //信号量,此处用于控制并发的线程数 
  15.  final Semaphore semaphore = new Semaphore(threadTotal); 
  16.  //闭锁,可实现计数器递减 
  17.  final CountDownLatch countDownLatch = new CountDownLatch(clientTotal); 
  18.  for (int i = 0; i < clientTotal ; i++) { 
  19.  executorService.execute(() -> { 
  20.  try { 
  21.     //执行此方法用于获取执行许可,当总计未释放的许可数不超过200时, 
  22.     //允许通行,否则线程阻塞等待,直到获取到许可。 
  23.  semaphore.acquire(); 
  24.  add(); 
  25.  //释放许可 
  26.  semaphore.release(); 
  27.  } catch (Exception e) { 
  28.  //log.error("exception", e); 
  29.  e.printStackTrace(); 
  30.  } 
  31.  //闭锁减一 
  32.  countDownLatch.countDown(); 
  33.  }); 
  34.  } 
  35.  countDownLatch.await();//线程阻塞,直到闭锁值为0时,阻塞才释放,继续往下执行 
  36.  executorService.shutdown(); 
  37.  log.info("count:{}"count); 
  38.  } 
  39.  private static void add() { 
  40.  count++; 
  41.  } 

如上方法模拟5000次请求,同时最大200个并发操作,观察最后的结果,发现每次的结果都有差别,和预期不符,得出结果部分如下:

  1. 22:18:26.449 [main] INFO modules.CountExample - count:4997 
  2. 22:18:26.449 [main] INFO modules.CountExample - count:5000 
  3. 22:18:26.449 [main] INFO modules.CountExample - count:4995 
  4. 22:18:26.449 [main] INFO modules.CountExample - count:4998 

最后结论:add 方法 非线程安全

那如何保证add方法 线程安全,将add方法进行如下修改即可:

  1. private static void add() { 
  2.  count.incrementAndGet(); 

执行结果如下:

  1. 22:18:26.449 [main] INFO modules.CountExample - count:5000 
  2. 22:18:26.449 [main] INFO modules.CountExample - count:5000 
  3. 22:18:26.449 [main] INFO modules.CountExample - count:5000 
  4. 22:18:26.449 [main] INFO modules.CountExample - count:5000 
  5. 22:18:26.449 [main] INFO modules.CountExample - count:5000 
  6. 22:18:26.449 [main] INFO modules.CountExample - count:5000 
  7. 22:18:26.449 [main] INFO modules.CountExample - count:5000 
  8. 22:18:26.449 [main] INFO modules.CountExample - count:500
  9. Java具有简单性、面向对象、分布式、健壮性、安全性、平台独立与可移植性、多线程、动态性等特点  。Java可以编写桌面应用程序、Web应用程序、分布式系统和嵌入式系统应用程序等。
赞(18)
踩(0)
分享到:
华为认证网络工程师 HCIE直播课视频教程