从零学Java 多线程(基础)

Java 多线程(基础)

文章目录

  • Java 多线程(基础)
    • 1 多线程
      • 1.1 多任务
      • 1.2 多线程
      • 1.3 普通方法调用和多线程
    • 2 进程和线程
      • 2.1 什么是进程(Process)?
      • 2.2 什么是线程(Thread)?
      • 2.3 进程和线程的区别
    • 3 线程的实现
      • 3.1 线程的组成
      • 3.2 线程执行特点
      • 3.3 线程的创建
        • 3.3.1 继承Thread类
        • 3.3.2 实现Runnable接口

1 多线程

1.1 多任务

在这里插入图片描述

  • 现实生活中太多这样同时做多件事的例子了,看起来是多个任务同时都在做,其实本质上我们的大脑在同一时间依旧只能做一件事

1.2 多线程

在这里插入图片描述

  • 原来是一条路,慢慢地因为车变多了,造成了道路堵塞,使通行效率变低。
    为了提高使用的效率,能够充分利用道路,于是加了多个车道,从此,妈妈再也不用担心道路堵塞了

1.3 普通方法调用和多线程

在这里插入图片描述

2 进程和线程

2.1 什么是进程(Process)?

进: 正在进行; 程: 程序

程序:程序是指令和数据的有序集合,其本身没有任何运行的含义,是一个静态的概念;

进程:进程是执行程序的一次执行过程,它是一个动态的概念。是系统资源分配的单位

  • 程序是静止的,只有真正运行时的程序,才被称为进程。

  • 目前操作系统都是支持多进程,可以同时执行多个进程,通过进程ID区分。

2.2 什么是线程(Thread)?

概念: 线程,又称轻量级进程(Light Weight Process)。

  • 进程中的一个执行路径,同时也是CPU的基本调度单位。
  • 进程由多个线程组成,彼此间完成不同的工作,抢占式执行,称为多线程。
  • 当然一个进程中至少有一个线程,不然没有存在的意义

注意:很多多线程都是模拟出来的,真正的多线程是指有多个CUP,即多核,如服务器。如果是模拟出来的多线程,即在一个CPU的情况下,在同一时间点,CPU只能执行一个代码,只是因为切换的太快,所以便产生了同时执行的错觉;

2.3 进程和线程的区别

  • 进程是操作系统资源分配的基本单位,而线程是CPU的基本调度单位。
  • 一个程序运行后至少有一个进程。
  • 一个进程可以包含多个线程,但是至少需要有一个线程。
  • 进程间不能共享数据段地址,但同进程的线程之间可以。

3 线程的实现

3.1 线程的组成

任何一个线程都具有基本的组成部分

  • CPU时间片:操作系统(OS)会为每个线程分配执行时间。
  • 运行内存:
    • 堆内存:存储线程需使用的对象,多个线程可以共享堆中的对象。
    • 栈内存:存储线程需使用的局部变量,每个线程都拥有独立的栈。
  • 线程的逻辑代码

3.2 线程执行特点

  • 线程抢占式执行,结果随机性。
    • 效率高
    • 可防止单一线程长时间独占CPU
  • 在单核CPU中,宏观上同时执行,微观上顺序执行;多核CPU中,可以真正并发执行。

注意:线程调用不一定立即执行,由CPU调度执行

3.3 线程的创建

运行程序时, JVM会自动创建主线程(main), main线程执行main方法

3.3.1 继承Thread类
  • 继承Thread类,重写run()方法,调用start() 方法
  • 适合没有资源共享

MyThread01:

public class MyThread01 extends Thread {
    @Override
    public void run() {
        for (int i = 0; i < 100; i++) {
            System.out.println("子线程...."+i);
        }
    }
}

Test:

public class Test {
    public static void main(String[] args) {
        //创建子线程
        MyThread01 thread01 = new MyThread01();
        //启动子线程
        thread01.start();
        //thread01.run();调用方法
        //main线程执行for
        for (int i = 0; i < 100; i++) {
            System.out.println("main线程......"+i);
        }
    }
}

面试题: 一个线程能启动两次吗?

答: 不能, 启动多次会出现IllegalThreadStateException错误

获取线程ID和名称

方式 1 :

//方式1 this.getName() this.getId()
System.out.println(this.getName()+"  "+this.getId()+"子线程...."+i);

方式 2 :

//方式2 Thread.currentThread().getName()/getId()
System.out.println(
    Thread.currentThread().getName()+"..."
    +Thread.currentThread().getId()+"子线程...."+i
);

