Java中的多线程的创建方式
首先理清几个基本概念:
程序:为完成特定任务,用某种语言编写的一组指令的集合。即一段静态的代码(还没运行起来)
进程:是程序的一次执行过程,也就是说程序运行起来了,加载到了内存中,并占用了cpu的资源。这是一个动态的过程:有自身的产生、存在和消亡的过程,这也是进程的生命周期。
线程:进程可进一步细化为线程,是一个程序内部的执行路径。
若一个进程同一时间并行执行多个线程,那么这个进程就是支持多线程的。
线程是cpu调度和执行的单位,每个线程拥有独立的运行栈和程序计数器(pc),线程切换的开销小。
多线程:是指程序当中包含多个执行单元,即在一个程序中可以同时运行多个不同的线程来执行不同的任务,也就是说允许单个线程创建多个并行执行的线程来完成各自的任务。
Java线程的分类:用户线程和守护线程
java的gc()垃圾回收线程就是一个守护线程
守护线程是用来服务用户线程的,通过start()方法前调用thread.serDaemon(True)可以把一个用户线程变成守护线程
并行和并发:
并行:多个cpu同时执行多个任务
并发:一个cpu同时执行多个任务
多线程的优点
提高应用程序的相应。
提高计算机cpu的利用率
改成程序结构。将复杂的进程分成多个线程,独立运行。
线程的创建与启动
多线程实现的原理
java的jvm允许程序运行多个线程,多线程可以通过Java中的java.lang.Thread类来体现
Thread类的特性
每个线程都是通过每个特定的Thread对象的run()方法来完成操作,经常把run()方法的主体称为线程体
通过Thread方法的start()方法来启动这个线程,而非调用run()
多线程的创建,方式一:继承于Thread类
1.创建一个继承于Thread类的子类
2.重写Thread类的run()方法
3.创建Thread类的子类的对象()
4.通过此对象调用start()来启动一个线程
代码实现:多线程执行同一段代码
package com.Thread.demo1; public class demo1 extends Thread{
@Override
public void run(){
for(int i = 0;i<100;i++){
if(i % 2 == 0){
System.out.println(Thread.currentThread().getName() + ":\t" + i);
}
}
}
public static void main(String[] args){
demo1 test1 = new demo1();
test1.start();
demo1 test2 = new demo1();
test2.start(); new demo1().start();
System.out.println("主线程");
}
}
多线程运行多代码:
currentThread()可以获取当前线程的引用,一般都是在没有线程对象又需要获得线程信息时通过 Thread.currentThread()获取当前代码段所在线程的引用。
public class demo2 {
public static void main(String[] args){
new Threadtest1().start();
new Threadtest2().start();
}
}
class Threadtest1 extends Thread{
@Override
public void run(){
for(int i = 0;i<100;i++){
if(i%2==0){
System.out.println(Thread.currentThread() + ":\t" + i);
}
}
}
}
class Threadtest2 extends Thread{
@Override
public void run(){
for(int i = 0;i<100;i++){
if(i % 2 !=0){
System.out.println(Thread.currentThread() + ":\t" + i);
}
}
}
}
运行结果:
多线程的创建方式二:实现runnable接口
1.创建一个实现runnable接口的类
2.实现类去实现runnable接口中的抽象方法:run()
3.创建实现类的对象
4.将此对象作为参数传到Thread类的构造器中,创建Thread类的对象
5.通过Thread类的对象调用start()方法
public class demo3 {
public static void main(String[] args){
RunnableTest runnableTest = new RunnableTest();
Thread t1 = new Thread(runnableTest);
t1.start();
Thread t2 = new Thread(runnableTest);
t2.start();
}
}
class RunnableTest implements Runnable{
@Override
public void run(){
for(int i = 0;i < 100;i++){
if(i % 2 == 0){
System.out.println(Thread.currentThread().getName() + " " + Thread.currentThread().getId() + ":\t" + i);
}
}
}
}
比较创建线程的两种方式
Java中只允许单进程,以children类来说,这个类如果存在父类(Person)的话,就不可以继承Thread类了,但是一个类可以实现多个接口,因此实现的方式没有类的单继承性的限制,用实现runnable接口的方式创建此类线程将更加实用!
实现runnable接口的方式天然具有共享数据的特性(不用static)
- 因为继承Thread的实现方式,需要创建多个子类的对象来进行多线程,如果子类中有变量A,而不使用static约束变量的话,每个子类的对象都会有自己独立的变量A,只有static约束A后,子类的对象才共享变量A。而实现Runnable接口的方式,只需要创建一个实现类的对象,要将这个对象传入Thread类并创建多个Thread类的对象来完成多线程,而这多个Thread类对象实际上就是调用一个实现类对象而已。实现的方式更适合来处理多个线程有共享数据的情况。
联系:Thread类中也实现了Runnable接口
相同点两种方式都需要重写run()方法,线程的执行逻辑都在run()方法中
多线程的创建方式,方式三:实现Callable接口
1.相比run()方法,可以有返回值
2.方法可以抛出异常
3.支持泛型的返回值
4.需要借助FutureTask类,比如获取返回结果
public class demo4 {
public static void main(String[] args){
NumSum numSum = new NumSum();
FutureTask<Integer> future = new FutureTask<Integer>(numSum);
new Thread(future).start();
try{
Integer sum = future.get();
System.out.println("总和为:" + sum); }catch (InterruptedException e){
e.printStackTrace();
}catch (ExecutionException e){
e.printStackTrace();
} }
} class NumSum implements Callable<Integer>{
@Override
public Integer call() throws Exception{
int sum = 0;
for (int i = 0;i<100;i++){
if(i%2==0){
sum+=i;
}
}
return sum;
}
}
Thread类中的常用方法
start()启动当前线程,调用当前线程的run()方法
run()通常需要重写Thread类中的此方法,将创建的线程要执行的操作声明在此方法中
currentThread()静态方法,返回当前代码执行的线程
getName()获取当前线程的名字
serName()设置当前线程的名字
yield()释放当前cpu的执行权
join()在线程a中调用线程b的join(), 此时线程a进入阻塞状态, 知道线程b完全执行完以后, 线程a才结束阻塞状态
stop()已过时,当执行此方法,强制结束当前线程
sleep(long miltime)让线程睡眠指定的毫秒数,在指定时间内,线城市阻塞状态
isAlive()判断当前线程是否存活
Java中的多线程的创建方式的更多相关文章
- 牛客网Java刷题知识点之什么是进程、什么是线程、什么是多线程、多线程的好处和弊端、多线程的创建方式、JVM中的多线程解析、多线程运行图解
不多说,直接上干货! 什么是进程? 正在进行中的程序(直译). 什么是线程? 就是进程中一个负责程序执行的控制单元(执行路径). 见 牛客网Java刷题知识点之进程和线程的区别 什么是多线程? 一个进 ...
- java中的多线程
什么是多线程? 首先得知道什么是线程? 线程是一组指令的集合,或者是程序的特殊段,它可以在程序里独立执行.也可以把它理解为代码运行的上下文.所以线程基本上是轻量级的进程,它负责在单个程序里执行多任务. ...
- 【转】Java中的多线程学习大总结
多线程作为Java中很重要的一个知识点,在此还是有必要总结一下的. 一.线程的生命周期及五种基本状态 关于Java中线程的生命周期,首先看一下下面这张较为经典的图: 上图中基本上囊括了Java中多线程 ...
- Java中的 多线程编程
Java 中的多线程编程 一.多线程的优缺点 多线程的优点: 1)资源利用率更好2)程序设计在某些情况下更简单3)程序响应更快 多线程的代价: 1)设计更复杂虽然有一些多线程应用程序比单线程的应用程序 ...
- java中的多线程 // 基础
java 中的多线程 简介 进程 : 指正在运行的程序,并具有一定的独立能力,即 当硬盘中的程序进入到内存中运行时,就变成了一个进程 线程 : 是进程中的一个执行单元,负责当前程序的执行.线程就是CP ...
- Java中的多线程技术全面详解
本文主要从整体上介绍Java中的多线程技术,对于一些重要的基础概念会进行相对详细的介绍,若有叙述不清晰或是不正确的地方,希望大家指出,谢谢大家:) 为什么使用多线程 并发与并行 我们知道,在单核机器上 ...
- 详细分析 Java 中实现多线程的方法有几种?(从本质上出发)
详细分析 Java 中实现多线程的方法有几种?(从本质上出发) 正确的说法(从本质上出发) 实现多线程的官方正确方法: 2 种. Oracle 官网的文档说明 方法小结 方法一: 实现 Runnabl ...
- Java中创建对象的几种方式
Java中创建对象的五种方式: 作为java开发者,我们每天创建很多对象,但是我们通常使用依赖注入的方式管理系统,比如:Spring去创建对象,然而这里有很多创建对象的方法:使用New关键字.使用Cl ...
- Java 中传统多线程
目录 Java 中传统多线程 线程初识 线程的概念 实现线程 线程的生命周期 常用API 线程同步 多线程共享数据的问题 线程同步及实现机制 线程间通讯 线程间通讯模型 线程中通讯的实现 @(目录) ...
随机推荐
- 一个小 Trick
平方变两次 一个状态 \(S\) 有一个贡献,所有状态 \(S\) 组成集合 \(U\) . 然后我们要统计下面这个东西 \[ans=\sum_{S\in U}f^2(S) \] 然后我们就可以看作是 ...
- 整除分块套杜教筛为什么是 O(n^2/3) 的
假设我们要筛一个东西叫做 \(f\) . 记 \[D(n)=\left\{n,\left\lfloor\dfrac n2\right\rfloor,\left\lfloor\dfrac n3\righ ...
- Redis 01 概述
参考源 https://www.bilibili.com/video/BV1S54y1R7SB?spm_id_from=333.999.0.0 版本 本文章基于 Redis 6.2.6 简介 NoSQ ...
- Vue3 发生错误:setup function returned a promise
当你组件中有 Promise 对象时,即 Axios.Ajax 这类的请求,然后把数据渲染到模板中就会报如下图的错误: 在这个异步组件外包裹一个 <Suspense> 组件.比如 App. ...
- MySQL事务概念与流程和索引控制
MySQL事务概念与流程和索引控制 视图 1.什么是视图 我们在执行SQL语句其实就是对表进行操作,所得到的其实也是一张表,而我们需要经常对这些表进行操作,拼接什么的都会产生一张虚拟表,我们可以基于该 ...
- RestTemplate用法
RestTemplate 用法 RestTemplate简介 RestTemplate 是一个同步的web http客户端请求模板工具,spring框架做的抽象模板, 常见的http客户端请求工具有: ...
- 【java】学习路径22-关于BigInteger类,大数字类
//int ooo = 19999999999999; //long ooo = 19999999999999; //这么大的整数,int和long都存不下的时候,我们就使用Math类下的BigInt ...
- 【java】学习路线7-继承、super方法、重写、重载
/*继承-java只有单继承如果你创建了很多个class,但是之间有很多相同的成员变量和成员方法,修改的时候又要多处修改好麻烦,此时就可以创建多一个类来存储这些重复的东西,统一管理.相当方便.*//* ...
- 基于ASP.NET Core 6.0的整洁架构
大家好,我是张飞洪,感谢您的阅读,我会不定期和你分享学习心得,希望我的文章能成为你成长路上的垫脚石,让我们一起精进. 本节将介绍基于ASP.NET Core的整洁架构的设计理念,同时基于理论落地的代码 ...
- K8S_三种Port区别总结
nodePort: 外部流量访问K8S集群中Service入口的一种方式 比如外部用户要访问k8s集群中的一个Web应用,那么我们可以配置对应service的type=NodePort,nodePor ...