多线程--线程的创建与启动

Posted xjs1874704478

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了多线程--线程的创建与启动相关的知识,希望对你有一定的参考价值。

创建一个线程,Java提供三种方法

(1)通过实现Runnable接口,重写run方法

(2)通过继承Thread类本身,--必须重写

(3)通过Callable和Future创建线程

Thread类的方法

void start()使该线程开始执行,Java虚拟机调用该线程的run方法。

final void setName(String name)改变线程名称,使之与参数name相同

setPriority(int priority)更改线程的优先级

join(long millisec)等待该线程终止的时间最长为milli毫秒

interrupt()中断线程

boolean isAlive()测试线程是否处于活动状态

上述方法被Thread对象调用

下面的方法是Thread类的静态方法static

yield()放弃CPU的使用权力

sleep(long millisec)休眠线程

currentThread()获得正在执行的线程对象

 

概念解析:

并行:是真正的同时。

并发:通过CPU调度算法,用户看上去同时执行,实际上从CPU层面上不是真正同时。

线程安全:并发情况下,多线程使用,线程调度顺序不影响任何结果。

同步:Java中的同步指的是通过人为的控制和调度,保证共享资源的多线程访问称为线程安全的--结果正确。

 

Thread和Runnable的区别

如果一个类继承Thread,则不合适资源共享,但是如果实现了Runnable接口,则很容易实现资源共享。

线程是CPU的调度单位,进程是操作系统的调度单位。

 

Method类中有个方法invoke()

public Object invoke(Object obj,Object ...args)

通俗点讲就是:对象调用的方法(对象.参数)


 

守护线程(DaemonThread)

守护线程具有最低的优先级;

是用来服务用户线程的。

(1)thread.setDaemon(true)必须在thread.start()之前设置。

(2)在Daemon线程中产生的新的线程也是Daemon的。

(3)守护线程不去访问固有的资源,如文件,数据库。

 

用户线程(userThread)

Thread和Runnable创建出来的线程。

 

线程的生命周期大体分为5种状态

1.New:通过new创建了Thread类或其子类对象。

2.Runnable: a)指Thread类对象调用了start()方法,这时的线程就等待时间片轮转到自己这,以便获得CPU;b)线程在处于Runnable状态时并没有运行完自己的run方法,时间片用完之后回到Runnable状态; c) 处于Blocked状态的线程结束当前的Blocked状态之后重新回到Runnable状态。

3.Running:线程已获CPU运行。

4.Dead:处于Running的线程,执行完run方法之后,就变成了Dead状态。

5.Blocked:处于Running状态的线程出于某种原因,让出当前CPU 给其他线程。


 

线程的创建与启动

继承Thread类:重写run方法

 1 package NiuMuThread;
 2 
 3 public class MyThread extends Thread{
 4 
 5     @Override
 6     public void run() {
 7         // TODO Auto-generated method stub
 8         for(int i=0;i<100;i++) {
 9 //            线程的静态方法-----当前线程----的名字
10             System.out.println(Thread.currentThread().getName()+":"+i);
11             try {
12                 Thread.sleep(100);
13             } catch (InterruptedException e) {
14                 // TODO Auto-generated catch block
15                 e.printStackTrace();
16             }
17         }
18     }
19 }

实现Runnable接口:重写run方法

 1 package NiuMuThread;
 2 
 3 public class MyThread2 implements Runnable{
 4 
 5     @Override
 6     public void run() {
 7         // TODO Auto-generated method stub
 8         for(int i=0;i<100;i++) {
 9 //            线程的静态方法-----当前线程----的名字
10             System.out.println(Thread.currentThread().getName()+":"+i);
11             try {
12                 Thread.sleep(100);
13             } catch (InterruptedException e) {
14                 // TODO Auto-generated catch block
15                 e.printStackTrace();
16             }
17         }
18     }
19 }

测试:

 1 package NiuMuThread;
 2 
 3 public class Test {
 4 
 5     public static void main(String[] args) {
 6         //第一个线程
 7         MyThread thread=new MyThread();
 8         //第三个线程
 9         MyThread2 r=new MyThread2();
10         
11         Thread th2=new Thread(r);
12         thread.start();//启动线程
13         th2.start();
14         
15         //第二个线程
16         for(int i=0;i<100;i++) {
17 //            线程的静态方法-----当前线程----的名字
18             System.out.println(Thread.currentThread().getName()+":"+i);
19             try {
20                 Thread.sleep(100);
21             } catch (InterruptedException e) {
22                 // TODO Auto-generated catch block
23                 e.printStackTrace();
24             }
25         }
26     }
27 }

注意:通过实现Runnable接口的,需要 new Thread(runnable),才能创建线程对象

***在线程的Thread对象上调用start()方法,而不是run()或者别的方法。

 

一些常见的问题

1.线程的名字,一个运行中的线程总是有名字的,名字有两个来源,一是虚拟机自己给的名字,或者是你自己定的名字。在没指定线程名字的情况下,虚拟机会为线程指定名字,主线程的名字是main,非主线程的名字不确定。

2.线程都可以设置名字,也可以获取线程的名字。

3.获取当前线程的对象的方法是:Thread.currentThread();

4.在上面代码中,只能保证:每个线程都将启动,每个线程都将运行直到完成。一系列线程以某种顺序启动并不意味着将按该顺序执行。对于任何一组启动的线程来说,调度程序不能保证其执行次序,持续时间也无法保证。

5.当线程目标run()方法结束时线程完成。

6.一旦线程启动,它就永远不能在重新启动。只有一个新的线程可以被重新启动,并且只能一次。一个可运行的线程或者死线程可以被重新启动。

7.线程的调度是JVM的一部分,在一个CPU的机器上,实际上一次只能运行一个线程。一次只有一个线程栈运行。JVM线程调度程序决定实际运行哪个处于可运行状态的线程。

众多可运行线程中的某一个会被选中作为当前线程。可运行线程被选择的顺序是没有保障的。

8.尽管通常采用队列形式,但这是没有保障的。队列形式是指当一个线程完成“一轮”时,它移到可运行队列的尾部等待,直到它最终排队到该队列的前端为止,它才能被再次选中。事实上,我们把它称为可运行池而不是一个可运行队列,目的是帮助认识线程并不都是以某种有保障的顺序排列。

9.尽管我们没有无法控制线程调度程序,但可以通过别的方式来影响线程的调度方式。

 

以上是关于多线程--线程的创建与启动的主要内容,如果未能解决你的问题,请参考以下文章

多线程--线程的创建与启动

C++并发与多线程 2_线程启动结束,创建线程多种方法,join,detach

八.多进程与多线程

多个请求是多线程吗

深度解析多线程的创建方式和正确启动多线程

Java多线程概述及创建