Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
openanolis
dragonwell8_jdk
提交
f6dea072
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看板
提交
f6dea072
编写于
4月 26, 2013
作者:
P
plevart
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
7123493: (proxy) Proxy.getProxyClass doesn't scale under high load
Reviewed-by: mchung
上级
659b4b81
变更
2
显示空白变更内容
内联
并排
Showing
2 changed file
with
580 addition
and
206 deletion
+580
-206
src/share/classes/java/lang/reflect/Proxy.java
src/share/classes/java/lang/reflect/Proxy.java
+201
-206
src/share/classes/java/lang/reflect/WeakCache.java
src/share/classes/java/lang/reflect/WeakCache.java
+379
-0
未找到文件。
src/share/classes/java/lang/reflect/Proxy.java
浏览文件 @
f6dea072
...
@@ -25,18 +25,14 @@
...
@@ -25,18 +25,14 @@
package
java.lang.reflect
;
package
java.lang.reflect
;
import
java.lang.ref.Reference
;
import
java.lang.ref.WeakReference
;
import
java.lang.ref.WeakReference
;
import
java.security.AccessController
;
import
java.security.AccessController
;
import
java.security.PrivilegedAction
;
import
java.security.PrivilegedAction
;
import
java.util.Arrays
;
import
java.util.Arrays
;
import
java.util.Collections
;
import
java.util.IdentityHashMap
;
import
java.util.HashMap
;
import
java.util.HashSet
;
import
java.util.Map
;
import
java.util.Map
;
import
java.util.Set
;
import
java.util.concurrent.atomic.AtomicLong
;
import
java.util.List
;
import
java.util.function.BiFunction
;
import
java.util.WeakHashMap
;
import
sun.misc.ProxyGenerator
;
import
sun.misc.ProxyGenerator
;
import
sun.misc.VM
;
import
sun.misc.VM
;
import
sun.reflect.CallerSensitive
;
import
sun.reflect.CallerSensitive
;
...
@@ -232,27 +228,15 @@ public class Proxy implements java.io.Serializable {
...
@@ -232,27 +228,15 @@ public class Proxy implements java.io.Serializable {
private
static
final
long
serialVersionUID
=
-
2222568056686623797L
;
private
static
final
long
serialVersionUID
=
-
2222568056686623797L
;
/** prefix for all proxy class names */
private
final
static
String
proxyClassNamePrefix
=
"$Proxy"
;
/** parameter types of a proxy class constructor */
/** parameter types of a proxy class constructor */
private
final
static
Class
[]
constructorParams
=
private
static
final
Class
<?>
[]
constructorParams
=
{
InvocationHandler
.
class
};
{
InvocationHandler
.
class
};
/** maps a class loader to the proxy class cache for that loader */
/**
private
static
Map
<
ClassLoader
,
Map
<
List
<
String
>,
Object
>>
loaderToCache
* a cache of proxy classes
=
new
WeakHashMap
<>();
*/
private
static
final
WeakCache
<
ClassLoader
,
Class
<?>[],
Class
<?>>
/** marks that a particular proxy class is currently being generated */
proxyClassCache
=
new
WeakCache
<>(
new
KeyFactory
(),
new
ProxyClassFactory
());
private
static
Object
pendingGenerationMarker
=
new
Object
();
/** next number to use for generation of unique proxy class names */
private
static
long
nextUniqueNumber
=
0
;
private
static
Object
nextUniqueNumberLock
=
new
Object
();
/** set of all generated proxy classes, for isProxyClass implementation */
private
static
Map
<
Class
<?>,
Void
>
proxyClasses
=
Collections
.
synchronizedMap
(
new
WeakHashMap
<
Class
<?>,
Void
>());
/**
/**
* the invocation handler for this proxy instance.
* the invocation handler for this proxy instance.
...
@@ -423,131 +407,190 @@ public class Proxy implements java.io.Serializable {
...
@@ -423,131 +407,190 @@ public class Proxy implements java.io.Serializable {
throw
new
IllegalArgumentException
(
"interface limit exceeded"
);
throw
new
IllegalArgumentException
(
"interface limit exceeded"
);
}
}
Class
<?>
proxyClass
=
null
;
// If the proxy class defined by the given loader implementing
// the given interfaces exists, this will simply return the cached copy;
// otherwise, it will create the proxy class via the ProxyClassFactory
return
proxyClassCache
.
get
(
loader
,
interfaces
);
}
/* collect interface names to use as key for proxy class cache */
/*
String
[]
interfaceNames
=
new
String
[
interfaces
.
length
];
* a key used for proxy class with 0 implemented interfaces
*/
private
static
final
Object
key0
=
new
Object
();
// for detecting duplicates
/*
Set
<
Class
<?>>
interfaceSet
=
new
HashSet
<>();
* Key1 and Key2 are optimized for the common use of dynamic proxies
* that implement 1 or 2 interfaces.
*/
for
(
int
i
=
0
;
i
<
interfaces
.
length
;
i
++)
{
/*
/*
* Verify that the class loader resolves the name of this
* a key used for proxy class with 1 implemented interface
* interface to the same Class object.
*/
*/
String
interfaceName
=
interfaces
[
i
].
getName
();
private
static
final
class
Key1
extends
WeakReference
<
Class
<?>>
{
Class
<?>
interfaceClass
=
null
;
private
final
int
hash
;
try
{
interfaceClass
=
Class
.
forName
(
interfaceName
,
false
,
loader
);
Key1
(
Class
<?>
intf
)
{
}
catch
(
ClassNotFoundException
e
)
{
super
(
intf
);
this
.
hash
=
intf
.
hashCode
();
}
}
if
(
interfaceClass
!=
interfaces
[
i
])
{
throw
new
IllegalArgumentException
(
@Override
interfaces
[
i
]
+
" is not visible from class loader"
);
public
int
hashCode
()
{
return
hash
;
}
}
/*
@Override
* Verify that the Class object actually represents an
public
boolean
equals
(
Object
obj
)
{
* interface.
Class
<?>
intf
;
*/
return
this
==
obj
||
if
(!
interfaceClass
.
isInterface
())
{
obj
!=
null
&&
throw
new
IllegalArgumentException
(
obj
.
getClass
()
==
Key1
.
class
&&
interfaceClass
.
getName
()
+
" is not an interface"
);
(
intf
=
get
())
!=
null
&&
intf
==
((
Key1
)
obj
).
get
();
}
}
}
/*
/*
* Verify that this interface is not a duplicate.
* a key used for proxy class with 2 implemented interfaces
*/
*/
if
(
interfaceSet
.
contains
(
interfaceClass
))
{
private
static
final
class
Key2
extends
WeakReference
<
Class
<?>>
{
throw
new
IllegalArgumentException
(
private
final
int
hash
;
"repeated interface: "
+
interfaceClass
.
getName
());
private
final
WeakReference
<
Class
<?>>
ref2
;
Key2
(
Class
<?>
intf1
,
Class
<?>
intf2
)
{
super
(
intf1
);
hash
=
31
*
intf1
.
hashCode
()
+
intf2
.
hashCode
();
ref2
=
new
WeakReference
<
Class
<?>>(
intf2
);
}
}
interfaceSet
.
add
(
interfaceClass
);
interfaceNames
[
i
]
=
interfaceName
;
@Override
public
int
hashCode
()
{
return
hash
;
}
}
/*
@Override
* Using string representations of the proxy interfaces as
public
boolean
equals
(
Object
obj
)
{
* keys in the proxy class cache (instead of their Class
Class
<?>
intf1
,
intf2
;
* objects) is sufficient because we require the proxy
return
this
==
obj
||
* interfaces to be resolvable by name through the supplied
obj
!=
null
&&
* class loader, and it has the advantage that using a string
obj
.
getClass
()
==
Key2
.
class
&&
* representation of a class makes for an implicit weak
(
intf1
=
get
())
!=
null
&&
* reference to the class.
intf1
==
((
Key2
)
obj
).
get
()
&&
*/
(
intf2
=
ref2
.
get
())
!=
null
&&
List
<
String
>
key
=
Arrays
.
asList
(
interfaceNames
);
intf2
==
((
Key2
)
obj
).
ref2
.
get
();
}
}
/*
/*
* Find or create the proxy class cache for the class loader.
* a key used for proxy class with any number of implemented interfaces
* (used here for 3 or more only)
*/
*/
Map
<
List
<
String
>,
Object
>
cache
;
private
static
final
class
KeyX
{
synchronized
(
loaderToCache
)
{
private
final
int
hash
;
cache
=
loaderToCache
.
get
(
loader
);
private
final
WeakReference
<
Class
<?>>[]
refs
;
if
(
cache
==
null
)
{
cache
=
new
HashMap
<>();
KeyX
(
Class
<?>[]
interfaces
)
{
loaderToCache
.
put
(
loader
,
cache
);
hash
=
Arrays
.
hashCode
(
interfaces
);
refs
=
new
WeakReference
[
interfaces
.
length
];
for
(
int
i
=
0
;
i
<
interfaces
.
length
;
i
++)
{
refs
[
i
]
=
new
WeakReference
<>(
interfaces
[
i
]);
}
}
/*
}
* This mapping will remain valid for the duration of this
* method, without further synchronization, because the mapping
@Override
* will only be removed if the class loader becomes unreachable.
public
int
hashCode
()
{
return
hash
;
}
@Override
public
boolean
equals
(
Object
obj
)
{
return
this
==
obj
||
obj
!=
null
&&
obj
.
getClass
()
==
KeyX
.
class
&&
equals
(
refs
,
((
KeyX
)
obj
).
refs
);
}
private
static
boolean
equals
(
WeakReference
<
Class
<?>>[]
refs1
,
WeakReference
<
Class
<?>>[]
refs2
)
{
if
(
refs1
.
length
!=
refs2
.
length
)
{
return
false
;
}
for
(
int
i
=
0
;
i
<
refs1
.
length
;
i
++)
{
Class
<?>
intf
=
refs1
[
i
].
get
();
if
(
intf
==
null
||
intf
!=
refs2
[
i
].
get
())
{
return
false
;
}
}
return
true
;
}
}
/**
* A function that maps an array of interfaces to an optimal key where
* Class objects representing interfaces are weakly referenced.
*/
*/
private
static
final
class
KeyFactory
implements
BiFunction
<
ClassLoader
,
Class
<?>[],
Object
>
{
@Override
public
Object
apply
(
ClassLoader
classLoader
,
Class
<?>[]
interfaces
)
{
switch
(
interfaces
.
length
)
{
case
1
:
return
new
Key1
(
interfaces
[
0
]);
// the most frequent
case
2
:
return
new
Key2
(
interfaces
[
0
],
interfaces
[
1
]);
case
0
:
return
key0
;
default
:
return
new
KeyX
(
interfaces
);
}
}
}
}
/*
/**
* Look up the list of interfaces in the proxy class cache using
* A factory function that generates, defines and returns the proxy class given
* the key. This lookup will result in one of three possible
* the ClassLoader and array of interfaces.
* kinds of values:
* null, if there is currently no proxy class for the list of
* interfaces in the class loader,
* the pendingGenerationMarker object, if a proxy class for the
* list of interfaces is currently being generated,
* or a weak reference to a Class object, if a proxy class for
* the list of interfaces has already been generated.
*/
*/
synchronized
(
cache
)
{
private
static
final
class
ProxyClassFactory
implements
BiFunction
<
ClassLoader
,
Class
<?>[],
Class
<?>>
{
// prefix for all proxy class names
private
static
final
String
proxyClassNamePrefix
=
"$Proxy"
;
// next number to use for generation of unique proxy class names
private
static
final
AtomicLong
nextUniqueNumber
=
new
AtomicLong
();
@Override
public
Class
<?>
apply
(
ClassLoader
loader
,
Class
<?>[]
interfaces
)
{
Map
<
Class
<?>,
Boolean
>
interfaceSet
=
new
IdentityHashMap
<>(
interfaces
.
length
);
for
(
Class
<?>
intf
:
interfaces
)
{
/*
/*
* Note that we need not worry about reaping the cache for
* Verify that the class loader resolves the name of this
* entries with cleared weak references because if a proxy class
* interface to the same Class object.
* has been garbage collected, its class loader will have been
* garbage collected as well, so the entire cache will be reaped
* from the loaderToCache map.
*/
*/
do
{
Class
<?>
interfaceClass
=
null
;
Object
value
=
cache
.
get
(
key
);
if
(
value
instanceof
Reference
)
{
proxyClass
=
(
Class
<?>)
((
Reference
)
value
).
get
();
}
if
(
proxyClass
!=
null
)
{
// proxy class already generated: return it
return
proxyClass
;
}
else
if
(
value
==
pendingGenerationMarker
)
{
// proxy class being generated: wait for it
try
{
try
{
cache
.
wait
();
interfaceClass
=
Class
.
forName
(
intf
.
getName
(),
false
,
loader
);
}
catch
(
InterruptedException
e
)
{
}
catch
(
ClassNotFoundException
e
)
{
}
if
(
interfaceClass
!=
intf
)
{
throw
new
IllegalArgumentException
(
intf
+
" is not visible from class loader"
);
}
/*
/*
* The class generation that we are waiting for should
* Verify that the Class object actually represents an
* take a small, bounded time, so we can safely ignore
* interface.
* thread interrupts here.
*/
*/
if
(!
interfaceClass
.
isInterface
())
{
throw
new
IllegalArgumentException
(
interfaceClass
.
getName
()
+
" is not an interface"
);
}
}
continue
;
}
else
{
/*
/*
* No proxy class for this list of interfaces has been
* Verify that this interface is not a duplicate.
* generated or is being generated, so we will go and
* generate it now. Mark it as pending generation.
*/
*/
cache
.
put
(
key
,
pendingGenerationMarker
);
if
(
interfaceSet
.
put
(
interfaceClass
,
Boolean
.
TRUE
)
!=
null
)
{
break
;
throw
new
IllegalArgumentException
(
"repeated interface: "
+
interfaceClass
.
getName
());
}
}
}
while
(
true
);
}
}
try
{
String
proxyPkg
=
null
;
// package to define proxy class in
String
proxyPkg
=
null
;
// package to define proxy class in
int
accessFlags
=
Modifier
.
PUBLIC
|
Modifier
.
FINAL
;
int
accessFlags
=
Modifier
.
PUBLIC
|
Modifier
.
FINAL
;
...
@@ -556,11 +599,11 @@ public class Proxy implements java.io.Serializable {
...
@@ -556,11 +599,11 @@ public class Proxy implements java.io.Serializable {
* proxy class will be defined in the same package. Verify that
* proxy class will be defined in the same package. Verify that
* all non-public proxy interfaces are in the same package.
* all non-public proxy interfaces are in the same package.
*/
*/
for
(
int
i
=
0
;
i
<
interfaces
.
length
;
i
++
)
{
for
(
Class
<?>
intf
:
interfaces
)
{
int
flags
=
int
erfaces
[
i
]
.
getModifiers
();
int
flags
=
int
f
.
getModifiers
();
if
(!
Modifier
.
isPublic
(
flags
))
{
if
(!
Modifier
.
isPublic
(
flags
))
{
accessFlags
=
Modifier
.
FINAL
;
accessFlags
=
Modifier
.
FINAL
;
String
name
=
int
erfaces
[
i
]
.
getName
();
String
name
=
int
f
.
getName
();
int
n
=
name
.
lastIndexOf
(
'.'
);
int
n
=
name
.
lastIndexOf
(
'.'
);
String
pkg
=
((
n
==
-
1
)
?
""
:
name
.
substring
(
0
,
n
+
1
));
String
pkg
=
((
n
==
-
1
)
?
""
:
name
.
substring
(
0
,
n
+
1
));
if
(
proxyPkg
==
null
)
{
if
(
proxyPkg
==
null
)
{
...
@@ -577,19 +620,11 @@ public class Proxy implements java.io.Serializable {
...
@@ -577,19 +620,11 @@ public class Proxy implements java.io.Serializable {
proxyPkg
=
ReflectUtil
.
PROXY_PACKAGE
+
"."
;
proxyPkg
=
ReflectUtil
.
PROXY_PACKAGE
+
"."
;
}
}
{
/*
/*
* Choose a name for the proxy class to generate.
* Choose a name for the proxy class to generate.
*/
*/
long
num
;
long
num
=
nextUniqueNumber
.
getAndIncrement
();
synchronized
(
nextUniqueNumberLock
)
{
num
=
nextUniqueNumber
++;
}
String
proxyName
=
proxyPkg
+
proxyClassNamePrefix
+
num
;
String
proxyName
=
proxyPkg
+
proxyClassNamePrefix
+
num
;
/*
* Verify that the class loader hasn't already
* defined a class with the chosen name.
*/
/*
/*
* Generate the specified proxy class.
* Generate the specified proxy class.
...
@@ -597,7 +632,7 @@ public class Proxy implements java.io.Serializable {
...
@@ -597,7 +632,7 @@ public class Proxy implements java.io.Serializable {
byte
[]
proxyClassFile
=
ProxyGenerator
.
generateProxyClass
(
byte
[]
proxyClassFile
=
ProxyGenerator
.
generateProxyClass
(
proxyName
,
interfaces
,
accessFlags
);
proxyName
,
interfaces
,
accessFlags
);
try
{
try
{
proxyClass
=
defineClass0
(
loader
,
proxyName
,
return
defineClass0
(
loader
,
proxyName
,
proxyClassFile
,
0
,
proxyClassFile
.
length
);
proxyClassFile
,
0
,
proxyClassFile
.
length
);
}
catch
(
ClassFormatError
e
)
{
}
catch
(
ClassFormatError
e
)
{
/*
/*
...
@@ -610,27 +645,6 @@ public class Proxy implements java.io.Serializable {
...
@@ -610,27 +645,6 @@ public class Proxy implements java.io.Serializable {
throw
new
IllegalArgumentException
(
e
.
toString
());
throw
new
IllegalArgumentException
(
e
.
toString
());
}
}
}
}
// add to set of all generated proxy classes, for isProxyClass
proxyClasses
.
put
(
proxyClass
,
null
);
}
finally
{
/*
* We must clean up the "pending generation" state of the proxy
* class cache entry somehow. If a proxy class was successfully
* generated, store it in the cache (with a weak reference);
* otherwise, remove the reserved entry. In all cases, notify
* all waiters on reserved entries in this cache.
*/
synchronized
(
cache
)
{
if
(
proxyClass
!=
null
)
{
cache
.
put
(
key
,
new
WeakReference
<
Class
<?>>(
proxyClass
));
}
else
{
cache
.
remove
(
key
);
}
cache
.
notifyAll
();
}
}
return
proxyClass
;
}
}
/**
/**
...
@@ -757,21 +771,6 @@ public class Proxy implements java.io.Serializable {
...
@@ -757,21 +771,6 @@ public class Proxy implements java.io.Serializable {
}
}
}
}
private
static
Object
newInstance
(
Constructor
<?>
cons
,
InvocationHandler
h
)
{
try
{
return
cons
.
newInstance
(
new
Object
[]
{
h
}
);
}
catch
(
IllegalAccessException
|
InstantiationException
e
)
{
throw
new
InternalError
(
e
.
toString
(),
e
);
}
catch
(
InvocationTargetException
e
)
{
Throwable
t
=
e
.
getCause
();
if
(
t
instanceof
RuntimeException
)
{
throw
(
RuntimeException
)
t
;
}
else
{
throw
new
InternalError
(
t
.
toString
(),
t
);
}
}
}
/**
/**
* Returns true if and only if the specified class was dynamically
* Returns true if and only if the specified class was dynamically
* generated to be a proxy class using the {@code getProxyClass}
* generated to be a proxy class using the {@code getProxyClass}
...
@@ -787,11 +786,7 @@ public class Proxy implements java.io.Serializable {
...
@@ -787,11 +786,7 @@ public class Proxy implements java.io.Serializable {
* @throws NullPointerException if {@code cl} is {@code null}
* @throws NullPointerException if {@code cl} is {@code null}
*/
*/
public
static
boolean
isProxyClass
(
Class
<?>
cl
)
{
public
static
boolean
isProxyClass
(
Class
<?>
cl
)
{
if
(
cl
==
null
)
{
return
Proxy
.
class
.
isAssignableFrom
(
cl
)
&&
proxyClassCache
.
containsValue
(
cl
);
throw
new
NullPointerException
();
}
return
proxyClasses
.
containsKey
(
cl
);
}
}
/**
/**
...
...
src/share/classes/java/lang/reflect/WeakCache.java
0 → 100644
浏览文件 @
f6dea072
/*
* Copyright (c) 2013, 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.
*/
package
java.lang.reflect
;
import
java.lang.ref.ReferenceQueue
;
import
java.lang.ref.WeakReference
;
import
java.util.Objects
;
import
java.util.concurrent.ConcurrentHashMap
;
import
java.util.concurrent.ConcurrentMap
;
import
java.util.function.BiFunction
;
import
java.util.function.Supplier
;
/**
* Cache mapping pairs of {@code (key, sub-key) -> value}. Keys and values are
* weakly but sub-keys are strongly referenced. Keys are passed directly to
* {@link #get} method which also takes a {@code parameter}. Sub-keys are
* calculated from keys and parameters using the {@code subKeyFactory} function
* passed to the constructor. Values are calculated from keys and parameters
* using the {@code valueFactory} function passed to the constructor.
* Keys can be {@code null} and are compared by identity while sub-keys returned by
* {@code subKeyFactory} or values returned by {@code valueFactory}
* can not be null. Sub-keys are compared using their {@link #equals} method.
* Entries are expunged from cache lazily on each invocation to {@link #get},
* {@link #containsValue} or {@link #size} methods when the WeakReferences to
* keys are cleared. Cleared WeakReferences to individual values don't cause
* expunging, but such entries are logically treated as non-existent and
* trigger re-evaluation of {@code valueFactory} on request for their
* key/subKey.
*
* @author Peter Levart
* @param <K> type of keys
* @param <P> type of parameters
* @param <V> type of values
*/
final
class
WeakCache
<
K
,
P
,
V
>
{
private
final
ReferenceQueue
<
K
>
refQueue
=
new
ReferenceQueue
<>();
// the key type is Object for supporting null key
private
final
ConcurrentMap
<
Object
,
ConcurrentMap
<
Object
,
Supplier
<
V
>>>
map
=
new
ConcurrentHashMap
<>();
private
final
ConcurrentMap
<
Supplier
<
V
>,
Boolean
>
reverseMap
=
new
ConcurrentHashMap
<>();
private
final
BiFunction
<
K
,
P
,
?>
subKeyFactory
;
private
final
BiFunction
<
K
,
P
,
V
>
valueFactory
;
/**
* Construct an instance of {@code WeakCache}
*
* @param subKeyFactory a function mapping a pair of
* {@code (key, parameter) -> sub-key}
* @param valueFactory a function mapping a pair of
* {@code (key, parameter) -> value}
* @throws NullPointerException if {@code subKeyFactory} or
* {@code valueFactory} is null.
*/
public
WeakCache
(
BiFunction
<
K
,
P
,
?>
subKeyFactory
,
BiFunction
<
K
,
P
,
V
>
valueFactory
)
{
this
.
subKeyFactory
=
Objects
.
requireNonNull
(
subKeyFactory
);
this
.
valueFactory
=
Objects
.
requireNonNull
(
valueFactory
);
}
/**
* Look-up the value through the cache. This always evaluates the
* {@code subKeyFactory} function and optionally evaluates
* {@code valueFactory} function if there is no entry in the cache for given
* pair of (key, subKey) or the entry has already been cleared.
*
* @param key possibly null key
* @param parameter parameter used together with key to create sub-key and
* value (should not be null)
* @return the cached value (never null)
* @throws NullPointerException if {@code parameter} passed in or
* {@code sub-key} calculated by
* {@code subKeyFactory} or {@code value}
* calculated by {@code valueFactory} is null.
*/
public
V
get
(
K
key
,
P
parameter
)
{
Objects
.
requireNonNull
(
parameter
);
expungeStaleEntries
();
Object
cacheKey
=
CacheKey
.
valueOf
(
key
,
refQueue
);
// lazily install the 2nd level valuesMap for the particular cacheKey
ConcurrentMap
<
Object
,
Supplier
<
V
>>
valuesMap
=
map
.
get
(
cacheKey
);
if
(
valuesMap
==
null
)
{
ConcurrentMap
<
Object
,
Supplier
<
V
>>
oldValuesMap
=
map
.
putIfAbsent
(
cacheKey
,
valuesMap
=
new
ConcurrentHashMap
<>());
if
(
oldValuesMap
!=
null
)
{
valuesMap
=
oldValuesMap
;
}
}
// create subKey and retrieve the possible Supplier<V> stored by that
// subKey from valuesMap
Object
subKey
=
Objects
.
requireNonNull
(
subKeyFactory
.
apply
(
key
,
parameter
));
Supplier
<
V
>
supplier
=
valuesMap
.
get
(
subKey
);
Factory
factory
=
null
;
while
(
true
)
{
if
(
supplier
!=
null
)
{
// supplier might be a Factory or a CacheValue<V> instance
V
value
=
supplier
.
get
();
if
(
value
!=
null
)
{
return
value
;
}
}
// else no supplier in cache
// or a supplier that returned null (could be a cleared CacheValue
// or a Factory that wasn't successful in installing the CacheValue)
// lazily construct a Factory
if
(
factory
==
null
)
{
factory
=
new
Factory
(
key
,
parameter
,
subKey
,
valuesMap
);
}
if
(
supplier
==
null
)
{
supplier
=
valuesMap
.
putIfAbsent
(
subKey
,
factory
);
if
(
supplier
==
null
)
{
// successfully installed Factory
supplier
=
factory
;
}
// else retry with winning supplier
}
else
{
if
(
valuesMap
.
replace
(
subKey
,
supplier
,
factory
))
{
// successfully replaced
// cleared CacheEntry / unsuccessful Factory
// with our Factory
supplier
=
factory
;
}
else
{
// retry with current supplier
supplier
=
valuesMap
.
get
(
subKey
);
}
}
}
}
/**
* Checks whether the specified non-null value is already present in this
* {@code WeakCache}. The check is made using identity comparison regardless
* of whether value's class overrides {@link Object#equals} or not.
*
* @param value the non-null value to check
* @return true if given {@code value} is already cached
* @throws NullPointerException if value is null
*/
public
boolean
containsValue
(
V
value
)
{
Objects
.
requireNonNull
(
value
);
expungeStaleEntries
();
return
reverseMap
.
containsKey
(
new
LookupValue
<>(
value
));
}
/**
* Returns the current number of cached entries that
* can decrease over time when keys/values are GC-ed.
*/
public
int
size
()
{
expungeStaleEntries
();
return
reverseMap
.
size
();
}
private
void
expungeStaleEntries
()
{
CacheKey
<
K
>
cacheKey
;
while
((
cacheKey
=
(
CacheKey
<
K
>)
refQueue
.
poll
())
!=
null
)
{
cacheKey
.
expungeFrom
(
map
,
reverseMap
);
}
}
/**
* A factory {@link Supplier} that implements the lazy synchronized
* construction of the value and installment of it into the cache.
*/
private
final
class
Factory
implements
Supplier
<
V
>
{
private
final
K
key
;
private
final
P
parameter
;
private
final
Object
subKey
;
private
final
ConcurrentMap
<
Object
,
Supplier
<
V
>>
valuesMap
;
Factory
(
K
key
,
P
parameter
,
Object
subKey
,
ConcurrentMap
<
Object
,
Supplier
<
V
>>
valuesMap
)
{
this
.
key
=
key
;
this
.
parameter
=
parameter
;
this
.
subKey
=
subKey
;
this
.
valuesMap
=
valuesMap
;
}
@Override
public
synchronized
V
get
()
{
// serialize access
// re-check
Supplier
<
V
>
supplier
=
valuesMap
.
get
(
subKey
);
if
(
supplier
!=
this
)
{
// something changed while we were waiting:
// might be that we were replaced by a CacheValue
// or were removed because of failure ->
// return null to signal WeakCache.get() to retry
// the loop
return
null
;
}
// else still us (supplier == this)
// create new value
V
value
=
null
;
try
{
value
=
Objects
.
requireNonNull
(
valueFactory
.
apply
(
key
,
parameter
));
}
finally
{
if
(
value
==
null
)
{
// remove us on failure
valuesMap
.
remove
(
subKey
,
this
);
}
}
// the only path to reach here is with non-null value
assert
value
!=
null
;
// wrap value with CacheValue (WeakReference)
CacheValue
<
V
>
cacheValue
=
new
CacheValue
<>(
value
);
// try replacing us with CacheValue (this should always succeed)
if
(
valuesMap
.
replace
(
subKey
,
this
,
cacheValue
))
{
// put also in reverseMap
reverseMap
.
put
(
cacheValue
,
Boolean
.
TRUE
);
}
else
{
throw
new
AssertionError
(
"Should not reach here"
);
}
// successfully replaced us with new CacheValue -> return the value
// wrapped by it
return
value
;
}
}
/**
* Common type of value suppliers that are holding a referent.
* The {@link #equals} and {@link #hashCode} of implementations is defined
* to compare the referent by identity.
*/
private
interface
Value
<
V
>
extends
Supplier
<
V
>
{}
/**
* An optimized {@link Value} used to look-up the value in
* {@link WeakCache#containsValue} method so that we are not
* constructing the whole {@link CacheValue} just to look-up the referent.
*/
private
static
final
class
LookupValue
<
V
>
implements
Value
<
V
>
{
private
final
V
value
;
LookupValue
(
V
value
)
{
this
.
value
=
value
;
}
@Override
public
V
get
()
{
return
value
;
}
@Override
public
int
hashCode
()
{
return
System
.
identityHashCode
(
value
);
// compare by identity
}
@Override
public
boolean
equals
(
Object
obj
)
{
return
obj
==
this
||
obj
instanceof
Value
&&
this
.
value
==
((
Value
<?>)
obj
).
get
();
// compare by identity
}
}
/**
* A {@link Value} that weakly references the referent.
*/
private
static
final
class
CacheValue
<
V
>
extends
WeakReference
<
V
>
implements
Value
<
V
>
{
private
final
int
hash
;
CacheValue
(
V
value
)
{
super
(
value
);
this
.
hash
=
System
.
identityHashCode
(
value
);
// compare by identity
}
@Override
public
int
hashCode
()
{
return
hash
;
}
@Override
public
boolean
equals
(
Object
obj
)
{
V
value
;
return
obj
==
this
||
obj
instanceof
Value
&&
// cleared CacheValue is only equal to itself
(
value
=
get
())
!=
null
&&
value
==
((
Value
<?>)
obj
).
get
();
// compare by identity
}
}
/**
* CacheKey containing a weakly referenced {@code key}. It registers
* itself with the {@code refQueue} so that it can be used to expunge
* the entry when the {@link WeakReference} is cleared.
*/
private
static
final
class
CacheKey
<
K
>
extends
WeakReference
<
K
>
{
// a replacement for null keys
private
static
final
Object
NULL_KEY
=
new
Object
();
static
<
K
>
Object
valueOf
(
K
key
,
ReferenceQueue
<
K
>
refQueue
)
{
return
key
==
null
// null key means we can't weakly reference it,
// so we use a NULL_KEY singleton as cache key
?
NULL_KEY
// non-null key requires wrapping with a WeakReference
:
new
CacheKey
<>(
key
,
refQueue
);
}
private
final
int
hash
;
private
CacheKey
(
K
key
,
ReferenceQueue
<
K
>
refQueue
)
{
super
(
key
,
refQueue
);
this
.
hash
=
System
.
identityHashCode
(
key
);
// compare by identity
}
@Override
public
int
hashCode
()
{
return
hash
;
}
@Override
public
boolean
equals
(
Object
obj
)
{
K
key
;
return
obj
==
this
||
obj
!=
null
&&
obj
.
getClass
()
==
this
.
getClass
()
&&
// cleared CacheKey is only equal to itself
(
key
=
this
.
get
())
!=
null
&&
// compare key by identity
key
==
((
CacheKey
<
K
>)
obj
).
get
();
}
void
expungeFrom
(
ConcurrentMap
<?,
?
extends
ConcurrentMap
<?,
?>>
map
,
ConcurrentMap
<?,
Boolean
>
reverseMap
)
{
// removing just by key is always safe here because after a CacheKey
// is cleared and enqueue-ed it is only equal to itself
// (see equals method)...
ConcurrentMap
<?,
?>
valuesMap
=
map
.
remove
(
this
);
// remove also from reverseMap if needed
if
(
valuesMap
!=
null
)
{
for
(
Object
cacheValue
:
valuesMap
.
values
())
{
reverseMap
.
remove
(
cacheValue
);
}
}
}
}
}
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录