美文网首页Java
创建线程的3种方法

创建线程的3种方法

作者: migill | 来源:发表于2019-12-10 18:47 被阅读0次

    转载 Java创建线程的3中方法

    • 继承Thread类创建线程类
    • 通过Runnable接口创建线程类
    • 通过Callable和Future创建线程
    1、继承Thread类创建线程类
    • 定义Thread类的子类,并重写该类的run方法,该run方法的方法体就代表了线程要完成的任务。因此把run()方法称为执行体。
    • 创建Thread类的实例,即创建了线程对象。
    • 调用线程对象的start()方法来启动该线程。
    public class ThreadSimple extends Thread {
        int i = 0;
    
        //重写run方法,run方法的方法体就是现场执行体
        public void run() {
            for (; i < 100; i++) {
                System.out.println(getName() + "  " + i);
            }
        }
    
        public static void main(String[] args) {
            for (int i = 0; i < 100; i++) {
                System.out.println(Thread.currentThread().getName() + "  : " + i);
                if (i == 20) {
                    new ThreadSimple().start();
                }
            }
        }
    }
    
    2、通过Runnable接口创建线程类
    • 定义runnable接口的实现类,并重写该接口的run()方法,该run()方法的方法体同样是该线程的线程执行体。
    • 创建Runnable实现类的实例,并依此实例作为Thread的target来创建Thread对象,该Thread对象才是真正的线程对象。
    • 调用线程对象的start()方法来启动该线程。
    public class ThreadSimple implements Runnable {
    
        private int i;
    
        public void run() {
            for (i = 0; i < 100; i++) {
                System.out.println(Thread.currentThread().getName() + " " + i);
            }
        }
    
        public static void main(String[] args) {
            for (int i = 0; i < 100; i++) {
                System.out.println(Thread.currentThread().getName() + " " + i);
                if (i == 20) {
                    ThreadSimple threadSimple = new ThreadSimple();
                    new Thread(threadSimple, "新线程1").start();
                }
            }
        }
    
    } 
    
    3、通过Callable和Future创建线程

    具体是创建Callable接口的实现类,并实现call()方法。并使用FutureTask类来包装Callable实现类的对象,且以此FutureTask对象作为Thread对象的target来创建线程。
    我们发现FutureTask类实际上是同时实现了Runnable和Future接口,由此才使得其具有Future和Runnable双重特性。通过Runnable特性,可以作为Thread对象的target,而Future特性,使得其可以取得新创建线程中的call()方法的返回值。

    public class ThreadSimple implements Callable<Integer> {
    
    
        public static void main(String[] args) {
    
            Callable<Integer> myCallable = new ThreadSimple();    // 创建myCallable对象
            FutureTask<Integer> ft = new FutureTask<Integer>(myCallable); //使用FutureTask来包装myCallable对象
    
            for (int i = 0; i < 31; i++) {
                System.out.println(Thread.currentThread().getName() + " " + i);
                if (i == 30) {
                    Thread thread = new Thread(ft);   //FutureTask对象作为Thread对象的target创建新的线程
                    thread.start();                      //线程进入到就绪状态
                }
            }
    
            System.out.println("主线程for循环执行完毕..");
    
            try {
                int sum = ft.get();            //取得新创建的新线程中的call()方法返回的结果
                System.out.println("sum = " + sum);
            } catch (InterruptedException e) {
                e.printStackTrace();
            } catch (ExecutionException e) {
                e.printStackTrace();
            }
    
        }
    
        private int i = 0;
    
        // 与run()方法不同的是,call()方法具有返回值
        @Override
        public Integer call() {
            int sum = 0;
            for (; i < 100; i++) {
                System.out.println(Thread.currentThread().getName() + " " + i);
                sum += i;
            }
            return sum;
        }
    }
    

    执行下此程序,我们发现sum = 4950永远都是最后输出的。而“主线程for循环执行完毕..”则很可能是在子线程循环中间输出。由CPU的线程调度机制,我们知道,“主线程for循环执行完毕..”的输出时机是没有任何问题的,那么为什么sum =4950会永远最后输出呢?
    原因在于通过ft.get()方法获取子线程call()方法的返回值时,当子线程此方法还未执行完毕,ft.get()方法会一直阻塞,直到call()方法执行完毕才能取到返回值。

    上述主要讲解了三种常见的线程创建方式,对于线程的启动而言,都是调用线程对象的start()方法,需要特别注意的是:不能对同一线程对象两次调用start()方法。


    Java的线程是不允许启动两次的,第二次调用必然会抛出IllegalThreadStateException,这是一种运行时异常,多次调用start被认为是编程错误。
    线程是一个动态执行的过程,它有一个从产生到死亡的过程,共五种状态:新建(newThread)、就绪(runnable)、运行(running)、死亡(dead)、堵塞(blocked)
    在第二次调用start()方法的时候,线程可能处于终止或者其他(非NEW)状态,但是不论如何,都是不可以再次启动的。

    Callable接口和Runnable接口的区别?
    Callable接口的线程执行体是call()方法,而Runnable接口的线程执行体是run()方法
    call()方法可以有返回值,也可以抛异常。run()方法不能。可以把Callable接口看做Runnable接口的增强版

    相关文章

      网友评论

        本文标题:创建线程的3种方法

        本文链接:https://www.haomeiwen.com/subject/ekpggctx.html