多线程 -- run()方法和start()方法的区别

本文详细解析了Java线程的启动与运行机制,包括直接调用run()方法与使用start()方法的区别,以及线程如何在不同场景下执行任务。通过实例代码展示线程的创建、执行流程及打印日志的输出顺序。

一. 直接调用run()方法

public class Two_Thread implements Runnable {


	public static void main(String[] args) {
		// TODO Auto-generated method stub
		Two_Thread R = new Two_Thread();
		R.run(); // 直接调用run()方法, 而不是start()方法启动线程
		System.out.println(Thread.currentThread().getName()+" back to main");
	}

	@Override
	public void run() {
		// TODO Auto-generated method stub
	
		increate();
	}

	private  void increate() { 
		// TODO Auto-generated method stub
		for(int i=0;i<5;i++){

			System.out.println(Thread.currentThread().getName()+" i = "+i);
		   
		     try {
	 				Thread.sleep(1000);
	 			} catch (InterruptedException e) {
	 				// TODO Auto-generated catch block
	 				e.printStackTrace();
	 			}
		}
	
	}

}
无论什么时候,打印的log都是以下不变:

main i = 0
main i = 1
main i = 2
main i = 3
main i = 4
main back to main

由log可以看出,run()方法是通过main线程启动的,并且它不不具有任何内在的线程能力(即总是执行完run方法,才打印main back to main)。如果要实现线程行为,必须采用start()方法。


二.添加一个线程t1. 并且去掉线程休眠 Thread.sleep(1000);

public class Two_Thread implements Runnable {


	public static void main(String[] args) {
		// TODO Auto-generated method stub
		Two_Thread R = new Two_Thread();
		Thread t1 = new Thread(R,"1"); //用Runnable对象1初始化
		t1.start(); //通过start()方法启动线程
		R.run();//调用Runable对象中的run()方法
		System.out.println(Thread.currentThread().getName()+" back to main");
	}

	@Override
	public void run() {
		// TODO Auto-generated method stub
	
		increate();
	}

	private  void increate() { 
		// TODO Auto-generated method stub
		for(int i=0;i<5;i++){

			System.out.println(Thread.currentThread().getName()+" i = "+i);
		   
		}
	
	}

}

此时打印的log是随机的:但是无论什么时候,这句“main back to main”log,一定是在执行完main i = 4之后才会执行。因为在主线程中,这俩句是顺序执行的。

情况1:

main i = 0
1 i = 0
main i = 1
1 i = 1
main i = 2
1 i = 2
main i = 3
1 i = 3
main i = 4
1 i = 4
main back to main


情况2:

1 i = 0
main i = 0
1 i = 1
main i = 1
1 i = 2
main i = 2
1 i = 3
main i = 3
main i = 4
main back to main
1 i = 4

### Java多线程中 `start()` 方法调用 `run()` 方法的机制 `Thread.start()` 方法是 Java 中启动新线程的关键入口,它负责通知 JVM 创建并启动一个新的线程。尽管表面上看起来简单,但实际上它的内部实现非常复杂,涉及到底层的操作系统交互以及 JNI(Java Native Interface)的支持。 #### 1. `start()` 方法的作用 当调用 `thread.start()` 时,JVM 并不会立即执行 `run()` 方法中的代码,而是通过本地方法触发操作系统的线程创建过程[^1]。具体来说: - **线程创建阶段**:`start()` 方法会请求操作系统分配新的线程资源,并将其加入到调度队列中。 - **线程初始化阶段**:在成功创建线程之后,JVM 将准备执行目标线程的任务逻辑。 - **任务执行阶段**:一旦新线程被调度器选中运行,JVM 自动调用该线程实例的 `run()` 方法作为其初始任务。 需要注意的是,开发者无需手动调用 `run()` 方法;如果直接调用 `thread.run()`,那么实际上只是在当前线程上下文中顺序执行了 `run()` 方法的内容,而不是开启真正的并发行为[^2]。 #### 2. 底层实现细节 以下是关于 `start()` `run()` 方法之间关系的一些重要技术要点: - **JNI 层面**:`start()` 方法最终会映射到底层 C/C++ 实现,在那里完成实际的新线程建立工作。此部分通常依赖于平台特定 API 来管理硬件级线程支持。 - **线程状态转换**:从用户视角来看,调用了 `start()` 后,线程的状态由 NEW 变更为 RUNNABLE 或 BLOCKED (取决于是否有锁竞争)[^3]。此时即使还没有被执行,也已经具备随时可运行条件。 - **同步与互斥处理**:假如多个线程尝试同时修改共享数据结构,则可能需要用到 synchronized 关键字来保护临界区代码片段以防竞态发生[^3]。这一步骤虽然不直接影响 start-run 流程本身,但在设计健壮程序时非常重要。 #### 示例代码展示 下面给出一段简单的例子说明如何正确使用 `start()` 方法间接触达 `run()` 函数体: ```java public class MyRunnable implements Runnable { public void run() { System.out.println(Thread.currentThread().getName()+" is running."); } public static void main(String[] args) throws InterruptedException { Thread t1 = new Thread(new MyRunnable(), "t1"); // 使用 start() 开启独立线程 t1.start(); // 主线程继续自己的流程... Thread.sleep(50); System.out.println("Main thread finished."); } } ``` 在这个案例里,“t1”代表我们自定义的一个子线程对象。“is running.”消息来自另一个 CPU 上下文环境下的打印语句——即证明确实形成了异步效果! --- ### 总结 综上所述,`Thread.start()` 不仅是一个普通的函数调用那么简单,它是整个 Java 多线程体系的基础构件之一。通过对原生接口封装后的抽象层次提升使得我们可以更方便快捷地构建跨平台应用软件解决方案的同时还隐藏了许多繁琐低级别的细节问题给程序员带来便利性体验。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值