博客园  :: 首页  :: 新随笔  :: 联系 :: 订阅 订阅  :: 管理

Java线程:创建与启动

Posted on 2019-09-22 13:27  v薛定谔的猫v  阅读(99)  评论(0编辑  收藏  举报

一、定义线程
        1、扩展java.lang.Thread类。

        此类中有个run()方法,应该注意其用法:public void run()

        如果该线程是使用独立的Runnable运行对象构造的,则调用该Runnable对象的run方法;否则,该方法不执行任何操作并返回。

        Thread的子类应该重写该方法。

2、实现java.lang.Runnable接口。

        void run()

        使用实现接口Runnable的对象创建一个线程时,启动该线程将导致在独立执行的线程中调用对象的run方法。

        方法run的常规协定是,它可能执行任何所需的操作。

二、实例化线程
        1、如果是扩展java.lang.Thread类的线程,则直接new即可。

        2、如果是实现了java.lang.Runnable接口的类,则用Thread的构造方法:


Thread(Runnabletarget)  
Thread(Runnabletarget, String name)  
Thread(ThreadGroupgroup, Runnable target)  
Thread(ThreadGroupgroup, Runnable target, String name)  
Thread(ThreadGroupgroup, Runnable target, String name, long stackSize)  
 

        其中:

        Runnable target:实现了Runnable接口的类的实例。 

Thread类也实现了Runnable接口,因此,从Thread类继承的类的实例也可以作为target传入这个构造方法。
直接实现Runnable接口类的实例。
线程池建立多线程。
        String name:线程的名子。这个名子可以在建立Thread实例后通过Thread类的setName方法设置。默认线程名:Thread-N,N是线程建立的顺序,是一个不重复的正整数。

        ThreadGroup group:当前建立的线程所属的线程组。如果不指定线程组,所有的线程都被加到一个默认的线程组中。

        long stackSize:线程栈的大小,这个值一般是CPU页面的整数倍。如x86的页面大小是4KB.在x86平台下,默认的线程栈大小是12KB。

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

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

        在调用start()方法之后:发生了一系列复杂的事情——

        启动新的执行线程(具有新的调用栈);

        该线程从新状态转移到可运行状态;

        当该线程获得机会执行时,其目标run()方法将运行。

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

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

/** 
 * 实现Runnable接口的类 
 */  
public class RunnableImpl implements Runnable{  
    private Stringname;  
    public RunnableImpl(String name) {  
       this.name = name;  
    }  
    @Override  
    public void run() {  
       for (int i = 0; i < 5; i++) {  
           for(long k=0;k<100000000;k++);  
           System.out.println(name+":"+i);  
       }       
    }  
}  
   
/** 
 * 测试Runnable类实现的多线程程序 
 */  
public class TestRunnable {  
   
    public static void main(String[] args) {  
       RunnableImpl ri1=new RunnableImpl("李白");  
       RunnableImpl ri2=new RunnableImpl("屈原");  
       Thread t1=new Thread(ri1);  
       Thread t2=new Thread(ri2);  
       t1.start();  
       t2.start();  
    }  
}  

执行结果:
[java] view plain copy
屈原:0  
李白:0  
屈原:1  
李白:1  
屈原:2  
李白:2  
李白:3  
屈原:3  
李白:4  
屈原:4  
2、扩展Thread类实现的多线程例子

/** 
 * 测试扩展Thread类实现的多线程程序 
 */  
public class TestThread extends Thread {  
    public TestThread(String name){  
       super(name);  
    }  
    @Override  
    public void run() {  
       for(int i=0;i<5;i++){  
           for(long k=0;k<100000000;k++);  
           System.out.println(this.getName()+":"+i);  
       }  
    }  
    public static void main(String[] args){  
       Thread t1=new TestThread("李白");  
       Thread t2=new TestThread("屈原");  
       t1.start();  
       t2.start();        
    }  
}  

 

 执行结果:
[java] view plain copy
屈原:0  
李白:0  
屈原:1  
李白:1  
屈原:2  
李白:2  
屈原:3  
屈原:4  
李白:3  
李白:4  
        对于上面的多线程程序代码来说,输出的结果是不确定的。其中的一条语句for(long k=0;k<100000000;k++);是用来模拟一个非常耗时的操作的。

五、一些常见问题
        1、线程的名字,一个运行中的线程总是有名字的,名字有两个来源,一个是虚拟机自己给的名字,一个是你自己的定的名字。在没有指定线程名字的情况下,虚拟机总会为线程指定名字,并且主线程的名字总是mian,非主线程的名字不确定。

        2、线程都可以设置名字,也可以获取线程的名字,连主线程也不例外。

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

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

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

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

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

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

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

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