当前位置: 首页 > news >正文

多线程原理和常用方法以及Thread和Runnable的区别

文章目录

    • 🍦多线程原理
      • 🍧随机性打印
      • 🍨多线程内存图解
    • 🍩Thread类的常用方法
      • 🍪获取线程名称 getName()
      • 🎂设置线程名称 setName() 或者 new Thread("线程名字")
      • 🍰使当前正在执行的线程以指定的毫秒数暂停 sleep(long millis)
    • 🥧创建多线程程序的第二种方式-实现Runnable接口
    • 🍫Thread和Runnable的区别
    • 🍬使用匿名内部类开启线程

🍦多线程原理

多线程是一种并发执行的编程技术,它允许程序同时执行多个线程(线程是进程内的独立执行单元)。多线程的主要目的是提高程序的执行效率和响应速度。下

🍧随机性打印

CPU有了两条执行的路径,CPU就有了选择 ,一会执行main方法 一会执行run方法。
也可以说两个线程,一个main线程 一个run线程 一起请求CPU的执行权(执行时间)谁抢到了就执行对应的代码
多线程随机性打印结果

🍨多线程内存图解

  1. main方法的第一步创建对象,创建对象开辟堆内存存储在堆内存中(地址值赋值给变量名0x11)
  2. mt.run()调用时 run方法被压栈进来 其实是一个单线程的程序(main线程,会先执行完run方法再执行主线程中的去其他方法)
  3. mt.start()调用时会开辟一个新的栈空间。执行run方法(run方法就不是在main线程执行,而是在新的栈空间执行,如果再start会再开辟一个栈空间再多一个线程)

对cpu而言,cpu就有了选择的权利 可以执行main方法、也可以执行两个run方法。
多线程好处:多线程执行时,在栈内存中,其实每一个执行线程都有一片自己所属的栈内存空间,多个线程互不影响 进行方法的压栈和弹栈。
多线程内存图解

🍩Thread类的常用方法

🍪获取线程名称 getName()

public static void main(String[] args) {//创建Thread类的子类对象MyThread mt = new MyThread();//调用start方法,开启新线程,执行run方法mt.start();new MyThread().start();new MyThread().start();//链式编程System.out.println(Thread.currentThread().getName());
}/**获取线程的名称:1.使用Thread类中的方法getName()String getName() 返回该线程的名称。2.可以先获取到当前正在执行的线程,使用线程中的方法getName()获取线程的名称static Thread currentThread() 返回对当前正在执行的线程对象的引用。* @author zjq*/
// 定义一个Thread类的子类
public class MyThread extends Thread{//重写Thread类中的run方法,设置线程任务@Overridepublic void run() {//获取线程名称//String name = getName();//System.out.println(name);//链式编程System.out.println(Thread.currentThread().getName());}
}

输出如下:

main
Thread-2
Thread-0
Thread-1

🎂设置线程名称 setName() 或者 new Thread(“线程名字”)

  1. 使用Thread类中的方法setName(名字)
    void setName(String name) 改变线程名称,使之与参数 name 相同。

  2. 创建一个带参数的构造方法,参数传递线程的名称;调用父类的带参构造方法,把线程名称传递给父类,让父类(Thread)给子线程起一个名字

    Thread(String name) 分配新的 Thread 对象。
    代码案例:

//开启多线程
MyThread mt = new MyThread();
mt.setName("小强");
mt.start();//开启多线程
new MyThread("旺财").start();

🍰使当前正在执行的线程以指定的毫秒数暂停 sleep(long millis)

代码案例:

