zoukankan      html  css  js  c++  java
  • 多线程-ThreadLocal,InheritableThreadLocal

    ThreadLocal

    变量值得共享可以使用public static变量的形式,所有的线程都使用同一个public static变量。如果想实现每一个线程都有自己的共享变量该如何解决呢?JDK中提供的ThreadLocal正是解决这样的问题。 
    ThreadLocal主要解决的就是每个线程绑定自己的值,可以将ThreadLocal类比喻成全局存放数据的盒子,盒子中可以存储每个线程的私有数据。ThreadLocal解决的是变量在不同线程键的隔离性,也就是不同 线程拥有自己的值,不同线程中的值是可以放入ThreadLocal类中进行保存的。

    package org.github.lujiango;
    
    public class Test03 {
        public static ThreadLocal<Object> tl = new ThreadLocal<Object>();
    
        static class ThreadA extends Thread {
            @Override
            public void run() {
                try {
                    tl.set("ThreadA");
                    Thread.sleep(200);
                    System.out.println("ThreadA get value = " + tl.get());
    
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        }
    
        static class ThreadB extends Thread {
            @Override
            public void run() {
                try {
                    tl.set("ThreadB");
                    Thread.sleep(200);
                    System.out.println("ThreadB get value = " + tl.get());
    
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        }
    
        public static void main(String[] args) {
            try {
                ThreadA a = new ThreadA();
                ThreadB b = new ThreadB();
                a.start();
                b.start();
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    
    }
    

      第一次调用ThreadLocal类的get()方法返回值是null,可以自定义protected T initialValue()方法,设置默认值。

    package org.github.lujiango;
    
    class ThreadLocalExt extends ThreadLocal<String> {
        @Override
        protected String initialValue() {
            return "defalut";
        }
    }
    
    public class Test03 {
        public static ThreadLocalExt tl = new ThreadLocalExt();
    
        static class ThreadA extends Thread {
            @Override
            public void run() {
                try {
                    System.out.println("ThreadA get value = " + tl.get());
                    tl.set("ThreadA");
                    Thread.sleep(200);
                    System.out.println("ThreadA get value = " + tl.get());
    
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        }
    
        static class ThreadB extends Thread {
            @Override
            public void run() {
                try {
                    System.out.println("ThreadB get value = " + tl.get());
                    tl.set("ThreadB");
                    Thread.sleep(200);
                    System.out.println("ThreadB get value = " + tl.get());
    
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        }
    
        public static void main(String[] args) {
            try {
                ThreadA a = new ThreadA();
                ThreadB b = new ThreadB();
                a.start();
                b.start();
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    
    }
    

    InheritableThreadLocal

    InheritableThreadLocal可以在子线程中取得父线程继承下来的值,即可以让子线程从父进程中取得值。

    package org.github.lujiango;
     
    public class Test04 {
        static InheritableThreadLocal<String> itl = new InheritableThreadLocal<String>();
     
        static class ThreadA extends Thread {
            @Override
            public void run() {
                try {
                    System.out.println("ThreadA get: " + itl.get());
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        }
     
        public static void main(String[] args) {
     
            try {
                itl.set("Main");
                Thread.sleep(1000);
                ThreadA a = new ThreadA();
                a.start();
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
     
    }
    

    InheritableThreadLocal即可以设置默认值,也可以在继承父线程值的同时修改。

    package org.github.lujiango;
     
    class InheritableThreadLocalExt extends InheritableThreadLocal<String> {
        @Override
        protected String initialValue() {
            return "defalut";
        }
     
        @Override
        protected String childValue(String parentValue) {
            return parentValue + " child";
        }
    }
     
    public class Test04 {
        static InheritableThreadLocalExt itl = new InheritableThreadLocalExt();
     
        static class ThreadA extends Thread {
            @Override
            public void run() {
                try {
                    Thread.sleep(1000);
                    System.out.println("ThreadA get: " + itl.get());
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        }
     
        public static void main(String[] args) {
     
            try {
                System.out.println(itl.get());
                itl.set("Main1");
                Thread.sleep(1000);
                ThreadA a = new ThreadA();
                a.start();
                itl.set("Main2");
                System.out.println(itl.get());
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
     
    }
    

    注:如果子线程在取得值得同时,主线程将InheritableThreadLocal中的值进行更改,那么子线程取到的值还是旧值。

  • 相关阅读:
    验证数字范围的小插件
    解决EJB懒加载问题
    JS获取按键的代码,Js如何屏蔽用户的按键,Js获取用户按键对应的ASII码(兼容所有浏览器)
    struts2标签之<s:select>
    c#(winform)中自定义ListItem类方便ComboBox和ListBox添加项完全解决
    辞职前须慎重考虑
    怎样把PDF文件在WinForm窗口中显示出来
    加载报表失败
    经典正则表达式 Javascript
    无法生成项目输出组“内容文件来自...
  • 原文地址:https://www.cnblogs.com/lujiango/p/7580790.html
Copyright © 2011-2022 走看看