亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Java ThreadLocal詳解_動力節(jié)點Java學院整理

 更新時間:2017年06月15日 14:55:22   作者:海子   
ThreadLocal,很多地方叫做線程本地變量,也有些地方叫做線程本地存儲,本文會詳細的介紹一下,有興趣的可以了解一下

一.對ThreadLocal的理解

ThreadLocal,很多地方叫做線程本地變量,也有些地方叫做線程本地存儲,其實意思差不多??赡芎芏嗯笥讯贾繲hreadLocal為變量在每個線程中都創(chuàng)建了一個副本,那么每個線程可以訪問自己內(nèi)部的副本變量。

這句話從字面上看起來很容易理解,但是真正理解并不是那么容易。

我們還是先來看一個例子:

class ConnectionManager {
   
  private static Connection connect = null;
   
  public static Connection openConnection() {
    if(connect == null){
      connect = DriverManager.getConnection();
    }
    return connect;
  }
   
  public static void closeConnection() {
    if(connect!=null)
      connect.close();
  }
}

假設(shè)有這樣一個數(shù)據(jù)庫鏈接管理類,這段代碼在單線程中使用是沒有任何問題的,但是如果在多線程中使用呢?很顯然,在多線程中使用會存在線程安全問題:

第一,這里面的2個方法都沒有進行同步,很可能在openConnection方法中會多次創(chuàng)建connect;

第二,由于connect是共享變量,那么必然在調(diào)用connect的地方需要使用到同步來保障線程安全,因為很可能一個線程在使用connect進行數(shù)據(jù)庫操作,而另外一個線程調(diào)用closeConnection關(guān)閉鏈接。

所以出于線程安全的考慮,必須將這段代碼的兩個方法進行同步處理,并且在調(diào)用connect的地方需要進行同步處理。

這樣將會大大影響程序執(zhí)行效率,因為一個線程在使用connect進行數(shù)據(jù)庫操作的時候,其他線程只有等待。

那么大家來仔細分析一下這個問題,這地方到底需不需要將connect變量進行共享?事實上,是不需要的。假如每個線程中都有一個connect變量,各個線程之間對connect變量的訪問實際上是沒有依賴關(guān)系的,即一個線程不需要關(guān)心其他線程是否對這個connect進行了修改的。

到這里,可能會有朋友想到,既然不需要在線程之間共享這個變量,可以直接這樣處理,在每個需要使用數(shù)據(jù)庫連接的方法中具體使用時才創(chuàng)建數(shù)據(jù)庫鏈接,然后在方法調(diào)用完畢再釋放這個連接。比如下面這樣:

class ConnectionManager {
   
  private Connection connect = null;
   
  public Connection openConnection() {
    if(connect == null){
      connect = DriverManager.getConnection();
    }
    return connect;
  }
   
  public void closeConnection() {
    if(connect!=null)
      connect.close();
  }
}
 
 
class Dao{
  public void insert() {
    ConnectionManager connectionManager = new ConnectionManager();
    Connection connection = connectionManager.openConnection();
     
    //使用connection進行操作
     
    connectionManager.closeConnection();
  }
}

這樣處理確實也沒有任何問題,由于每次都是在方法內(nèi)部創(chuàng)建的連接,那么線程之間自然不存在線程安全問題。但是這樣會有一個致命的影響:導致服務(wù)器壓力非常大,并且嚴重影響程序執(zhí)行性能。由于在方法中需要頻繁地開啟和關(guān)閉數(shù)據(jù)庫連接,這樣不盡嚴重影響程序執(zhí)行效率,還可能導致服務(wù)器壓力巨大。

那么這種情況下使用ThreadLocal是再適合不過的了,因為ThreadLocal在每個線程中對該變量會創(chuàng)建一個副本,即每個線程內(nèi)部都會有一個該變量,且在線程內(nèi)部任何地方都可以使用,線程之間互不影響,這樣一來就不存在線程安全問題,也不會嚴重影響程序執(zhí)行性能。

