`
lhgyy00
  • 浏览: 145200 次
  • 性别: Icon_minigender_2
  • 来自: 北京
社区版块
存档分类
最新评论

java基础 ThreadLocal

 
阅读更多

今天学习了ThreadLocal,和大家分享下,理解的不是很透彻。

API中的解释是:

该类提供了线程局部 变量。这些变量不同于它们的普通对应物,因为访问某个变量(通过其 getset 方法)的每个线程都有自己的局部变量,它独立于变量的初始化副本。ThreadLocal 实例通常是类中的 private static 字段,它们希望将状态与某一个线程(例如,用户 ID 或事务 ID)相关联。

网上一些说是(1)为了解决多线程并发的问题,我没有看出来;

另有一些说法是(2)ThreadLocal和多线程并发没有什么关系。ThreadLocal模式是为了解决单线程内的跨类跨方法调用的,有点AOP的味道;

此处待慢慢学习,不做评论。

 

实现原理,这边写了一个简单的SmipleThreadLocal的例子,代码粗糙,但是和ThreadLocal的设计思路是一样的

 

 

 

import java.util.Collections;
import java.util.HashMap;
import java.util.Map;

/**
 * @author Eva

 * threadLocal 的雏形
 */
public class SimpleThreadLocal {
 private Map valueMap = Collections.synchronizedMap(new HashMap());

 public void set(Object newValue) {
  valueMap.put(Thread.currentThread(), newValue);// 键为线程对象,值为本线程的变量副本
 }

 public Object get() {
  Thread thread = Thread.currentThread();
  Object o = valueMap.get(thread);// 本线程对应的变量
  if (o == null && !valueMap.containsKey(thread)) {
   o = initialValue();// 如果map中不存在,则初始化并且放在Map中
   valueMap.put(thread, o);
  }
  return o;

 }

 public Object initialValue() {
  return null;
 }

 public void remove() {
  valueMap.remove(Thread.currentThread());
 }
}

 

 

ThreadLocal的接口方法:

  • void set(Object value)

设置当前线程的线程局部变量的值。

  • public Object get()

该方法返回当前线程所对应的线程局部变量。

  • public void remove()

将当前线程局部变量的值删除,目的是为了减少内存的占用,该方法是JDK 5.0新增的方法。需要指出的是,当线程结束后,对应该线程的局部变量将自动被垃圾回收,所以显式调用该方法清除线程的局部变量并不是必须的操作,但它可以加快内存回收的速度。

  • protected Object initialValue()

返回该线程局部变量的初始值,该方法是一个protected的方法,显然是为了让子类覆盖而设计的。这个方法是一个延迟调用方法,在线程第1次调用get()或set(Object)时才执行,并且仅执行1次。ThreadLocal中的缺省实现直接返回一个null。

值得一提的是,在JDK5.0中,ThreadLocal已经支持泛型,该类的类名已经变为ThreadLocal<T>。API方法也相应进行了调整,新版本的API方法分别是void set(T value)、T get()以及T initialValue()。

 

ThreadLocal的使用

(1)自己写了一个多线程的:

public class SequenceNumber {
 // 通过匿名内部类ThreadLocal的initalValue方法指定初始值
 private static ThreadLocal seqNum = new ThreadLocal() {
  protected Object initialValue() {
   return new Integer(0);
  };
 };

 // 获得下一个序列值
 public int getNextNum() { 
  seqNum.set(new Integer(Integer.parseInt(seqNum.get().toString()) + 1));
  return Integer.parseInt(seqNum.get().toString());
 }

 public static void main(String[] args) {
  SequenceNumber sn = new SequenceNumber();
  TestClient t1 = new TestClient(sn);
  TestClient t2 = new TestClient(sn);
  TestClient t3 = new TestClient(sn);
  t1.start();
  t2.start();
  t3.start();
 }

 private static class TestClient extends Thread {
  private SequenceNumber sn;

  public TestClient(SequenceNumber sn) {
   this.sn = sn;
  }

  public void run() {
   for (int i = 0; i < 3; i++) {
    System.out.println("thread[" + Thread.currentThread().getName()
      + "]===sn[" + sn.getNextNum()+ "]");
   }
  }
 }
}

输出结果:

//------------------------------------------------------------------

thread[Thread-1]===sn[1]
thread[Thread-1]===sn[2]
thread[Thread-1]===sn[3]
thread[Thread-2]===sn[1]
thread[Thread-2]===sn[2]
thread[Thread-2]===sn[3]
thread[Thread-0]===sn[1]
thread[Thread-0]===sn[2]
thread[Thread-0]===sn[3]
//------------------------------------------------------------------

 

 

 

 

但是本人觉得这个不足以说明他是解决多线程并发的问题。

 

 

 

 (2)下面的两个是在网上摘抄的:

  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()就可以获取该线程上次获取过的连接实例。

(3) 当要给线程初始化一个特殊值时,需要自己实现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);

  }

  }

  }

 

 

 

 

分享到:
评论

相关推荐

    java基础面试考察点.pdf

    Java 基础面试考察点 Java 基础面试考察点是 Java 开发人员必须掌握的知识点,涵盖了 Java 基础知识、多线程、JVM 相关知识点等几个方面。 Java 基础知识点 1. Java 内部集合类:ArrayList 和 LinkedList 的区别...

    Java并发编程中ThreadLocal的原理与应用分析

    适合人群:具备一定Java基础,特别是对并发编程有一定认识的研发人员或进阶程序员。 使用场景及目标:主要用于理解和解决在高并发环境下线程安全问题的方法和技术。通过本文的学习,能够掌握ThreadLocal的基本用法...

    java面试基础

    此外,线程和并发编程是Java中的重要部分,你需要熟悉线程的基本操作,如synchronized关键字、volatile、ThreadLocal,以及并发工具类如Semaphore、CountDownLatch和CyclicBarrier。 在工具方面,了解如何使用常用...

    传智博客JAVA基础笔记个人总结

    传智博客的JAVA基础笔记个人总结是一份全面的学习资源,旨在帮助初学者或有一定基础的开发者巩固和提升Java编程技能。这份笔记是基于传智播客的课程内容,结合了牛牧老师和水镜老师的讲解精华,提供了27天的学习计划...

    threadLocal

    1. 多线程:理解ThreadLocal的使用必须建立在对多线程的理解基础上,包括线程的创建、执行、同步机制等。 2. 并发编程:ThreadLocal是解决并发问题的一种策略,它提供了一种避免共享状态的方式,减少了锁的使用。 3....

    Quartz-ThreadLocal.rar

    Quartz 是一个开源的作业调度框架,用于在 Java 应用程序中安排任务。...同时,还可以深入学习 Java 基础,提升编程能力。记得实践是检验真理的唯一标准,动手操作这些 demo,会让你对这些知识点有更深入的理解。

    ThreadLocal简单Demo

    **线程局部变量(ThreadLocal)** 在Java编程中,`ThreadLocal`是...以上就是关于`ThreadLocal`及其内部类`ThreadLocalMap`的基础知识,它们在多线程编程中起到关键作用,帮助开发者实现高效、安全的线程局部变量管理。

    一份通俗易懂、风趣幽默的Java学习指南内容涵盖Java基础、Java并发编程、Java虚拟机、Java企业级开发、Java面试

    1. **Java基础**:这是所有Java学习的起点,包括语法基础、数据类型、控制结构、类与对象、接口、异常处理等。学习者应掌握面向对象编程的基本概念,理解封装、继承和多态性,并熟练运用Java的基础API,如集合框架、...

    java基础.docx

    Java基础是程序开发中的核心部分,对于任何Java开发者来说,理解和掌握这些基本概念至关重要。本文主要探讨了几个关键的知识点,包括设计模式和Java基础知识,尤其是多线程。 设计模式是软件开发中的一种最佳实践,...

    Java基础知识点 - 内容比较全面

    下面将详细讲解标题和描述中提及的一些核心Java基础知识。 1. **Java中的引用概念**:在Java中,引用是对象的一个指针,它指向内存中的对象实例。有强引用、软引用、弱引用和虚引用四种类型,分别对应不同的内存...

    Java在面试中的高频考点Java基础、Java集合、Java多线程与并发编程、Java虚拟机、数据库、计算机基础、框架和中间件

    1. **Java基础**:Java的基础包括语法、数据类型、运算符、流程控制、类与对象、封装、继承和多态等。面试时,面试官可能会询问异常处理、访问修饰符、构造函数、接口和内部类的理解及其使用场景。 2. **Java集合**...

    Java知识点总结,面试必备,java基础、java集合、JVM、Java并发

    1. **Java基础**:Java的基础知识包括语法特性,如类、对象、封装、继承、多态等面向对象编程概念。此外,还有异常处理、输入/输出流、集合框架的基础知识,以及Java标准库中的常用类和接口,如String、ArrayList、...

    免费分享 Java面试笔记 面试八股文 计算机网络基础

    本资源适用于运营商、Java开发校招面试基础巩固,包含计算机网络、Java基础、Java集合、Java并发编程、JVM、MySQL、Spring、MyBatis、Redis、Rocket MQ的经典面试题目,涵盖每个知识点的各个方面,能够很好的提升...

    JAVA语言自学全套教程百度云链接

    1. **Java基础**:从安装Java开发环境(JDK)开始,介绍Java的历史和特点。接着会深入讲解基本语法,如变量、数据类型、运算符、流程控制语句(如if,for,while),以及方法的定义与调用。 2. **面向对象编程**:...

    java笔试题大集合及答案

    1. **Java基础知识**:这部分通常包括Java语法、面向对象编程、异常处理、IO流、集合框架等内容。你需要理解类、对象、继承、封装、多态等核心概念,熟练掌握异常处理机制,熟悉各种IO操作,以及List、Set、Map等...

    Java基础面试题集锦-20230713.docx

    【Java基础面试题集锦】 Java作为一门广泛应用于企业级应用开发、互联网服务以及移动应用的语言,其基础知识的掌握对于成为一名合格的Java开发者至关重要。Java之父张孝祥的整理,旨在帮助程序员回顾和巩固核心的...

    2016 Java 基础 面试题

    ### 2016 Java 基础面试题解析 #### 1. **throw 和 throws 的区别** 在Java中,`throw`关键字用于手动抛出一个异常,而`throws`...以上就是2016年Java基础面试题的相关知识点解析,希望对准备面试的朋友们有所帮助。

    java基础125道笔试题

    ### Java基础125道笔试题概览 #### 1. 面向对象基础 **描述:** 面向对象编程是Java的核心特性之一,这里主要涉及类、对象、继承、封装、多态等概念。 **知识点:** - 类与对象的概念及其区别。 - 继承的基本原理...

    编程学习总结 包括 Java基础、Java并发、JVM、MySQL、Redis、Spring、MyBatis、RocketMQ

    1. **Java基础**:Java的基础知识涵盖语法、类、对象、接口、异常处理、集合框架等。理解这些基本概念是进一步深入学习Java的前提。例如,了解如何声明变量、编写方法、创建类与对象,以及掌握集合类如ArrayList、...

Global site tag (gtag.js) - Google Analytics