Java通过卖票理解多线程

发布时间 - 2026-01-11 03:12:38    点击率:

        以卖票的例子来介绍多线程和资源共享,下面我们来看看为什么要用卖票作为例子。

  卖票是包含一系列动作的过程,有各种操作,例如查询票、收钱、数钱、出票等,其中有一个操作是每次卖掉一张,就将总的票数减去1。有10张票,如果一个人卖票,先做查票、收钱、数钱等各种操作,再将总的票数减去1,效率很低。如果多个人卖票,每个人都是做同样的操作,数钱、检查钱,最后将总的票数减1,这样效率高。但是有一个问题,如果出现两个人同时将总的票数减掉了1,例如,A、B两个人同时读取到票的总数是10,A从中减去1,同时B也从中减去1,总数显示是9,其实票只有8张。导致数据错误。

  按照正常逻辑,同一时刻只允许一个人来从总票数中减去1,A读取总票数,再减去1的过程中,B必须等待,等A操作完了,B才能进行。其实票就是共享资源,一次只能由一个人访问。这里就要用到同步机制,即锁机制,使用关键词synchronized将读取总的票数,并减去1的操作锁定,使得一次只能由一个人访问。每个售票员就是一个线程,多个售票员进行同一项卖票任务。

  synchronized原理是,执行synchronized部分代码的时候必须需要对象锁,而一个对象只有一个锁,只有执行完synchronized里面的代码后释放锁,其他线程才可以获得锁,那么就保证了同一时刻只有一个线程访问synchronized里面的代码。使得资源共享的关键是,只有一个实例,synchronized使用的是同一把锁,用实例的锁或者定义一个实例。这就需要使用实现Runnable接口的方式,实现多线程,这样传入的是一个实例。继承Thread的方式,传入的是多个实例,每个实例都有一个锁,那就无法实现控制。

具体代码如下:

package com.test;
public class SaleTickets implements Runnable
{
  private int ticketCount = 10;// 总的票数,这个是共享资源,多个线程都会访问
  Object mutex = new Object();// 锁,自己定义的,或者使用实例的锁
  /**
   * 卖票
   */
  public void sellTicket()
  {
    synchronized (mutex)// 当操作的是共享数据时,
            // 用同步代码块进行包围起来,执行里面的代码需要mutex的锁,但是mutex只有一个锁。这样在执行时,只能有一个线程执行同步代码块里面的内容
    {
      if (ticketCount > 0)
      {
        ticketCount--;
        System.out.println(Thread.currentThread().getName()
            + "正在卖票,还剩" + ticketCount + "张票");
      }
      else
      {
        System.out.println("票已经卖完!");
        return;
      }
    }
  }
  public void run()
  {
    while (ticketCount > 0)// 循环是指线程不停的去卖票
    {
      sellTicket();
      /**
       * 在同步代码块里面睡觉,和不睡效果是一样 的,作用只是自已不执行,也不让线程执行。sleep不释放锁,抱着锁睡觉。其他线程拿不到锁,也不能执行同步代码。wait()可以释放锁
       * 所以把睡觉放到同步代码块的外面,这样卖完一张票就睡一会,让其他线程再卖,这样所有的线程都可以卖票
       */
      try
      {
        Thread.sleep(100);
      }
      catch (InterruptedException e)
      {
        e.printStackTrace();
      }
    }
  }
}

下面是调用:

package com.test;
public class TicketMain
{
  public static void main(String[] args)
  {
    SaleTickets runTicekt = new SaleTickets();//只定义了一个实例,这就只有一个Object mutex = new Object();即一个锁。
    Thread th1 = new Thread(runTicekt, "窗口1");//每个线程等其他线程释放该锁后,才能执行
    Thread th2 = new Thread(runTicekt, "窗口2");
    Thread th3 = new Thread(runTicekt, "窗口3");
    Thread th4 = new Thread(runTicekt, "窗口4");
    th1.start();
    th2.start();
    th3.start();
    th4.start();
  }
}

输出:

窗口1正在卖票,还剩9张票
窗口4正在卖票,还剩8张票
窗口3正在卖票,还剩7张票
窗口2正在卖票,还剩6张票
窗口3正在卖票,还剩5张票
窗口2正在卖票,还剩4张票
窗口1正在卖票,还剩3张票
窗口4正在卖票,还剩2张票
窗口3正在卖票,还剩1张票
窗口1正在卖票,还剩0张票
票已经卖完!

  这是多个线程,完成同一个任务的情况,即多个线程调用同一个实例,通过实现Runable接口实现。多个线程可以异步的做这个任务中其他事情,但是对于共享资源的访问只能以同步的方式操作,即一个接一个访问共享资源,其他资源可以并行访问。

  另一种实现多线程的方式是继承Thread,调用的时候需要传递多个实例,这是多个线程,多个实例的情况,每个线程独立处理一个实例,各个线程不能实现资源共享。