修改线程名称

方式 1 :

//方式1 thread.setName()
thread01.setName("线程-01");
thread02.setName("线程-02");

方式 2 :

//方式2 构造方法
//重写构造方法
public MyThread01(String name) {
    super(name);
}

//创建子线程时,修改
MyThread01 thread01 = new MyThread01("线程-01");
MyThread01 thread02 = new MyThread01("线程-02");

课堂案例

实现4个窗口各卖100张票?

内存分析:

在这里插入图片描述

SaleTicket:

package StageOne.day19.MyThread;

/**
 * @author 胡昊龙
 * @version 1.0
 * @description: TODO
 * @date 2024/1/12 15:22
 */
public class SaleTicket extends Thread{
    //总票数
    private int count = 100;
    //重写构造方法
    public SaleTicket(String name) {
        super(name);
    }

    @Override
    public void run() {
        while (count > 0) {
            System.out.println(Thread.currentThread().getName() +
                    "卖了第" + count + "张票");
            count--;
        }
    }
}

Test:

public class Test2 {
    public static void main(String[] args) {
        SaleTicket s1 = new SaleTicket("窗口1");
        SaleTicket s2 = new SaleTicket("窗口2");
        SaleTicket s3 = new SaleTicket("窗口3");
        SaleTicket s4 = new SaleTicket("窗口4");
        s1.start();
        s2.start();
        s3.start();
        s4.start();
    }
}
3.3.2 实现Runnable接口

实现Runnable接口

  • 适合有资源共享
  • 操作相同,共享资源类实现Runnable接口
  • 操作不同,操作类分开实现Runnable接口,构造方法传递共享资源

MyRunnable:

package StageOne.day19.MyRunnable;

/**
 * @author 胡昊龙
 * @version 1.0
 * @description: TODO
 * @date 2024/1/12 16:02
 */
public class MyRunnable implements Runnable{
    //重写run()方法
    @Override
    public void run() {
        for (int i = 0; i < 100; i++) {
            System.out.println(
                    Thread.currentThread().getName()+"子线程"+"..."+i
            );
        }
    }
}

Test:

public class Test {
    public static void main(String[] args) {
        //创建可运行对象
        MyRunnable runnable = new MyRunnable();
        //创建线程对象
        Thread t1 = new Thread(runnable,"线程-01");
        Thread t2 = new Thread(runnable,"线程-02");
        Thread t3 = new Thread(runnable,"线程-03");
        //启动线程
        t1.start();
        t2.start();
        t3.start();
        //main
        for (int i = 0; i < 100; i++) {
            System.out.println(
                    "主线程..............."+i
            );
        }
    }
}

课堂案例 1

实现4个窗口共卖100张票?

内存分析:

Ticket:

public class Ticket implements Runnable{
    private int count = 100;
    @Override
    public void run() {
        while (count > 0) {
            System.out.println(Thread.currentThread().getName() +
                    "卖了第" + count + "张票");
            count--;
        }
    }
}

Test:

public class Test01 {
    public static void main(String[] args) {
        //创建票对象(可运行对象)
        Ticket ticket = new Ticket();
        //创建线程对象
        Thread t1 = new Thread(ticket,"窗口1");
        Thread t2 = new Thread(ticket,"窗口2");
        Thread t3 = new Thread(ticket,"窗口3");
        Thread t4 = new Thread(ticket,"窗口4");
        //启动线程
        t1.start();
        t2.start();
        t3.start();
        t4.start();
    }
}

课堂案例 2

你和你女朋友共用一张银行卡,你向卡中存钱,你女朋友从卡中取钱,使用程序模拟过程?

BankCard:

public class BankCard {
    private double money;

    public double getMoney() {
        return money;
    }

    public void setMoney(double money) {
        this.money = money;
    }
}

Save:

public class Save implements Runnable{
    private BankCard card;

    public Save(BankCard card) {
        this.card = card;
    }

    @Override
    public void run() {
        for (int i = 0; i < 10; i++) {
            card.setMoney(card.getMoney()+1000);
            System.out.println(Thread.currentThread().getName()+"第"+i+"次存钱存了1000元," +
                    " 当前余额为" + card.getMoney());
        }
    }
}

Withdraw:

public class Withdraw implements Runnable{
    private BankCard card;

    public Withdraw(BankCard card) {
        this.card = card;
    }

