Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
openanolis
dragonwell8_jdk
提交
d6fde63a
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看板
提交
d6fde63a
编写于
10月 11, 2013
作者:
A
alanb
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
8019526: (fs) Files.lines, etc without Charset parameter
Reviewed-by: psandoz, henryjen
上级
84621ebc
变更
3
隐藏空白更改
内联
并排
Showing
3 changed file
with
503 addition
and
260 deletion
+503
-260
src/share/classes/java/nio/file/Files.java
src/share/classes/java/nio/file/Files.java
+179
-17
test/java/nio/file/Files/BytesAndLines.java
test/java/nio/file/Files/BytesAndLines.java
+270
-214
test/java/nio/file/Files/StreamTest.java
test/java/nio/file/Files/StreamTest.java
+54
-29
未找到文件。
src/share/classes/java/nio/file/Files.java
浏览文件 @
d6fde63a
...
@@ -43,9 +43,10 @@ import java.nio.channels.SeekableByteChannel;
...
@@ -43,9 +43,10 @@ import java.nio.channels.SeekableByteChannel;
import
java.nio.charset.Charset
;
import
java.nio.charset.Charset
;
import
java.nio.charset.CharsetDecoder
;
import
java.nio.charset.CharsetDecoder
;
import
java.nio.charset.CharsetEncoder
;
import
java.nio.charset.CharsetEncoder
;
import
java.nio.charset.StandardCharsets
;
import
java.nio.file.attribute.BasicFileAttributeView
;
import
java.nio.file.attribute.BasicFileAttributeView
;
import
java.nio.file.attribute.BasicFileAttributes
;
import
java.nio.file.attribute.BasicFileAttributes
;
import
java.nio.file.attribute.DosFileAttributes
;
import
java.nio.file.attribute.DosFileAttributes
;
// javadoc
import
java.nio.file.attribute.FileAttribute
;
import
java.nio.file.attribute.FileAttribute
;
import
java.nio.file.attribute.FileAttributeView
;
import
java.nio.file.attribute.FileAttributeView
;
import
java.nio.file.attribute.FileOwnerAttributeView
;
import
java.nio.file.attribute.FileOwnerAttributeView
;
...
@@ -104,8 +105,7 @@ public final class Files {
...
@@ -104,8 +105,7 @@ public final class Files {
return
()
->
{
return
()
->
{
try
{
try
{
c
.
close
();
c
.
close
();
}
}
catch
(
IOException
e
)
{
catch
(
IOException
e
)
{
throw
new
UncheckedIOException
(
e
);
throw
new
UncheckedIOException
(
e
);
}
}
};
};
...
@@ -2550,7 +2550,7 @@ public final class Files {
...
@@ -2550,7 +2550,7 @@ public final class Files {
* checkExec} is invoked to check execute access to the file.
* checkExec} is invoked to check execute access to the file.
*/
*/
public
static
boolean
isExecutable
(
Path
path
)
{
public
static
boolean
isExecutable
(
Path
path
)
{
return
isAccessible
(
path
,
AccessMode
.
EXECUTE
);
return
isAccessible
(
path
,
AccessMode
.
EXECUTE
);
}
}
// -- Recursive operations --
// -- Recursive operations --
...
@@ -2782,6 +2782,37 @@ public final class Files {
...
@@ -2782,6 +2782,37 @@ public final class Files {
return
new
BufferedReader
(
reader
);
return
new
BufferedReader
(
reader
);
}
}
/**
* Opens a file for reading, returning a {@code BufferedReader} to read text
* from the file in an efficient manner. Bytes from the file are decoded into
* characters using the {@link StandardCharsets#UTF_8 UTF-8} {@link Charset
* charset}.
*
* <p> This method works as if invoking it were equivalent to evaluating the
* expression:
* <pre>{@code
* Files.newBufferedReader(path, StandardCharsets.UTF_8)
* }</pre>
*
* @param path
* the path to the file
*
* @return a new buffered reader, with default buffer size, to read text
* from the file
*
* @throws IOException
* if an I/O error occurs opening the file
* @throws SecurityException
* In the case of the default provider, and a security manager is
* installed, the {@link SecurityManager#checkRead(String) checkRead}
* method is invoked to check read access to the file.
*
* @since 1.8
*/
public
static
BufferedReader
newBufferedReader
(
Path
path
)
throws
IOException
{
return
newBufferedReader
(
path
,
StandardCharsets
.
UTF_8
);
}
/**
/**
* Opens or creates a file for writing, returning a {@code BufferedWriter}
* Opens or creates a file for writing, returning a {@code BufferedWriter}
* that may be used to write text to the file in an efficient manner.
* that may be used to write text to the file in an efficient manner.
...
@@ -2827,6 +2858,41 @@ public final class Files {
...
@@ -2827,6 +2858,41 @@ public final class Files {
return
new
BufferedWriter
(
writer
);
return
new
BufferedWriter
(
writer
);
}
}
/**
* Opens or creates a file for writing, returning a {@code BufferedWriter}
* to write text to the file in an efficient manner. The text is encoded
* into bytes for writing using the {@link StandardCharsets#UTF_8 UTF-8}
* {@link Charset charset}.
*
* <p> This method works as if invoking it were equivalent to evaluating the
* expression:
* <pre>{@code
* Files.newBufferedWriter(path, StandardCharsets.UTF_8, options)
* }</pre>
*
* @param path
* the path to the file
* @param options
* options specifying how the file is opened
*
* @return a new buffered writer, with default buffer size, to write text
* to the file
*
* @throws IOException
* if an I/O error occurs opening or creating the file
* @throws UnsupportedOperationException
* if an unsupported option is specified
* @throws SecurityException
* In the case of the default provider, and a security manager is
* installed, the {@link SecurityManager#checkWrite(String) checkWrite}
* method is invoked to check write access to the file.
*
* @since 1.8
*/
public
static
BufferedWriter
newBufferedWriter
(
Path
path
,
OpenOption
...
options
)
throws
IOException
{
return
newBufferedWriter
(
path
,
StandardCharsets
.
UTF_8
,
options
);
}
/**
/**
* Reads all bytes from an input stream and writes them to an output stream.
* Reads all bytes from an input stream and writes them to an output stream.
*/
*/
...
@@ -3025,9 +3091,7 @@ public final class Files {
...
@@ -3025,9 +3091,7 @@ public final class Files {
* @throws OutOfMemoryError
* @throws OutOfMemoryError
* if an array of the required size cannot be allocated
* if an array of the required size cannot be allocated
*/
*/
private
static
byte
[]
read
(
InputStream
source
,
int
initialSize
)
private
static
byte
[]
read
(
InputStream
source
,
int
initialSize
)
throws
IOException
{
throws
IOException
{
int
capacity
=
initialSize
;
int
capacity
=
initialSize
;
byte
[]
buf
=
new
byte
[
capacity
];
byte
[]
buf
=
new
byte
[
capacity
];
int
nread
=
0
;
int
nread
=
0
;
...
@@ -3131,9 +3195,7 @@ public final class Files {
...
@@ -3131,9 +3195,7 @@ public final class Files {
*
*
* @see #newBufferedReader
* @see #newBufferedReader
*/
*/
public
static
List
<
String
>
readAllLines
(
Path
path
,
Charset
cs
)
public
static
List
<
String
>
readAllLines
(
Path
path
,
Charset
cs
)
throws
IOException
{
throws
IOException
{
try
(
BufferedReader
reader
=
newBufferedReader
(
path
,
cs
))
{
try
(
BufferedReader
reader
=
newBufferedReader
(
path
,
cs
))
{
List
<
String
>
result
=
new
ArrayList
<>();
List
<
String
>
result
=
new
ArrayList
<>();
for
(;;)
{
for
(;;)
{
...
@@ -3146,6 +3208,37 @@ public final class Files {
...
@@ -3146,6 +3208,37 @@ public final class Files {
}
}
}
}
/**
* Read all lines from a file. Bytes from the file are decoded into characters
* using the {@link StandardCharsets#UTF_8 UTF-8} {@link Charset charset}.
*
* <p> This method works as if invoking it were equivalent to evaluating the
* expression:
* <pre>{@code
* Files.readAllLines(path, StandardCharsets.UTF_8)
* }</pre>
*
* @param path
* the path to the file
*
* @return the lines from the file as a {@code List}; whether the {@code
* List} is modifiable or not is implementation dependent and
* therefore not specified
*
* @throws IOException
* if an I/O error occurs reading from the file or a malformed or
* unmappable byte sequence is read
* @throws SecurityException
* In the case of the default provider, and a security manager is
* installed, the {@link SecurityManager#checkRead(String) checkRead}
* method is invoked to check read access to the file.
*
* @since 1.8
*/
public
static
List
<
String
>
readAllLines
(
Path
path
)
throws
IOException
{
return
readAllLines
(
path
,
StandardCharsets
.
UTF_8
);
}
/**
/**
* Writes bytes to a file. The {@code options} parameter specifies how the
* Writes bytes to a file. The {@code options} parameter specifies how the
* the file is created or opened. If no options are present then this method
* the file is created or opened. If no options are present then this method
...
@@ -3262,6 +3355,45 @@ public final class Files {
...
@@ -3262,6 +3355,45 @@ public final class Files {
return
path
;
return
path
;
}
}
/**
* Write lines of text to a file. Characters are encoded into bytes using
* the {@link StandardCharsets#UTF_8 UTF-8} {@link Charset charset}.
*
* <p> This method works as if invoking it were equivalent to evaluating the
* expression:
* <pre>{@code
* Files.write(path, lines, StandardCharsets.UTF_8, options);
* }</pre>
*
* @param path
* the path to the file
* @param lines
* an object to iterate over the char sequences
* @param options
* options specifying how the file is opened
*
* @return the path
*
* @throws IOException
* if an I/O error occurs writing to or creating the file, or the
* text cannot be encoded as {@code UTF-8}
* @throws UnsupportedOperationException
* if an unsupported option is specified
* @throws SecurityException
* In the case of the default provider, and a security manager is
* installed, the {@link SecurityManager#checkWrite(String) checkWrite}
* method is invoked to check write access to the file.
*
* @since 1.8
*/
public
static
Path
write
(
Path
path
,
Iterable
<?
extends
CharSequence
>
lines
,
OpenOption
...
options
)
throws
IOException
{
return
write
(
path
,
lines
,
StandardCharsets
.
UTF_8
,
options
);
}
// -- Stream APIs --
// -- Stream APIs --
/**
/**
...
@@ -3431,9 +3563,11 @@ public final class Files {
...
@@ -3431,9 +3563,11 @@ public final class Files {
* if an I/O error is thrown when accessing the starting file.
* if an I/O error is thrown when accessing the starting file.
* @since 1.8
* @since 1.8
*/
*/
public
static
Stream
<
Path
>
walk
(
Path
start
,
int
maxDepth
,
public
static
Stream
<
Path
>
walk
(
Path
start
,
int
maxDepth
,
FileVisitOption
...
options
)
FileVisitOption
...
options
)
throws
IOException
{
throws
IOException
{
FileTreeIterator
iterator
=
new
FileTreeIterator
(
start
,
maxDepth
,
options
);
FileTreeIterator
iterator
=
new
FileTreeIterator
(
start
,
maxDepth
,
options
);
try
{
try
{
return
StreamSupport
.
stream
(
Spliterators
.
spliteratorUnknownSize
(
iterator
,
Spliterator
.
DISTINCT
),
false
)
return
StreamSupport
.
stream
(
Spliterators
.
spliteratorUnknownSize
(
iterator
,
Spliterator
.
DISTINCT
),
false
)
...
@@ -3484,9 +3618,7 @@ public final class Files {
...
@@ -3484,9 +3618,7 @@ public final class Files {
* @see #walk(Path, int, FileVisitOption...)
* @see #walk(Path, int, FileVisitOption...)
* @since 1.8
* @since 1.8
*/
*/
public
static
Stream
<
Path
>
walk
(
Path
start
,
public
static
Stream
<
Path
>
walk
(
Path
start
,
FileVisitOption
...
options
)
throws
IOException
{
FileVisitOption
...
options
)
throws
IOException
{
return
walk
(
start
,
Integer
.
MAX_VALUE
,
options
);
return
walk
(
start
,
Integer
.
MAX_VALUE
,
options
);
}
}
...
@@ -3547,7 +3679,8 @@ public final class Files {
...
@@ -3547,7 +3679,8 @@ public final class Files {
int
maxDepth
,
int
maxDepth
,
BiPredicate
<
Path
,
BasicFileAttributes
>
matcher
,
BiPredicate
<
Path
,
BasicFileAttributes
>
matcher
,
FileVisitOption
...
options
)
FileVisitOption
...
options
)
throws
IOException
{
throws
IOException
{
FileTreeIterator
iterator
=
new
FileTreeIterator
(
start
,
maxDepth
,
options
);
FileTreeIterator
iterator
=
new
FileTreeIterator
(
start
,
maxDepth
,
options
);
try
{
try
{
return
StreamSupport
.
stream
(
Spliterators
.
spliteratorUnknownSize
(
iterator
,
Spliterator
.
DISTINCT
),
false
)
return
StreamSupport
.
stream
(
Spliterators
.
spliteratorUnknownSize
(
iterator
,
Spliterator
.
DISTINCT
),
false
)
...
@@ -3561,7 +3694,7 @@ public final class Files {
...
@@ -3561,7 +3694,7 @@ public final class Files {
}
}
/**
/**
* Read all lines from a file as a {@code Stream}.
Unlike {@link
* Read all lines from a file as a {@code Stream}. Unlike {@link
* #readAllLines(Path, Charset) readAllLines}, this method does not read
* #readAllLines(Path, Charset) readAllLines}, this method does not read
* all lines into a {@code List}, but instead populates lazily as the stream
* all lines into a {@code List}, but instead populates lazily as the stream
* is consumed.
* is consumed.
...
@@ -3619,4 +3752,33 @@ public final class Files {
...
@@ -3619,4 +3752,33 @@ public final class Files {
throw
e
;
throw
e
;
}
}
}
}
/**
* Read all lines from a file as a {@code Stream}. Bytes from the file are
* decoded into characters using the {@link StandardCharsets#UTF_8 UTF-8}
* {@link Charset charset}.
*
* <p> This method works as if invoking it were equivalent to evaluating the
* expression:
* <pre>{@code
* Files.lines(path, StandardCharsets.UTF_8)
* }</pre>
*
* @param path
* the path to the file
*
* @return the lines from the file as a {@code Stream}
*
* @throws IOException
* if an I/O error occurs opening the file
* @throws SecurityException
* In the case of the default provider, and a security manager is
* installed, the {@link SecurityManager#checkRead(String) checkRead}
* method is invoked to check read access to the file.
*
* @since 1.8
*/
public
static
Stream
<
String
>
lines
(
Path
path
)
throws
IOException
{
return
lines
(
path
,
StandardCharsets
.
UTF_8
);
}
}
}
test/java/nio/file/Files/BytesAndLines.java
浏览文件 @
d6fde63a
/*
/*
* Copyright (c) 2011, Oracle and/or its affiliates. All rights reserved.
* Copyright (c) 2011,
2013,
Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
*
* This code is free software; you can redistribute it and/or modify it
* This code is free software; you can redistribute it and/or modify it
...
@@ -22,262 +22,318 @@
...
@@ -22,262 +22,318 @@
*/
*/
/* @test
/* @test
* @bug 7006126 8020669 8024788
* @bug 7006126 8020669 8024788
8019526
* @build BytesAndLines PassThroughFileSystem
* @build BytesAndLines PassThroughFileSystem
* @run
main
BytesAndLines
* @run
testng
BytesAndLines
* @summary Unit test for methods for Files readAllBytes, readAllLines and
* @summary Unit test for methods for Files readAllBytes, readAllLines and
* and write methods.
* and write methods.
*/
*/
import
java.nio.file.*
;
import
java.nio.ByteBuffer
;
import
static
java
.
nio
.
file
.
Files
.*;
import
java.nio.CharBuffer
;
import
java.io.*
;
import
java.nio.file.Files
;
import
java.util.*
;
import
java.nio.file.Path
;
import
java.nio.charset.*
;
import
java.nio.file.Paths
;
import
java.nio.file.OpenOption
;
import
static
java
.
nio
.
file
.
StandardOpenOption
.*;
import
java.nio.charset.Charset
;
import
java.nio.charset.CharacterCodingException
;
import
java.nio.charset.MalformedInputException
;
import
java.nio.charset.UnmappableCharacterException
;
import
static
java
.
nio
.
charset
.
StandardCharsets
.*;
import
java.util.Arrays
;
import
java.util.ArrayList
;
import
java.util.Collections
;
import
java.util.List
;
import
java.util.Random
;
import
java.util.concurrent.Callable
;
import
java.io.IOException
;
import
org.testng.annotations.AfterClass
;
import
org.testng.annotations.BeforeClass
;
import
org.testng.annotations.Test
;
import
static
org
.
testng
.
Assert
.*;
@Test
(
groups
=
"unit"
)
public
class
BytesAndLines
{
public
class
BytesAndLines
{
static
final
Random
rand
=
new
Random
();
static
final
Charset
US_ASCII
=
Charset
.
forName
(
"US-ASCII"
);
// data for text files
private
static
final
String
EN_STRING
=
"The quick brown fox jumps over the lazy dog"
;
private
static
final
String
JA_STRING
=
"\u65e5\u672c\u8a9e\u6587\u5b57\u5217"
;
// used for random byte content
private
static
Random
RAND
=
new
Random
();
// file used by most tests
private
Path
tmpfile
;
@BeforeClass
void
setup
()
throws
IOException
{
tmpfile
=
Files
.
createTempFile
(
"blah"
,
null
);
}
public
static
void
main
(
String
[]
args
)
throws
IOException
{
@AfterClass
testReadAndWriteBytes
();
void
cleanup
()
throws
IOException
{
testReadLines
();
Files
.
deleteIfExists
(
tmpfile
);
testWriteLines
();
}
}
/**
/**
*
Test readAllBytes(Path) and write(Path, byte[], OpenOption...)
*
Returns a byte[] of the given size with random content
*/
*/
static
void
testReadAndWriteBytes
()
throws
IOException
{
private
byte
[]
genBytes
(
int
size
)
{
// exercise methods with various sizes
byte
[]
arr
=
new
byte
[
size
];
testReadAndWriteBytes
(
0
);
RAND
.
nextBytes
(
arr
);
for
(
int
i
=
0
;
i
<
100
;
i
++)
{
return
arr
;
testReadAndWriteBytes
(
rand
.
nextInt
(
32000
));
}
}
// NullPointerException
/**
* Exercise NullPointerException
*/
public
void
testNulls
()
{
Path
file
=
Paths
.
get
(
"foo"
);
Path
file
=
Paths
.
get
(
"foo"
);
byte
[]
bytes
=
new
byte
[
100
];
List
<
String
>
lines
=
Collections
.
emptyList
();
List
<
String
>
lines
=
Collections
.
emptyList
();
checkNullPointerException
(()
->
Files
.
readAllBytes
(
null
));
checkNullPointerException
(()
->
Files
.
write
(
null
,
bytes
));
checkNullPointerException
(()
->
Files
.
write
(
file
,
(
byte
[])
null
));
checkNullPointerException
(()
->
Files
.
write
(
file
,
bytes
,
(
OpenOption
[])
null
));
checkNullPointerException
(()
->
Files
.
write
(
file
,
bytes
,
new
OpenOption
[]
{
null
}
));
checkNullPointerException
(()
->
Files
.
readAllLines
(
null
));
checkNullPointerException
(()
->
Files
.
readAllLines
(
file
,
(
Charset
)
null
));
checkNullPointerException
(()
->
Files
.
readAllLines
(
null
,
Charset
.
defaultCharset
()));
checkNullPointerException
(()
->
Files
.
write
(
null
,
lines
));
checkNullPointerException
(()
->
Files
.
write
(
file
,
(
List
<
String
>)
null
));
checkNullPointerException
(()
->
Files
.
write
(
file
,
lines
,
(
OpenOption
[])
null
));
checkNullPointerException
(()
->
Files
.
write
(
file
,
lines
,
new
OpenOption
[]
{
null
}
));
checkNullPointerException
(()
->
Files
.
write
(
null
,
lines
,
Charset
.
defaultCharset
()));
checkNullPointerException
(()
->
Files
.
write
(
file
,
null
,
Charset
.
defaultCharset
()));
checkNullPointerException
(()
->
Files
.
write
(
file
,
lines
,
(
Charset
)
null
));
checkNullPointerException
(()
->
Files
.
write
(
file
,
lines
,
Charset
.
defaultCharset
(),
(
OpenOption
[])
null
));
checkNullPointerException
(()
->
Files
.
write
(
file
,
lines
,
Charset
.
defaultCharset
(),
new
OpenOption
[]
{
null
}
));
}
private
void
checkNullPointerException
(
Callable
<?>
c
)
{
try
{
try
{
readAllBytes
(
null
);
c
.
call
();
throw
new
RuntimeException
(
"NullPointerException expected"
);
fail
(
"NullPointerException expected"
);
}
catch
(
NullPointerException
ignore
)
{
}
}
catch
(
NullPointerException
ignore
)
{
try
{
}
catch
(
Exception
e
)
{
write
(
null
,
lines
,
Charset
.
defaultCharset
());
fail
(
e
+
" not expected"
);
throw
new
RuntimeException
(
"NullPointerException expected"
);
}
}
catch
(
NullPointerException
ignore
)
{
}
}
try
{
write
(
file
,
null
,
Charset
.
defaultCharset
());
/**
throw
new
RuntimeException
(
"NullPointerException expected"
);
* Exercise Files.readAllBytes(Path) on varied file sizes
}
catch
(
NullPointerException
ignore
)
{
}
*/
try
{
public
void
testReadAllBytes
()
throws
IOException
{
write
(
file
,
lines
,
null
);
int
size
=
0
;
throw
new
RuntimeException
(
"NullPointerException expected"
);
while
(
size
<=
16
*
1024
)
{
}
catch
(
NullPointerException
ignore
)
{
}
testReadAllBytes
(
size
);
try
{
size
+=
512
;
write
(
file
,
lines
,
Charset
.
defaultCharset
(),
(
OpenOption
[])
null
);
}
throw
new
RuntimeException
(
"NullPointerException expected"
);
}
}
catch
(
NullPointerException
ignore
)
{
}
try
{
OpenOption
[]
opts
=
{
null
};
write
(
file
,
lines
,
Charset
.
defaultCharset
(),
opts
);
throw
new
RuntimeException
(
"NullPointerException expected"
);
}
catch
(
NullPointerException
ignore
)
{
}
private
void
testReadAllBytes
(
int
size
)
throws
IOException
{
// write bytes to file (random content)
byte
[]
expected
=
genBytes
(
size
);
Files
.
write
(
tmpfile
,
expected
);
// check expected bytes are read
byte
[]
read
=
Files
.
readAllBytes
(
tmpfile
);
assertTrue
(
Arrays
.
equals
(
read
,
expected
),
"Bytes read not the same as written"
);
}
/**
* Linux specific test to exercise Files.readAllBytes on /proc. This is
* special because file sizes are reported as 0 even though the file
* has content.
*/
public
void
testReadAllBytesOnProcFS
()
throws
IOException
{
// read from procfs
// read from procfs
if
(
System
.
getProperty
(
"os.name"
).
equals
(
"Linux"
))
{
if
(
System
.
getProperty
(
"os.name"
).
equals
(
"Linux"
))
{
// Refer to the Linux proc(5) man page for details about /proc/self/stat file
Path
statFile
=
Paths
.
get
(
"/proc/self/stat"
);
// procfs reports it to be zero sized, even though data can be read from it
byte
[]
data
=
Files
.
readAllBytes
(
statFile
);
String
statFile
=
"/proc/self/stat"
;
Path
pathStat
=
Paths
.
get
(
statFile
);
byte
[]
data
=
Files
.
readAllBytes
(
pathStat
);
assertTrue
(
data
.
length
>
0
,
"Files.readAllBytes('"
+
statFile
+
"') failed to read"
);
assertTrue
(
data
.
length
>
0
,
"Files.readAllBytes('"
+
statFile
+
"') failed to read"
);
}
}
// test readAllBytes on custom file system
Path
myfile
=
PassThroughFileSystem
.
create
().
getPath
(
file
.
toString
());
for
(
int
size
=
0
;
size
<=
1024
;
size
+=
512
)
{
byte
[]
b1
=
new
byte
[
size
];
rand
.
nextBytes
(
b1
);
Files
.
write
(
myfile
,
b1
);
byte
[]
b2
=
Files
.
readAllBytes
(
myfile
);
assertTrue
(
Arrays
.
equals
(
b1
,
b2
),
"bytes not equal"
);
}
}
}
/**
static
void
testReadAndWriteBytes
(
int
size
)
throws
IOException
{
* Exercise Files.readAllBytes(Path) on custom file system. This is special
Path
path
=
createTempFile
(
"blah"
,
null
);
* because readAllBytes was originally implemented to use FileChannel
* and so may not be supported by custom file system providers.
*/
public
void
testReadAllBytesOnCustomFS
()
throws
IOException
{
Path
myfile
=
PassThroughFileSystem
.
create
().
getPath
(
"myfile"
);
try
{
try
{
boolean
append
=
rand
.
nextBoolean
();
int
size
=
0
;
while
(
size
<=
1024
)
{
byte
[]
b1
=
new
byte
[
size
];
byte
[]
b1
=
genBytes
(
size
);
rand
.
nextBytes
(
b1
);
Files
.
write
(
myfile
,
b1
);
byte
[]
b2
=
Files
.
readAllBytes
(
myfile
);
byte
[]
b2
=
(
append
)
?
new
byte
[
size
]
:
new
byte
[
0
];
assertTrue
(
Arrays
.
equals
(
b1
,
b2
),
"bytes not equal"
);
rand
.
nextBytes
(
b2
);
size
+=
512
;
// write method should create file if it doesn't exist
if
(
rand
.
nextBoolean
())
delete
(
path
);
// write bytes to file
Path
target
=
write
(
path
,
b1
);
assertTrue
(
target
==
path
,
"Unexpected path"
);
assertTrue
(
size
(
path
)
==
b1
.
length
,
"Unexpected file size"
);
// append bytes to file (might be 0 bytes)
write
(
path
,
b2
,
StandardOpenOption
.
APPEND
);
assertTrue
(
size
(
path
)
==
b1
.
length
+
b2
.
length
,
"Unexpected file size"
);
// read entire file
byte
[]
read
=
readAllBytes
(
path
);
// check bytes are correct
byte
[]
expected
;
if
(
append
)
{
expected
=
new
byte
[
b1
.
length
+
b2
.
length
];
System
.
arraycopy
(
b1
,
0
,
expected
,
0
,
b1
.
length
);
System
.
arraycopy
(
b2
,
0
,
expected
,
b1
.
length
,
b2
.
length
);
}
else
{
expected
=
b1
;
}
}
assertTrue
(
Arrays
.
equals
(
read
,
expected
),
"Bytes read not the same as bytes written"
);
}
finally
{
}
finally
{
deleteIfExists
(
path
);
Files
.
deleteIfExists
(
myfile
);
}
}
}
}
/**
/**
*
Test readAllLines(Path,Charset)
*
Exercise Files.write(Path, byte[], OpenOption...) on various sizes
*/
*/
static
void
testReadLines
()
throws
IOException
{
public
void
testWriteBytes
()
throws
IOException
{
Path
tmpfile
=
createTempFile
(
"blah"
,
"txt"
);
int
size
=
0
;
try
{
while
(
size
<
16
*
1024
)
{
List
<
String
>
lines
;
testWriteBytes
(
size
,
false
);
testWriteBytes
(
size
,
true
);
size
+=
512
;
}
}
private
void
testWriteBytes
(
int
size
,
boolean
append
)
throws
IOException
{
byte
[]
bytes
=
genBytes
(
size
);
Path
result
=
Files
.
write
(
tmpfile
,
bytes
);
assertTrue
(
result
==
tmpfile
);
if
(
append
)
{
Files
.
write
(
tmpfile
,
bytes
,
APPEND
);
assertTrue
(
Files
.
size
(
tmpfile
)
==
size
*
2
);
}
byte
[]
expected
;
if
(
append
)
{
expected
=
new
byte
[
size
<<
1
];
System
.
arraycopy
(
bytes
,
0
,
expected
,
0
,
bytes
.
length
);
System
.
arraycopy
(
bytes
,
0
,
expected
,
bytes
.
length
,
bytes
.
length
);
}
else
{
expected
=
bytes
;
}
byte
[]
read
=
Files
.
readAllBytes
(
tmpfile
);
assertTrue
(
Arrays
.
equals
(
read
,
expected
),
"Bytes read not the same as written"
);
}
// zero lines
/**
assertTrue
(
size
(
tmpfile
)
==
0
,
"File should be empty"
);
* Exercise Files.readAllLines(Path, Charset)
lines
=
readAllLines
(
tmpfile
,
US_ASCII
);
*/
public
void
testReadAllLines
()
throws
IOException
{
// zero lines
Files
.
write
(
tmpfile
,
new
byte
[
0
]);
List
<
String
>
lines
=
Files
.
readAllLines
(
tmpfile
,
US_ASCII
);
assertTrue
(
lines
.
isEmpty
(),
"No line expected"
);
assertTrue
(
lines
.
isEmpty
(),
"No line expected"
);
// one line
// one line
byte
[]
hi
=
{
(
byte
)
'h'
,
(
byte
)
'i'
};
byte
[]
hi
=
{
(
byte
)
'h'
,
(
byte
)
'i'
};
write
(
tmpfile
,
hi
);
Files
.
write
(
tmpfile
,
hi
);
lines
=
readAllLines
(
tmpfile
,
US_ASCII
);
lines
=
Files
.
readAllLines
(
tmpfile
,
US_ASCII
);
assertTrue
(
lines
.
size
()
==
1
,
"One line expected"
);
assertTrue
(
lines
.
size
()
==
1
,
"One line expected"
);
assertTrue
(
lines
.
get
(
0
).
equals
(
"hi"
),
"'Hi' expected"
);
assertTrue
(
lines
.
get
(
0
).
equals
(
"hi"
),
"'Hi' expected"
);
// two lines using platform's line separator
// two lines using platform's line separator
List
<
String
>
expected
=
Arrays
.
asList
(
"hi"
,
"there"
);
List
<
String
>
expected
=
Arrays
.
asList
(
"hi"
,
"there"
);
write
(
tmpfile
,
expected
,
US_ASCII
);
Files
.
write
(
tmpfile
,
expected
,
US_ASCII
);
assertTrue
(
size
(
tmpfile
)
>
0
,
"File is empty"
);
assertTrue
(
Files
.
size
(
tmpfile
)
>
0
,
"File is empty"
);
lines
=
readAllLines
(
tmpfile
,
US_ASCII
);
lines
=
Files
.
readAllLines
(
tmpfile
,
US_ASCII
);
assertTrue
(
lines
.
equals
(
expected
),
"Unexpected lines"
);
assertTrue
(
lines
.
equals
(
expected
),
"Unexpected lines"
);
// MalformedInputException
// MalformedInputException
byte
[]
bad
=
{
(
byte
)
0xff
,
(
byte
)
0xff
};
byte
[]
bad
=
{
(
byte
)
0xff
,
(
byte
)
0xff
};
write
(
tmpfile
,
bad
);
Files
.
write
(
tmpfile
,
bad
);
try
{
try
{
readAllLines
(
tmpfile
,
US_ASCII
);
Files
.
readAllLines
(
tmpfile
,
US_ASCII
);
throw
new
RuntimeException
(
"MalformedInputException expected"
);
fail
(
"MalformedInputException expected"
);
}
catch
(
MalformedInputException
ignore
)
{
}
}
catch
(
MalformedInputException
ignore
)
{
}
}
// NullPointerException
try
{
readAllLines
(
null
,
US_ASCII
);
throw
new
RuntimeException
(
"NullPointerException expected"
);
}
catch
(
NullPointerException
ignore
)
{
}
try
{
readAllLines
(
tmpfile
,
null
);
throw
new
RuntimeException
(
"NullPointerException expected"
);
}
catch
(
NullPointerException
ignore
)
{
}
// read from procfs
if
(
System
.
getProperty
(
"os.name"
).
equals
(
"Linux"
))
{
// Refer to the Linux proc(5) man page for details about /proc/self/status file
// procfs reports this file to be zero sized, even though data can be read from it
String
statusFile
=
"/proc/self/status"
;
Path
pathStatus
=
Paths
.
get
(
statusFile
);
lines
=
Files
.
readAllLines
(
pathStatus
,
US_ASCII
);
assertTrue
(
lines
.
size
()
>
0
,
"Files.readAllLines('"
+
pathStatus
+
"') failed to read"
);
}
}
finally
{
/**
delete
(
tmpfile
);
* Linux specific test to exercise Files.readAllLines(Path) on /proc. This
* is special because file sizes are reported as 0 even though the file
* has content.
*/
public
void
testReadAllLinesOnProcFS
()
throws
IOException
{
if
(
System
.
getProperty
(
"os.name"
).
equals
(
"Linux"
))
{
Path
statFile
=
Paths
.
get
(
"/proc/self/stat"
);
List
<
String
>
lines
=
Files
.
readAllLines
(
statFile
);
assertTrue
(
lines
.
size
()
>
0
,
"Files.readAllLines('"
+
statFile
+
"') failed to read"
);
}
}
}
}
/**
/**
*
Test write(Path,Iterable<? extends CharSequence>,Charset,OpenOption...
)
*
Exercise Files.readAllLines(Path
)
*/
*/
static
void
testWriteLines
()
throws
IOException
{
public
void
testReadAllLinesUTF8
()
throws
IOException
{
Path
tmpfile
=
createTempFile
(
"blah"
,
"txt"
);
Files
.
write
(
tmpfile
,
encodeAsUTF8
(
EN_STRING
+
"\n"
+
JA_STRING
));
List
<
String
>
lines
=
Files
.
readAllLines
(
tmpfile
);
assertTrue
(
lines
.
size
()
==
2
,
"Read "
+
lines
.
size
()
+
" lines instead of 2"
);
assertTrue
(
lines
.
get
(
0
).
equals
(
EN_STRING
));
assertTrue
(
lines
.
get
(
1
).
equals
(
JA_STRING
));
// a sample of malformed sequences
testReadAllLinesMalformedUTF8
((
byte
)
0xFF
);
// one-byte sequence
testReadAllLinesMalformedUTF8
((
byte
)
0xC0
,
(
byte
)
0x80
);
// invalid first byte
testReadAllLinesMalformedUTF8
((
byte
)
0xC2
,
(
byte
)
0x00
);
// invalid second byte
}
private
byte
[]
encodeAsUTF8
(
String
s
)
throws
CharacterCodingException
{
// not using s.getBytes here so as to catch unmappable characters
ByteBuffer
bb
=
UTF_8
.
newEncoder
().
encode
(
CharBuffer
.
wrap
(
s
));
byte
[]
result
=
new
byte
[
bb
.
limit
()];
bb
.
get
(
result
);
assertTrue
(
bb
.
remaining
()
==
0
);
return
result
;
}
private
void
testReadAllLinesMalformedUTF8
(
byte
...
bytes
)
throws
IOException
{
Files
.
write
(
tmpfile
,
bytes
);
try
{
try
{
// write method should create file if it doesn't exist
Files
.
readAllLines
(
tmpfile
);
if
(
rand
.
nextBoolean
())
fail
(
"MalformedInputException expected"
);
delete
(
tmpfile
);
}
catch
(
MalformedInputException
ignore
)
{
}
}
// zero lines
Path
result
=
write
(
tmpfile
,
Collections
.<
String
>
emptyList
(),
US_ASCII
);
assert
(
size
(
tmpfile
)
==
0
);
assert
(
result
==
tmpfile
);
// two lines
List
<
String
>
lines
=
Arrays
.
asList
(
"hi"
,
"there"
);
write
(
tmpfile
,
lines
,
US_ASCII
);
List
<
String
>
actual
=
readAllLines
(
tmpfile
,
US_ASCII
);
assertTrue
(
actual
.
equals
(
lines
),
"Unexpected lines"
);
// append two lines
write
(
tmpfile
,
lines
,
US_ASCII
,
StandardOpenOption
.
APPEND
);
List
<
String
>
expected
=
new
ArrayList
<
String
>();
expected
.
addAll
(
lines
);
expected
.
addAll
(
lines
);
assertTrue
(
expected
.
size
()
==
4
,
"List should have 4 elements"
);
actual
=
readAllLines
(
tmpfile
,
US_ASCII
);
assertTrue
(
actual
.
equals
(
expected
),
"Unexpected lines"
);
// UnmappableCharacterException
try
{
String
s
=
"\u00A0\u00A1"
;
write
(
tmpfile
,
Arrays
.
asList
(
s
),
US_ASCII
);
throw
new
RuntimeException
(
"UnmappableCharacterException expected"
);
}
catch
(
UnmappableCharacterException
ignore
)
{
}
// NullPointerException
try
{
write
(
null
,
lines
,
US_ASCII
);
throw
new
RuntimeException
(
"NullPointerException expected"
);
}
catch
(
NullPointerException
ignore
)
{
}
try
{
write
(
tmpfile
,
null
,
US_ASCII
);
throw
new
RuntimeException
(
"NullPointerException expected"
);
}
catch
(
NullPointerException
ignore
)
{
}
try
{
write
(
tmpfile
,
lines
,
null
);
throw
new
RuntimeException
(
"NullPointerException expected"
);
}
catch
(
NullPointerException
ignore
)
{
}
try
{
write
(
tmpfile
,
lines
,
US_ASCII
,
(
OpenOption
[])
null
);
throw
new
RuntimeException
(
"NullPointerException expected"
);
}
catch
(
NullPointerException
ignore
)
{
}
try
{
OpenOption
[]
opts
=
{
(
OpenOption
)
null
};
write
(
tmpfile
,
lines
,
US_ASCII
,
opts
);
throw
new
RuntimeException
(
"NullPointerException expected"
);
}
catch
(
NullPointerException
ignore
)
{
}
}
finally
{
/**
delete
(
tmpfile
);
* Exercise Files.write(Path, Iterable<? extends CharSequence>, Charset, OpenOption...)
}
*/
public
void
testWriteLines
()
throws
IOException
{
// zero lines
Path
result
=
Files
.
write
(
tmpfile
,
Collections
.<
String
>
emptyList
(),
US_ASCII
);
assert
(
Files
.
size
(
tmpfile
)
==
0
);
assert
(
result
==
tmpfile
);
// two lines
List
<
String
>
lines
=
Arrays
.
asList
(
"hi"
,
"there"
);
Files
.
write
(
tmpfile
,
lines
,
US_ASCII
);
List
<
String
>
actual
=
Files
.
readAllLines
(
tmpfile
,
US_ASCII
);
assertTrue
(
actual
.
equals
(
lines
),
"Unexpected lines"
);
// append two lines
Files
.
write
(
tmpfile
,
lines
,
US_ASCII
,
APPEND
);
List
<
String
>
expected
=
new
ArrayList
<>();
expected
.
addAll
(
lines
);
expected
.
addAll
(
lines
);
assertTrue
(
expected
.
size
()
==
4
,
"List should have 4 elements"
);
actual
=
Files
.
readAllLines
(
tmpfile
,
US_ASCII
);
assertTrue
(
actual
.
equals
(
expected
),
"Unexpected lines"
);
// UnmappableCharacterException
try
{
String
s
=
"\u00A0\u00A1"
;
Files
.
write
(
tmpfile
,
Arrays
.
asList
(
s
),
US_ASCII
);
fail
(
"UnmappableCharacterException expected"
);
}
catch
(
UnmappableCharacterException
ignore
)
{
}
}
}
static
void
assertTrue
(
boolean
expr
,
String
errmsg
)
{
/**
if
(!
expr
)
* Exercise Files.write(Path, Iterable<? extends CharSequence>, OpenOption...)
throw
new
RuntimeException
(
errmsg
);
*/
public
void
testWriteLinesUTF8
()
throws
IOException
{
List
<
String
>
lines
=
Arrays
.
asList
(
EN_STRING
,
JA_STRING
);
Files
.
write
(
tmpfile
,
lines
);
List
<
String
>
actual
=
Files
.
readAllLines
(
tmpfile
,
UTF_8
);
assertTrue
(
actual
.
equals
(
lines
),
"Unexpected lines"
);
}
}
}
}
test/java/nio/file/Files/StreamTest.java
浏览文件 @
d6fde63a
...
@@ -22,11 +22,10 @@
...
@@ -22,11 +22,10 @@
*/
*/
/* @test
/* @test
* @bug 8006884
* @bug 8006884 8019526
* @summary Unit test for java.nio.file.Files
* @library ..
* @build PassThroughFileSystem FaultyFileSystem
* @build PassThroughFileSystem FaultyFileSystem
* @run testng StreamTest
* @run testng StreamTest
* @summary Unit test for java.nio.file.Files methods that return a Stream
*/
*/
import
java.io.IOException
;
import
java.io.IOException
;
...
@@ -43,11 +42,13 @@ import java.nio.file.Path;
...
@@ -43,11 +42,13 @@ import java.nio.file.Path;
import
java.nio.file.Paths
;
import
java.nio.file.Paths
;
import
java.nio.file.attribute.BasicFileAttributes
;
import
java.nio.file.attribute.BasicFileAttributes
;
import
java.util.Arrays
;
import
java.util.Arrays
;
import
java.util.Collections
;
import
java.util.Iterator
;
import
java.util.Iterator
;
import
java.util.List
;
import
java.util.List
;
import
java.util.Objects
;
import
java.util.Objects
;
import
java.util.Set
;
import
java.util.Set
;
import
java.util.TreeSet
;
import
java.util.TreeSet
;
import
java.util.concurrent.Callable
;
import
java.util.function.BiPredicate
;
import
java.util.function.BiPredicate
;
import
java.util.stream.Stream
;
import
java.util.stream.Stream
;
import
java.util.stream.Collectors
;
import
java.util.stream.Collectors
;
...
@@ -316,56 +317,80 @@ public class StreamTest {
...
@@ -316,56 +317,80 @@ public class StreamTest {
try
{
try
{
// zero lines
// zero lines
assertTrue
(
Files
.
size
(
tmpfile
)
==
0
,
"File should be empty"
);
assertTrue
(
Files
.
size
(
tmpfile
)
==
0
,
"File should be empty"
);
try
(
Stream
<
String
>
s
=
Files
.
lines
(
tmpfile
))
{
checkLines
(
s
,
Collections
.
emptyList
());
}
try
(
Stream
<
String
>
s
=
Files
.
lines
(
tmpfile
,
US_ASCII
))
{
try
(
Stream
<
String
>
s
=
Files
.
lines
(
tmpfile
,
US_ASCII
))
{
assertEquals
(
s
.
mapToInt
(
l
->
1
).
reduce
(
0
,
Integer:
:
sum
),
0
,
"No line expected"
);
checkLines
(
s
,
Collections
.
emptyList
()
);
}
}
// one line
// one line
byte
[]
hi
=
{
(
byte
)
'h'
,
(
byte
)
'i'
};
List
<
String
>
oneLine
=
Arrays
.
asList
(
"hi"
);
Files
.
write
(
tmpfile
,
hi
);
Files
.
write
(
tmpfile
,
oneLine
,
US_ASCII
);
try
(
Stream
<
String
>
s
=
Files
.
lines
(
tmpfile
))
{
checkLines
(
s
,
oneLine
);
}
try
(
Stream
<
String
>
s
=
Files
.
lines
(
tmpfile
,
US_ASCII
))
{
try
(
Stream
<
String
>
s
=
Files
.
lines
(
tmpfile
,
US_ASCII
))
{
List
<
String
>
lines
=
s
.
collect
(
Collectors
.
toList
());
checkLines
(
s
,
oneLine
);
assertTrue
(
lines
.
size
()
==
1
,
"One line expected"
);
assertTrue
(
lines
.
get
(
0
).
equals
(
"hi"
),
"'Hi' expected"
);
}
}
// two lines using platform's line separator
// two lines using platform's line separator
List
<
String
>
expected
=
Arrays
.
asList
(
"hi"
,
"there"
);
List
<
String
>
twoLines
=
Arrays
.
asList
(
"hi"
,
"there"
);
Files
.
write
(
tmpfile
,
expected
,
US_ASCII
);
Files
.
write
(
tmpfile
,
twoLines
,
US_ASCII
);
assertTrue
(
Files
.
size
(
tmpfile
)
>
0
,
"File is empty"
);
try
(
Stream
<
String
>
s
=
Files
.
lines
(
tmpfile
))
{
checkLines
(
s
,
twoLines
);
}
try
(
Stream
<
String
>
s
=
Files
.
lines
(
tmpfile
,
US_ASCII
))
{
try
(
Stream
<
String
>
s
=
Files
.
lines
(
tmpfile
,
US_ASCII
))
{
List
<
String
>
lines
=
s
.
collect
(
Collectors
.
toList
());
checkLines
(
s
,
twoLines
);
assertTrue
(
lines
.
equals
(
expected
),
"Unexpected lines"
);
}
}
// MalformedInputException
// MalformedInputException
byte
[]
bad
=
{
(
byte
)
0xff
,
(
byte
)
0xff
};
byte
[]
bad
=
{
(
byte
)
0xff
,
(
byte
)
0xff
};
Files
.
write
(
tmpfile
,
bad
);
Files
.
write
(
tmpfile
,
bad
);
try
(
Stream
<
String
>
s
=
Files
.
lines
(
tmpfile
))
{
checkMalformedInputException
(
s
);
}
try
(
Stream
<
String
>
s
=
Files
.
lines
(
tmpfile
,
US_ASCII
))
{
try
(
Stream
<
String
>
s
=
Files
.
lines
(
tmpfile
,
US_ASCII
))
{
try
{
checkMalformedInputException
(
s
);
List
<
String
>
lines
=
s
.
collect
(
Collectors
.
toList
());
throw
new
RuntimeException
(
"UncheckedIOException expected"
);
}
catch
(
UncheckedIOException
ex
)
{
assertTrue
(
ex
.
getCause
()
instanceof
MalformedInputException
,
"MalformedInputException expected"
);
}
}
}
// NullPointerException
// NullPointerException
try
{
checkNullPointerException
(()
->
Files
.
lines
(
null
));
Files
.
lines
(
null
,
US_ASCII
);
checkNullPointerException
(()
->
Files
.
lines
(
null
,
US_ASCII
));
throw
new
RuntimeException
(
"NullPointerException expected"
);
checkNullPointerException
(()
->
Files
.
lines
(
tmpfile
,
null
));
}
catch
(
NullPointerException
ignore
)
{
}
try
{
Files
.
lines
(
tmpfile
,
null
);
throw
new
RuntimeException
(
"NullPointerException expected"
);
}
catch
(
NullPointerException
ignore
)
{
}
}
finally
{
}
finally
{
Files
.
delete
(
tmpfile
);
Files
.
delete
(
tmpfile
);
}
}
}
}
private
void
checkLines
(
Stream
<
String
>
s
,
List
<
String
>
expected
)
{
List
<
String
>
lines
=
s
.
collect
(
Collectors
.
toList
());
assertTrue
(
lines
.
size
()
==
expected
.
size
(),
"Unexpected number of lines"
);
assertTrue
(
lines
.
equals
(
expected
),
"Unexpected content"
);
}
private
void
checkMalformedInputException
(
Stream
<
String
>
s
)
{
try
{
List
<
String
>
lines
=
s
.
collect
(
Collectors
.
toList
());
fail
(
"UncheckedIOException expected"
);
}
catch
(
UncheckedIOException
ex
)
{
IOException
cause
=
ex
.
getCause
();
assertTrue
(
cause
instanceof
MalformedInputException
,
"MalformedInputException expected"
);
}
}
private
void
checkNullPointerException
(
Callable
<?>
c
)
{
try
{
c
.
call
();
fail
(
"NullPointerException expected"
);
}
catch
(
NullPointerException
ignore
)
{
}
catch
(
Exception
e
)
{
fail
(
e
+
" not expected"
);
}
}
public
void
testDirectoryIteratorException
()
throws
IOException
{
public
void
testDirectoryIteratorException
()
throws
IOException
{
Path
dir
=
testFolder
.
resolve
(
"dir2"
);
Path
dir
=
testFolder
.
resolve
(
"dir2"
);
Path
trigger
=
dir
.
resolve
(
"DirectoryIteratorException"
);
Path
trigger
=
dir
.
resolve
(
"DirectoryIteratorException"
);
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录