public static void main(String[] args) {//模拟秒表for (int i = 1; i <=60 ; i++) {System.out.println(i);//使用Thread类的sleep方法让程序睡眠1秒钟try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}}}

🥧创建多线程程序的第二种方式-实现Runnable接口

实现Runnable接口实现多线程的步骤:

  1. 创建一个Runnable接口的实现类
  2. 在实现类中重写Runnable接口的run方法,设置线程任务
  3. 创建一个Runnable接口的实现类对象
  4. 创建Thread类对象,构造方法中传递Runnable接口的实现类对象
  5. 调用Thread类中的start方法,开启新的线程执行run方法

代码案例如下:

/*** 1.创建一个Runnable接口的实现类* @author zjq*/
public class RunnableImpl implements Runnable{//2.在实现类中重写Runnable接口的run方法,设置线程任务@Overridepublic void run() {for (int i = 0; i <20 ; i++) {System.out.println(Thread.currentThread().getName()+"-->"+i);}}
}public static void main(String[] args) {//3.创建一个Runnable接口的实现类对象RunnableImpl run = new RunnableImpl();//4.创建Thread类对象,构造方法中传递Runnable接口的实现类对象Thread t = new Thread(run);//打印线程名称//5.调用Thread类中的start方法,开启新的线程执行run方法t.start();for (int i = 0; i <20 ; i++) {System.out.println(Thread.currentThread().getName()+"-->"+i);}
}

🍫Thread和Runnable的区别

实现Runnable接口创建多线程程序的好处:

  1. 避免了单继承的局限性

一个类只能继承一个类(一个人只能有一个亲爹),类继承了Thread类就不能继承其他的类。
实现了Runnable接口,还可以继承其他的类,实现其他的接口。

  1. 增强了程序的扩展性,降低了程序的耦合性(解耦)

实现Runnable接口的方式,把设置线程任务和开启新线程进行了分离(解耦)。
实现类中,重写了run方法:用来设置线程任务。
创建Thread类对象,调用start方法:用来开启新线程。

🍬使用匿名内部类开启线程

匿名内部类开启线程可以简化代码的编码。
代码案例如下:

/**匿名内部类方式实现线程的创建匿名:没有名字内部类:写在其他类内部的类匿名内部类作用:简化代码把子类继承父类,重写父类的方法,创建子类对象合一步完成把实现类实现类接口,重写接口中的方法,创建实现类对象合成一步完成匿名内部类的最终产物:子类/实现类对象,而这个类没有名字格式:new 父类/接口(){重复父类/接口中的方法};* @author zjq*/
public class Demo01InnerClassThread {public static void main(String[] args) {//线程的父类是Thread// new MyThread().start();new Thread(){//重写run方法,设置线程任务@Overridepublic void run() {for (int i = 0; i <20 ; i++) {System.out.println(Thread.currentThread().getName()+"-->"+"詹");}}}.start();//线程的接口Runnable//Runnable r = new RunnableImpl();//多态Runnable r = new Runnable(){//重写run方法,设置线程任务@Overridepublic void run() {for (int i = 0; i <20 ; i++) {System.out.println(Thread.currentThread().getName()+"-->"+"线程");}}};new Thread(r).start();//简化接口的方式new Thread(new Runnable(){//重写run方法,设置线程任务@Overridepublic void run() {for (int i = 0; i <20 ; i++) {System.out.println(Thread.currentThread().getName()+"-->"+"zjq");}}}).start();}
}

本文内容到此结束了,
如有收获欢迎点赞👍收藏💖关注✔️,您的鼓励是我最大的动力。
如有错误❌疑问💬欢迎各位指出。
主页:共饮一杯无的博客汇总👨‍💻

保持热爱,奔赴下一场山海。🏃🏃🏃

相关文章:

  • uniapp 在线预览PDF
  • Matlab数学建模算法详解之混合整数线性规划 (MILP) 算法(附完整实现代码)
  • 【Vue3+Ts项目】硅谷甄选 — 搭建后台管理系统模板
  • Docker容器间网络共享
  • JavaScript 的 闭包
  • 【Linux】vim-多模式的文本编辑器
  • 键盘打字盲打练习系列之指法练习——2
  • 前端知识笔记(十二)—————前端面试容易问到的问题总结
  • 肖sir __数据库练习__001
  • 【数据库】基于排序算法的去重,集合与包的并,差,交,连接操作实现原理,执行代价以及优化
  • 机器学习笔记 - 如何在Python中对网格和点云进行体素化?
  • Git常用命令#更改用户名
  • web前端之css变量的妙用、通过JavaScrip改变css文件中的属性值、querySelector、setProperty
  • python实现多线程并发测试并生成excel报告
  • 使用 kubeadm 部署 Kubernetes 集群(二)k8s环境安装
  • Android框架之Volley
  • Bytom交易说明(账户管理模式)
  • C++类的相互关联
  • css布局,左右固定中间自适应实现
  • es6(二):字符串的扩展
  • in typeof instanceof ===这些运算符有什么作用
  • IOS评论框不贴底(ios12新bug)
  • k8s如何管理Pod
  • Linux各目录及每个目录的详细介绍
  • Python代码面试必读 - Data Structures and Algorithms in Python
  • Redis 中的布隆过滤器
  • 分布式任务队列Celery
  • 关于extract.autodesk.io的一些说明
  • 基于HAProxy的高性能缓存服务器nuster
  • ------- 计算机网络基础
  • 盘点那些不知名却常用的 Git 操作
  • 让你的分享飞起来——极光推出社会化分享组件
  • 我的面试准备过程--容器(更新中)
  • ​LeetCode解法汇总2583. 二叉树中的第 K 大层和
  • #每日一题合集#牛客JZ23-JZ33
  • (02)vite环境变量配置
  • (14)Hive调优——合并小文件
  • (2)STM32单片机上位机
  • (3)(3.2) MAVLink2数据包签名(安全)
  • (编程语言界的丐帮 C#).NET MD5 HASH 哈希 加密 与JAVA 互通
  • (十一)图像的罗伯特梯度锐化
  • (转)JAVA中的堆栈
  • .Net Core 中间件验签
  • .NET面试题(二)
  • .NET微信公众号开发-2.0创建自定义菜单
  • []sim300 GPRS数据收发程序
  • [2017][note]基于空间交叉相位调制的两个连续波在few layer铋Bi中的全光switch——
  • [2021ICPC济南 L] Strange Series (Bell 数 多项式exp)
  • [cb]UIGrid+UIStretch的自适应
  • [FC][常见Mapper IRQ研究]
  • [Fri 26 Jun 2015 ~ Thu 2 Jul 2015] Deep Learning in arxiv
  • [Java][算法 双指针]Day 02---LeetCode 热题 100---04~07
  • [JS]数据类型
  • [LeetCode] Binary Tree Preorder Traversal 二叉树的先序遍历
  • [LeetCode] Ransom Note 赎金条