Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
openanolis
dragonwell8_jdk
提交
84ddca8a
D
dragonwell8_jdk
项目概览
openanolis
/
dragonwell8_jdk
通知
4
Star
2
Fork
0
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
0
列表
看板
标记
里程碑
合并请求
0
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
D
dragonwell8_jdk
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
0
Issue
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
提交
Issue看板
提交
84ddca8a
编写于
12月 11, 2012
作者:
A
akhil
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
8003246: Add InitialValue Supplier to ThreadLocal
Reviewed-by: mduigou, forax, dl, chegar, briangoetz
上级
ffabb003
变更
2
显示空白变更内容
内联
并排
Showing
2 changed file
with
171 addition
and
10 deletion
+171
-10
src/share/classes/java/lang/ThreadLocal.java
src/share/classes/java/lang/ThreadLocal.java
+44
-10
test/java/lang/ThreadLocal/ThreadLocalSupplierTest.java
test/java/lang/ThreadLocal/ThreadLocalSupplierTest.java
+127
-0
未找到文件。
src/share/classes/java/lang/ThreadLocal.java
浏览文件 @
84ddca8a
...
...
@@ -25,19 +25,21 @@
package
java.lang
;
import
java.lang.ref.*
;
import
java.util.Objects
;
import
java.util.concurrent.atomic.AtomicInteger
;
import
java.util.function.Supplier
;
/**
* This class provides thread-local variables. These variables differ from
* their normal counterparts in that each thread that accesses one (via its
*
<tt>get</tt> or <tt>set</tt>
method) has its own, independently initialized
* copy of the variable.
<tt>ThreadLocal</tt>
instances are typically private
*
{@code get} or {@code set}
method) has its own, independently initialized
* copy of the variable.
{@code ThreadLocal}
instances are typically private
* static fields in classes that wish to associate state with a thread (e.g.,
* a user ID or Transaction ID).
*
* <p>For example, the class below generates unique identifiers local to each
* thread.
* A thread's id is assigned the first time it invokes
<tt>ThreadId.get()</tt>
* A thread's id is assigned the first time it invokes
{@code ThreadId.get()}
* and remains unchanged on subsequent calls.
* <pre>
* import java.util.concurrent.atomic.AtomicInteger;
...
...
@@ -61,7 +63,7 @@ import java.util.concurrent.atomic.AtomicInteger;
* }
* </pre>
* <p>Each thread holds an implicit reference to its copy of a thread-local
* variable as long as the thread is alive and the
<tt>ThreadLocal</tt>
* variable as long as the thread is alive and the
{@code ThreadLocal}
* instance is accessible; after a thread goes away, all of its copies of
* thread-local instances are subject to garbage collection (unless other
* references to these copies exist).
...
...
@@ -108,14 +110,14 @@ public class ThreadLocal<T> {
* thread-local variable. This method will be invoked the first
* time a thread accesses the variable with the {@link #get}
* method, unless the thread previously invoked the {@link #set}
* method, in which case the
<tt>initialValue</tt>
method will not
* method, in which case the
{@code initialValue}
method will not
* be invoked for the thread. Normally, this method is invoked at
* most once per thread, but it may be invoked again in case of
* subsequent invocations of {@link #remove} followed by {@link #get}.
*
* <p>This implementation simply returns
<tt>null</tt>
; if the
* <p>This implementation simply returns
{@code null}
; if the
* programmer desires thread-local variables to have an initial
* value other than
<tt>null</tt>, <tt>ThreadLocal</tt>
must be
* value other than
{@code null}, {@code ThreadLocal}
must be
* subclassed, and this method overridden. Typically, an
* anonymous inner class will be used.
*
...
...
@@ -125,8 +127,22 @@ public class ThreadLocal<T> {
return
null
;
}
/**
* Creates a thread local variable. The initial value of the variable is
* determined by invoking the {@code get} method on the {@code Supplier}.
*
* @param supplier the supplier to be used to determine the initial value
* @return a new thread local variable
* @throws NullPointerException if the specified supplier is null
* @since 1.8
*/
public
static
<
T
>
ThreadLocal
<
T
>
withInitial
(
Supplier
<?
extends
T
>
supplier
)
{
return
new
SuppliedThreadLocal
<>(
supplier
);
}
/**
* Creates a thread local variable.
* @see #withInitial(java.util.function.Supplier)
*/
public
ThreadLocal
()
{
}
...
...
@@ -195,7 +211,7 @@ public class ThreadLocal<T> {
* reinitialized by invoking its {@link #initialValue} method,
* unless its value is {@linkplain #set set} by the current thread
* in the interim. This may result in multiple invocations of the
*
<tt>initialValue</tt>
method in the current thread.
*
{@code initialValue}
method in the current thread.
*
* @since 1.5
*/
...
...
@@ -250,6 +266,24 @@ public class ThreadLocal<T> {
throw
new
UnsupportedOperationException
();
}
/**
* An extension of ThreadLocal that obtains its initial value from
* the specified {@code Supplier}.
*/
static
final
class
SuppliedThreadLocal
<
T
>
extends
ThreadLocal
<
T
>
{
private
final
Supplier
<?
extends
T
>
supplier
;
SuppliedThreadLocal
(
Supplier
<?
extends
T
>
supplier
)
{
this
.
supplier
=
Objects
.
requireNonNull
(
supplier
);
}
@Override
protected
T
initialValue
()
{
return
supplier
.
get
();
}
}
/**
* ThreadLocalMap is a customized hash map suitable only for
* maintaining thread local values. No operations are exported
...
...
@@ -599,9 +633,9 @@ public class ThreadLocal<T> {
* @param i a position known NOT to hold a stale entry. The
* scan starts at the element after i.
*
* @param n scan control:
<tt>log2(n)</tt>
cells are scanned,
* @param n scan control:
{@code log2(n)}
cells are scanned,
* unless a stale entry is found, in which case
*
<tt>log2(table.length)-1</tt>
additional cells are scanned.
*
{@code log2(table.length)-1}
additional cells are scanned.
* When called from insertions, this parameter is the number
* of elements, but when from replaceStaleEntry, it is the
* table length. (Note: all this could be changed to be either
...
...
test/java/lang/ThreadLocal/ThreadLocalSupplierTest.java
0 → 100644
浏览文件 @
84ddca8a
/*
* Copyright (c) 2012, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
import
java.util.concurrent.atomic.AtomicInteger
;
import
java.util.function.Supplier
;
import
org.testng.annotations.Test
;
import
static
org
.
testng
.
Assert
.*;
/**
* @test
* @run testng ThreadLocalSupplierTest
* @summary tests ThreadLocal.withInitial(<Supplier>).
* Adapted from java.lang.Basic functional test of ThreadLocal
*
* @author Jim Gish <jim.gish@oracle.com>
*/
@Test
public
class
ThreadLocalSupplierTest
{
static
final
class
IntegerSupplier
implements
Supplier
<
Integer
>
{
private
final
AtomicInteger
supply
=
new
AtomicInteger
(
0
);
@Override
public
Integer
get
()
{
return
supply
.
getAndIncrement
();
}
public
int
numCalls
()
{
return
supply
.
intValue
();
}
}
static
IntegerSupplier
theSupply
=
new
IntegerSupplier
();
static
final
class
MyThreadLocal
extends
ThreadLocal
<
Integer
>
{
private
final
ThreadLocal
<
Integer
>
delegate
;
public
volatile
boolean
everCalled
;
public
MyThreadLocal
(
Supplier
<
Integer
>
supplier
)
{
delegate
=
ThreadLocal
.<
Integer
>
withInitial
(
supplier
);
}
@Override
public
Integer
get
()
{
return
delegate
.
get
();
}
@Override
protected
synchronized
Integer
initialValue
()
{
// this should never be called since we are using the factory instead
everCalled
=
true
;
return
null
;
}
}
/**
* Our one and only ThreadLocal from which we get thread ids using a
* supplier which simply increments a counter on each call of get().
*/
static
MyThreadLocal
threadLocal
=
new
MyThreadLocal
(
theSupply
);
public
void
testMultiThread
()
throws
Exception
{
final
int
threadCount
=
500
;
final
Thread
th
[]
=
new
Thread
[
threadCount
];
final
boolean
visited
[]
=
new
boolean
[
threadCount
];
// Create and start the threads
for
(
int
i
=
0
;
i
<
threadCount
;
i
++)
{
th
[
i
]
=
new
Thread
()
{
@Override
public
void
run
()
{
final
int
threadId
=
threadLocal
.
get
();
assertFalse
(
visited
[
threadId
],
"visited["
+
threadId
+
"]="
+
visited
[
threadId
]);
visited
[
threadId
]
=
true
;
// check the get() again
final
int
secondCheckThreadId
=
threadLocal
.
get
();
assertEquals
(
secondCheckThreadId
,
threadId
);
}
};
th
[
i
].
start
();
}
// Wait for the threads to finish
for
(
int
i
=
0
;
i
<
threadCount
;
i
++)
{
th
[
i
].
join
();
}
assertEquals
(
theSupply
.
numCalls
(),
threadCount
);
// make sure the provided initialValue() has not been called
assertFalse
(
threadLocal
.
everCalled
);
// Check results
for
(
int
i
=
0
;
i
<
threadCount
;
i
++)
{
assertTrue
(
visited
[
i
],
"visited["
+
i
+
"]="
+
visited
[
i
]);
}
}
public
void
testSimple
()
{
final
String
expected
=
"OneWithEverything"
;
final
ThreadLocal
<
String
>
threadLocal
=
ThreadLocal
.<
String
>
withInitial
(()
->
expected
);
assertEquals
(
expected
,
threadLocal
.
get
());
}
}
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录