    @Override
    public void run() {
        for (int i = 0; i < 10; i++) {
            if (card.getMoney()>=1000) {
                card.setMoney(card.getMoney()-1000);
                System.out.println(Thread.currentThread().getName()+"取走了1000元"
                +"当前余额为"+card.getMoney());
            } else {
                System.out.println("当前余额不足...");
                i--;//取款失败不计数,保证把钱取完
            }
        }
    }
}

Test:

public class Test {
    public static void main(String[] args) {
        //卡
        BankCard card = new BankCard();
        //存钱
        Save save = new Save(card);
        //取钱
        Withdraw withdraw = new Withdraw(card);
        //线程
        Thread t1 = new Thread(save,"小明");
        Thread t2 = new Thread(withdraw,"小红");
        t1.start();
        t2.start();
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/321917.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Kubernetes (十二) 存储——Volumes配置管理

一. 卷的概念 官方地址&#xff1a;卷 | Kuberneteshttps://v1-24.docs.kubernetes.io/zh-cn/docs/concepts/storage/volumes/ 二. 卷的类型及使用 …

OpenGauss源码分析-SQL引擎

所讨论文件大多位于src\common\backend\parser文件夹下 总流程 start_xact_command()&#xff1a;开始一个事务。pg_parse_query()&#xff1a;对查询语句进行词法和语法分析&#xff0c;生成一个或者多个初始的语法分析树。进入foreach (parsetree_item, parsetree_list)循环…

NLP论文阅读记录 - 2021 | WOS 使用分层多尺度抽象建模和动态内存进行抽象文本摘要

文章目录 前言0、论文摘要一、Introduction1.3本文贡献 二.前提三.本文方法四 实验效果4.1数据集4.2 对比模型4.3实施细节4.4评估指标4.5 实验结果4.6 细粒度分析 五 总结思考 前言 Abstractive Text Summarization with Hierarchical Multi-scale Abstraction Modeling and Dy…

ecmaScript-定义变量

简言 了解定义变量的三种方式&#xff1a;var、let、const。 正文 var var 语句 用于声明一个函数范围或全局范围的变量&#xff0c;并可将其初始化为一个值&#xff08;可选&#xff09;。 特点&#xff1a; 无论在何处声明变量&#xff0c;都会在执行任何代码之前进行变…

【Spring 篇】深入解析SpringMVC的组件魅力

SpringMVC&#xff0c;这个名字在Java Web开发者的耳边仿佛是一首动听的旋律&#xff0c;携着轻盈的氛围&#xff0c;带给我们一种愉悦的编程体验。但是&#xff0c;当我们深入探寻这个框架时&#xff0c;它的魅力远不止表面的简单&#xff0c;它由许多组件构成&#xff0c;每个…

【STM32】HAL库的STOP低功耗模式UART串口唤醒,第一个接收字节出错的问题(已解决)

【STM32】HAL库的STOP低功耗模式UART串口唤醒&#xff0c;第一个接收字节出错的问题&#xff08;已解决&#xff09; 文章目录 BUG复现调试代码推测原因及改进方案尝试中断时钟供电外设唤醒方式校验码硬件问题 切换到STOP0模式尝试结论和猜想解决方案附录&#xff1a;Cortex-M…

网站万词霸屏推广系统源码:实现关键词推广,轻松提高关键词排名,带完整的安装部署教程

现如今&#xff0c;互联网的快速发展&#xff0c;网站推广成为企业网络营销的重要手段。而关键词排名作为网站推广的关键因素&#xff0c;一直备受关注。罗峰给大家分享一款网站万词霸屏推广系统源码&#xff0c;该系统可实现关键词推广&#xff0c;有效提高关键词排名&#xf…

Go模板后端渲染时vue单页面冲突处理

go后端模版语法是通过 {{}} &#xff0c;vue也是通过双花括号来渲染的&#xff0c;如果使用go渲染vue的html页面的时候就会报错&#xff0c;因为分别不出来哪个是vue的&#xff0c;哪个是go的&#xff0c;既可以修改go的模板语法 template.New("output").Delims(&qu…

在线直径测量仪 使用范围广 透明、柔软、易形变…

在线直径测量仪&#xff0c;用于截面为圆形的产品的外径尺寸检测&#xff0c;并且由于其是光学检测设备&#xff0c;进行非接触式的尺寸检测&#xff0c;因此对其材质温度等都没有要求&#xff0c;对硬质的、柔软的、熔融的、透明的、高温的等各种易检测不易检测的轧材均可进行…

【C++】异常机制

异常 一、传统的处理错误的方式二、C异常概念三、异常的使用1. 异常的抛出和捕获&#xff08;1&#xff09;异常的抛出和匹配原则&#xff08;2&#xff09;在函数调用链中异常栈展开匹配原则 2. 异常的重新抛出3. 异常安全4. 异常规范 四、自定义异常体系五、C 标准库的异常体…

uniapp+vue3打包问题记录

**背景&#xff1a;**打包app出现问题&#xff0c;只显示底部导航的文字&#xff0c;其他一片空白 1. pages.json文件&#xff1a;tabBar中的iconPath图标格式不支持svg&#xff0c;只支持&#xff1a;png, jpg, jpeg的格式&#xff0c;当图片改为.png的时候可以正常显示 2. …

212. 单词搜索 II(字典树的另一种类型)

大致思路是&#xff1a; 根据words列表建立字典树&#xff0c;其中注意在单词末尾&#xff0c;将原来的isEnd变量换成存储这个单词的变量&#xff0c;方便存储到ans中&#xff0c;另外&#xff0c;字典树的字节点由原来的Trie数组变为hashmap&#xff0c;方便检索字母。 建立…

1.14寒假集训

A: 解题思路&#xff1a;按照题目意思模拟即可&#xff0c;只要不满足条件就输出“No”然后结束循环&#xff0c;否则最后输出“Yes”。 下面是c代码&#xff1a; #include<iostream> using namespace std; int main() { int n,arr[100000],index 0; cin >…

Vue中的v-model

聚沙成塔每天进步一点点 本文内容 ⭐ 专栏简介基本用法文本输入框复选框下拉框 原理解析文本输入框的原理复选框和下拉框的原理 ⭐ 写在最后 ⭐ 专栏简介 Vue学习之旅的奇妙世界 欢迎大家来到 Vue 技能树参考资料专栏&#xff01;创建这个专栏的初衷是为了帮助大家更好地应对 V…

CAN/CANFD数据记录仪汽车电子售后神器

CAN数据记录仪是一种用于采集和存储CAN总线数据的工具&#xff0c;广泛应用于汽车、轨道车辆、工业控制等大数据量且不易排查故障的系统中。它可以实时存储总线上的数据&#xff0c;方便后续的研究和分析。解决工程师售后难点。 在选择CAN数据记录仪时&#xff0c;需要根据实…

堆排序——高效解决TOP-K问题

. 个人主页&#xff1a;晓风飞 专栏&#xff1a;数据结构|Linux|C语言 路漫漫其修远兮&#xff0c;吾将上下而求索 文章目录 引言什么是堆&#xff1f;建堆堆排序&#xff1a;排序的最终结果 堆排序实现函数声明交换函数 Swap下沉调整 DnAdd堆排序函数 HeapSort主函数 文件中找…

一天吃透Java并发面试八股文

内容摘自我的学习网站&#xff1a;topjavaer.cn 分享50道Java并发高频面试题。 线程池 线程池&#xff1a;一个管理线程的池子。 为什么平时都是使用线程池创建线程&#xff0c;直接new一个线程不好吗&#xff1f; 嗯&#xff0c;手动创建线程有两个缺点 不受控风险频繁创…

通过离散点拟合曲线

文章目录 使用离散点拟合曲线参考代码路径:作者源码:测试代码效果图:k3k4k5 使用离散点拟合曲线 参考代码路径: https://www.bragitoff.com/2015/09/c-program-for-polynomial-fit-least-squares/ https://gist.github.com/Thileban/272a67f2affdc14a5f69ad3220e5c24b https:/…

PID横向控制和仿真实现

文章目录 1. PID介绍2. PID横向控制原理3. 算法和仿真实现 1. PID介绍 PID是一种常见的控制算法&#xff0c;全称为Proportional-Integral-Derivative&#xff0c;即比例-积分-微分控制器。PID控制器是一种线性控制器&#xff0c;它将设定值与实际值进行比较&#xff0c;根据误…

基于51单片机的模拟量输入输出通道实验

实验一 模拟量输入输出通道实验&#xff08;C51&#xff09; 一、实验目的&#xff1a; 1、了解A/D、D/A转换的基本原理。 2、了解A/D转换芯片ADC0809、D/A转换芯片DAC0832的性能及编程方法。 3、掌握过程通道中A/D转换与D/A转换与计算机的接口方法。 4、了解计算机如何进…