线程的创建和启动
java使用Thread类代表线程,所有的线程对象都必须是Thread类或者其子类的实例。
java创建线程有三种方法:
一:继承Thread类创建线程类
通过Thread类来创建并启动多线程的步骤如下
- 定义Therad类的子类,并重写该类的Thread方法,该run()方法体就代表了线程需要完成的任务,因此把run()方法称为线程执行体;
- 创建Thread子类的实例,即创建线程对象; 3) 调用线程的start()方法来启动该线程;
start()方法的作用:启动一个分支线程,在JVM中开辟一个新的空间,并执行这个方法瞬间就结束了,这个方法的作用是为了开启一个新栈空间,只要新的栈空间开发出来,start()方法就结束了,线程启动成功,启动成功的线程会自动调用run()方法,并且run()方法,在分支栈的栈底部,
public class createThread extends Thread{
private int i;
//重写run()方法,run()方法体就是多线程执行体
public void run(){
//这段程序运行在分支栈中
for (int j = 0; j < 100; j++) {
System.out.println("分支栈——,》"+j+ getName());
}
}
//这是mai'n方法,这里的代码在主线程中运行,属于主线程
public static void main(String[] args) {
//新建一个分支线程对象
createThread ct=new createThread();
//启动线程
ct.start();
//这里的代码还是在主线程中运行
for (int i = 0; i < 1000; i++) {
System.out.println(Thread.currentThread());
}
}
}
二:实现Runnable接口创建线程类
实现Runnable接口创建线程类的步骤如下:
- 定义Runnable接口的实现类,并重写该接口中的run()方法,run()方法时该线程的执行体。
- 创建Runnable实现类的实例,并以此作为Thread的target构造方法来创建Thread对象,该Thread方法才是真正的线程类。
- 调用线程对象的start()方法来启动线程
public class createRunnabe implements Runnable {
public static void main(String[] args) {
/* //创建一个可以运行的对象;
createRunnabe cr=new createRunnabe();
//将可运行的对象封装成一个线程对象
Thread thread=new Thread(cr);
//启动线程*/
Thread thread=new Thread(new createRunnabe()); //合并代码
thread.start();
for (int i = 0; i <100 ; i++) {
System.out.println("主线程--》"+i);
}
}
@Override
public void run() {
for (int i = 0; i < 100; i++) {
System.out.println("分支线程-->"+i);
}
}
}
注意:
Runnable对象仅仅作为Thread对象的Target,Runnable实现类里包含的run()方法仅仅作为线程的执行体,而实际上线程对象依旧是Thread实例,所以该Thread线程负责执行其target的run()方法。
三:使用接口Callable和接口Future创建线程
Callable接口提供了一个call()方法作为线程的执行体,但是call()方法比run()方法功能更强大。
Call方法有返回值;
Call方法可以声明抛出的异常;
因此可以提供一个Callable接口的实现类作为Thread的target,代替runnable接口的run()执行体,但是Callable接口是java5的特性,不是Runnable接口的子接口,所以Callable接口不能直接作为Thread的target.
java5提供了Futuru接口来代表Callable接口里的Call()方法的返回值,并为Future接口提供了一个实现类FutureTask,该实现类实现了FutureTask实现了Future接口,并实现了Callable接口的实现类—可以作为Thread类的target.
创建并启动有返回值的线程的步骤:
- 创建Callable接口的实现类,并实现Call()方法,该call()方法将作为线程执行体,且该Call()方法有返回值,在创建Callable实现类的实例,这里使用匿名内部类实现Callable接口,也可以使用Lambda表达式实现接口。(应为Callable接口是函数式接口所以可以使用Lambda表达式来创建Callable对象)
- 使用FutureTast类来包装Callable对象,该FutureTusk对象封装了Callable对象的Call()方法的返回值。
- 使用FutureTask对象作为Thread对象的target创建并启动新线程。
public class ThirdThread {
public static void main(String[] args) {
/*FutureTask(Callable<V> callable)
创建一个 FutureTask会在运行,执行给定的 Callable。*/
//采用匿名内部类来实现Callable接口中的call()方法
FutureTask futuretask = new FutureTask(new Callable(){
@Override
public Object call() throws Exception {
for (int i = 0; i < 100; i++) {
System.out.println(Thread.currentThread().getName() + " :" + i);
}
//call方法可以有返回值
return this;
}
});
Thread thread=new Thread(futuretask);
//启动分支此线程,实质上是以Callable接口来创建并启动线程
thread.start();
for (int i = 0; i < 100; i++) {
System.out.println(Thread.currentThread().getName()+": "+i);
}
}
}
创建线程的三种方式对比
通过继承Thread类,实现Runnable,实现Callable接口都可以实现多线程,不过实现Runnable接口和实现Callable接口的方式基本相同,只是Callable接口里定义的方法有返回值,可以声明并抛出异常。因此这两种实现接口的方式差不多:
实现接口的方式和继承Thread类的主要差别:
优势:
线程实现了Callable、Runnable接口还可以继承其他类,提高了程序的扩展性。
实现接口的情况下,多个线程可以共享一个target对象非常适合于多个线程来处理同一份资源,从而可以将CPU代码、数据分开,形成清晰的模型.劣势:
编程稍微复杂,如果需要访问当前线程,则必须使用Thread.CurrentThread()方法。
本文详细介绍了Java中创建线程的三种主要方法:继承Thread类、实现Runnable接口和使用Callable与Future接口。每种方法都有其特点和适用场景,文章通过代码示例深入解析了这些方法的具体应用。

被折叠的 条评论
为什么被折叠?



