CONAN ZONE

          你越掙扎我就越興奮

          BlogJava 首頁 新隨筆 聯系 聚合 管理
            0 Posts :: 282 Stories :: 0 Comments :: 0 Trackbacks

          EasyDBO的數據庫連接部分,為了給每個連接提供上下文,程序用到了一個關鍵的類——ThreadLocal。

          什么是ThreadLocal?

          顧名思義它是local variable(線程局部變量)。它的功用非常簡單,就是為每一個使用該變量的線程都提供一個變量值的副本,是每一個線程都可以獨立地改變自己的副本,而不會和其它線程的副本沖突。從線程的角度看,就好像每一個線程都完全擁有該變量。

          使用場景

          1. To keep state with a thread (user-id, transaction-id, logging-id)
          2. To cache objects which you need frequently

          ThreadLocal類

          它主要由四個方法組成initialValue(),get(),set(T),remove(),其中值得注意的是initialValue(),該方法是一個protected的方法,顯然是為了子類重寫而特意實現的。該方法返回當前線程在該線程局部變量的初始值,這個方法是一個延遲調用方法,在一個線程第1次調用get()或者set(Object)時才執行,并且僅執行1次。ThreadLocal中的確實實現直接返回一個null:

          ThreadLocal的原理

          ThreadLocal是如何做到為每一個線程維護變量的副本的呢?其實實現的思路很簡單,在ThreadLocal類中有一個Map,用于存儲每一個線程的變量的副本。比如下面的示例實現:

          public class ThreadLocal
          {
          private Map values = Collections.synchronizedMap(new HashMap());
          public Object get()
          {
          Thread curThread = Thread.currentThread();
          Object o = values.get(curThread);
          if (o == null && !values.containsKey(curThread))
          {
          o = initialValue();
          values.put(curThread, o);
          }
          return o;
          }

           public void set(Object newValue)
          {
          values.put(Thread.currentThread(), newValue);
          }

           public Object initialValue()
          {
          return null;
          }
          }

          ThreadLocal 的使用

          使用方法一:

          Hibernate的文檔時看到了關于使ThreadLocal管理多線程訪問的部分。具體代碼如下

          1.  public static final ThreadLocal session = new ThreadLocal();
          2.  public static Session currentSession() {
          3.      Session s = (Session)session.get();
          4.      //open a new session,if this session has none
          5.   if(s == null){
          6.      s = sessionFactory.openSession();
          7.      session.set(s);
          8.   }
                return s;
          9. }

          我們逐行分析
          1。 初始化一個ThreadLocal對象,ThreadLocal有三個成員方法 get()、set()、initialvalue()。
              如果不初始化initialvalue,則initialvalue返回null。
          3。session的get根據當前線程返回其對應的線程內部變量,也就是我們需要的net.sf.hibernate.Session(相當于對應每個數據庫連接).多線程情況下共享數據庫鏈接是不安全的。ThreadLocal保證了每個線程都有自己的s(數據庫連接)。
          5。如果是該線程初次訪問,自然,s(數據庫連接)會是null,接著創建一個Session,具體就是行6。
          6。創建一個數據庫連接實例 s
          7。保存該數據庫連接s到ThreadLocal中。
          8。如果當前線程已經訪問過數據庫了,則從session中get()就可以獲取該線程上次獲取過的連接實例。

          使用方法二

          當要給線程初始化一個特殊值時,需要自己實現ThreadLocal的子類并重寫該方法,通常使用一個內部匿名類對ThreadLocal進行子類化,EasyDBO中創建jdbc連接上下文就是這樣做的:

          public class JDBCContext{
           private static Logger logger = Logger.getLogger(JDBCContext.class);
           private DataSource ds;
           protected Connection connection;
           private boolean isValid = true;
           private static ThreadLocal jdbcContext;
           
           private JDBCContext(DataSource ds){
            this.ds = ds;
            createConnection();  
           }
           public static JDBCContext getJdbcContext(javax.sql.DataSource ds)
           {  
            if(jdbcContext==null)jdbcContext=new JDBCContextThreadLocal(ds);
            JDBCContext context = (JDBCContext) jdbcContext.get();
            if (context == null) {
             context = new JDBCContext(ds);
            }
            return context;
           }

           private static class JDBCContextThreadLocal extends ThreadLocal {
            public javax.sql.DataSource ds;
            public JDBCContextThreadLocal(javax.sql.DataSource ds)
            {
             this.ds=ds;
            }
            protected synchronized Object initialValue() {
             return new JDBCContext(ds);
            }
           }
          }

          使用單例模式,不同的線程調用getJdbcContext()獲得自己的jdbcContext,都是通過JDBCContextThreadLocal 內置子類來獲得JDBCContext對象的線程局部變量,這個變量是該線程所獨有的。

          posted on 2009-01-13 12:58 CONAN 閱讀(228) 評論(0)  編輯  收藏 所屬分類: J2SEJAVA
          主站蜘蛛池模板: 舒兰市| 巩义市| 泸溪县| 白朗县| 南安市| 绵竹市| 桃园县| 靖宇县| 连江县| 东明县| 湘阴县| 西和县| 伊宁市| 洛扎县| 扶风县| 泸州市| 江北区| 资溪县| 阿拉善左旗| 来安县| 张家川| 惠州市| 汾阳市| 腾冲县| 池州市| 桐梓县| 九台市| 随州市| 油尖旺区| 济阳县| 宁明县| 桃园县| 玉环县| 万盛区| 丹寨县| 石河子市| 包头市| 和林格尔县| 连州市| 西平县| 浮梁县|