JAVA进阶篇——线程+实例演示(附商品销售实例源码)

x33g5p2x  于2021-11-22 转载在 Java  
字(4.7k)|赞(0)|评价(0)|浏览(602)

何为线程?

简单来说,打开你电脑的任务管理器,在进程选项卡中,可以看到你电脑上一个一个正在运行的程序,我们可以把这里的每一个程序理解为一个进程,而进程是受系统管理的基本单元。

那么啥是线程呢?我们可以理解为一个进程里面包含了很多个线程。而一个线程可以当作这个进程的一个子任务。就比如说你的微信,在你刷朋友圈的同时,也可以接收消息,这就是两个线程在运行。

我们为什么要用线程呢?我们来看看下面这张图。

那么我们用了线程以后呢?

由图可知,任务二并不需要按照我们平常所写地Java代码从上到下执行,需要等待10S,等任务一执行完毕以后才执行任务二,线程带来的好处显而易见,能够极大地提升程序地运行效率。简单来说,线程是异步的。

实现多线程编程的方式主要有两种,一种是继承(extends)Thread类,一种是实现Runnable接口,h还有一种是使用FutureTask的方式创建。

这里举一个例子,如果你是商家,有自己的销售平台,当库存量为n时,是不是每当有人下单m件商品时,库存量就得减少m呀。那么在这样的情况下,用线程编程是怎么实现的呢?

线程Thread可以调用currentThread()方法的getName()获取到当前线程的名字。接下来我们来看看这段代码。

  1. package com.gantiexia.threadTest;
  2. /** * @author GanTieXia * @date 2021/9/17 20:52 */
  3. public class ShareVariableThread extends Thread{
  4. public int count = 5;
  5. @Override
  6. public void run(){
  7. // 获取到线程拿到count时的值并保存下载做输出使用
  8. int beforeCount = count;
  9. // 操作count
  10. count--;
  11. // 输出
  12. System.out.println("线程操作前count的值:" + beforeCount + "。线程" + Thread.currentThread().getName() + "操作后的值:" + count);
  13. }
  14. public static void main(String[] args) {
  15. // 创建五个线程
  16. Thread a = new ShareVariableThread();
  17. Thread b = new ShareVariableThread();
  18. Thread c = new ShareVariableThread();
  19. Thread d = new ShareVariableThread();
  20. Thread e = new ShareVariableThread();
  21. // 启动线程
  22. a.start();
  23. b.start();
  24. c.start();
  25. d.start();
  26. e.start();
  27. }
  28. }

相信大家在看这部分的代码时,肯定会预想到答案,count肯定会一次递减输出,但是真的是这样吗,我们运行看一看。

如图,每一个线程在被创建的时候,其实都是new了一个本类的对象,也就是说,count在每个线程中都出现了一次,每个线程都在自己减少自己的count的值,并没有做到变量共享的情况。

于是我们把代码改成这样:

  1. package com.gantiexia.threadTest;
  2. import com.gantiexia.myhashmap.ThreadAa;
  3. /** * @author GanTieXia * @date 2021/9/17 20:51 */
  4. public class PrivateVariableThread extends Thread{
  5. private int count = 5;
  6. @Override
  7. public void run(){
  8. // 获取到线程拿到count时的值并保存下载做输出使用
  9. int beforeCount = count;
  10. // 操作count
  11. count--;
  12. // 输出
  13. System.out.println("线程操作前count的值:" + beforeCount + "。线程" + Thread.currentThread().getName() + "操作后的值:" + count);
  14. }
  15. public static void main(String[] args) {
  16. PrivateVariableThread thread = new PrivateVariableThread();
  17. // 创建五个线程
  18. Thread a = new Thread(thread,"A");
  19. Thread b = new Thread(thread,"B");
  20. Thread c = new Thread(thread,"C");
  21. Thread d = new Thread(thread,"D");
  22. Thread e = new Thread(thread,"E");
  23. // 启动线程
  24. a.start();
  25. b.start();
  26. c.start();
  27. d.start();
  28. e.start();
  29. }
  30. }

看到这里,那这样总是在共享变量了吧?答案会不会如我们锁期待的一样呢?我们来看看。

可以看到,得到的结果并不是我们想要的结果,线程A和线程B操作后的count值都为3了,虽然每次线程锁拿到的值都是正确的,但是输出的时候却不正确了,这就造成了非线程不安全的问题,这是为什么呢?

在JVM中,count–的操作步骤要分为这几步:
①获取到count的值。
②计算count–的值。
③对count重新赋值。

在这三个步骤执行的过程中,如果有多个线程同时访问,那么一定会出现非线程安全的问题。这个时候,我们就要在run()方法上加上我们的synchronized锁。

  1. @Override
  2. synchronized public void run(){
  3. // 获取到线程拿到count时的值并保存下载做输出使用
  4. int beforeCount = count;
  5. // 操作count
  6. count--;
  7. // 输出
  8. System.out.println("线程操作前count的值:" + beforeCount + "。线程" + Thread.currentThread().getName() + "操作后的值:" + count);
  9. }

