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

Java并发基础01:揭秘传统线程技术中创建线程的两种方式

欢迎关注我的微信公众号:程序员私房菜(id:eson_15)

传统的线程技术中有两种创建线程的方式:一是继承Thread类,并重写run()方法;二是实现Runnable接口,覆盖接口中的run()方法,并把Runnable接口的实现扔给Thread。这两种方式大部分人可能都知道,但是为什么这样玩就可以呢?下面我们来详细分析一下这两种方法的来龙去脉。

1. 揭秘Thread中run()

上面我们看到这两种方式都跟run()方法有关,所以我们来看一下Thread的源码中run()方法到底都干了什么:

@Override
public void run() {
    if (target != null) {
        target.run();
    }
}
复制代码

我们可以看出,run()方法中很简单,只有一个if语句,如果target不为空就执行target的run()方法,否则什么也不干,那么这target到底是何方神圣呢?我们点击进去可以看到:

private Runnable target;
复制代码

原来target就是Runnable接口,我们再点进Runnable看看:

@FunctionalInterface
public interface Runnable {
    public abstract void run();
}
复制代码

Runnable中就一个方法,也是run()方法!好了,现在再回到Thread类的run()方法中,如果target不为空,即实现了Runnable接口,也即实现了Runnable中的run()方法,那么我们就使用该接口中的run()方法;如果target为空,即没有实现Runnable接口,那我们什么也不做,即线程创建后立马就消失了。

所以到这里,大家就明白了为什么创建线程有上面两种方式了。第一种:你不是要先进行if判断么?我现在不判断了,我把你的if干掉,我在run()方法中自己写代码,想干啥就干啥,即重写Thread中的run()方法,;第二种:你不是要先进行if判断么?行,给你一个Runnable接口让你判断,但你还是得调用我Runnable中的run()方法啊,那我重写我Runnable中的run()方法不就行了!  

知道了来龙去脉后,下面就针对这两种传统的方式写个实例。

2. 创建方式1:继承Thread类

只要两步即可创建并开启一个线程:

  • 继承Thread类,并实现run()方法;
  • 调用start()方法开启线程。

由于只要实现一个run()方法即可,所以我们可以使用java中的匿名内部类来实现,如下:

public class TraditionalThread {

	public static void main(String[] args) {
		
		/********** 第一种方法:继承Thread类,覆写run()方法 **************/
		Thread thread1 = new Thread(){

			@Override
			public void run() {
				try {
					Thread.sleep(500);//让线程休息500毫秒
				} catch (InterruptedException e) {
					e.printStackTrace();
				}
				System.out.println(Thread.currentThread().getName());//打印出当前线程名
			}
		};
		thread1.start();//开启线程
	}
}
复制代码

3. 创建方式2:实现Runnable接口

只要两步即可创建并开启一个线程:

  • 实现Runnable接口,并实现run()方法;
  • 调用start()方法开启线程。

由于只要实现一个run()方法即可,所以我们也可以使用java中的匿名内部类来实现,如下:

public class TraditionalThread {

	public static void main(String[] args) {
		
		/********** 第二种方法:实现Runnable接口,扔给Thread **************/
		Thread thread2 = new Thread(new Runnable() {
			
			@Override
			public void run() {
				try {
					Thread.sleep(500);
				} catch (InterruptedException e) {
					e.printStackTrace();
				}
				System.out.println(Thread.currentThread().getName());
				
			}
		});
		thread2.start();
	}
}
复制代码

4. 两种方式同时使用

如果有个哥们比较给力,他两种方式同时使用了,即:既实现了Thread类中的run()方法,又给Thread扔了一个实现了run()方法的Runnable。如下所示:

public class TraditionalThread {

	public static void main(String[] args) {
		//这哥们的代码写的比较给力
		new Thread(new Runnable() {
			
			@Override
			public void run() {
				try {
					Thread.sleep(500);
				} catch (InterruptedException e) {
					e.printStackTrace();
				}
				System.out.println("Runnable:" + Thread.currentThread().getName());
			}
		}){

			@Override
			public void run() {
				try {
					Thread.sleep(500);
				} catch (InterruptedException e) {
					e.printStackTrace();
				}
				System.out.println("Thread:" + Thread.currentThread().getName());
			}
			
		}.start();
	}

}
复制代码