但是要注意,雖然ThreadLocal能夠解決上面說的問題,但是由于在每個線程中都創(chuàng)建了副本,所以要考慮它對資源的消耗,比如內(nèi)存的占用會比不使用ThreadLocal要大。

二.深入解析ThreadLocal類

在上面談到了對ThreadLocal的一些理解,那我們下面來看一下具體ThreadLocal是如何實現(xiàn)的。

先了解一下ThreadLocal類提供的幾個方法:

public T get() { }
public void set(T value) { }
public void remove() { }
protected T initialValue() { }

get()方法是用來獲取ThreadLocal在當前線程中保存的變量副本,set()用來設(shè)置當前線程中變量的副本,remove()用來移除當前線程中變量的副本,initialValue()是一個protected方法,一般是用來在使用時進行重寫的,它是一個延遲加載方法,下面會詳細說明。

首先我們來看一下ThreadLocal類是如何為每個線程創(chuàng)建一個變量的副本的。

先看下get方法的實現(xiàn):

第一句是取得當前線程,然后通過getMap(t)方法獲取到一個map,map的類型為ThreadLocalMap。然后接著下面獲取到<key,value>鍵值對,注意這里獲取鍵值對傳進去的是  this,而不是當前線程t。

如果獲取成功,則返回value值。

如果map為空,則調(diào)用setInitialValue方法返回value。

我們上面的每一句來仔細分析:

首先看一下getMap方法中做了什么:

可能大家沒有想到的是,在getMap中,是調(diào)用當期線程t,返回當前線程t中的一個成員變量threadLocals。

那么我們繼續(xù)取Thread類中取看一下成員變量threadLocals是什么:

實際上就是一個ThreadLocalMap,這個類型是ThreadLocal類的一個內(nèi)部類,我們繼續(xù)取看ThreadLocalMap的實現(xiàn):

可以看到ThreadLocalMap的Entry繼承了WeakReference,并且使用ThreadLocal作為鍵值。

然后再繼續(xù)看setInitialValue方法的具體實現(xiàn):

很容易了解,就是如果map不為空,就設(shè)置鍵值對,為空,再創(chuàng)建Map,看一下createMap的實現(xiàn):

至此,可能大部分朋友已經(jīng)明白了ThreadLocal是如何為每個線程創(chuàng)建變量的副本的:

首先,在每個線程Thread內(nèi)部有一個ThreadLocal.ThreadLocalMap類型的成員變量threadLocals,這個threadLocals就是用來存儲實際的變量副本的,鍵值為當前ThreadLocal變量,value為變量副本(即T類型的變量)。

初始時,在Thread里面,threadLocals為空,當通過ThreadLocal變量調(diào)用get()方法或者set()方法,就會對Thread類中的threadLocals進行初始化,并且以當前ThreadLocal變量為鍵值,以ThreadLocal要保存的副本變量為value,存到threadLocals。

然后在當前線程里面,如果要使用副本變量,就可以通過get方法在threadLocals里面查找。

下面通過一個例子來證明通過ThreadLocal能達到在每個線程中創(chuàng)建變量副本的效果:

public class Test {
  ThreadLocal<Long> longLocal = new ThreadLocal<Long>();
  ThreadLocal<String> stringLocal = new ThreadLocal<String>();
 
   
  public void set() {
    longLocal.set(Thread.currentThread().getId());
    stringLocal.set(Thread.currentThread().getName());
  }
   
  public long getLong() {
    return longLocal.get();
  }
   
  public String getString() {
    return stringLocal.get();
  }
   
  public static void main(String[] args) throws InterruptedException {
    final Test test = new Test();
     
     
    test.set();
    System.out.println(test.getLong());
    System.out.println(test.getString());
   
     
    Thread thread1 = new Thread(){
      public void run() {
        test.set();
        System.out.println(test.getLong());
        System.out.println(test.getString());
      };
    };
    thread1.start();
    thread1.join();
     
    System.out.println(test.getLong());
    System.out.println(test.getString());
  }
}