这样我们的运行结果又是怎样的呢?

那么此刻,我们就得到了我们想要的结果了。

实例演示

再回到我们刚才上面所讲到的问题:

这里举一个例子,如果你是商家,有自己的销售平台,当库存量为n时,是不是每当有人下单m件商品时,库存量就得减少m呀。那么在这样的情况下,用线程编程是怎么实现的呢?

我们来用线程简单模拟一下这个情景:

  1. package com.gantiexia.threadTest;
  2. /** * @author GanTieXia * @date 2021/9/17 22:48 */
  3. public class SaleThread extends Thread{
  4. // 商品初始库存量为100件
  5. public int INVENTORY_COUNT = 100;
  6. @Override
  7. synchronized public void run(){
  8. // 获取到线程拿到count时的值并保存下来做输出使用
  9. int beforeCount = INVENTORY_COUNT;
  10. // 假设每个人的下单数量为[1,10]件之间
  11. int saleCount = (int) ((Math.random()*10) + 1);
  12. // 减少库存量
  13. INVENTORY_COUNT = INVENTORY_COUNT - saleCount;
  14. // 库存量大于0的时候才可以继续卖出
  15. if(INVENTORY_COUNT >= 0){
  16. // 输出
  17. System.out.println("剩余库存量:" + beforeCount + "件。顾客" + Thread.currentThread().getName() + "购买"+ saleCount +"件商品。剩余库存量:" + INVENTORY_COUNT + "件。");
  18. } else if(INVENTORY_COUNT < 0) {
  19. // 当库存不够时
  20. System.out.println("剩余库存量:" + beforeCount + "件,顾客需求量:"+ saleCount + "件。库存不足,请尽快补充库存!");
  21. // 销售不成功,库存量为这个顾客操作时的剩余库存量
  22. INVENTORY_COUNT = beforeCount;
  23. }
  24. }
  25. public static void main(String[] args) {
  26. SaleThread thread = new SaleThread();
  27. // 假设此处有26个顾客产生了购买行为
  28. for(int i=0;i<26;i++) {
  29. // 顾客姓名我们用大写英文字母表示
  30. char customer = (char)('A'+i);
  31. // 转换成字符转用来创建线程时设置顾客姓名
  32. String customerName = String.valueOf(customer);
  33. // 开始发生购买行为
  34. Thread purchaseBehavior = new Thread(thread,customerName);
  35. // 发生购买行为
  36. purchaseBehavior.start();
  37. }
  38. }
  39. }

如上代码运行后,注意每次运行的结果都是不一样的,着重看一下结尾的输出。 运行后的输出为:

  1. 剩余库存量:100件。顾客A购买4件商品。剩余库存量:96件。
  2. 剩余库存量:96件。顾客D购买2件商品。剩余库存量:94件。
  3. 剩余库存量:94件。顾客E购买10件商品。剩余库存量:84件。
  4. 剩余库存量:84件。顾客R购买10件商品。剩余库存量:74件。
  5. 剩余库存量:74件。顾客B购买5件商品。剩余库存量:69件。
  6. 剩余库存量:69件。顾客F购买7件商品。剩余库存量:62件。
  7. 剩余库存量:62件。顾客I购买6件商品。剩余库存量:56件。
  8. 剩余库存量:56件。顾客K购买5件商品。剩余库存量:51件。
  9. 剩余库存量:51件。顾客J购买4件商品。剩余库存量:47件。
  10. 剩余库存量:47件。顾客L购买6件商品。剩余库存量:41件。
  11. 剩余库存量:41件。顾客M购买2件商品。剩余库存量:39件。
  12. 剩余库存量:39件。顾客N购买2件商品。剩余库存量:37件。
  13. 剩余库存量:37件。顾客O购买10件商品。剩余库存量:27件。
  14. 剩余库存量:27件。顾客P购买6件商品。剩余库存量:21件。
  15. 剩余库存量:21件。顾客S购买5件商品。剩余库存量:16件。
  16. 剩余库存量:16件。顾客U购买4件商品。剩余库存量:12件。
  17. 剩余库存量:12件。顾客T购买7件商品。剩余库存量:5件。
  18. 剩余库存量:5件,顾客需求量:8件。库存不足,请尽快补充库存!
  19. 剩余库存量:5件,顾客需求量:8件。库存不足,请尽快补充库存!
  20. 剩余库存量:5件。顾客Y购买5件商品。剩余库存量:0件。
  21. 剩余库存量:0件,顾客需求量:1件。库存不足,请尽快补充库存!
  22. 剩余库存量:0件,顾客需求量:9件。库存不足,请尽快补充库存!
  23. 剩余库存量:0件,顾客需求量:7件。库存不足,请尽快补充库存!
  24. 剩余库存量:0件,顾客需求量:9件。库存不足,请尽快补充库存!
  25. 剩余库存量:0件,顾客需求量:8件。库存不足,请尽快补充库存!
  26. 剩余库存量:0件,顾客需求量:4件。库存不足,请尽快补充库存!

相关文章

最新文章

更多