2015-04-06 71 views
0

我正在学习java线程,我去了关于本地线程值的课程,我决定做一个使用它们的程序。 我在构造函数(线程构造函数)中设置本地线程值以显示它在屏幕上以检查它是否工作,但是当我启动线程时,本地值变为NULL,因此我得到NullPointerException。 我错过了什么,或者我只能在启动方法中设置本地线程值?在构造函数中设置ThreadLocal变量是否工作?

package practice; 
import static java.lang.System.out; 
import java.util.Scanner; 
class Try 
{ 
    Thread a1,a2,a3; 
    int x=0; 
    synchronized void change(int who){ 
     out.println("who called"+who); 
     out.println("x initial="+x); 
     x++; 
     out.println("x after="+x); 
    } 
    class now implements Runnable{ 
     ThreadLocal<Integer> id=new ThreadLocal<Integer>(); 
     public void run(){ 
      for(int i=1;i<=25;i++){ 
      out.println("im running id="+id.get()); 
      change(id.get()); 
      out.println("after call me="+id.get()); 
      } 
     } 
     now(int givenid){ 
      out.println("my givenid is "+givenid); 
      id.set(givenid); 
      out.println("my id is "+id.get()); 
     } 
    } 
    public static void main(String[] args) 
    {  
     new Try(); 
    } 


    Try(){ 

    a1=new Thread(new now(1)); 
    a2=new Thread(new now(2)); 
    a1.start(); 
    a2.start(); 
    } 

} 
+0

哪里有你用setLength()或initialValue()方法的threadLocal来初始化threadlocal? – 2015-04-06 12:21:31

+0

不要说“ThreadLocal _variable_”。虽然确实有一个变量,并且它的类型是'ThreadLocal ',那么线程局部性不是变量的属性;它是变量引用的_object_的一个属性。 ThreadLocal _object_是get()和set()方法的对象,它们根据哪个线程调用它们来引用不同的位置。 – 2015-04-06 13:56:03

回答

0

我真的很喜欢Banthar的答案,但我只是想说明,仅仅加入了几个印有你的代码会发生什么...

package threads; 

import static java.lang.System.out; 

class Main { 
    Thread a1, a2, a3; 
    int x = 0; 

    public Main() { 
     out.println("Current thread (in Main's constructor): " + Thread.currentThread().getName()); 
     a1 = new Thread(new MyRunnable(1)); 
     a2 = new Thread(new MyRunnable(2)); 
     a1.start(); 
     a2.start(); 
    } 

    private synchronized void change(int who) { 
     out.println("Current thread (in change() method): " + Thread.currentThread().getName()); 
     out.println("who called" + who); 
     out.println("x initial=" + x); 
     x++; 
     out.println("x after=" + x); 
    } 

    private class MyRunnable implements Runnable { 
     ThreadLocal<Integer> id = new ThreadLocal<Integer>(); 

     public MyRunnable(int givenid) { 
      out.println("Current thread (in MyRunnable's constructor): " + Thread.currentThread().getName()); 
      out.println("my givenid is " + givenid); 
      id.set(givenid); 
      out.println("my id is " + id.get()); 
     } 

     public void run() { 
      out.println("Current thread (in run() method): " + Thread.currentThread().getName()); 
      out.println("im running id=" + id.get()); 
      change(id.get()); 
      out.println("after call me=" + id.get()); 
     } 
    } 

    public static void main(String[] args) { 
     new Main(); 
    } 
} 

,输出是:

Current thread (in Main's constructor): main 
Current thread (in MyRunnable's constructor): main 
my givenid is 1 
my id is 1 
Current thread (in MyRunnable's constructor): main 
my givenid is 2 
my id is 2 
Current thread (in run() method): Thread-0 
Current thread (in run() method): Thread-1 
im running id=null 
im running id=null 

对不起,有点改变名字。

正如你所看到的实例MyRunnable(你now类)当您在主要线程设置的ThreadLocal。这就是为什么你不能通过其他线程访问它。

0

这是线程本地人应该如何工作。本地线程就像一个从线程到对象的映射。每个线程都会看到它自己的价值。您正在设置主线程中的值,只有该线程才会看到此值。

我不确定你想在这里实现什么。线程本地通常用于缓存非线程安全对象,这些对象分配起来非常昂贵,并且通过参数传递它们是不可行的。它们通常会存储在静态字段中。例如:

private static ThreadLocal<SimpleDateFormat> FORMAT = new ThreadLocal<SimpleDateFormat>(); 

public static Date parse(final String date) throws ParseException { 
    SimpleDateFormat format = FORMAT.get(); 
    if (format == null) { 
     format = new SimpleDateFormat(); 
     FORMAT.set(format); 
    } 
    return format.parse(date); 
} 

这将缓存每个线程的一个实例SimpleDateFormat

0

您需要在ThreadLocal内设置“包装”值。看看javadocs。您需要致电set()或使用initialValue(),然后您的ThreadLocal将返回除null之外的任何内容。

相关问题