??xml version="1.0" encoding="utf-8" standalone="yes"?>
]]>
and rowid not in (select min(rowid) from 表名 group by id having count(id )>1)
判断d的单元格是否为HSSFCell.CELL_TYPE_NUMERICcdQ然后利用cell.getNumericCellValue()Q读取该单元格的数据?/p>
getNumericCellValue()d的数据类型是double型,
因此Q需要重新进行数据{换:HSSFDateUtil.getJavaDate(d).toLocaleString()。其中d出的doublecd数据?/p>
xQ则成功的将excel表内的数据读取出来?/p>
单元格的格式d有以下几U:
HSSFCell.CELL_TYPE_BLANK;
HSSFCell.CELL_TYPE_BOOLEAN;
HSSFCell.CELL_TYPE_ERROR;
HSSFCell.CELL_TYPE_FORMULA;
HSSFCell.CELL_TYPE_NUMERIC;
HSSFCell.CELL_TYPE_STRING;
PSQ在所有例子中正则表达式匹配结果包含在源文本中的【和】之_有的例子会用java来实玎ͼ如果是java本n正则表达式的用法Q会在相应的地方说明。所有java例子都在JDK1.6.0_13下测试通过?/span>
一、有多少个匹?/span>
前面几篇讲的都是匚w一个字W,但是一个字W或字符集合要匹配多ơ,应该怎么做呢Q比如要匚w一个电子邮件地址Q用之前说到的方法,可能有h会写出像\w@\w\.\wq样的正则表辑ּQ但q个只能匚w到像a@b.cq样的地址Q明显是不正的Q接下来来看看如何匚w电子邮g地址?/span>
首先要知道电子邮件地址的组成:以字母数字或下划U开头的一l字W,后面跟@W号Q再后面是域名,即用户名@域名地址。不q这也跟具体的邮服务提供商有关Q有的在用户名中也允?字符?/span>
1、匹配一个或多个字符
要想匚w同一个字W(或字W集合)的多ơ重复,只要单地l这个字W(或字W集合)加上一?字符作ؓ后缀可以了?匚w一个或多个字符Q至一个)。如Qa匚wa本nQa+匹配一个或多个q箋出现的aQ[0-9]+匚w多个q箋的数字?/span>
注意Q在l一个字W集合加?后缀的时候,必须?攑֜字符集合的外面,否则׃是重复匹配了。如[0-9+]q样pC数字或+号了Q虽然语法上正确Q但不是我们惌的了?/span>
文本QHello, mhmyqn@qq.com or mhmyqn@126.com is my email.
正则表达式:\w+@(\w+\.)+\w+
l果QHello, 【mhmyqn@qq.com?or 【mhmyqn@126.com?is my email.
分析Q\w+可以匚w一个或多个字符Q而子表达?\w+\.)+可匹配像xxxx.edu.q样的字W串Q而最后不会是.字符l尾Q所以后面还会有一个\w+。像mhmyqn@xxxx.edu.cnq样的邮件地址也会匚w到?/span>
2、匹配零个或多个字符
匚w零个或多个字W用元W?Q它的用法和+完全一P只要把它攑֜一下字W或字符集合的后面,可以匹配该字符Q或字符集合Q连l出现零ơ或多次。如正则表达式ab*c可以匚wac、abc、abbbbbc{?/span>
3、匹配零个或一个字W?/span>
匚w零个或一个字W用元字符?。像上一说到的匚w一个空白行使用正则表达式\r\n\r\nQ但在Unix和Linux中不需要\rQ就可以使用元字W?Q\r?\n\r?\nq样既可匚wwindows中的I白行,也可匚wUnix和Linux中的I白行。下面来看一个匹配http或https协议的URL的例子:
文本QThe URL is http://www.mikan.com, to connect securely use https://www.mikan.cominstead.
正则表达式:https?://(\w+\.)+\w+
l果QThe URL is 【http://www.mikan.com? to connect securely use 【https://www.mikan.com?instead.
分析Q这个模式以https?开_表示?之前的一个字W可以有Q也可以没有Q所以它能匹配http或httpsQ后面部分和前一个例子一栗?/span>
二、匹配的重复ơ数
正则表达式里???解决了很多问题,但是Q?/span>
1Q??匚w的字W个数没有上限。我们无法ؓ它们匹配的字符个数讑֮一个最大倹{?/span>
2Q???臛_匚w一个或零个字符。我们无法ؓ它们匹配的字符个数另行讑֮一个最倹{?/span>
3Q如果只使用*?Q我们无法把它们匹配的字符个数讑֮Z个精的数字?/span>
正则表达式里提供了一个用来设定重复次数的语法Q重复次数要用{和}字符来给出,把数值写在它们中间?/span>
1、ؓ重复匚wơ数讑֮一个精?/span>
如果想ؓ重复匚wơ数讑֮一个精的|把那个数字写在{和}之间卛_。如{4}表示它前面的那个字符Q或字符集合Q必d原始文本中连l重复出?ơ才是一个匹配,如果只出C3ơ,也不是一个匹配?/span>
如前面几中说到的匹配页面中颜色的例子,可以用重复ơ数来匹配:#[[:xdigit:]]{6}?[0-9a-fA-F]{6}QPOSIX字符在java中是#\\p{XDigit}{6}?/span>
2、ؓ重复匚wơ数讑֮一个区?/span>
{}语法q可以用来ؓ重复匚wơ数讑֮一个区_也就是ؓ重复匚wơ数讑֮一个最值和最大倹{这U区间必M{n, m}q样的Ş式给出,其中n>=m>=0。如查日期格式是否正(不检查日期的有效性)的正则表辑ּQ如日期2012-08-12?012-8-12Q:\d{4}-\d{1,2}-\d{1,2}?/span>
3、匹配至重复多次
{}语法的最后一U用法是l出一个最的重复ơ数Q但不必l出最大重复次敎ͼQ如{3,}表示臛_重复3ơ。注意:{3,}中一定要有逗号Q而且逗号后不能有I格。否则会出错?/span>
来看一个例子,使用正则表达式把所有金额大?100的金额找出来Q?/span>
文本Q?/span>
$25.36
$125.36
$205.0
$2500.44
$44.30
正则表达式:$\d{3,}\.\d{2}
l果Q?/span>
$25.36
?125.36?/span>
?205.0?/span>
?2500.44?/span>
$44.30
+??可以表示成重复次敎ͼ
+{h于{1,}
*{h于{0,}
?{h于{0,1}
三、防止过度匹?/span>
?只能匚w零个或一个字W,{n}和{n,m}也有匚w重复ơ数的上限,但是??、{n,}都没有上限|q样有时会导致过度匹配的现象?/span>
来看匚w一个html标签的例?/span>
文本Q?/span>
Yesterday is <b>history</b>,tomorrow is a <B>mystery</B>, but today is a <b>gift</b>.
正则表达式:<[Bb]>.*</[Bb]>
l果Q?/span>
Yesterday is ?lt;b>history</b>,tomorrow is a <B>mystery</B>, but today is a <b>gift</b>?
分析Q?lt;[Bb]>匚w<b>标签Q不区分大小写)Q?lt;/[Bb]>匚w</b>标签Q不区分大小写)。但l果却不是预期的那样有三个,W一?lt;/b>标签之后Q一直到最后一?lt;/b>之间的东西全部匹配出来了?/span>
Z么会q样呢?因ؓ*?都是贪婪型的元字W,它们在匹配时的行为模式是多多益善Q它们会可能从一D|本的开头一直匹配到q段文本的末,而不是从q段文本的开头匹配到到W一个匹配时为止?/span>
当不需要这U贪婪行为时Q可以用这些元字符的懒惰型版本。懒惰意思是匚w可能少的字W,与贪婪型相反。懒惰型元字W只需要给贪婪型元字符加上一?后缀卛_。下面是贪婪型元字符的对应懒惰型版本Q?/span>
* *?
+ +?
{n,} {n,}?
所以上面的例子中,正则表达式只需要改?lt;[Bb]>.*?</[Bb]>卛_Q结果如下:
<b>history</b>
<B>mystery</B>
<b>gift</b>
四、ȝ
正则表达式的真下威力体现在重复次数匹配方面。这里介l了+??几种元字W的用法Q如果要_的确定匹配次敎ͼ使用{}。元字符分贪婪型和懒惰型两种Q在需要防止过度匹配的场合下,请用懒惰型元字W来构造正则表辑ּ
一、对Ҏ字符q行转义
元字W是一些在正则表达式里有着Ҏ含义的字W。因为元字符在正则表辑ּ里有着Ҏ的含义,所以这些字W就无法用来代表它们本n。在元字W前面加上一个反斜杠可以对它进行{义,q样得到的{义序列将匚w那个字符本n而不是它Ҏ的元字符含义。如Q如果想要匹配[和]Q就必须对它q行转义Q\[和\]?/span>
对元字符转义需要用到斜杠\字符Q这意味着\字符本向也是一个元字符Q要匚w\字符本nQ必{义成\\。如匚wwindows文g路径?/span>
二、匹配空白字W?/span>
元字W大致可以分ZU:一U是用来匚w文本的(?Q,另一U是正则表达式的语法所要求的(如[和]Q?/span>
在进行正则表辑ּ搜烦的时候,我们l常会遇到需要对原始文本中里的非打印I白字符q行匚w的情c比如说Q我们可能需要把所有的制表W找出来Q或者我们需要把换行W找出来Q这cdW很难被直接输入C个正则表辑ּ里,q时我们可以使用如下列出的特D元字符来输入它们:
\b 回退Qƈ删除Q一个字W(Backspace键)
\f 换页W?/span>
\n 换行W?/span>
\r 回RW?/span>
\t 制表W(Tab键)
\v 垂直制表W?/span>
来看一个例子,把文件中的空白行LQ?/span>
文本Q?/span>
8 5 4 1 6 3 2 7 9
7 6 2 9 5 8 3 4 1
9 3 1 4 2 7 8 5 6
6 9 3 8 7 5 1 2 4
5 1 8 3 4 2 6 9 7
2 4 7 6 1 9 5 3 8
3 26 7 8 4 9 1 5
4 8 9 5 3 1 7 6 2
1 7 5 2 9 6 4 8 3
正则表达式:\r\n\r\n
分析Q\r\n匚w一个回?换行l合Qwindows操作pȝ中把它作为文本行的结束标{。用正则表辑ּ\r\n\r\nq行的搜索将匚w两个q箋的行标{,而这正好是空白行?/span>
注意QUnix和Linux操作pȝ中只使用一个换行符来结束一个文本行Q换句话_在Unix或Linuxpȝ中匹配空白行只用\n\n卛_Q不需要加上\r。同旉用于windows和Unix/Linux的正则表辑ּ应该包括一个可先的\r和一个必d配的\nQ即\r?\n\r?\nQ这会在后面的文章中讲到?/span>
Java代码如下Q?/span>
原内容:
8 5 4 1 6 3 2 7 9
7 6 2 9 5 8 3 4 1
9 3 1 4 2 7 8 5 6
6 9 3 8 7 5 1 2 4
5 1 8 3 4 2 6 9 7
2 4 7 6 1 9 5 3 8
3 2 6 7 8 4 9 1 5
4 8 9 5 3 1 7 6 2
1 7 5 2 9 6 4 8 3
处理后:-----------------------------
8 5 4 1 6 3 2 7 9
7 6 2 9 5 8 3 4 1
9 3 1 4 2 7 8 5 6
6 9 3 8 7 5 1 2 4
5 1 8 3 4 2 6 9 7
2 4 7 6 1 9 5 3 8
3 2 6 7 8 4 9 1 5
4 8 9 5 3 1 7 6 2
1 7 5 2 9 6 4 8 3
三、匹配特定的字符cd
字符集合Q匹配多个字W中的某一个)是最常见的匹配Ş式,而一些常用的字符集合可以用特D元字符来代ѝ这些元字符匚w的是某一cd的字W(cd字符Q,cd字符q不是必不可的Q因为可以通过逐一列D有关字符或通过定义一个字W区间来匚w某一cdW,但是使用它们构造出来的正则表达式简明易懂,在实际应用中很常用?/span>
1、匹配数字与非数?/span>
\d M一个数字,{h于[0-9]或[0123456789]
\D M一个非数字Q等价于[^0-9]或[^0123456789]
2、匹配字母和数字与非字母和数?/span>
字母QA-Z不区分大写Q、数字、下划线是一U常用的字符集合Q可用如下类元字W:
\w M一个字母(不区分大写Q、数字、下划线Q等价于[0-9a-zA-Z_]
\W M一个非字母数字和下划线Q等价于[^0-9a-zA-Z_]
3、匹配空白字W与非空白字W?/span>
\s M一下空白字W,{h于[\f\n\r\t\v]
\S M一下空白字W,{h于[^\f\n\r\t\v]
注意Q退格元字符\b没有不在\s的范围之内?/span>
4、匹配十六进制或八进制数?/span>
十六q制Q用前缀\x来给出,如:\x0A对应于ASCII字符10Q换行符Q,其效果等价于\n?/span>
八进Ӟ用前~\0来给出,数值本w可以是两位或三位数字,如:\011对应于ASCII字符9Q制表符Q,其效果等价于\t?/span>
四、用POSIX字符c?/span>
POSIX字符cL很多正则表达式实现都支持的一U简写Ş式。Java也支持它Q但JavaScript不支持。POSIX字符如下所C:
[:alnum:] M一个字母或数字Q等价于[a-zA-Z0-9]
[:alpha:] M一个字母,{h于[a-zA-Z]
[:blank:] I格或制表符Q等价于[\t]
[:cntrl:] ASCII控制字符QASCII 0?1Q再加上ASCII 127Q?/span>
[:digit:] M一个数字,{h于[0-9]
[:graph:] M一个可打印字符Q但不包括空?/span>
[:lower:] M一个小写字母,{h于[a-z]
[:print:] M一个可打印字符
[:punct:] 既不属于[:alnum:]和[:cntrl:]的Q何一个字W?/span>
[:space:] M一个空白字W,包括I格Q等价于[^\f\n\r\t\v]
[:upper:] M一个大写字母,{h于[A-Z]
[:xdigit:] M一个十六进制数字,{h于[a-fA-F0-9]
POSIX字符和之前见q的元字W不太一P我们来看一个前面利用正则表辑ּ来匹配网中的颜色的例子Q?/span>
文本Q?lt;span style="background-color:#3636FF;height:30px;width:60px;">试</span>
正则表达式:#[[:xdigit:]] [[:xdigit:]] [[:xdigit:]] [[:xdigit:]] [[:xdigit:]] [[:xdigit:]]
l果Q?lt;span style="background-color:?3636FF?height:30px;width:60px;">试</span>
注意Q这里用的模式以[[开头、以]]l束Q这是用POSIX字符cL必须的,POSIX字符必须括在[:?]之间Q外层[和]字符用来定义一个集合,内层的[和]字符是POSIX字符cLw的l成部分?/span>
在java中的POSIX字符表示有所不同Q不是包括在[:?]之间Q而是以\p开_包括在{和}之间Q且大小写有区别Q同时增加了\p{ASCII}Q如下所C:
\p{Alnum} 字母数字字符Q[\p{Alpha}\p{Digit}]
\p{Alpha} 字母字符Q[\p{Lower}\p{Upper}]
\p{ASCII} 所?ASCIIQ[\x00-\x7F]
\p{Blank} I格或制表符Q[ \t]
\p{Cntrl} 控制字符Q[\x00-\x1F\x7F]
\p{Digit} 十进制数字:[0-9]
\p{Graph} 可见字符Q[\p{Alnum}\p{Punct}]
\p{Lower} 写字母字符Q[a-z]
\p{Print} 可打印字W:[\p{Graph}\x20]
\p{Punct} 标点W号Q?"#$%&'()*+,-./:;<=>?@[\]^_`{|}~
\p{Space} I白字符Q[ \t\n\x0B\f\r]
\p{Upper} 大写字母字符Q[A-Z]
\p{XDigit} 十六q制数字Q[0-9a-fA-F]
public class Test1 { public static void main(String[] args) { // TODO 自动生成Ҏ存根 } public void f1() { System.out.println("f1"); } //无法被子c覆盖的Ҏ public final void f2() { System.out.println("f2"); } public void f3() { System.out.println("f3"); } private void f4() { System.out.println("f4"); } } public class Test2 extends Test1 { public void f1(){ System.out.println("Test1父类Ҏf1被覆?"); } public static void main(String[] args) { Test2 t=new Test2(); t.f1(); t.f2(); //调用从父cȝ承过来的finalҎ t.f3(); //调用从父cȝ承过来的Ҏ //t.f4(); //调用p|Q无法从父类l承获得 } } |
3、final变量Q常量)
用final修饰的成员变量表C常量,g旦给定就无法改变Q?
final修饰的变量有三种Q静态变量、实例变量和局部变量,分别表示三种cd的常量?
从下面的例子中可以看出,一旦给final变量初值后Q值就不能再改变了?
另外Qfinal变量定义的时候,可以先声明,而不l初|q中变量也称为finalI白Q无Z么情况,~译器都保I白final在用之前必被初始化。但是,finalI白在final关键字final的用上提供了更大的灉|性,为此Q一个类中的final数据成员可以实C对象而有所不同Q却有保持其恒定不变的特征?/p>
package org.leizhimin; public class Test3 { private final String S="final实例变量S"; private final int A=100; public final int B=90; public static final int C=80; private static final int D=70; public final int E; //finalI白,必须在初始化对象的时候赋初? public Test3(int x){ E=x; } /** * @param args */ public static void main(String[] args) { Test3 t=new Test3(2); //t.A=101; //出错,final变量的g旦给定就无法改变 //t.B=91; //出错,final变量的g旦给定就无法改变 //t.C=81; //出错,final变量的g旦给定就无法改变 //t.D=71; //出错,final变量的g旦给定就无法改变 System.out.println(t.A); System.out.println(t.B); System.out.println(t.C); //不推荐用对象方式讉K静态字D? System.out.println(t.D); //不推荐用对象方式讉K静态字D? System.out.println(Test3.C); System.out.println(Test3.D); //System.out.println(Test3.E); //出错,因ؓE为finalI白,依据不同对象值有所不同. System.out.println(t.E); Test3 t1=new Test3(3); System.out.println(t1.E); //finalI白变量E依据对象的不同而不? } private void test(){ System.out.println(new Test3(1).A); System.out.println(Test3.C); System.out.println(Test3.D); } public void test2(){ final int a; //finalI白,在需要的时候才赋? final int b=4; //局部常?-final用于局部变量的情Ş final int c; //finalI白,一直没有给赋? a=3; //a=4; 出错,已经l赋qg. //b=2; 出错,已经l赋qg. } } |
4、final参数
当函数参CؓfinalcdӞ你可以读取用该参数Q但是无法改变该参数的倹{?/p>
public class Test4 { public static void main(String[] args) { new Test4().f1(2); } public void f1(final int i){ //i++; //i是finalcd?g允许改变? System.out.print(i); } } |
public class Test5 { private static int a; private int b; static{ Test5.a=3; System.out.println(a); Test5 t=new Test5(); t.f(); t.b=1000; System.out.println(t.b); } static{ Test5.a=4; System.out.println(a); } public static void main(String[] args) { // TODO 自动生成Ҏ存根 } static{ Test5.a=5; System.out.println(a); } public void f(){ System.out.println("hhahhahah"); } } |
思义它是local variableQ线E局部变量)。它的功用非常简单,是为每一个用该变量的线E都提供一个变量值的副本Q是每一个线E都可以独立地改变自q副本Q而不会和其它U程的副本冲H。从U程的角度看Q就好像每一个线E都完全拥有该变量?/p>
使用场景
ThreadLocalc?/strong>
它主要由四个Ҏl成initialValue()Qget()Qset(T)Qremove()Q其中值得注意的是initialValue()Q该Ҏ是一个protected的方法,昄是ؓ了子c重写而特意实现的。该Ҏq回当前U程在该U程局部变量的初始|q个Ҏ是一个gq调用方法,在一个线E第1ơ调用get()或者set(Object)时才执行Qƈ且仅执行1ơ。ThreadLocal中的实实现直接q回一个nullQ?/p>
ThreadLocal的原?/p>
ThreadLocal是如何做Cؓ每一个线E维护变量的副本的呢Q其实实现的思\很简单,在ThreadLocalcM有一个MapQ用于存储每一个线E的变量的副本。比如下面的CZ实现Q?/p>
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 的?/strong>
使用Ҏ一Q?/p>
Hibernate的文档时看到了关于ThreadLocal理多线E访问的部分。具体代码如?
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对象QThreadLocal有三个成员方?get()、set()、initialvalue()?
如果不初始化initialvalueQ则initialvalueq回null?
3。session的getҎ当前U程q回其对应的U程内部变量Q也是我们需要的net.sf.hibernate.SessionQ相当于对应每个数据库连接).多线E情况下׃n数据库链接是不安全的。ThreadLocal保证了每个线E都有自qsQ数据库q接Q?
5。如果是该线E初ơ访问,自然QsQ数据库q接Q会是nullQ接着创徏一个SessionQ具体就是行6?
6。创Z个数据库q接实例 s
7。保存该数据库连接s到ThreadLocal中?
8。如果当前线E已l访问过数据库了Q则从session中get()可以获取该U程上次获取q的q接实例?
使用Ҏ?/p>
当要l线E初始化一个特D值时Q需要自己实现ThreadLocal的子cdƈ重写该方法,通常使用一个内部匿名类对ThreadLocalq行子类化,EasyDBO中创建jdbcq接上下文就是这样做的:
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);
}
}
}
使用单例模式Q不同的U程调用getJdbcContext()获得自己的jdbcContextQ都是通过JDBCContextThreadLocal 内置子类来获得JDBCContext对象的线E局部变量,q个?/p>
ThreadLocal目的是保存一些线E别的全局变量Q比如connectionQ或者事务上下文Q避免这些值需要一直通过函数参数的方式一路传递?span id="more-109">
2. 常见用法
举例其中一U常见用法:
public class Test2 { public static void main(String[] args) throws InterruptedException { testThreadLocal(); } private static void testThreadLocal() { Util.setGlobalName("zili.dengzl"); new Foo().printName(); } } class Foo{ public void printName(){ System.out.println("globalName="+Util.getGlobalName()); } } class Util { private static final ThreadLocal<String> globalName = new ThreadLocal<String>(); public static String getGlobalName() { return globalName.get(); } public static void setGlobalName(String name) { globalName.set(name); } }
3.实现分析
要实C面这L功能Q最单的x是用一个Map<Thread,T>,如下Q?/p>
class MockThreadLocal<T> { private Map<Thread, T> map = new HashMap<Thread, T>(); public T get() { return (T) map.get(Thread.currentThread()); } public void set(T value) { map.put(Thread.currentThread(), value); } }
q样也能实现ThreadLocal的效果,但是有一个问题,当对应的U程消失后,map中对应的U程值ƈ不会被回Ӟ从而造成内存泄露?/p>
事实上ThreadLocal是这样做的:
每个Thread都有一个threadLocalMapQkey是threadLocal对象Qvalue是具体用的倹{ThreadLocal对象的get是先取得当前的ThreadQ然后从q个Thread的threadLcoalMap中取出倹{setcM?/p>
下面看下具体代码Q?/p>
/** * Returns the value in the current thread's copy of this * thread-local variable. If the variable has no value for the * current thread, it is first initialized to the value returned * by an invocation of the {@link #initialValue} method. * * @return the current thread's value of this thread-local */ public T get() { Thread t = Thread.currentThread(); ThreadLocalMap map = getMap(t); if (map != null) { ThreadLocalMap.Entry e = map.getEntry(this); if (e != null) return (T)e.value; } return setInitialValue(); }
注意q里如果取到没有该线E对应的|会调用setInitialValue();Q最l调用initialValue()生成一个|q也是我们很多场景下要overrideq个Ҏ的原因;
下面看一下getMap(Thread t)ҎQ?/p>
ThreadLocalMap getMap(Thread t) { return t.threadLocals; }
在ThreadcMQ?/p>
/* ThreadLocal values pertaining to this thread. This map is maintained * by the ThreadLocal class. */ ThreadLocal.ThreadLocalMap threadLocals = null;
由此可见Q所有的ThreadLocal的信息,最l是兌到Thread上的Q线E消失后Q对应的Thread对象也被回收Q这时对应的ThreadLocal对象(该线E部?也会被回收?/p>
q里Z么是一个ThreadLocalMap呢,因ؓ一个线E可以有多个ThreadLocal变量Q通过map.getEntry(this)取得对应的某个具体的变量?/p>
private Entry getEntry(ThreadLocal key) { int i = key.threadLocalHashCode & (table.length - 1); Entry e = table[i]; if (e != null && e.get() == key) return e; else return getEntryAfterMiss(key, i, e); }
最后要注意的一ҎQThreadLocalMap的Entry是一个weakReferenceQ?/p>
/** * The entries in this hash map extend WeakReference, using * its main ref field as the key (which is always a * ThreadLocal object). Note that null keys (i.e. entry.get() * == null) mean that the key is no longer referenced, so the * entry can be expunged from table. Such entries are referred to * as "stale entries" in the code that follows. */ static class Entry extends WeakReference<ThreadLocal> { /** The value associated with this ThreadLocal. */ Object value; Entry(ThreadLocal k, Object v) { super(k); value = v; } }
q里主要因ؓThreadLocalMap的key是ThreadLocal对象Q如果某个ThreadLocal对象所有的强引用没有了Q会利用weakref的功能把他回收掉Q然后复用这个entry?/p>
考虑一下如果不用weakReference会出C么情况:假设某个对象是这样引用的
private final ThreadLocal<String> globalName = new ThreadLocal<String>();
注意没有staticQ然后这个对象被不断的new出来Q然后死掉,每次ThreadLocalmap中都会多Z个entryQ然后这个entry强引用一个ThreadLocal对象QThreadLocalMap本n没有办法确定哪个entry是不用了的,如果恰好q个U程是线E池中的Q会存活很久Q那杯具了?/p>
ThreadLocalMap用了weakReferenceQ失d引用的ThreadLocal对象会在下次gc时被回收Q然后ThreadLocalMap本n在get和set的时候会考察key为空的EntryQƈ复用它或者清除,从而避免内存泄霌Ӏ?/p>
q样看来QHashMap也有一L问题Q但Z么hashMap不这样呢Q因为hashMap的put是业务代码操作的Q因此如果有长期存活的HashMapQ(比如static的)业务代码putq去有义务去removeQ但ThreadLocal的put操作时ThreadLocalcd的,业务代码不知道,因此也不会去做removeQ而ThreadLocalMap本n不知道引用他的某个entry的key的对象什么时候死掉了Q那么如果不用弱引用Q就不知道这个ThreadLocal对象什么时候需要回收了?/p>
附:
q里补充一下weakReference的用法供参考(当强引用不存在时Q下ơ垃圑֛收会回收弱引用所引用的对象)Q?/p>
Object o = new Object(); WeakReference<Object> ref = new WeakReference<Object>(o); System.out.println(ref.get()); o=null; System.gc(); System.out.println(ref.get());
l果输出Q?/p>
java.lang.Object@de6ced null
4.1 Z么一般的ThreadLocal用法都要加staticQ如下:
class Test { private static final ThreadLocal<String> globalName = new ThreadLocal<String>(); }
class TestThreadLocal{ public static void main(String[] args) { Test t1 = new Test(); Test t2 = new Test(); t1.pool.set("a"); System.out.println(t1.pool.get()); System.out.println(t2.pool.get()); } } class Test{ public ThreadLocal pool = new ThreadLocal(); }
输出会是:aQnull
原因无需多解释了。唯一需要啰嗦的一ҎQ就一般情况都是单例,上面那个weakreferenceq是必要的,因ؓ作ؓ框架代码Q不能保证正怋用的情况下一个线E有很多ThreadLocalQ如果不用weakreferenceQ就会有内存泄漏的风险,特别是针对线E池中的U程?/p>