這段代碼的輸出結(jié)果為:

從這段代碼的輸出結(jié)果可以看出,在main線程中和thread1線程中,longLocal保存的副本值和stringLocal保存的副本值都不一樣。最后一次在main線程再次打印副本值是為了證明在main線程中和thread1線程中的副本值確實是不同的。

總結(jié)一下:
  1)實際的通過ThreadLocal創(chuàng)建的副本是存儲在每個線程自己的threadLocals中的;

  2)為何threadLocals的類型ThreadLocalMap的鍵值為ThreadLocal對象,因為每個線程中可有多個threadLocal變量,就像上面代碼中的longLocal和stringLocal;

  3)在進行g(shù)et之前,必須先set,否則會報空指針異常;

如果想在get之前不需要調(diào)用set就能正常訪問的話,必須重寫initialValue()方法。

因為在上面的代碼分析過程中,我們發(fā)現(xiàn)如果沒有先set的話,即在map中查找不到對應(yīng)的存儲,則會通過調(diào)用setInitialValue方法返回i,而在setInitialValue方法中,有一個語句是T value = initialValue(), 而默認情況下,initialValue方法返回的是null。

看下面這個例子:

public class Test {
  ThreadLocal<Long> longLocal = new ThreadLocal<Long>();
  ThreadLocal<String> stringLocal = new ThreadLocal<String>();
 
  public void set() {
    longLocal.set(Thread.currentThread().getId());
    stringLocal.set(Thread.currentThread().getName());
  }
   
  public long getLong() {
    return longLocal.get();
  }
   
  public String getString() {
    return stringLocal.get();
  }
   
  public static void main(String[] args) throws InterruptedException {
    final Test test = new Test();
     
    System.out.println(test.getLong());
    System.out.println(test.getString());
 
    Thread thread1 = new Thread(){
      public void run() {
        test.set();
        System.out.println(test.getLong());
        System.out.println(test.getString());
      };
    };
    thread1.start();
    thread1.join();
     
    System.out.println(test.getLong());
    System.out.println(test.getString());
  }
}

在main線程中,沒有先set,直接get的話,運行時會報空指針異常。

但是如果改成下面這段代碼,即重寫了initialValue方法:

public class Test {
  ThreadLocal<Long> longLocal = new ThreadLocal<Long>(){
    protected Long initialValue() {
      return Thread.currentThread().getId();
    };
  };
  ThreadLocal<String> stringLocal = new ThreadLocal<String>(){;
    protected String initialValue() {
      return Thread.currentThread().getName();
    };
  };
 
   
  public void set() {
    longLocal.set(Thread.currentThread().getId());
    stringLocal.set(Thread.currentThread().getName());
  }
   
  public long getLong() {
    return longLocal.get();
  }
   
  public String getString() {
    return stringLocal.get();
  }
   
  public static void main(String[] args) throws InterruptedException {
    final Test test = new Test();
 
    test.set();
    System.out.println(test.getLong());
    System.out.println(test.getString());
   
     
    Thread thread1 = new Thread(){
      public void run() {
        test.set();
        System.out.println(test.getLong());
        System.out.println(test.getString());
      };
    };
    thread1.start();
    thread1.join();
     
    System.out.println(test.getLong());
    System.out.println(test.getString());
  }
}

就可以直接不用先set而直接調(diào)用get了。

三.ThreadLocal的應(yīng)用場景

最常見的ThreadLocal使用場景為 用來解決 數(shù)據(jù)庫連接、Session管理等。

如:

private static ThreadLocal<Connection> connectionHolder
= new ThreadLocal<Connection>() {
public Connection initialValue() {
  return DriverManager.getConnection(DB_URL);
}
};
 