现在又会执行哪个呢?我们运行一下上面的程序就会发现,它会打印出Thread的信息,所以运行的是Thread的run()方法,知道结论了,但是为啥呢?

从面向对象的思想去考虑:上面一段代码其实是新new了一个对象(子对象)继承了Thread对象(父对象),在子对象里重写了父类的run()方法,父对象中扔了个Runnable进去,父对象中的run()方法就是最初的带有if判断的run()方法。

好了,现在执行start()后,肯定先在子类中找run()方法,找到了,父类的run()方法自然就被干掉了,所以会打印出Thread:,如果我们现在假设子类中没有重写run()方法,那么必然要去父类找run()方法,父类的run()方法中就得判断是否有Runnable传进来,现在有一个,所以执行Runnable中的run()方法,那么就会打印Runnable:出来。

OK,传统的创建线程的两种方式就总结这么多~如有错误之处,欢迎指正~我们一起进步!

也欢迎大家关注我的微信公众号:程序员私房菜。我会持续输出更多文章。

相关文章:

  • iOS上的实时远程配置
  • 【译】 WebSocket 协议第六章——发送与接收消息(Sending and Receiving Data)
  • shell各种括号用途总结
  • 简易的RPC调用框架(大神写的)
  • 捕捉Web页面子类错误堆栈中的信息
  • (原創) 如何解决make kernel时『clock skew detected』的warning? (OS) (Linux)
  • (原創) 如何刪除Windows Live Writer留在本機的文章? (Web) (Windows Live Writer)
  • 2010年9月blog汇总:敏捷个人和模型驱动开发
  • Eclipse:应该掌握的快捷键
  • 图像处理时用的卷积函数
  • asp.net web api
  • 各浏览器对页面外部资源加载的策略
  • 收藏 c#小函数
  • 解决Page.FindControl方法找不到指定控件 转
  • 二台电脑之间数据库文件进行备份
  • [ 一起学React系列 -- 8 ] React中的文件上传
  • [deviceone开发]-do_Webview的基本示例
  • Javascript编码规范
  • Java新版本的开发已正式进入轨道,版本号18.3
  • JS进阶 - JS 、JS-Web-API与DOM、BOM
  • Kibana配置logstash,报表一体化
  • Mysql5.6主从复制
  • MySQL-事务管理(基础)
  • spring boot下thymeleaf全局静态变量配置
  • Spring声明式事务管理之一:五大属性分析
  • Vultr 教程目录
  • 浏览器缓存机制分析
  • 听说你叫Java(二)–Servlet请求
  • 赢得Docker挑战最佳实践
  • 在weex里面使用chart图表
  • Hibernate主键生成策略及选择
  • #1014 : Trie树
  • #DBA杂记1
  • #git 撤消对文件的更改
  • #include<初见C语言之指针(5)>
  • (C语言)共用体union的用法举例
  • (WSI分类)WSI分类文献小综述 2024
  • (zz)子曾经曰过:先有司,赦小过,举贤才
  • (附源码)ssm失物招领系统 毕业设计 182317
  • (附源码)计算机毕业设计SSM智慧停车系统
  • (汇总)os模块以及shutil模块对文件的操作
  • (七)理解angular中的module和injector,即依赖注入
  • (三维重建学习)已有位姿放入colmap和3D Gaussian Splatting训练
  • (转载)跟我一起学习VIM - The Life Changing Editor
  • .net 8 发布了,试下微软最近强推的MAUI
  • .net 验证控件和javaScript的冲突问题
  • .NET关于 跳过SSL中遇到的问题
  • .sdf和.msp文件读取
  • /*在DataTable中更新、删除数据*/
  • @Autowired和@Resource装配
  • [].shift.call( arguments ) 和 [].slice.call( arguments )
  • [Android Pro] AndroidX重构和映射
  • [bzoj1006]: [HNOI2008]神奇的国度(最大势算法)
  • [CSS]中子元素在父元素中居中
  • [GN] 后端接口已经写好 初次布局前端需要的操作(例)