Fork me on GitHub

java线程:线程启动

概念和原理

1、操作系统中线程和进程的概念
  现在的操作系统是多任务操作系统。多线程是实现多任务的一种方式。
  进程 是指一个内存中运行的应用程序,每个进程都有自己独立的一块内存空间,一个进程中可以启动多个线程。比如在Windows系统中,一个运行的exe就是一个进程。
  线程 是指进程中的一个执行流程,一个进程中可以运行多个线程。比如java.exe进程中可以运行很多线程。线程总是属于某个进程,进程中的多个线程共享进程的内存。
  “同时”执行是人的感觉,在线程之间 实际上轮换执行

2、Java中的线程
  在Java中,“线程”指两件不同的事情:
  1).java.lang.Thread类的一个实例;
  2).线程的执行。
  线程总体分两类:用户线程守候线程。当所有用户线程执行完毕的时候,JVM自动关闭。但是守候线程却不独立于JVM,守候线程一般是由操作系统或者用户自己创建的

创建和启动

实例化线程

  1).如果是扩展java.lang.Thread类的线程,则直接new即可。
  2).如果是实现了java.lang.Runnable接口的类,则用Thread的构造方法:

1
2
3
4
5
Thread(Runnable target)
Thread(Runnable target, String name)
Thread(ThreadGroup group, Runnable target)
Thread(ThreadGroup group, Runnable target, String name)
Thread(ThreadGroup group, Runnable target, String name, long stackSize)

Runnable 接口与 Thread 类之间的区别
  Runnable 接口必须实现 run 方法,而 Thread 类中的run 方法是一个空方法,可以不重写
  Runnable 接口的实现类并不是真正的线程类,只是线程运行的目标类。要想以线程的方式执行 run 方法,必须依靠 Thread 类
  Runnable 接口适合于资源的共享: 如果线程类显式继承了一个类,如child继承了parent类,就不能在继承Thread类,得用Runnable 接口

启动线程

  在线程的Thread对象上 调用start()方法,而不是run()或者别的方法。
  在调用start()方法之前:线程处于新状态中,新状态指有一个Thread对象,但还没有一个真正的线程。

在调用start()方法之后:发生了一系列复杂的事情
  1.启动新的执行线程(具有新的调用栈);
  2.该线程从新状态转移到可运行状态;
  3.当该线程获得机会执行时,其目标run()方法将运行。

  注意:对Java来说,run()方法没有任何特别之处。像main()方法一样,它只是新线程知道调用的方法名称(和签名)。因此,在Runnable上或者Thread上调用run方法是合法的,但并不启动新的线程。

实例

1、实现Runnable接口的多线程例子

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
public class TestRunnable {
public static void main(String[] args) {
//2、创建对应Runna接口的实现类对象
MyRunnable mr = new MyRunnable();
//3、创建Thread对象,利用Thread(Runnable target)
Thread thread1 = new Thread(mr,"线程1");
Thread thread2 = new Thread(mr, "线程2");
//4、启动线程
thread1.start();
thread2.start();
}
static class MyRunnable implements Runnable{
int i=0;
//1、创建实现Runnable接口的实现类:必须实现run()方法
public void run() {
for (; i<10; i++){
System.out.println(Thread.currentThread().getName()+":"+i);
}
}
}
}

  结果每次顺序可能都不一致:线程2:0 , 线程2:1 , 线程2:2 , 线程2:3 , 线程1:0 , 线程2:4 , 线程1:5 , 线程2:6 , 线程2:8 , 线程1:7 , 线程2:9

2、扩展Thread类实现的多线程例子

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
/**
* 在java中,Thread类代表一个线程
*/
public class ThreadTest {
public static void main(String[] args) {
//1.创建线程对象
Thread thread1 = new MyThread("线程1");
Thread thread2 = new MyThread("线程2");
//2、调用线程对象的start()方法启动线程,而不是run方法
thread1.start();
thread2.start();
for (int i=0;i<10; i++){
System.out.println(Thread.currentThread().getName()+":"+i);
}
}
}
class MyThread extends Thread{
public MyThread(String name){
super(name);
}
/**
* 线程体在run()方法中
*/
public void run(){
String threadName = Thread.currentThread().getName();
for (int i=0;i<10; i++){
System.out.println(threadName+":"+i);
}
}
}

  结果:main:0 , 线程1:0 , 线程1:1 , 线程1:2 , 线程1:3 , 线程1:4 , 线程1:5 , 线程1:6 , 线程1:7 , 线程1:8 , 线程1:9 , main:1 , main:2 , main:3 , main:4 , main:5 , main:6 , main:7 , main:8 , main:9 , 线程2:0 , 线程2:1 , 线程2:2 , 线程2:3 , 线程2:4 , 线程2:5 , 线程2:6 , 线程2:7 , 线程2:8 , 线程2:9

3、一些常见问题
  1).线程的名字,一个运行中的线程总是有名字的,名字有两个来源,一个是虚拟机自己给的名字,一个是你自己的定的名字。在没有指定线程名字的情况下,虚拟机总会为线程指定名字,并且主线程的名字总是mian,非主线程的名字不确定。
  2).线程都可以设置名字,也可以获取线程的名字,连主线程也不例外。
  3).获取当前线程的对象的方法是:Thread.currentThread();
  4).在上面的代码中,只能保证:每个线程都将启动,每个线程都将运行直到完成。一系列线程以某种顺序启动并不意味着将按该顺序执行。对于任何一组启动的线程来说,调度程序不能保证其执行次序,持续时间也无法保证。
  5).当线程目标run()方法结束时该线程完成。
  6).一旦线程启动,它就永远不能再重新启动。只有一个新的线程可以被启动,并且只能一次。一个可运行的线程或死线程可以被重新启动。
  7).线程的调度是JVM的一部分,在一个CPU的机器上上,实际上一次只能运行一个线程。一次只有一个线程栈执行。JVM线程调度程序决定实际运行哪个处于可运行状态的线程。
众多可运行线程中的某一个会被选中做为当前线程。可运行线程被选择运行的顺序是没有保障的。
  8).尽管通常采用队列形式,但这是没有保障的。队列形式是指当一个线程完成“一轮”时,它移到可运行队列的尾部等待,直到它最终排队到该队列的前端为止,它才能被再次选中。事实上,我们把它称为可运行池而不是一个可运行队列,目的是帮助认识线程并不都是以某种有保障的顺序排列唱呢个一个队列的事实。
  9).尽管我们没有无法控制线程调度程序,但可以通过别的方式来影响线程调度的方式。

-----------------本文结束,感谢您的阅读-----------------