Skip to content
项目
群组
代码片段
帮助
正在加载...
帮助
为 GitLab 提交贡献
登录/注册
切换导航
H
h2database
项目
项目
详情
活动
周期分析
仓库
仓库
文件
提交
分支
标签
贡献者
分枝图
比较
统计图
议题
0
议题
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
CI / CD
CI / CD
流水线
作业
计划
统计图
Wiki
Wiki
代码片段
代码片段
成员
成员
折叠边栏
关闭边栏
活动
分枝图
统计图
创建新议题
作业
提交
议题看板
打开侧边栏
Administrator
h2database
Commits
aac17564
Unverified
提交
aac17564
authored
7 年前
作者:
Noel Grandin
提交者:
GitHub
7 年前
浏览文件
操作
浏览文件
下载
差异文件
Merge pull request #878 from katzyn/datetime
Fix IYYY in TO_CHAR and reimplement TRUNCATE without a Calendar
上级
296a1608
8075f8dd
隐藏空白字符变更
内嵌
并排
正在显示
4 个修改的文件
包含
85 行增加
和
60 行删除
+85
-60
Function.java
h2/src/main/org/h2/expression/Function.java
+8
-25
ToChar.java
h2/src/main/org/h2/util/ToChar.java
+49
-34
ValueTimestampTimeZone.java
h2/src/main/org/h2/value/ValueTimestampTimeZone.java
+13
-1
TestFunctions.java
h2/src/test/org/h2/test/db/TestFunctions.java
+15
-0
没有找到文件。
h2/src/main/org/h2/expression/Function.java
浏览文件 @
aac17564
...
...
@@ -17,7 +17,6 @@ import java.sql.ResultSet;
import
java.sql.SQLException
;
import
java.text.SimpleDateFormat
;
import
java.util.ArrayList
;
import
java.util.Calendar
;
import
java.util.GregorianCalendar
;
import
java.util.HashMap
;
import
java.util.Locale
;
...
...
@@ -1233,32 +1232,16 @@ public class Function extends Expression implements FunctionCall {
}
case
TRUNCATE:
{
if
(
v0
.
getType
()
==
Value
.
TIMESTAMP
)
{
java
.
sql
.
Timestamp
d
=
v0
.
getTimestamp
();
Calendar
c
=
DateTimeUtils
.
createGregorianCalendar
();
c
.
setTime
(
d
);
c
.
set
(
Calendar
.
HOUR_OF_DAY
,
0
);
c
.
set
(
Calendar
.
MINUTE
,
0
);
c
.
set
(
Calendar
.
SECOND
,
0
);
c
.
set
(
Calendar
.
MILLISECOND
,
0
);
result
=
ValueTimestamp
.
fromMillis
(
c
.
getTimeInMillis
());
result
=
ValueTimestamp
.
fromDateValueAndNanos
(((
ValueTimestamp
)
v0
).
getDateValue
(),
0
);
}
else
if
(
v0
.
getType
()
==
Value
.
DATE
)
{
ValueDate
vd
=
(
ValueDate
)
v0
;
Calendar
c
=
DateTimeUtils
.
createGregorianCalendar
();
c
.
setTime
(
vd
.
getDate
());
c
.
set
(
Calendar
.
HOUR_OF_DAY
,
0
);
c
.
set
(
Calendar
.
MINUTE
,
0
);
c
.
set
(
Calendar
.
SECOND
,
0
);
c
.
set
(
Calendar
.
MILLISECOND
,
0
);
result
=
ValueTimestamp
.
fromMillis
(
c
.
getTimeInMillis
());
result
=
ValueTimestamp
.
fromDateValueAndNanos
(((
ValueDate
)
v0
).
getDateValue
(),
0
);
}
else
if
(
v0
.
getType
()
==
Value
.
TIMESTAMP_TZ
)
{
ValueTimestampTimeZone
ts
=
(
ValueTimestampTimeZone
)
v0
;
result
=
ValueTimestampTimeZone
.
fromDateValueAndNanos
(
ts
.
getDateValue
(),
0
,
ts
.
getTimeZoneOffsetMins
());
}
else
if
(
v0
.
getType
()
==
Value
.
STRING
)
{
ValueString
vd
=
(
ValueString
)
v0
;
Calendar
c
=
DateTimeUtils
.
createGregorianCalendar
();
c
.
setTime
(
ValueTimestamp
.
parse
(
vd
.
getString
(),
session
.
getDatabase
().
getMode
()).
getDate
());
c
.
set
(
Calendar
.
HOUR_OF_DAY
,
0
);
c
.
set
(
Calendar
.
MINUTE
,
0
);
c
.
set
(
Calendar
.
SECOND
,
0
);
c
.
set
(
Calendar
.
MILLISECOND
,
0
);
result
=
ValueTimestamp
.
fromMillis
(
c
.
getTimeInMillis
());
ValueTimestamp
ts
=
ValueTimestamp
.
parse
(
v0
.
getString
(),
session
.
getDatabase
().
getMode
());
result
=
ValueTimestamp
.
fromDateValueAndNanos
(
ts
.
getDateValue
(),
0
);
}
else
{
double
d
=
v0
.
getDouble
();
int
p
=
v1
==
null
?
0
:
v1
.
getInt
();
...
...
This diff is collapsed.
Click to expand it.
h2/src/main/org/h2/util/ToChar.java
浏览文件 @
aac17564
...
...
@@ -18,6 +18,7 @@ import java.util.TimeZone;
import
org.h2.api.ErrorCode
;
import
org.h2.message.DbException
;
import
org.h2.value.Value
;
import
org.h2.value.ValueTimestampTimeZone
;
/**
* Emulates Oracle's TO_CHAR function.
...
...
@@ -663,20 +664,20 @@ public class ToChar {
// Long/short date/time format
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"DL"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"DL"
)
!=
null
)
{
String
day
=
getNames
(
WEEKDAYS
)[
DateTimeUtils
.
getSundayDayOfWeek
(
dateValue
)];
String
month
=
getNames
(
MONTHS
)[
monthOfYear
-
1
];
output
.
append
(
day
).
append
(
", "
).
append
(
month
).
append
(
' '
).
append
(
dayOfMonth
).
append
(
", "
);
StringUtils
.
appendZeroPadded
(
output
,
4
,
posYear
);
i
+=
2
;
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"DS"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"DS"
)
!=
null
)
{
StringUtils
.
appendZeroPadded
(
output
,
2
,
monthOfYear
);
output
.
append
(
'/'
);
StringUtils
.
appendZeroPadded
(
output
,
2
,
dayOfMonth
);
output
.
append
(
'/'
);
StringUtils
.
appendZeroPadded
(
output
,
4
,
posYear
);
i
+=
2
;
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"TS"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"TS"
)
!=
null
)
{
output
.
append
(
h12
).
append
(
':'
);
StringUtils
.
appendZeroPadded
(
output
,
2
,
minute
);
output
.
append
(
':'
);
...
...
@@ -687,10 +688,10 @@ public class ToChar {
// Day
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"DDD"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"DDD"
)
!=
null
)
{
output
.
append
(
DateTimeUtils
.
getDayOfYear
(
dateValue
));
i
+=
3
;
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"DD"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"DD"
)
!=
null
)
{
StringUtils
.
appendZeroPadded
(
output
,
2
,
dayOfMonth
);
i
+=
2
;
}
else
if
((
cap
=
containsAt
(
format
,
i
,
"DY"
))
!=
null
)
{
...
...
@@ -704,99 +705,113 @@ public class ToChar {
}
output
.
append
(
cap
.
apply
(
day
));
i
+=
3
;
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"D"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"D"
)
!=
null
)
{
output
.
append
(
DateTimeUtils
.
getSundayDayOfWeek
(
dateValue
));
i
+=
1
;
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"J"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"J"
)
!=
null
)
{
output
.
append
(
DateTimeUtils
.
absoluteDayFromDateValue
(
dateValue
)
-
JULIAN_EPOCH
);
i
+=
1
;
// Hours
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"HH24"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"HH24"
)
!=
null
)
{
StringUtils
.
appendZeroPadded
(
output
,
2
,
hour
);
i
+=
4
;
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"HH12"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"HH12"
)
!=
null
)
{
StringUtils
.
appendZeroPadded
(
output
,
2
,
h12
);
i
+=
4
;
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"HH"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"HH"
)
!=
null
)
{
StringUtils
.
appendZeroPadded
(
output
,
2
,
h12
);
i
+=
2
;
// Minutes
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"MI"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"MI"
)
!=
null
)
{
StringUtils
.
appendZeroPadded
(
output
,
2
,
minute
);
i
+=
2
;
// Seconds
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"SSSSS"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"SSSSS"
)
!=
null
)
{
int
seconds
=
(
int
)
(
timeNanos
/
1_000_000_000
);
output
.
append
(
seconds
);
i
+=
5
;
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"SS"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"SS"
)
!=
null
)
{
StringUtils
.
appendZeroPadded
(
output
,
2
,
second
);
i
+=
2
;
// Fractional seconds
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"FF1"
,
"FF2"
,
"FF3"
,
"FF4"
,
"FF5"
,
"FF6"
,
"FF7"
,
"FF8"
,
"FF9"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"FF1"
,
"FF2"
,
"FF3"
,
"FF4"
,
"FF5"
,
"FF6"
,
"FF7"
,
"FF8"
,
"FF9"
)
!=
null
)
{
int
x
=
format
.
charAt
(
i
+
2
)
-
'0'
;
int
ff
=
(
int
)
(
nanos
*
Math
.
pow
(
10
,
x
-
9
));
StringUtils
.
appendZeroPadded
(
output
,
x
,
ff
);
i
+=
3
;
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"FF"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"FF"
)
!=
null
)
{
StringUtils
.
appendZeroPadded
(
output
,
9
,
nanos
);
i
+=
2
;
// Time zone
}
else
if
((
cap
=
containsAt
(
format
,
i
,
"TZR"
))
!=
null
)
{
TimeZone
tz
=
TimeZone
.
getDefault
();
}
else
if
(
containsAt
(
format
,
i
,
"TZR"
)
!=
null
)
{
TimeZone
tz
=
value
instanceof
ValueTimestampTimeZone
?
((
ValueTimestampTimeZone
)
value
).
getTimeZone
()
:
TimeZone
.
getDefault
();
output
.
append
(
tz
.
getID
());
i
+=
3
;
}
else
if
((
cap
=
containsAt
(
format
,
i
,
"TZD"
))
!=
null
)
{
TimeZone
tz
=
TimeZone
.
getDefault
();
}
else
if
(
containsAt
(
format
,
i
,
"TZD"
)
!=
null
)
{
TimeZone
tz
=
value
instanceof
ValueTimestampTimeZone
?
((
ValueTimestampTimeZone
)
value
).
getTimeZone
()
:
TimeZone
.
getDefault
();
boolean
daylight
=
tz
.
inDaylightTime
(
new
java
.
util
.
Date
());
output
.
append
(
tz
.
getDisplayName
(
daylight
,
TimeZone
.
SHORT
));
i
+=
3
;
// Week
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"IW"
,
"WW"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"IW"
,
"WW"
)
!=
null
)
{
output
.
append
(
DateTimeUtils
.
getWeekOfYear
(
dateValue
,
0
,
1
));
i
+=
2
;
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"W"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"W"
)
!=
null
)
{
int
w
=
1
+
dayOfMonth
/
7
;
output
.
append
(
w
);
i
+=
1
;
// Year
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"Y,YYY"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"Y,YYY"
)
!=
null
)
{
output
.
append
(
new
DecimalFormat
(
"#,###"
).
format
(
posYear
));
i
+=
5
;
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"SYYYY"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"SYYYY"
)
!=
null
)
{
// Should be <= 0, but Oracle prints negative years with off-by-one difference
if
(
year
<
0
)
{
output
.
append
(
'-'
);
}
StringUtils
.
appendZeroPadded
(
output
,
4
,
posYear
);
i
+=
5
;
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"YYYY"
,
"IYYY"
,
"RRRR"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"YYYY"
,
"RRRR"
)
!=
null
)
{
StringUtils
.
appendZeroPadded
(
output
,
4
,
posYear
);
i
+=
4
;
}
else
if
((
cap
=
containsAt
(
format
,
i
,
"YYY"
,
"IYY"
))
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"IYYY"
)
!=
null
)
{
StringUtils
.
appendZeroPadded
(
output
,
4
,
Math
.
abs
(
DateTimeUtils
.
getIsoWeekYear
(
dateValue
)));
i
+=
4
;
}
else
if
(
containsAt
(
format
,
i
,
"YYY"
)
!=
null
)
{
StringUtils
.
appendZeroPadded
(
output
,
3
,
posYear
%
1000
);
i
+=
3
;
}
else
if
((
cap
=
containsAt
(
format
,
i
,
"YY"
,
"IY"
,
"RR"
))
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"IYY"
)
!=
null
)
{
StringUtils
.
appendZeroPadded
(
output
,
3
,
Math
.
abs
(
DateTimeUtils
.
getIsoWeekYear
(
dateValue
))
%
1000
);
i
+=
3
;
}
else
if
(
containsAt
(
format
,
i
,
"YY"
,
"RR"
)
!=
null
)
{
StringUtils
.
appendZeroPadded
(
output
,
2
,
posYear
%
100
);
i
+=
2
;
}
else
if
((
cap
=
containsAt
(
format
,
i
,
"I"
,
"Y"
))
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"IY"
)
!=
null
)
{
StringUtils
.
appendZeroPadded
(
output
,
2
,
Math
.
abs
(
DateTimeUtils
.
getIsoWeekYear
(
dateValue
))
%
100
);
i
+=
2
;
}
else
if
(
containsAt
(
format
,
i
,
"Y"
)
!=
null
)
{
output
.
append
(
posYear
%
10
);
i
+=
1
;
}
else
if
(
containsAt
(
format
,
i
,
"I"
)
!=
null
)
{
output
.
append
(
Math
.
abs
(
DateTimeUtils
.
getIsoWeekYear
(
dateValue
))
%
10
);
i
+=
1
;
// Month / quarter
...
...
@@ -811,35 +826,35 @@ public class ToChar {
String
month
=
getNames
(
SHORT_MONTHS
)[
monthOfYear
-
1
];
output
.
append
(
cap
.
apply
(
month
));
i
+=
3
;
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"MM"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"MM"
)
!=
null
)
{
StringUtils
.
appendZeroPadded
(
output
,
2
,
monthOfYear
);
i
+=
2
;
}
else
if
((
cap
=
containsAt
(
format
,
i
,
"RM"
))
!=
null
)
{
output
.
append
(
cap
.
apply
(
toRomanNumeral
(
monthOfYear
)));
i
+=
2
;
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"Q"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"Q"
)
!=
null
)
{
int
q
=
1
+
((
monthOfYear
-
1
)
/
3
);
output
.
append
(
q
);
i
+=
1
;
// Local radix character
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"X"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"X"
)
!=
null
)
{
char
c
=
DecimalFormatSymbols
.
getInstance
().
getDecimalSeparator
();
output
.
append
(
c
);
i
+=
1
;
// Format modifiers
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"FM"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"FM"
)
!=
null
)
{
fillMode
=
!
fillMode
;
i
+=
2
;
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"FX"
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"FX"
)
!=
null
)
{
i
+=
2
;
// Literal text
}
else
if
(
(
cap
=
containsAt
(
format
,
i
,
"\""
)
)
!=
null
)
{
}
else
if
(
containsAt
(
format
,
i
,
"\""
)
!=
null
)
{
for
(
i
=
i
+
1
;
i
<
format
.
length
();
i
++)
{
char
c
=
format
.
charAt
(
i
);
if
(
c
!=
'"'
)
{
...
...
This diff is collapsed.
Click to expand it.
h2/src/main/org/h2/value/ValueTimestampTimeZone.java
浏览文件 @
aac17564
...
...
@@ -15,6 +15,7 @@ import org.h2.api.ErrorCode;
import
org.h2.api.TimestampWithTimeZone
;
import
org.h2.message.DbException
;
import
org.h2.util.DateTimeUtils
;
import
org.h2.util.StringUtils
;
/**
* Implementation of the TIMESTAMP WITH TIME ZONE data type.
...
...
@@ -158,7 +159,18 @@ public class ValueTimestampTimeZone extends Value {
if
(
offset
==
0
)
{
return
DateTimeUtils
.
UTC
;
}
return
new
SimpleTimeZone
(
offset
*
60000
,
Integer
.
toString
(
offset
));
StringBuilder
b
=
new
StringBuilder
(
9
);
b
.
append
(
"GMT"
);
if
(
offset
<
0
)
{
b
.
append
(
'-'
);
offset
=
-
offset
;
}
else
{
b
.
append
(
'+'
);
}
StringUtils
.
appendZeroPadded
(
b
,
2
,
offset
/
60
);
b
.
append
(
':'
);
StringUtils
.
appendZeroPadded
(
b
,
2
,
offset
%
60
);
return
new
SimpleTimeZone
(
offset
*
60000
,
b
.
toString
());
}
@Override
...
...
This diff is collapsed.
Click to expand it.
h2/src/test/org/h2/test/db/TestFunctions.java
浏览文件 @
aac17564
...
...
@@ -1486,6 +1486,17 @@ public class TestFunctions extends TestBase implements AggregateFunction {
assertResult
(
"0 BC"
,
stat
,
"SELECT TO_CHAR(X, 'Y BC') FROM U"
);
assertResult
(
"1979 A.D."
,
stat
,
"SELECT TO_CHAR(X, 'YYYY B.C.') FROM T"
);
assertResult
(
"2013"
,
stat
,
"SELECT TO_CHAR(DATE '2013-12-30', 'YYYY') FROM DUAL"
);
assertResult
(
"013"
,
stat
,
"SELECT TO_CHAR(DATE '2013-12-30', 'YYY') FROM DUAL"
);
assertResult
(
"13"
,
stat
,
"SELECT TO_CHAR(DATE '2013-12-30', 'YY') FROM DUAL"
);
assertResult
(
"3"
,
stat
,
"SELECT TO_CHAR(DATE '2013-12-30', 'Y') FROM DUAL"
);
// ISO week year
assertResult
(
"2014"
,
stat
,
"SELECT TO_CHAR(DATE '2013-12-30', 'IYYY') FROM DUAL"
);
assertResult
(
"014"
,
stat
,
"SELECT TO_CHAR(DATE '2013-12-30', 'IYY') FROM DUAL"
);
assertResult
(
"14"
,
stat
,
"SELECT TO_CHAR(DATE '2013-12-30', 'IY') FROM DUAL"
);
assertResult
(
"4"
,
stat
,
"SELECT TO_CHAR(DATE '2013-12-30', 'I') FROM DUAL"
);
assertResult
(
"0001"
,
stat
,
"SELECT TO_CHAR(DATE '-0001-01-01', 'IYYY') FROM DUAL"
);
assertResult
(
"0005"
,
stat
,
"SELECT TO_CHAR(DATE '-0004-01-01', 'IYYY') FROM DUAL"
);
assertResult
(
"08:12 AM"
,
stat
,
"SELECT TO_CHAR(X, 'HH:MI AM') FROM T"
);
assertResult
(
"08:12 A.M."
,
stat
,
"SELECT TO_CHAR(X, 'HH:MI A.M.') FROM T"
);
assertResult
(
"02:04 P.M."
,
stat
,
"SELECT TO_CHAR(X, 'HH:MI A.M.') FROM U"
);
...
...
@@ -1608,6 +1619,10 @@ public class TestFunctions extends TestBase implements AggregateFunction {
assertResult
(
expected
,
stat
,
"SELECT TO_CHAR(X, 'TS') FROM T"
);
assertResult
(
tzLongName
,
stat
,
"SELECT TO_CHAR(X, 'TZR') FROM T"
);
assertResult
(
tzShortName
,
stat
,
"SELECT TO_CHAR(X, 'TZD') FROM T"
);
assertResult
(
"GMT+10:30"
,
stat
,
"SELECT TO_CHAR(TIMESTAMP WITH TIME ZONE '2010-01-01 0:00:00+10:30', 'TZR')"
);
assertResult
(
"GMT+10:30"
,
stat
,
"SELECT TO_CHAR(TIMESTAMP WITH TIME ZONE '2010-01-01 0:00:00+10:30', 'TZD')"
);
expected
=
String
.
format
(
"%f"
,
1.1
).
substring
(
1
,
2
);
assertResult
(
expected
,
stat
,
"SELECT TO_CHAR(X, 'X') FROM T"
);
expected
=
String
.
format
(
"%,d"
,
1979
);
...
...
This diff is collapsed.
Click to expand it.
编写
预览
Markdown
格式
0%
重试
或
添加新文件
添加附件
取消
您添加了
0
人
到此讨论。请谨慎行事。
请先完成此评论的编辑!
取消
请
注册
或者
登录
后发表评论