总结

以上是本文关于通过卖票实例理解多线程的全部内容,希望对大家有所帮助。


# java  # 多线程卖票  # 多线程的理解  # Java利用future及时获取多线程运行结果  # Java多线程阻塞与唤醒代码示例  # Java多线程ForkJoinPool实例详解  # 浅谈Java的两种多线程实现方式  # Java基于Socket实现简单的多线程回显服务器功能示例  # java 多线程Thread与runnable的区别  # 简单了解Java多线程实现的四种方式  # 关键词  # 多个  # 还剩  # 的是  # 只有一个  # 多线程  # 这是  # 资源共享  # 卖完  # 这就  # 数钱  # 收钱  # 有一个  # 都是  # 总票数  # 都有  # 那就  # 每个人  # 是指  # 抱着 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 南京网站制作费用,南京远驱官方网站?  如何选择PHP开源工具快速搭建网站?  网站建设整体流程解析,建站其实很容易!  如何在橙子建站中快速调整背景颜色?  在线制作视频的网站有哪些,电脑如何制作视频短片?  如何在万网ECS上快速搭建专属网站?  Laravel任务队列怎么用_Laravel Queues异步处理任务提升应用性能  Laravel如何实现RSS订阅源功能_Laravel动态生成网站XML格式订阅内容【教程】  Microsoft Edge如何解决网页加载问题 Edge浏览器加载问题修复  移动端手机网站制作软件,掌上时代,移动端网站的谷歌SEO该如何做?  中山网站推广排名,中山信息港登录入口?  ,交易猫的商品怎么发布到网站上去?  Laravel如何配置任务调度?(Cron Job示例)  高端网站建设与定制开发一站式解决方案 中企动力  大学网站设计制作软件有哪些,如何将网站制作成自己app?  如何快速使用云服务器搭建个人网站?  如何用PHP快速搭建高效网站?分步指南  JavaScript如何实现继承_有哪些常用方法  如何快速重置建站主机并恢复默认配置?  Python自然语言搜索引擎项目教程_倒排索引查询优化案例  ChatGPT怎么生成Excel公式_ChatGPT公式生成方法【指南】  如何在IIS中配置站点IP、端口及主机头?  javascript中闭包概念与用法深入理解  Android GridView 滑动条设置一直显示状态(推荐)  Win11怎么关闭透明效果_Windows11辅助功能视觉效果设置  如何用低价快速搭建高质量网站?  Laravel怎么自定义错误页面_Laravel修改404和500页面模板  如何快速启动建站代理加盟业务?  如何用西部建站助手快速创建专业网站?  Laravel如何集成Inertia.js与Vue/React?(安装配置)  Laravel如何监控和管理失败的队列任务_Laravel失败任务处理与监控  SQL查询语句优化的实用方法总结  Laravel如何使用Collections进行数据处理?(实用方法示例)  如何在宝塔面板创建新站点?  深圳网站制作平台,深圳市做网站好的公司有哪些?  Laravel中的Facade(门面)到底是什么原理  中山网站制作网页,中山新生登记系统登记流程?  android nfc常用标签读取总结  Android自定义控件实现温度旋转按钮效果  C++用Dijkstra(迪杰斯特拉)算法求最短路径  Python函数文档自动校验_规范解析【教程】  轻松掌握MySQL函数中的last_insert_id()  高防网站服务器:DDoS防御与BGP线路的AI智能防护方案  Laravel怎么配置S3云存储驱动_Laravel集成阿里云OSS或AWS S3存储桶【教程】  广州网站制作公司哪家好一点,广州欧莱雅百库网络科技有限公司官网?  Laravel如何创建和注册中间件_Laravel中间件编写与应用流程  Laravel怎么实现一对多关联查询_Laravel Eloquent模型关系定义与预加载【实战】  Laravel怎么进行浏览器测试_Laravel Dusk自动化浏览器测试入门  零服务器AI建站解决方案:快速部署与云端平台低成本实践  Laravel如何使用Blade模板引擎?(完整语法和示例)