【JavaEE初阶】多线程(1)

2024-09-07 14:36
文章标签 java 多线程 初阶 ee

本文主要是介绍【JavaEE初阶】多线程(1),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

欢迎关注个人主页:逸狼


创造不易,可以点点赞吗~

如有错误,欢迎指出~



目录

并发编程 

线程 与 进程

创建线程

写法1 

写法2

写法3

写法4

写法5

Thread类的常见构造方法

前台/后台 线程


并发编程 

并发编程: 通过写特殊的代码 把多个cpu的核心都利用起来 

多进程编程就是一种典型的并发编程 ,但多进程编程最大的问题是 进程太'重'了(创建进程/销毁进程,时间/空间开销比较大,一旦需求场景需要频繁的创建销毁进程 ,开销就非常明显了(最典型的就是 服务器开发(针对每个发送请求的客户端,都需要创建一个单独的进程,由这个进程负责给客户端提供服务)))

线程 与 进程

为解决进程开销比较大的问题,发明了'线程'(可以理解为更轻量化的进程,也能解决 并发编程的问题,创建/销毁的开销要比进程更低,因此多线程的编程就成为了当下最主流的并发编程方式(通过多线程可以充分利用好多核cpu))

 进程在系统中是通过PCB这样的 数据结构来 描述 的,通过链表的形式来 组织 的, 线程同样也是通过PCB来描述的(一个进程,是一组PCB,一个线程,是一个PCB,存在包含关系:一个进程中,可以包含多个线程)

  • 进程是系统  "资源分配" 的基本单位
  • 线程是系统  "调度执行" 的基本单位

一个可执行程序,运行的时候,操作系统就会创建进程,给这个程序分配各种系统资源(cpu,内存,硬盘,网络带宽....),同时也会在这个进程中创建一个或多个 线程,这些线程再去cpu上执行.

  • 一个进程要么包含一个,要么包含多个 线程,不能没有
  • 同一个进程中的这些线程 ,共用同一份系统资源

线程比进程 更轻量化 主要在于 创建线程省去了"分配资源" 的过程,销毁线程 也省去了'释放资源'的过程 (一旦创建进程,同时也会创建第一个线程--->负责分配资源 ,创建后面的线程就不必分配资源了)  

随着线程数目的增加,每个线程要负责完成的工作量就变少了,这些线程同时开始工作,总的消耗时间会进一步减少, 但是如果线程的数目太多,超出了 cpu核心数目,此时就无法再完成所有线程的"并发"执行.势必会存在严重的 '竞争'

  • 多个线程之间 ,可能会相互影响,线程安全问题,一个线程一旦抛出异常,也可能会把其他线程也一起带走.
  • 多个进程之间, 一般不会相互影响,一个进程崩溃了,不会影响其他进程(进程的 隔离性)

创建线程