public static Connection getConnection() {
return connectionHolder.get();
}
private static final ThreadLocal threadSession = new ThreadLocal();
 
public static Session getSession() throws InfrastructureException {
  Session s = (Session) threadSession.get();
  try {
    if (s == null) {
      s = getSessionFactory().openSession();
      threadSession.set(s);
    }
  } catch (HibernateException ex) {
    throw new InfrastructureException(ex);
  }
  return s;
}

相關(guān)文章

  • java數(shù)組與以逗號分隔開的字符串的相互轉(zhuǎn)換操作

    java數(shù)組與以逗號分隔開的字符串的相互轉(zhuǎn)換操作

    這篇文章主要介紹了java數(shù)組與以逗號分隔開的字符串的相互轉(zhuǎn)換操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-09-09
  • SpringBoot+MyBatis-Plus實現(xiàn)分頁功能

    SpringBoot+MyBatis-Plus實現(xiàn)分頁功能

    在SpringBoot項目中,結(jié)合MyBatis-Plus(簡稱MP)可以非常方便地實現(xiàn)分頁功能,MP為開發(fā)者提供了分頁插件PaginationInterceptor,只需簡單配置即可使用,本文給大家介紹了SpringBoot+MyBatis-Plus實現(xiàn)分頁功能,文中通過代碼示例給大家介紹的非常詳細,需要的朋友可以參考下
    2024-01-01
  • Spring Boot 實例代碼之通過接口安全退出

    Spring Boot 實例代碼之通過接口安全退出

    這篇文章主要介紹了Spring Boot 實例代碼之通過接口安全退出的相關(guān)資料,需要的朋友可以參考下
    2017-09-09
  • Java子類實例化總是默認調(diào)用父類的無參構(gòu)造操作

    Java子類實例化總是默認調(diào)用父類的無參構(gòu)造操作

    這篇文章主要介紹了Java子類實例化總是默認調(diào)用父類的無參構(gòu)造操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-10-10
  • Hibernate使用hbm.xml配置映射關(guān)系解析

    Hibernate使用hbm.xml配置映射關(guān)系解析

    這篇文章主要介紹了Hibernate使用hbm.xml配置映射關(guān)系解析,具有一定參考價值,需要的朋友可以了解下。
    2017-11-11
  • springboot如何讀取sftp的文件

    springboot如何讀取sftp的文件

    這篇文章主要介紹了springboot如何讀取sftp的文件,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-11-11
  • SpringSecurity+OAuth2.0?搭建認證中心和資源服務(wù)中心流程分析

    SpringSecurity+OAuth2.0?搭建認證中心和資源服務(wù)中心流程分析

    OAuth?2.0?主要用于在互聯(lián)網(wǎng)上安全地委托授權(quán),廣泛應(yīng)用于身份驗證和授權(quán)場景,這篇文章介紹SpringSecurity+OAuth2.0?搭建認證中心和資源服務(wù)中心,感興趣的朋友一起看看吧
    2024-01-01
  • Java ArrayList的底層實現(xiàn)方法

    Java ArrayList的底層實現(xiàn)方法

    今天小編就為大家分享一篇Java ArrayList的底層實現(xiàn)方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-10-10
  • 使用vscode搭建javaweb項目的詳細步驟

    使用vscode搭建javaweb項目的詳細步驟

    我個人是很喜歡VsCode的,開源免費、功能全面,所以為了方便,我把我?guī)缀跛械倪\行都集成到了VsCode上來,JavaWeb也不例外,下面這篇文章主要給大家介紹了關(guān)于使用vscode搭建javaweb項目的相關(guān)資料,需要的朋友可以參考下
    2022-11-11
  • Java騷操作之CountDownLatch代碼詳解

    Java騷操作之CountDownLatch代碼詳解

    這篇文章主要介紹了Java騷操作之CountDownLatch代碼詳解,本文通過實例圖文相結(jié)合給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-02-02

最新評論