在传统的同步开发模式下,当我们调用一个函数时,通过这个函数的参数将数据传入,并通过这个函数的返回值来返回最终的计算结果。但在多线程的异步开发模式下,数据的传递和返回和同步开发模式有很大的区别。由于线程的运行和结束是不可预料的,因此,在传递和返回数据时就无法象函数一样通过函数参数和return语句来返回数据。

9.1、通过构造方法传递数据 


在创建线程时,必须要建立一个Thread类的或其子类的实例。因此,我们不难想到在调用start方法之前通过线程类的构造方法将数据传入线程。并将传入的数据使用类变量保存起来,以便线程使用(其实就是在run方法中使用)。下面的代码演示了如何通过构造方法来传递数据: 

 


[java] view plain copy

 



1.    
2. package mythread;   
3. public class MyThread1 extends Thread   
4. {   
5. private String name;   
6. public MyThread1(String name)   
7. {   
8. this.name = name;   
9. }   
10. public void run()   
11. {   
12. System.out.println("hello " + name);   
13. }   
14. public static void main(String[] args)   
15. {   
16. Thread thread = new MyThread1("world");   
17. thread.start();   
18. }   
19. }


由于这种方法是在创建线程对象的同时传递数据的,因此,在线程运行之前这些数据就就已经到位了,这样就不会造成数据在线程运行后才传入的现象。如果要传递更复杂的数据,可以使用集合、类等数据结构。使用构造方法来传递数据虽然比较安全,但如果要传递的数据比较多时,就会造成很多不便。由于Java没有默认参数,要想实现类似默认参数的效果,就得使用重载,这样不但使构造方法本身过于复杂,又会使构造方法在数量上大增。因此,要想避免这种情况,就得通过类方法或类变量来传递数据。 

 

9.2、通过变量和方法传递数据 


向对象中传入数据一般有两次机会,第一次机会是在建立对象时通过构造方法将数据传入,另外一次机会就是在类中定义一系列的public的方法或变量(也可称之为字段)。然后在建立完对象后,通过对象实例逐个赋值。下面的代码是对MyThread1类的改版,使用了一个setName方法来设置 name变量: 

 


[java] view plain copy

 



1.    
2. package mythread;   
3. public class MyThread2 implements Runnable   
4. {   
5. private String name;   
6. public void setName(String name)   
7. {   
8. this.name = name;   
9. }   
10. public void run()   
11. {   
12. System.out.println("hello " + name);   
13. }   
14. public static void main(String[] args)   
15. {   
16. MyThread2 myThread = new MyThread2();   
17. myThread.setName("world");   
18. Thread thread = new Thread(myThread);   
19. thread.start();   
20. }   
21. }




3.通过回调函数传递数据

上面讨论的两种向线程中传递数据的方法是最常用的。但这两种方法都是main方法中主动将数据传入线程类的。这对于线程来说,是被动接收这些数据的。然而,在有些应用中需要在线程运行的过程中动态地获取数据,如在下面代码的run方法中产生了3个随机数,然后通过Work类的process方法求这三个随机数的和,并通过Data类的value将结果返回。从这个例子可以看出,在返回value之前,必须要得到三个随机数。也就是说,这个 value是无法事先就传入线程类的。 

package cn.qblank.thread;

import java.util.Random;

class Data{
    public int value = 0;
}

class Work{
    public void process(Data data,Integer[] numbers){
        for(int n: numbers){
            data.value += n;
        }
    }
}

public class ThreadData extends Thread{
    private Work work;
    public ThreadData(Work work){
        this.work = work;
    }

    @Override
    public void run() {
        Data data = new Data();
        Random random = new Random();
        int n1 = random.nextInt(10);
        int n2 = random.nextInt(100);
        int n3 = random.nextInt(50);
        work.process(data,new Integer[]{n1,n2,n3});
        System.out.println(String.valueOf(n1) + "+" + String.valueOf(n2) + "+"
                + String.valueOf(n3) + "=" + data.value);
    }
    public static void main(String[] args) {
        Thread t = new ThreadData(new Work());
        t.start();
    }
}

java 线程runable 怎么传参数_数据