008-ThreadLocal
一、基本用法
ThreadLocal是一个容器,用于存放线程的局部变量。如果ThreadLocalVariable(线程局部变量)更加好理解。
在Jdk 1.2 java.lang.ThreadLocal开始使用,他是为解决多线程并发设计的.
示例序列号生成,保证每个线程生成唯一序列号
编写Sequence接口
package com.lhx.chapter4.threadlocaltest;
public interface Sequence {
int getNumber();
}
编写线程类
package com.lhx.chapter4.threadlocaltest;
public class ClientThread extends Thread {
private Sequence sequence;
public ClientThread(Sequence sequence){
this.sequence=sequence;
}
@Override
public void run() {
for(int i=0;i<3;i++){
System.out.println(Thread.currentThread().getName()+"=>"+sequence.getNumber());
}
}
}
实现方式一
package com.lhx.chapter4.threadlocaltest;
public class SequenceA implements Sequence {
private static int number = 0;
@Override
public int getNumber() {
number=number+1;
return number;
}
public static void main(String[] args) {
Sequence sequence = new SequenceA();
ClientThread clientThread1 = new ClientThread(sequence);
ClientThread clientThread2 = new ClientThread(sequence);
ClientThread clientThread3 = new ClientThread(sequence);
clientThread1.start();
clientThread2.start();
clientThread3.start();
}
}
Thread-0=>1
Thread-1=>2
Thread-0=>3
Thread-1=>4
Thread-0=>5
Thread-1=>6
Thread-2=>7
Thread-2=>8
Thread-2=>9
由输出可以线程间共享一个Static变量,无法保证线程安全。
示例2,
package com.lhx.chapter4.threadlocaltest;
public class SequenceB implements Sequence {
private static ThreadLocal<Integer> numberContainer = new ThreadLocal<Integer>(){
@Override
protected Integer initialValue() {
return 0;
}
};
@Override
public int getNumber() {
numberContainer.set(numberContainer.get()+1);
return numberContainer.get();
}
public static void main(String[] args) {
Sequence sequence = new SequenceB();
ClientThread clientThread1 = new ClientThread(sequence);
ClientThread clientThread2 = new ClientThread(sequence);
ClientThread clientThread3 = new ClientThread(sequence);
clientThread1.start();
clientThread2.start();
clientThread3.start();
}
}
输出值:
Thread-0=>1
Thread-0=>2
Thread-0=>3
Thread-1=>1
Thread-1=>2
Thread-1=>3
Thread-2=>1
Thread-2=>2
Thread-2=>3
可以看出每个线程变量独立开来,也就是说ThreadLocal为每个线程提供了一个副本。
ThreadLocal的API,
public void set(T value)将值放入线程局部变量中
public T get() 从线程局部变量中获取值
public void remove()从线程局部变量中移除
protected T initialValue()返回线程局部变量的初始值(默认为null),必须默认实现。
自定义实现ThreadLocal
package com.lhx.chapter4.threadlocaltest; import java.util.Collections;
import java.util.HashMap;
import java.util.Map; public class MyThreadLocal<T> {
private Map<Thread,T> container=Collections.synchronizedMap(new HashMap<Thread,T>());
public void set(T value){
container.put(Thread.currentThread(),value);
}
public T get(){
Thread thread = Thread.currentThread();
T value = container.get(thread);
if(value==null&&!container.containsKey(thread)){
value=initialValue();
container.put(thread,value);
}
return value;
}
public void remove(){
container.remove(Thread.currentThread());
} protected T initialValue(){
return null;
} }
用法一致
二、使用案例
通过ThreadLocal存放Jdbc Connection,以达到事务控制能力。
示例,修改价格,记录日志
update product set price=? where id=?
insert into log(crerated,description)values(?,?)
编写工具类DBUtil
package com.lhx.test.dbutil1; import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.SQLException; public class DBUtil {
private static final String driver = "com.mysql.jdbc.Driver";
private static final String url = "jdbc:mysql://localhost:3306/demo";
private static final String username = "root";
private static final String password = "root"; private static Connection conn = null; public static Connection getConnection() {
try {
Class.forName(driver);
conn = DriverManager.getConnection(url, username, password);
} catch (Exception e) {
e.printStackTrace();
}
return conn;
} public static void colseConnection() {
try {
if(conn!=null){
conn.close();
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
编写ProductService
package com.lhx.test.dbutil1;
public interface ProductService {
void updateProductPrice(long productId,int price);
}
编写ProductService实现类编写ProductServiceImpl
package com.lhx.test.dbutil1; import java.sql.Connection;
import java.sql.PreparedStatement;
import java.text.SimpleDateFormat;
import java.util.Date; public class ProductServiceImpl implements ProductService { private static final String Update_Product_sql = "update product set price=? where id=?";
private static final String Insert_Log_sql = "insert into log(crerated,description)values(?,?)"; @Override
public void updateProductPrice(long productId, int price) {
try {
Connection connection = DBUtil.getConnection();
connection.setAutoCommit(false);
updateProduct(connection,Update_Product_sql,productId,price);
insertLog(connection,Update_Product_sql,"Create Product");
connection.commit();
} catch (Exception e) {
e.printStackTrace();
}finally {
DBUtil.colseConnection();
}
}
private void updateProduct(Connection connection,String sql,long productId,int price) throws Exception{
PreparedStatement pstmt = connection.prepareStatement(sql);
pstmt.setInt(1,price);
pstmt.setLong(2,productId);
int rows = pstmt.executeUpdate();
if(rows!=0){
System.out.println("update product success");
}
}
private void insertLog(Connection connection,String sql,String msg) throws Exception{
PreparedStatement pstmt = connection.prepareStatement(sql);
pstmt.setString(1,new SimpleDateFormat("yyyy-MM-dd HH:mm:ss.sss").format(new Date()));
pstmt.setString(2,msg);
int rows = pstmt.executeUpdate();
if(rows!=0){
System.out.println("insert log success");
}
}
}
个人测试
public static void main(String[] args) {
ProductService productService = new ProductServiceImpl();
productService.updateProductPrice(1,3000);
}
并发测试
package com.lhx.test.dbutil1;
import com.lhx.test.dbutil1.ProductService;
public class ClientThread extends Thread {
private ProductService productService;
public ClientThread(ProductService productService) {
this.productService = productService;
}
@Override
public void run() {
System.out.println(Thread.currentThread().getName());
productService.updateProductPrice(1, 3000);
}
}
public static void main(String[] args) {
for (int i = 0; i < 10; i++) {
ProductService productService = new ProductServiceImpl();
ClientThread clientThread = new ClientThread(productService);
clientThread.start();
}
}
使用ThreadLocal编写工具类
package com.lhx.test.dbutil2; import java.sql.Connection;
import java.sql.DriverManager; public class DBUtil {
private static final String driver = "com.mysql.jdbc.Driver";
private static final String url = "jdbc:mysql://localhost:3306/demo";
private static final String username = "root";
private static final String password = "root"; private static ThreadLocal<Connection> conContainer = new ThreadLocal<>(); public static Connection getConnection() {
Connection conn = conContainer.get();
try {
if (conn == null) {
Class.forName(driver);
conn = DriverManager.getConnection(url, username, password);
}
} catch (Exception e) {
e.printStackTrace();
}
finally {
conContainer.set(conn);
}
return conn;
} public static void colseConnection() {
Connection conn = conContainer.get();
try {
if (conn != null) {
conn.close();
}
} catch (Exception e) {
e.printStackTrace();
}
finally {
conContainer.remove();
}
}
}
008-ThreadLocal的更多相关文章
- ThreadLocal简单理解
在java开源项目的代码中看到一个类里ThreadLocal的属性: private static ThreadLocal<Boolean> clientMode = new Thread ...
- Android线程管理之ThreadLocal理解及应用场景
前言: 最近在学习总结Android的动画效果,当学到Android属性动画的时候大致看了下源代码,里面的AnimationHandler存取使用了ThreadLocal,激起了我很大的好奇心以及兴趣 ...
- Threadlocal使用Case
Threadlocal能够为每个线程分配一份单独的副本,使的线程与线程之间能够独立的访问各自副本.Threadlocal 内部维护一个Map,key为线程的名字,value为对应操作的副本. /** ...
- 多线程映射工具——ThreadLocal
ThreadLocal相当于一个Map<Thread, T>,各线程使用自己的线程对象Thread.currentThread()作为键存取数据,但ThreadLocal实际上是一个包装了 ...
- ThreadLocal 工作原理、部分源码分析
1.大概去哪里看 ThreadLocal 其根本实现方法,是在Thread里面,有一个ThreadLocal.ThreadLocalMap属性 ThreadLocal.ThreadLocalMap t ...
- ThreadLocal<T>的是否有设计问题
一.吐槽 ThreadLocal<T>明显是.NET从JAVA中来的一个概念,但是这种设计是否出现了问题. 很明显,在JAVA中threadLocal直接是Thread的成员,当然随着th ...
- 理解ThreadLocal —— 一个map的key
作用: 当工作于多线程中的对象使用ThreadLocal维护变量时,threadLocal为每个使用该变量的线程分配一个独立的变量副本. 接口方法: protected T initialValue( ...
- JavaSe:ThreadLocal
JDK中有一个ThreadLocal类,使用很方便,但是却很容易出现问题.究其原因, 就是对ThreadLocal理解不到位.最近项目中,出现了内存泄漏的问题.其中就有同事在使用ThreadLocal ...
- 0041 Java学习笔记-多线程-线程池、ForkJoinPool、ThreadLocal
什么是线程池 创建线程,因为涉及到跟操作系统交互,比较耗费资源.如果要创建大量的线程,而每个线程的生存期又很短,这时候就应该使用线程池了,就像数据库的连接池一样,预先开启一定数量的线程,有任务了就将任 ...
- ThreadLocal 源码剖析
ThreadLocal是Java语言提供的用于支持线程局部变量的类.所谓的线程局部变量,就是仅仅只能被本线程访问,不能在线程之间进行共享访问的变量(每个线程一个拷贝).在各个Java web的各种框架 ...
随机推荐
- Python - pandas 数据分析
pandas: powerful Python data analysis toolkit 官方文档: http://pandas.pydata.org/pandas-docs/stable/ 1. ...
- C#之正则表达式验证
/** 匹配身份证号 规则: 15位纯数字或者18位纯数字或者17位数字加一位x */ var regex = new System.Text.RegularExpressions.Regex(@&q ...
- css3中-moz、-ms、-webkit,-o分别代表的意思
这种方式在业界上统称:识别码.前缀 //-ms代表[ie]内核识别码 //-moz代表火狐[firefox]内核识别码 //-webkit代表谷歌[chrome]/苹果[safari]内核识别码 // ...
- SQL SERVER 存储过程示例
USE TEST_DEV; SET ANSI_NULLS ON GO SET QUOTED_IDENTIFIER ON GO -- ================================== ...
- 实战c++中的string系列--std::string与MFC中CString的转换
搞过MFC的人都知道cstring,给我们提供了非常多便利的方法. CString 是一种非常实用的数据类型. 它们非常大程度上简化了MFC中的很多操作,使得MFC在做字符串操作的时候方便了非常多.无 ...
- Alluxio部署(local模式)
下载(pre-build for Hadoop 2.7) http://www.alluxio.org/download` 解压 tar -xvf alluxio-1.3.0-hadoop2.7-bi ...
- mysql的两个备份语句
适合多引擎混合(例如:myisam与innodb混合)的备份命令如下: mysqldump -A -R --triggers --master-data=2 --single-transaction ...
- 1. 写出一个能创建多级目录的 PHP 函数(新浪网技术部)
function create_dir($path,$mode){ if (is_dir($path)){ echo "该目录已经存在"; }else{ if(mkdir($pat ...
- IOS 多线程的一些总结
IOS 多线程 有三种主要方法 (1)NSThread (2)NSOperation (3)** 下面简单介绍这三个方法 1.NSThread 调用方法如下: 如函数需要输入参数, ...
- iPhone SDK中多线程的使用方法以及注意事项
多线程iphonethreadapplication编程嵌入式 然现在大部分PC应用程序都支持多线程/多任务的开发方式,但是在iPhone上,Apple并不推荐使用多线程的编程方式.但是多线程编程毕竟 ...