Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
爱吃血肠
spring-framework
提交
356ef45e
S
spring-framework
项目概览
爱吃血肠
/
spring-framework
通知
1
Star
0
Fork
0
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
0
列表
看板
标记
里程碑
合并请求
0
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
S
spring-framework
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
0
Issue
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
提交
Issue看板
体验新版 GitCode,发现更多精彩内容 >>
提交
356ef45e
编写于
3月 14, 2018
作者:
J
Juergen Hoeller
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
ConcurrentReferenceHashMap properly handles getOrDefault for null values
Issue: SPR-16584
上级
cc12afde
变更
2
隐藏空白更改
内联
并排
Showing
2 changed file
with
96 addition
and
86 deletion
+96
-86
spring-core/src/main/java/org/springframework/util/ConcurrentReferenceHashMap.java
.../org/springframework/util/ConcurrentReferenceHashMap.java
+38
-21
spring-core/src/test/java/org/springframework/util/ConcurrentReferenceHashMapTests.java
...springframework/util/ConcurrentReferenceHashMapTests.java
+58
-65
未找到文件。
spring-core/src/main/java/org/springframework/util/ConcurrentReferenceHashMap.java
浏览文件 @
356ef45e
/*
/*
* Copyright 2002-201
7
the original author or authors.
* Copyright 2002-201
8
the original author or authors.
*
*
* Licensed under the Apache License, Version 2.0 (the "License");
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* you may not use this file except in compliance with the License.
...
@@ -54,6 +54,7 @@ import org.springframework.lang.Nullable;
...
@@ -54,6 +54,7 @@ import org.springframework.lang.Nullable;
* {@linkplain SoftReference soft entry references}.
* {@linkplain SoftReference soft entry references}.
*
*
* @author Phillip Webb
* @author Phillip Webb
* @author Juergen Hoeller
* @since 3.2
* @since 3.2
* @param <K> the key type
* @param <K> the key type
* @param <V> the value type
* @param <V> the value type
...
@@ -226,19 +227,30 @@ public class ConcurrentReferenceHashMap<K, V> extends AbstractMap<K, V> implemen
...
@@ -226,19 +227,30 @@ public class ConcurrentReferenceHashMap<K, V> extends AbstractMap<K, V> implemen
@Override
@Override
@Nullable
@Nullable
public
V
get
(
Object
key
)
{
public
V
get
(
@Nullable
Object
key
)
{
Reference
<
K
,
V
>
reference
=
getReference
(
key
,
Restructure
.
WHEN_NECESSARY
);
Entry
<
K
,
V
>
entry
=
getEntryIfAvailable
(
key
);
Entry
<
K
,
V
>
entry
=
(
reference
!=
null
?
reference
.
get
()
:
null
);
return
(
entry
!=
null
?
entry
.
getValue
()
:
null
);
return
(
entry
!=
null
?
entry
.
getValue
()
:
null
);
}
}
@Override
@Override
public
boolean
containsKey
(
Object
key
)
{
@Nullable
Reference
<
K
,
V
>
reference
=
getReference
(
key
,
Restructure
.
WHEN_NECESSARY
);
public
V
getOrDefault
(
@Nullable
Object
key
,
@Nullable
V
defaultValue
)
{
Entry
<
K
,
V
>
entry
=
(
reference
!=
null
?
reference
.
get
()
:
null
);
Entry
<
K
,
V
>
entry
=
getEntryIfAvailable
(
key
);
return
(
entry
!=
null
?
entry
.
getValue
()
:
defaultValue
);
}
@Override
public
boolean
containsKey
(
@Nullable
Object
key
)
{
Entry
<
K
,
V
>
entry
=
getEntryIfAvailable
(
key
);
return
(
entry
!=
null
&&
ObjectUtils
.
nullSafeEquals
(
entry
.
getKey
(),
key
));
return
(
entry
!=
null
&&
ObjectUtils
.
nullSafeEquals
(
entry
.
getKey
(),
key
));
}
}
@Nullable
private
Entry
<
K
,
V
>
getEntryIfAvailable
(
@Nullable
Object
key
)
{
Reference
<
K
,
V
>
reference
=
getReference
(
key
,
Restructure
.
WHEN_NECESSARY
);
return
(
reference
!=
null
?
reference
.
get
()
:
null
);
}
/**
/**
* Return a {@link Reference} to the {@link Entry} for the specified {@code key},
* Return a {@link Reference} to the {@link Entry} for the specified {@code key},
* or {@code null} if not found.
* or {@code null} if not found.
...
@@ -254,28 +266,28 @@ public class ConcurrentReferenceHashMap<K, V> extends AbstractMap<K, V> implemen
...
@@ -254,28 +266,28 @@ public class ConcurrentReferenceHashMap<K, V> extends AbstractMap<K, V> implemen
@Override
@Override
@Nullable
@Nullable
public
V
put
(
K
key
,
V
value
)
{
public
V
put
(
@Nullable
K
key
,
@Nullable
V
value
)
{
return
put
(
key
,
value
,
true
);
return
put
(
key
,
value
,
true
);
}
}
@Override
@Override
@Nullable
@Nullable
public
V
putIfAbsent
(
K
key
,
V
value
)
{
public
V
putIfAbsent
(
@Nullable
K
key
,
@Nullable
V
value
)
{
return
put
(
key
,
value
,
false
);
return
put
(
key
,
value
,
false
);
}
}
@Nullable
@Nullable
private
V
put
(
final
K
key
,
final
V
value
,
final
boolean
overwriteExisting
)
{
private
V
put
(
@Nullable
final
K
key
,
@Nullable
final
V
value
,
final
boolean
overwriteExisting
)
{
return
doTask
(
key
,
new
Task
<
V
>(
TaskOption
.
RESTRUCTURE_BEFORE
,
TaskOption
.
RESIZE
)
{
return
doTask
(
key
,
new
Task
<
V
>(
TaskOption
.
RESTRUCTURE_BEFORE
,
TaskOption
.
RESIZE
)
{
@Override
@Override
@Nullable
@Nullable
protected
V
execute
(
@Nullable
Reference
<
K
,
V
>
reference
,
@Nullable
Entry
<
K
,
V
>
entry
,
@Nullable
Entries
entries
)
{
protected
V
execute
(
@Nullable
Reference
<
K
,
V
>
reference
,
@Nullable
Entry
<
K
,
V
>
entry
,
@Nullable
Entries
entries
)
{
if
(
entry
!=
null
)
{
if
(
entry
!=
null
)
{
V
previous
Value
=
entry
.
getValue
();
V
old
Value
=
entry
.
getValue
();
if
(
overwriteExisting
)
{
if
(
overwriteExisting
)
{
entry
.
setValue
(
value
);
entry
.
setValue
(
value
);
}
}
return
previous
Value
;
return
old
Value
;
}
}
Assert
.
state
(
entries
!=
null
,
"No entries segment"
);
Assert
.
state
(
entries
!=
null
,
"No entries segment"
);
entries
.
add
(
value
);
entries
.
add
(
value
);
...
@@ -342,9 +354,9 @@ public class ConcurrentReferenceHashMap<K, V> extends AbstractMap<K, V> implemen
...
@@ -342,9 +354,9 @@ public class ConcurrentReferenceHashMap<K, V> extends AbstractMap<K, V> implemen
@Nullable
@Nullable
protected
V
execute
(
@Nullable
Reference
<
K
,
V
>
reference
,
@Nullable
Entry
<
K
,
V
>
entry
)
{
protected
V
execute
(
@Nullable
Reference
<
K
,
V
>
reference
,
@Nullable
Entry
<
K
,
V
>
entry
)
{
if
(
entry
!=
null
)
{
if
(
entry
!=
null
)
{
V
previous
Value
=
entry
.
getValue
();
V
old
Value
=
entry
.
getValue
();
entry
.
setValue
(
value
);
entry
.
setValue
(
value
);
return
previous
Value
;
return
old
Value
;
}
}
return
null
;
return
null
;
}
}
...
@@ -389,7 +401,7 @@ public class ConcurrentReferenceHashMap<K, V> extends AbstractMap<K, V> implemen
...
@@ -389,7 +401,7 @@ public class ConcurrentReferenceHashMap<K, V> extends AbstractMap<K, V> implemen
}
}
@Nullable
@Nullable
private
<
T
>
T
doTask
(
Object
key
,
Task
<
T
>
task
)
{
private
<
T
>
T
doTask
(
@Nullable
Object
key
,
Task
<
T
>
task
)
{
int
hash
=
getHash
(
key
);
int
hash
=
getHash
(
key
);
return
getSegmentForHash
(
hash
).
doTask
(
hash
,
key
,
task
);
return
getSegmentForHash
(
hash
).
doTask
(
hash
,
key
,
task
);
}
}
...
@@ -488,7 +500,7 @@ public class ConcurrentReferenceHashMap<K, V> extends AbstractMap<K, V> implemen
...
@@ -488,7 +500,7 @@ public class ConcurrentReferenceHashMap<K, V> extends AbstractMap<K, V> implemen
* @return the result of the operation
* @return the result of the operation
*/
*/
@Nullable
@Nullable
public
<
T
>
T
doTask
(
final
int
hash
,
final
Object
key
,
final
Task
<
T
>
task
)
{
public
<
T
>
T
doTask
(
final
int
hash
,
@Nullable
final
Object
key
,
final
Task
<
T
>
task
)
{
boolean
resize
=
task
.
hasOption
(
TaskOption
.
RESIZE
);
boolean
resize
=
task
.
hasOption
(
TaskOption
.
RESIZE
);
if
(
task
.
hasOption
(
TaskOption
.
RESTRUCTURE_BEFORE
))
{
if
(
task
.
hasOption
(
TaskOption
.
RESTRUCTURE_BEFORE
))
{
restructureIfNecessary
(
resize
);
restructureIfNecessary
(
resize
);
...
@@ -504,7 +516,7 @@ public class ConcurrentReferenceHashMap<K, V> extends AbstractMap<K, V> implemen
...
@@ -504,7 +516,7 @@ public class ConcurrentReferenceHashMap<K, V> extends AbstractMap<K, V> implemen
Entry
<
K
,
V
>
entry
=
(
reference
!=
null
?
reference
.
get
()
:
null
);
Entry
<
K
,
V
>
entry
=
(
reference
!=
null
?
reference
.
get
()
:
null
);
Entries
entries
=
new
Entries
()
{
Entries
entries
=
new
Entries
()
{
@Override
@Override
public
void
add
(
V
value
)
{
public
void
add
(
@Nullable
V
value
)
{
@SuppressWarnings
(
"unchecked"
)
@SuppressWarnings
(
"unchecked"
)
Entry
<
K
,
V
>
newEntry
=
new
Entry
<>((
K
)
key
,
value
);
Entry
<
K
,
V
>
newEntry
=
new
Entry
<>((
K
)
key
,
value
);
Reference
<
K
,
V
>
newReference
=
Segment
.
this
.
referenceManager
.
createReference
(
newEntry
,
hash
,
head
);
Reference
<
K
,
V
>
newReference
=
Segment
.
this
.
referenceManager
.
createReference
(
newEntry
,
hash
,
head
);
...
@@ -617,7 +629,7 @@ public class ConcurrentReferenceHashMap<K, V> extends AbstractMap<K, V> implemen
...
@@ -617,7 +629,7 @@ public class ConcurrentReferenceHashMap<K, V> extends AbstractMap<K, V> implemen
Entry
<
K
,
V
>
entry
=
currRef
.
get
();
Entry
<
K
,
V
>
entry
=
currRef
.
get
();
if
(
entry
!=
null
)
{
if
(
entry
!=
null
)
{
K
entryKey
=
entry
.
getKey
();
K
entryKey
=
entry
.
getKey
();
if
(
entryKey
==
key
||
entryKey
.
equals
(
key
))
{
if
(
ObjectUtils
.
nullSafeEquals
(
entryKey
,
key
))
{
return
currRef
;
return
currRef
;
}
}
}
}
...
@@ -688,27 +700,32 @@ public class ConcurrentReferenceHashMap<K, V> extends AbstractMap<K, V> implemen
...
@@ -688,27 +700,32 @@ public class ConcurrentReferenceHashMap<K, V> extends AbstractMap<K, V> implemen
*/
*/
protected
static
final
class
Entry
<
K
,
V
>
implements
Map
.
Entry
<
K
,
V
>
{
protected
static
final
class
Entry
<
K
,
V
>
implements
Map
.
Entry
<
K
,
V
>
{
@Nullable
private
final
K
key
;
private
final
K
key
;
@Nullable
private
volatile
V
value
;
private
volatile
V
value
;
public
Entry
(
K
key
,
V
value
)
{
public
Entry
(
@Nullable
K
key
,
@Nullable
V
value
)
{
this
.
key
=
key
;
this
.
key
=
key
;
this
.
value
=
value
;
this
.
value
=
value
;
}
}
@Override
@Override
@Nullable
public
K
getKey
()
{
public
K
getKey
()
{
return
this
.
key
;
return
this
.
key
;
}
}
@Override
@Override
@Nullable
public
V
getValue
()
{
public
V
getValue
()
{
return
this
.
value
;
return
this
.
value
;
}
}
@Override
@Override
public
V
setValue
(
V
value
)
{
@Nullable
public
V
setValue
(
@Nullable
V
value
)
{
V
previous
=
this
.
value
;
V
previous
=
this
.
value
;
this
.
value
=
value
;
this
.
value
=
value
;
return
previous
;
return
previous
;
...
@@ -800,7 +817,7 @@ public class ConcurrentReferenceHashMap<K, V> extends AbstractMap<K, V> implemen
...
@@ -800,7 +817,7 @@ public class ConcurrentReferenceHashMap<K, V> extends AbstractMap<K, V> implemen
* Add a new entry with the specified value.
* Add a new entry with the specified value.
* @param value the value to add
* @param value the value to add
*/
*/
public
abstract
void
add
(
V
value
);
public
abstract
void
add
(
@Nullable
V
value
);
}
}
...
...
spring-core/src/test/java/org/springframework/util/ConcurrentReferenceHashMapTests.java
浏览文件 @
356ef45e
/*
/*
* Copyright 2002-201
6
the original author or authors.
* Copyright 2002-201
8
the original author or authors.
*
*
* Licensed under the Apache License, Version 2.0 (the "License");
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* you may not use this file except in compliance with the License.
...
@@ -61,7 +61,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -61,7 +61,7 @@ public class ConcurrentReferenceHashMapTests {
@Test
@Test
public
void
shouldCreateWithDefaults
()
throws
Exception
{
public
void
shouldCreateWithDefaults
()
{
ConcurrentReferenceHashMap
<
Integer
,
String
>
map
=
new
ConcurrentReferenceHashMap
<>();
ConcurrentReferenceHashMap
<
Integer
,
String
>
map
=
new
ConcurrentReferenceHashMap
<>();
assertThat
(
map
.
getSegmentsSize
(),
is
(
16
));
assertThat
(
map
.
getSegmentsSize
(),
is
(
16
));
assertThat
(
map
.
getSegment
(
0
).
getSize
(),
is
(
1
));
assertThat
(
map
.
getSegment
(
0
).
getSize
(),
is
(
1
));
...
@@ -69,7 +69,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -69,7 +69,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldCreateWithInitialCapacity
()
throws
Exception
{
public
void
shouldCreateWithInitialCapacity
()
{
ConcurrentReferenceHashMap
<
Integer
,
String
>
map
=
new
ConcurrentReferenceHashMap
<>(
32
);
ConcurrentReferenceHashMap
<
Integer
,
String
>
map
=
new
ConcurrentReferenceHashMap
<>(
32
);
assertThat
(
map
.
getSegmentsSize
(),
is
(
16
));
assertThat
(
map
.
getSegmentsSize
(),
is
(
16
));
assertThat
(
map
.
getSegment
(
0
).
getSize
(),
is
(
2
));
assertThat
(
map
.
getSegment
(
0
).
getSize
(),
is
(
2
));
...
@@ -77,7 +77,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -77,7 +77,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldCreateWithInitialCapacityAndLoadFactor
()
throws
Exception
{
public
void
shouldCreateWithInitialCapacityAndLoadFactor
()
{
ConcurrentReferenceHashMap
<
Integer
,
String
>
map
=
new
ConcurrentReferenceHashMap
<>(
32
,
0.5f
);
ConcurrentReferenceHashMap
<
Integer
,
String
>
map
=
new
ConcurrentReferenceHashMap
<>(
32
,
0.5f
);
assertThat
(
map
.
getSegmentsSize
(),
is
(
16
));
assertThat
(
map
.
getSegmentsSize
(),
is
(
16
));
assertThat
(
map
.
getSegment
(
0
).
getSize
(),
is
(
2
));
assertThat
(
map
.
getSegment
(
0
).
getSize
(),
is
(
2
));
...
@@ -85,7 +85,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -85,7 +85,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldCreateWithInitialCapacityAndConcurrenyLevel
()
throws
Exception
{
public
void
shouldCreateWithInitialCapacityAndConcurrenyLevel
()
{
ConcurrentReferenceHashMap
<
Integer
,
String
>
map
=
new
ConcurrentReferenceHashMap
<>(
16
,
2
);
ConcurrentReferenceHashMap
<
Integer
,
String
>
map
=
new
ConcurrentReferenceHashMap
<>(
16
,
2
);
assertThat
(
map
.
getSegmentsSize
(),
is
(
2
));
assertThat
(
map
.
getSegmentsSize
(),
is
(
2
));
assertThat
(
map
.
getSegment
(
0
).
getSize
(),
is
(
8
));
assertThat
(
map
.
getSegment
(
0
).
getSize
(),
is
(
8
));
...
@@ -93,7 +93,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -93,7 +93,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldCreateFullyCustom
()
throws
Exception
{
public
void
shouldCreateFullyCustom
()
{
ConcurrentReferenceHashMap
<
Integer
,
String
>
map
=
new
ConcurrentReferenceHashMap
<>(
5
,
0.5f
,
3
);
ConcurrentReferenceHashMap
<
Integer
,
String
>
map
=
new
ConcurrentReferenceHashMap
<>(
5
,
0.5f
,
3
);
// concurrencyLevel of 3 ends up as 4 (nearest power of 2)
// concurrencyLevel of 3 ends up as 4 (nearest power of 2)
assertThat
(
map
.
getSegmentsSize
(),
is
(
4
));
assertThat
(
map
.
getSegmentsSize
(),
is
(
4
));
...
@@ -103,7 +103,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -103,7 +103,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldNeedNonNegativeInitialCapacity
()
throws
Exception
{
public
void
shouldNeedNonNegativeInitialCapacity
()
{
new
ConcurrentReferenceHashMap
<
Integer
,
String
>(
0
,
1
);
new
ConcurrentReferenceHashMap
<
Integer
,
String
>(
0
,
1
);
this
.
thrown
.
expect
(
IllegalArgumentException
.
class
);
this
.
thrown
.
expect
(
IllegalArgumentException
.
class
);
this
.
thrown
.
expectMessage
(
"Initial capacity must not be negative"
);
this
.
thrown
.
expectMessage
(
"Initial capacity must not be negative"
);
...
@@ -111,7 +111,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -111,7 +111,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldNeedPositiveLoadFactor
()
throws
Exception
{
public
void
shouldNeedPositiveLoadFactor
()
{
new
ConcurrentReferenceHashMap
<
Integer
,
String
>(
0
,
0.1f
,
1
);
new
ConcurrentReferenceHashMap
<
Integer
,
String
>(
0
,
0.1f
,
1
);
this
.
thrown
.
expect
(
IllegalArgumentException
.
class
);
this
.
thrown
.
expect
(
IllegalArgumentException
.
class
);
this
.
thrown
.
expectMessage
(
"Load factor must be positive"
);
this
.
thrown
.
expectMessage
(
"Load factor must be positive"
);
...
@@ -119,7 +119,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -119,7 +119,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldNeedPositiveConcurrencyLevel
()
throws
Exception
{
public
void
shouldNeedPositiveConcurrencyLevel
()
{
new
ConcurrentReferenceHashMap
<
Integer
,
String
>(
1
,
1
);
new
ConcurrentReferenceHashMap
<
Integer
,
String
>(
1
,
1
);
this
.
thrown
.
expect
(
IllegalArgumentException
.
class
);
this
.
thrown
.
expect
(
IllegalArgumentException
.
class
);
this
.
thrown
.
expectMessage
(
"Concurrency level must be positive"
);
this
.
thrown
.
expectMessage
(
"Concurrency level must be positive"
);
...
@@ -127,7 +127,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -127,7 +127,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldPutAndGet
()
throws
Exception
{
public
void
shouldPutAndGet
()
{
// NOTE we are using mock references so we don't need to worry about GC
// NOTE we are using mock references so we don't need to worry about GC
assertThat
(
this
.
map
.
size
(),
is
(
0
));
assertThat
(
this
.
map
.
size
(),
is
(
0
));
this
.
map
.
put
(
123
,
"123"
);
this
.
map
.
put
(
123
,
"123"
);
...
@@ -140,32 +140,40 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -140,32 +140,40 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldReplaceOnDoublePut
()
throws
Exception
{
public
void
shouldReplaceOnDoublePut
()
{
this
.
map
.
put
(
123
,
"321"
);
this
.
map
.
put
(
123
,
"321"
);
this
.
map
.
put
(
123
,
"123"
);
this
.
map
.
put
(
123
,
"123"
);
assertThat
(
this
.
map
.
get
(
123
),
is
(
"123"
));
assertThat
(
this
.
map
.
get
(
123
),
is
(
"123"
));
}
}
@Test
@Test
public
void
shouldPutNullKey
()
throws
Exception
{
public
void
shouldPutNullKey
()
{
assertThat
(
this
.
map
.
get
(
null
),
is
(
nullValue
()));
assertThat
(
this
.
map
.
getOrDefault
(
null
,
"456"
),
is
(
"456"
));
this
.
map
.
put
(
null
,
"123"
);
this
.
map
.
put
(
null
,
"123"
);
assertThat
(
this
.
map
.
get
(
null
),
is
(
"123"
));
assertThat
(
this
.
map
.
get
(
null
),
is
(
"123"
));
assertThat
(
this
.
map
.
getOrDefault
(
null
,
"456"
),
is
(
"123"
));
}
}
@Test
@Test
public
void
shouldPutNullValue
()
throws
Exception
{
public
void
shouldPutNullValue
()
{
assertThat
(
this
.
map
.
get
(
123
),
is
(
nullValue
()));
assertThat
(
this
.
map
.
getOrDefault
(
123
,
"456"
),
is
(
"456"
));
this
.
map
.
put
(
123
,
"321"
);
this
.
map
.
put
(
123
,
"321"
);
assertThat
(
this
.
map
.
get
(
123
),
is
(
"321"
));
assertThat
(
this
.
map
.
getOrDefault
(
123
,
"456"
),
is
(
"321"
));
this
.
map
.
put
(
123
,
null
);
this
.
map
.
put
(
123
,
null
);
assertThat
(
this
.
map
.
get
(
123
),
is
(
nullValue
()));
assertThat
(
this
.
map
.
get
(
123
),
is
(
nullValue
()));
assertThat
(
this
.
map
.
getOrDefault
(
123
,
"456"
),
is
(
nullValue
()));
}
}
@Test
@Test
public
void
shouldGetWithNoItems
()
throws
Exception
{
public
void
shouldGetWithNoItems
()
{
assertThat
(
this
.
map
.
get
(
123
),
is
(
nullValue
()));
assertThat
(
this
.
map
.
get
(
123
),
is
(
nullValue
()));
}
}
@Test
@Test
public
void
shouldApplySupplimentalHash
()
throws
Exception
{
public
void
shouldApplySupplimentalHash
()
{
Integer
key
=
123
;
Integer
key
=
123
;
this
.
map
.
put
(
key
,
"123"
);
this
.
map
.
put
(
key
,
"123"
);
assertThat
(
this
.
map
.
getSupplimentalHash
(),
is
(
not
(
key
.
hashCode
())));
assertThat
(
this
.
map
.
getSupplimentalHash
(),
is
(
not
(
key
.
hashCode
())));
...
@@ -173,7 +181,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -173,7 +181,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldGetFollowingNexts
()
throws
Exception
{
public
void
shouldGetFollowingNexts
()
{
// Use loadFactor to disable resize
// Use loadFactor to disable resize
this
.
map
=
new
TestWeakConcurrentCache
<>(
1
,
10.0f
,
1
);
this
.
map
=
new
TestWeakConcurrentCache
<>(
1
,
10.0f
,
1
);
this
.
map
.
put
(
1
,
"1"
);
this
.
map
.
put
(
1
,
"1"
);
...
@@ -187,7 +195,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -187,7 +195,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldResize
()
throws
Exception
{
public
void
shouldResize
()
{
this
.
map
=
new
TestWeakConcurrentCache
<>(
1
,
0.75f
,
1
);
this
.
map
=
new
TestWeakConcurrentCache
<>(
1
,
0.75f
,
1
);
this
.
map
.
put
(
1
,
"1"
);
this
.
map
.
put
(
1
,
"1"
);
assertThat
(
this
.
map
.
getSegment
(
0
).
getSize
(),
is
(
1
));
assertThat
(
this
.
map
.
getSegment
(
0
).
getSize
(),
is
(
1
));
...
@@ -217,7 +225,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -217,7 +225,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldPurgeOnGet
()
throws
Exception
{
public
void
shouldPurgeOnGet
()
{
this
.
map
=
new
TestWeakConcurrentCache
<>(
1
,
0.75f
,
1
);
this
.
map
=
new
TestWeakConcurrentCache
<>(
1
,
0.75f
,
1
);
for
(
int
i
=
1
;
i
<=
5
;
i
++)
{
for
(
int
i
=
1
;
i
<=
5
;
i
++)
{
this
.
map
.
put
(
i
,
String
.
valueOf
(
i
));
this
.
map
.
put
(
i
,
String
.
valueOf
(
i
));
...
@@ -232,7 +240,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -232,7 +240,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldPergeOnPut
()
throws
Exception
{
public
void
shouldPergeOnPut
()
{
this
.
map
=
new
TestWeakConcurrentCache
<>(
1
,
0.75f
,
1
);
this
.
map
=
new
TestWeakConcurrentCache
<>(
1
,
0.75f
,
1
);
for
(
int
i
=
1
;
i
<=
5
;
i
++)
{
for
(
int
i
=
1
;
i
<=
5
;
i
++)
{
this
.
map
.
put
(
i
,
String
.
valueOf
(
i
));
this
.
map
.
put
(
i
,
String
.
valueOf
(
i
));
...
@@ -248,28 +256,28 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -248,28 +256,28 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldPutIfAbsent
()
throws
Exception
{
public
void
shouldPutIfAbsent
()
{
assertThat
(
this
.
map
.
putIfAbsent
(
123
,
"123"
),
is
(
nullValue
()));
assertThat
(
this
.
map
.
putIfAbsent
(
123
,
"123"
),
is
(
nullValue
()));
assertThat
(
this
.
map
.
putIfAbsent
(
123
,
"123b"
),
is
(
"123"
));
assertThat
(
this
.
map
.
putIfAbsent
(
123
,
"123b"
),
is
(
"123"
));
assertThat
(
this
.
map
.
get
(
123
),
is
(
"123"
));
assertThat
(
this
.
map
.
get
(
123
),
is
(
"123"
));
}
}
@Test
@Test
public
void
shouldPutIfAbsentWithNullValue
()
throws
Exception
{
public
void
shouldPutIfAbsentWithNullValue
()
{
assertThat
(
this
.
map
.
putIfAbsent
(
123
,
null
),
is
(
nullValue
()));
assertThat
(
this
.
map
.
putIfAbsent
(
123
,
null
),
is
(
nullValue
()));
assertThat
(
this
.
map
.
putIfAbsent
(
123
,
"123"
),
is
(
nullValue
()));
assertThat
(
this
.
map
.
putIfAbsent
(
123
,
"123"
),
is
(
nullValue
()));
assertThat
(
this
.
map
.
get
(
123
),
is
(
nullValue
()));
assertThat
(
this
.
map
.
get
(
123
),
is
(
nullValue
()));
}
}
@Test
@Test
public
void
shouldPutIfAbsentWithNullKey
()
throws
Exception
{
public
void
shouldPutIfAbsentWithNullKey
()
{
assertThat
(
this
.
map
.
putIfAbsent
(
null
,
"123"
),
is
(
nullValue
()));
assertThat
(
this
.
map
.
putIfAbsent
(
null
,
"123"
),
is
(
nullValue
()));
assertThat
(
this
.
map
.
putIfAbsent
(
null
,
"123b"
),
is
(
"123"
));
assertThat
(
this
.
map
.
putIfAbsent
(
null
,
"123b"
),
is
(
"123"
));
assertThat
(
this
.
map
.
get
(
null
),
is
(
"123"
));
assertThat
(
this
.
map
.
get
(
null
),
is
(
"123"
));
}
}
@Test
@Test
public
void
shouldRemoveKeyAndValue
()
throws
Exception
{
public
void
shouldRemoveKeyAndValue
()
{
this
.
map
.
put
(
123
,
"123"
);
this
.
map
.
put
(
123
,
"123"
);
assertThat
(
this
.
map
.
remove
(
123
,
"456"
),
is
(
false
));
assertThat
(
this
.
map
.
remove
(
123
,
"456"
),
is
(
false
));
assertThat
(
this
.
map
.
get
(
123
),
is
(
"123"
));
assertThat
(
this
.
map
.
get
(
123
),
is
(
"123"
));
...
@@ -279,7 +287,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -279,7 +287,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldRemoveKeyAndValueWithExistingNull
()
throws
Exception
{
public
void
shouldRemoveKeyAndValueWithExistingNull
()
{
this
.
map
.
put
(
123
,
null
);
this
.
map
.
put
(
123
,
null
);
assertThat
(
this
.
map
.
remove
(
123
,
"456"
),
is
(
false
));
assertThat
(
this
.
map
.
remove
(
123
,
"456"
),
is
(
false
));
assertThat
(
this
.
map
.
get
(
123
),
is
(
nullValue
()));
assertThat
(
this
.
map
.
get
(
123
),
is
(
nullValue
()));
...
@@ -289,7 +297,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -289,7 +297,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldReplaceOldValueWithNewValue
()
throws
Exception
{
public
void
shouldReplaceOldValueWithNewValue
()
{
this
.
map
.
put
(
123
,
"123"
);
this
.
map
.
put
(
123
,
"123"
);
assertThat
(
this
.
map
.
replace
(
123
,
"456"
,
"789"
),
is
(
false
));
assertThat
(
this
.
map
.
replace
(
123
,
"456"
,
"789"
),
is
(
false
));
assertThat
(
this
.
map
.
get
(
123
),
is
(
"123"
));
assertThat
(
this
.
map
.
get
(
123
),
is
(
"123"
));
...
@@ -298,7 +306,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -298,7 +306,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldReplaceOldNullValueWithNewValue
()
throws
Exception
{
public
void
shouldReplaceOldNullValueWithNewValue
()
{
this
.
map
.
put
(
123
,
null
);
this
.
map
.
put
(
123
,
null
);
assertThat
(
this
.
map
.
replace
(
123
,
"456"
,
"789"
),
is
(
false
));
assertThat
(
this
.
map
.
replace
(
123
,
"456"
,
"789"
),
is
(
false
));
assertThat
(
this
.
map
.
get
(
123
),
is
(
nullValue
()));
assertThat
(
this
.
map
.
get
(
123
),
is
(
nullValue
()));
...
@@ -307,21 +315,21 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -307,21 +315,21 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldReplaceValue
()
throws
Exception
{
public
void
shouldReplaceValue
()
{
this
.
map
.
put
(
123
,
"123"
);
this
.
map
.
put
(
123
,
"123"
);
assertThat
(
this
.
map
.
replace
(
123
,
"456"
),
is
(
"123"
));
assertThat
(
this
.
map
.
replace
(
123
,
"456"
),
is
(
"123"
));
assertThat
(
this
.
map
.
get
(
123
),
is
(
"456"
));
assertThat
(
this
.
map
.
get
(
123
),
is
(
"456"
));
}
}
@Test
@Test
public
void
shouldReplaceNullValue
()
throws
Exception
{
public
void
shouldReplaceNullValue
()
{
this
.
map
.
put
(
123
,
null
);
this
.
map
.
put
(
123
,
null
);
assertThat
(
this
.
map
.
replace
(
123
,
"456"
),
is
(
nullValue
()));
assertThat
(
this
.
map
.
replace
(
123
,
"456"
),
is
(
nullValue
()));
assertThat
(
this
.
map
.
get
(
123
),
is
(
"456"
));
assertThat
(
this
.
map
.
get
(
123
),
is
(
"456"
));
}
}
@Test
@Test
public
void
shouldGetSize
()
throws
Exception
{
public
void
shouldGetSize
()
{
assertThat
(
this
.
map
.
size
(),
is
(
0
));
assertThat
(
this
.
map
.
size
(),
is
(
0
));
this
.
map
.
put
(
123
,
"123"
);
this
.
map
.
put
(
123
,
"123"
);
this
.
map
.
put
(
123
,
null
);
this
.
map
.
put
(
123
,
null
);
...
@@ -330,7 +338,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -330,7 +338,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldSupportIsEmpty
()
throws
Exception
{
public
void
shouldSupportIsEmpty
()
{
assertThat
(
this
.
map
.
isEmpty
(),
is
(
true
));
assertThat
(
this
.
map
.
isEmpty
(),
is
(
true
));
this
.
map
.
put
(
123
,
"123"
);
this
.
map
.
put
(
123
,
"123"
);
this
.
map
.
put
(
123
,
null
);
this
.
map
.
put
(
123
,
null
);
...
@@ -339,7 +347,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -339,7 +347,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldContainKey
()
throws
Exception
{
public
void
shouldContainKey
()
{
assertThat
(
this
.
map
.
containsKey
(
123
),
is
(
false
));
assertThat
(
this
.
map
.
containsKey
(
123
),
is
(
false
));
assertThat
(
this
.
map
.
containsKey
(
456
),
is
(
false
));
assertThat
(
this
.
map
.
containsKey
(
456
),
is
(
false
));
this
.
map
.
put
(
123
,
"123"
);
this
.
map
.
put
(
123
,
"123"
);
...
@@ -349,7 +357,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -349,7 +357,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldContainValue
()
throws
Exception
{
public
void
shouldContainValue
()
{
assertThat
(
this
.
map
.
containsValue
(
"123"
),
is
(
false
));
assertThat
(
this
.
map
.
containsValue
(
"123"
),
is
(
false
));
assertThat
(
this
.
map
.
containsValue
(
null
),
is
(
false
));
assertThat
(
this
.
map
.
containsValue
(
null
),
is
(
false
));
this
.
map
.
put
(
123
,
"123"
);
this
.
map
.
put
(
123
,
"123"
);
...
@@ -359,7 +367,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -359,7 +367,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldRemoveWhenKeyIsInMap
()
throws
Exception
{
public
void
shouldRemoveWhenKeyIsInMap
()
{
this
.
map
.
put
(
123
,
null
);
this
.
map
.
put
(
123
,
null
);
this
.
map
.
put
(
456
,
"456"
);
this
.
map
.
put
(
456
,
"456"
);
this
.
map
.
put
(
null
,
"789"
);
this
.
map
.
put
(
null
,
"789"
);
...
@@ -370,14 +378,14 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -370,14 +378,14 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldRemoveWhenKeyIsNotInMap
()
throws
Exception
{
public
void
shouldRemoveWhenKeyIsNotInMap
()
{
assertThat
(
this
.
map
.
remove
(
123
),
is
(
nullValue
()));
assertThat
(
this
.
map
.
remove
(
123
),
is
(
nullValue
()));
assertThat
(
this
.
map
.
remove
(
null
),
is
(
nullValue
()));
assertThat
(
this
.
map
.
remove
(
null
),
is
(
nullValue
()));
assertThat
(
this
.
map
.
isEmpty
(),
is
(
true
));
assertThat
(
this
.
map
.
isEmpty
(),
is
(
true
));
}
}
@Test
@Test
public
void
shouldPutAll
()
throws
Exception
{
public
void
shouldPutAll
()
{
Map
<
Integer
,
String
>
m
=
new
HashMap
<>();
Map
<
Integer
,
String
>
m
=
new
HashMap
<>();
m
.
put
(
123
,
"123"
);
m
.
put
(
123
,
"123"
);
m
.
put
(
456
,
null
);
m
.
put
(
456
,
null
);
...
@@ -390,7 +398,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -390,7 +398,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldClear
()
throws
Exception
{
public
void
shouldClear
()
{
this
.
map
.
put
(
123
,
"123"
);
this
.
map
.
put
(
123
,
"123"
);
this
.
map
.
put
(
456
,
null
);
this
.
map
.
put
(
456
,
null
);
this
.
map
.
put
(
null
,
"789"
);
this
.
map
.
put
(
null
,
"789"
);
...
@@ -402,7 +410,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -402,7 +410,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldGetKeySet
()
throws
Exception
{
public
void
shouldGetKeySet
()
{
this
.
map
.
put
(
123
,
"123"
);
this
.
map
.
put
(
123
,
"123"
);
this
.
map
.
put
(
456
,
null
);
this
.
map
.
put
(
456
,
null
);
this
.
map
.
put
(
null
,
"789"
);
this
.
map
.
put
(
null
,
"789"
);
...
@@ -414,7 +422,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -414,7 +422,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldGetValues
()
throws
Exception
{
public
void
shouldGetValues
()
{
this
.
map
.
put
(
123
,
"123"
);
this
.
map
.
put
(
123
,
"123"
);
this
.
map
.
put
(
456
,
null
);
this
.
map
.
put
(
456
,
null
);
this
.
map
.
put
(
null
,
"789"
);
this
.
map
.
put
(
null
,
"789"
);
...
@@ -423,13 +431,13 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -423,13 +431,13 @@ public class ConcurrentReferenceHashMapTests {
expected
.
add
(
"123"
);
expected
.
add
(
"123"
);
expected
.
add
(
null
);
expected
.
add
(
null
);
expected
.
add
(
"789"
);
expected
.
add
(
"789"
);
Collections
.
sort
(
actual
,
NULL_SAFE_STRING_SORT
);
actual
.
sort
(
NULL_SAFE_STRING_SORT
);
Collections
.
sort
(
expected
,
NULL_SAFE_STRING_SORT
);
expected
.
sort
(
NULL_SAFE_STRING_SORT
);
assertThat
(
actual
,
is
(
expected
));
assertThat
(
actual
,
is
(
expected
));
}
}
@Test
@Test
public
void
shouldGetEntrySet
()
throws
Exception
{
public
void
shouldGetEntrySet
()
{
this
.
map
.
put
(
123
,
"123"
);
this
.
map
.
put
(
123
,
"123"
);
this
.
map
.
put
(
456
,
null
);
this
.
map
.
put
(
456
,
null
);
this
.
map
.
put
(
null
,
"789"
);
this
.
map
.
put
(
null
,
"789"
);
...
@@ -441,7 +449,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -441,7 +449,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldGetEntrySetFollowingNext
()
throws
Exception
{
public
void
shouldGetEntrySetFollowingNext
()
{
// Use loadFactor to disable resize
// Use loadFactor to disable resize
this
.
map
=
new
TestWeakConcurrentCache
<>(
1
,
10.0f
,
1
);
this
.
map
=
new
TestWeakConcurrentCache
<>(
1
,
10.0f
,
1
);
this
.
map
.
put
(
1
,
"1"
);
this
.
map
.
put
(
1
,
"1"
);
...
@@ -455,7 +463,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -455,7 +463,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldRemoveViaEntrySet
()
throws
Exception
{
public
void
shouldRemoveViaEntrySet
()
{
this
.
map
.
put
(
1
,
"1"
);
this
.
map
.
put
(
1
,
"1"
);
this
.
map
.
put
(
2
,
"2"
);
this
.
map
.
put
(
2
,
"2"
);
this
.
map
.
put
(
3
,
"3"
);
this
.
map
.
put
(
3
,
"3"
);
...
@@ -470,7 +478,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -470,7 +478,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
@Test
@Test
public
void
shouldSetViaEntrySet
()
throws
Exception
{
public
void
shouldSetViaEntrySet
()
{
this
.
map
.
put
(
1
,
"1"
);
this
.
map
.
put
(
1
,
"1"
);
this
.
map
.
put
(
2
,
"2"
);
this
.
map
.
put
(
2
,
"2"
);
this
.
map
.
put
(
3
,
"3"
);
this
.
map
.
put
(
3
,
"3"
);
...
@@ -485,36 +493,21 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -485,36 +493,21 @@ public class ConcurrentReferenceHashMapTests {
@Test
@Test
@Ignore
(
"Intended for use during development only"
)
@Ignore
(
"Intended for use during development only"
)
public
void
shouldBeFasterThanSynchronizedMap
()
throws
Exception
{
public
void
shouldBeFasterThanSynchronizedMap
()
throws
Interrupted
Exception
{
Map
<
Integer
,
WeakReference
<
String
>>
synchronizedMap
=
Collections
.
synchronizedMap
(
new
WeakHashMap
<
Integer
,
WeakReference
<
String
>>());
Map
<
Integer
,
WeakReference
<
String
>>
synchronizedMap
=
Collections
.
synchronizedMap
(
new
WeakHashMap
<
Integer
,
WeakReference
<
String
>>());
StopWatch
mapTime
=
timeMultiThreaded
(
"SynchronizedMap"
,
synchronizedMap
,
StopWatch
mapTime
=
timeMultiThreaded
(
"SynchronizedMap"
,
synchronizedMap
,
v
->
new
WeakReference
<>(
String
.
valueOf
(
v
)));
new
ValueFactory
<
WeakReference
<
String
>>()
{
@Override
public
WeakReference
<
String
>
newValue
(
int
v
)
{
return
new
WeakReference
<>(
String
.
valueOf
(
v
));
}
});
System
.
out
.
println
(
mapTime
.
prettyPrint
());
System
.
out
.
println
(
mapTime
.
prettyPrint
());
this
.
map
.
setDisableTestHooks
(
true
);
this
.
map
.
setDisableTestHooks
(
true
);
StopWatch
cacheTime
=
timeMultiThreaded
(
"WeakConcurrentCache"
,
this
.
map
,
StopWatch
cacheTime
=
timeMultiThreaded
(
"WeakConcurrentCache"
,
this
.
map
,
String:
:
valueOf
);
new
ValueFactory
<
String
>()
{
@Override
public
String
newValue
(
int
v
)
{
return
String
.
valueOf
(
v
);
}
});
System
.
out
.
println
(
cacheTime
.
prettyPrint
());
System
.
out
.
println
(
cacheTime
.
prettyPrint
());
// We should be at least 4 time faster
// We should be at least 4 time faster
assertThat
(
cacheTime
.
getTotalTimeSeconds
(),
assertThat
(
cacheTime
.
getTotalTimeSeconds
(),
is
(
lessThan
(
mapTime
.
getTotalTimeSeconds
()
/
4.0
)));
is
(
lessThan
(
mapTime
.
getTotalTimeSeconds
()
/
4.0
)));
}
}
@Test
@Test
public
void
shouldSupportNullReference
()
throws
Exception
{
public
void
shouldSupportNullReference
()
{
// GC could happen during restructure so we must be able to create a reference for a null entry
// GC could happen during restructure so we must be able to create a reference for a null entry
map
.
createReferenceManager
().
createReference
(
null
,
1234
,
null
);
map
.
createReferenceManager
().
createReference
(
null
,
1234
,
null
);
}
}
...
@@ -558,7 +551,7 @@ public class ConcurrentReferenceHashMapTests {
...
@@ -558,7 +551,7 @@ public class ConcurrentReferenceHashMapTests {
}
}
private
static
interface
ValueFactory
<
V
>
{
private
interface
ValueFactory
<
V
>
{
V
newValue
(
int
k
);
V
newValue
(
int
k
);
}
}
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录