class MyThread extends Thread{//Thread类可以直接使用,不用导入任何包//重写run方法public void run(){System.out.println("hello thread");}
}public class Demo1 {public static void main(String[] args) {MyThread t =new MyThread();//创建线程t.start();//调用start就会再进程内部创建一个新的线程,新的线程就会执行刚才run里的代码}
}

注意:上述代码中,run方法并没有手动去调用, 但这个方法最终执行了,此时这样的方法称为'回调函数' 

这个代码运行起来 是一个进程,但这个进程中包含两个进程,主线程 和新线程 ,这两个线程并发/ 并行的在cpu上执行

写法1 

继承Tread ,重写run,通过start 实例启动线程

class MyThread extends Thread{//Thread类可以直接使用,不用导入任何包//重写run方法public void run(){while(true){System.out.println("hello thread");try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}}}
}public class Demo1 {public static void main(String[] args)  {MyThread t =new MyThread();//创建线程t.start();//调用start就会再进程内部创建一个新的线程,新的线程就会执行刚才run里的代码while(true){System.out.println("hello main");try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}}}
}

写法2

 实现Runnable ,重写run,通过Thread的实例,把Runnable的实例传进去,再调用start

class MyRunnable implements Runnable{@Overridepublic void run() {//描述线程要完成的逻辑是啥while(true){System.out.println("hello thread");try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}}}
}
public class Demo2 {public static void main(String[] args) throws InterruptedException {MyRunnable runnable= new MyRunnable();Thread t =new Thread(runnable);t.start();while(true){System.out.println("hello main");Thread.sleep(1000);}}
}

Runnable 是用来描述 要执行的任务是什么,通过 Thread创建线程,线程要执行的任务是通过Runnable来描述的,而不是通过Thread自己来描述的, 这里的runnable只是一个任务,并不是和'线程'这样的概念强相关,后续执行这个任务的载体,可以是线程,也可以是其他....(有利于代码的 解耦合)

写法3

继承Thread,重写run ,通过 匿名内部类来实现,本质上是写法1

package thread;public class Demo3 {public static void main(String[] args) throws InterruptedException {Thread t = new Thread() {@Overridepublic void run() {while (true) {System.out.println("hello thread");try {Thread.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}}}};t.start();while (true) {System.out.println("hello main");Thread.sleep(1000);}}
}

写法4

通过匿名内部类实现 本质上是写法2

public class Demo4 {public static void main(String[] args) throws InterruptedException {Thread t= new Thread(new Runnable() {@Overridepublic void run() {while(true){System.out.println("hello thread");try {Thread.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}}}});t.start();while(true){System.out.println("hello main");Thread.sleep(1000);}}
}

写法5

基于 lambda 表达式来创建线程(很多时候,写"匿名内部类"的目的不是为了写"类"而是为了写类里面的方法,lambda就是直接能够表示要写的run方法)

public class Demo5 {public static void main(String[] args) throws InterruptedException {Thread t=new Thread(()->{while(true){System.out.println("hello thread");try {Thread.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}}});t.start();while(true){System.out.println("hello main");Thread.sleep(1000);}}
}

Thread类的常见构造方法

ThreadGroup线程组 :把多个线程放到一组中,方便统一设置线程的一些属性, 现在很少使用线程组,线程的相关属性用的也不多,现在更多的会使用 线程池.

属性中的 ID和 状态是JVM自动分配的,不能手动设置

线程的状态有 阻塞和就绪

设置不同的优先级 会影响到系统的调度(基于'统计'规则的影响,肉眼很难观察到)

前台/后台 线程

  • 前台线程:  某个线程在执行的过程中,够阻止进程的结束
  • 后台线程:  某个线程在执行的过程中,不能阻止进程的结束(进程结束时会带走正在执行的后台线程)

一个进程中,前台线程可以有多个(创建的线程默认就是前台的),必须所有前台线程都结束,进程才结束

代码中常见的new Thread 对象,生命周期和内核中实际的线程 是不一定一样的,可能会出现Thread对象仍然存在,但是内核中的线程不存在的这样的情况(但不会出现Thread对象不存在,线程还存在 的这次情况)

调用start 之前,内核中还没有创建线程

线程的run执行完毕,内核的线程就无了,但Thread对象仍然存在

这篇关于【JavaEE初阶】多线程(1)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1145327

相关文章

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

SpringBoot基于注解实现数据库字段回填的完整方案

《SpringBoot基于注解实现数据库字段回填的完整方案》这篇文章主要为大家详细介绍了SpringBoot如何基于注解实现数据库字段回填的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解... 目录数据库表pom.XMLRelationFieldRelationFieldMapping基础的一些代

一篇文章彻底搞懂macOS如何决定java环境

《一篇文章彻底搞懂macOS如何决定java环境》MacOS作为一个功能强大的操作系统,为开发者提供了丰富的开发工具和框架,下面:本文主要介绍macOS如何决定java环境的相关资料,文中通过代码... 目录方法一:使用 which命令方法二:使用 Java_home工具(Apple 官方推荐)那问题来了,

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

Java中的.close()举例详解

《Java中的.close()举例详解》.close()方法只适用于通过window.open()打开的弹出窗口,对于浏览器的主窗口,如果没有得到用户允许是不能关闭的,:本文主要介绍Java中的.... 目录当你遇到以下三种情况时,一定要记得使用 .close():用法作用举例如何判断代码中的 input

Spring Gateway动态路由实现方案

《SpringGateway动态路由实现方案》本文主要介绍了SpringGateway动态路由实现方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随... 目录前沿何为路由RouteDefinitionRouteLocator工作流程动态路由实现尾巴前沿S