先介绍两个概念:
1. CPU的执行资格: 对于线程,可以被CPU处理的线程,但还在处理队列里面排队等候,则称该等候线程具有CPU的执行资格。
2. CPU的执行权: 对于线程,正在被CPU处理的线程,则称该线程具有CPU的执行权。
再看上图,从线程被创建,通过start()方法开启线程,这时,则有一个线程在运行,其他线程进入临时阻塞状态。
而执行sleep(time)或者wait()方法的线程进入了冻结状态,run()方法执行结束的线程消亡了。
分析下各自线程的CPU的执行资格和执行权
正在运行的线程:具备着执行资格和执行权。
临时阻塞状态的线程:具备着执行资格,但是没有执行权。
冻结的线程:没有执行资格,也没有执行权。
前面说过创建线程具有两种方式,下面介绍第二种:
实现Runnable接口
步骤:
- 定义类实现Runnable接口。
- 覆盖接口中的run()方法,将线程的任务代码封装到run()方法中。
- 通过Thread类创建线程对象,并将Runnable接口的子类对象作为Thread类的构造函数的参数进行传递。因为线程的任务都封装在Runnable接口子类对象的run()方法中,所以要在线程对象创建时就必须明确要运行的任务,故而将Runnable接口的子类对象作为Thread类的构造函数的参数进行传递。
- 调用线程对象的start()方法开启线程。
以下为示例代码:
/*通过接口实现线程*/
class Demo2 implements Runnable
{
private String name;
Demo2(String name)
{
this.name = name;
}
@Override
public void run()
{
show();
}
public void show()
{
for(int i = 0; i < 5; i++)
{
for(int j=-99999999; j<999999999; j++) {}
System.out.println(name+" : "+ i + "..." + Thread.currentThread().getName());
}
}
}
public class ThreadDemo3
{
public static void main(String[] args)
{
Demo2 demo1 = new Demo2("旺财");
Demo2 demo2 = new Demo2("xiaoqiang");
new Thread(demo1).start();
new Thread(demo2).start();
for(int i= 0; i < 5; i++)
{
for(int j=-99999999; j<999999999; j++) {}
System.out.println("Over..." + Thread.currentThread().getName());
}
}
}
则可得到执行结果:
Over…main
xiaoqiang : 0…Thread-1
旺财 : 0…Thread-0
Over…main
Over…main
Over…main
xiaoqiang : 1…Thread-1
旺财 : 1…Thread-0
xiaoqiang : 2…Thread-1
旺财 : 2…Thread-0
Over…main
旺财 : 3…Thread-0
旺财 : 4…Thread-0
xiaoqiang : 3…Thread-1
xiaoqiang : 4…Thread-1