Skip to content
项目
群组
代码片段
帮助
正在加载...
帮助
为 GitLab 提交贡献
登录/注册
切换导航
H
h2database
项目
项目
详情
活动
周期分析
仓库
仓库
文件
提交
分支
标签
贡献者
分枝图
比较
统计图
议题
0
议题
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
CI / CD
CI / CD
流水线
作业
计划
统计图
Wiki
Wiki
代码片段
代码片段
成员
成员
折叠边栏
关闭边栏
活动
分枝图
统计图
创建新议题
作业
提交
议题看板
打开侧边栏
Administrator
h2database
Commits
291b541c
Unverified
提交
291b541c
authored
12月 19, 2018
作者:
Evgenij Ryazanov
提交者:
GitHub
12月 19, 2018
浏览文件
操作
浏览文件
下载
差异文件
Merge pull request #1607 from katzyn/misc
Fixes for quantified comparison predicates and other changes
上级
3d9f22c6
6786598c
隐藏空白字符变更
内嵌
并排
正在显示
25 个修改的文件
包含
242 行增加
和
104 行删除
+242
-104
help.csv
h2/src/docsrc/help/help.csv
+20
-1
changelog.html
h2/src/docsrc/html/changelog.html
+6
-0
features.html
h2/src/docsrc/html/features.html
+0
-1
Parser.java
h2/src/main/org/h2/command/Parser.java
+75
-37
Mode.java
h2/src/main/org/h2/engine/Mode.java
+0
-7
Aggregate.java
h2/src/main/org/h2/expression/aggregate/Aggregate.java
+16
-1
Comparison.java
h2/src/main/org/h2/expression/condition/Comparison.java
+19
-0
ConditionInSelect.java
...c/main/org/h2/expression/condition/ConditionInSelect.java
+28
-16
ToChar.java
h2/src/main/org/h2/expression/function/ToChar.java
+3
-2
MVMap.java
h2/src/main/org/h2/mvstore/MVMap.java
+1
-1
MVStore.java
h2/src/main/org/h2/mvstore/MVStore.java
+1
-0
Page.java
h2/src/main/org/h2/mvstore/Page.java
+2
-0
Column.java
h2/src/main/org/h2/table/Column.java
+0
-8
ParserUtil.java
h2/src/main/org/h2/util/ParserUtil.java
+21
-12
Value.java
h2/src/main/org/h2/value/Value.java
+2
-1
ValueDecimal.java
h2/src/main/org/h2/value/ValueDecimal.java
+3
-2
BenchCThread.java
h2/src/test/org/h2/test/bench/BenchCThread.java
+2
-1
TestLIRSMemoryConsumption.java
h2/src/test/org/h2/test/db/TestLIRSMemoryConsumption.java
+1
-1
TestPersistentCommonTableExpressions.java
.../org/h2/test/db/TestPersistentCommonTableExpressions.java
+3
-3
TestPreparedStatement.java
h2/src/test/org/h2/test/jdbc/TestPreparedStatement.java
+0
-1
select.sql
h2/src/test/org/h2/test/scripts/dml/select.sql
+24
-0
any.sql
h2/src/test/org/h2/test/scripts/functions/aggregate/any.sql
+12
-0
testScript.sql
h2/src/test/org/h2/test/scripts/testScript.sql
+0
-6
TestSubqueryPerformanceOnLazyExecutionMode.java
...test/unit/TestSubqueryPerformanceOnLazyExecutionMode.java
+2
-2
dictionary.txt
h2/src/tools/org/h2/build/doc/dictionary.txt
+1
-1
没有找到文件。
h2/src/docsrc/help/help.csv
浏览文件 @
291b541c
...
@@ -2152,7 +2152,7 @@ ID<>2
...
@@ -2152,7 +2152,7 @@ ID<>2
"
"
"Other Grammar","Condition Right Hand Side","
"Other Grammar","Condition Right Hand Side","
compare { {
ALL
( select ) } | operand }
compare { {
{ ALL | ANY | SOME }
( select ) } | operand }
| IS [ NOT ] NULL
| IS [ NOT ] NULL
| IS [ NOT ] [ DISTINCT FROM ] operand
| IS [ NOT ] [ DISTINCT FROM ] operand
| BETWEEN operand AND operand
| BETWEEN operand AND operand
...
@@ -2162,6 +2162,18 @@ compare { { ALL ( select ) } | operand }
...
@@ -2162,6 +2162,18 @@ compare { { ALL ( select ) } | operand }
","
","
The right hand side of a condition.
The right hand side of a condition.
Quantified comparison predicate ALL returns TRUE if specified comparison operation between
left size of condition and each row from a subquery returns TRUE, including case when there are no rows.
ALL predicate returns FALSE if at least one such comparison returns FALSE.
Otherwise it returns NULL.
Quantified comparison predicates ANY and SOME return TRUE if specified comparison operation between
left size of condition and at least one row from a subquery returns TRUE.
ANY and SOME predicates return FALSE if all such comparisons return FALSE.
Otherwise it returns NULL.
Note that these predicates have priority over ANY and SOME aggregate functions with subquery on the right side.
Use parentheses around aggregate function.
The conditions ""IS [ NOT ]"" and ""IS [ NOT ] DISTINCT FROM"" are null-safe, meaning
The conditions ""IS [ NOT ]"" and ""IS [ NOT ] DISTINCT FROM"" are null-safe, meaning
NULL is considered the same as NULL, and the condition never evaluates to NULL.
NULL is considered the same as NULL, and the condition never evaluates to NULL.
...
@@ -2179,6 +2191,8 @@ ILIKE does a case-insensitive compare.
...
@@ -2179,6 +2191,8 @@ ILIKE does a case-insensitive compare.
When comparing with REGEXP, regular expression matching is used.
When comparing with REGEXP, regular expression matching is used.
See Java ""Matcher.find"" for details.
See Java ""Matcher.find"" for details.
","
","
VALUE > 10
A IS NOT DISTINCT FROM B
LIKE 'Jo%'
LIKE 'Jo%'
"
"
...
@@ -3469,8 +3483,13 @@ EVERY(ID>10)
...
@@ -3469,8 +3483,13 @@ EVERY(ID>10)
Returns true if any expression is true.
Returns true if any expression is true.
If no rows are selected, the result is NULL.
If no rows are selected, the result is NULL.
Aggregates are only allowed in select statements.
Aggregates are only allowed in select statements.
Note that if ANY or SOME aggregate function is placed on the right side of comparison operation
and argument of this function is a subquery additional parentheses around aggregate function are required,
otherwise it will be parsed as quantified comparison predicate.
","
","
ANY(NAME LIKE 'W%')
ANY(NAME LIKE 'W%')
A = (ANY((SELECT B FROM T)))
"
"
"Functions (Aggregate)","COUNT","
"Functions (Aggregate)","COUNT","
...
...
h2/src/docsrc/html/changelog.html
浏览文件 @
291b541c
...
@@ -21,6 +21,12 @@ Change Log
...
@@ -21,6 +21,12 @@ Change Log
<h2>
Next Version (unreleased)
</h2>
<h2>
Next Version (unreleased)
</h2>
<ul>
<ul>
<li>
Issue #1606: Quantified comparison predicate doesn't work correctly on primary key column
</li>
<li>
Issue #1057: Very slow execution with subquery and connection parameter LAZY_QUERY_EXECUTION=1
</li>
<li>
Issue #1072: Very slow execution with join and connection parameter LAZY_QUERY_EXECUTION=1
</li>
<li>
PR #1601: Return BIGINT from ROWNUM(), ROW_NUMBER() and rank functions
<li>
PR #1601: Return BIGINT from ROWNUM(), ROW_NUMBER() and rank functions
</li>
</li>
<li>
PR #1599: cleanup StringUtils.cache
<li>
PR #1599: cleanup StringUtils.cache
...
...
h2/src/docsrc/html/features.html
浏览文件 @
291b541c
...
@@ -959,7 +959,6 @@ or the SQL statement <code>SET MODE PostgreSQL</code>.
...
@@ -959,7 +959,6 @@ or the SQL statement <code>SET MODE PostgreSQL</code>.
</li><li>
Fixed-width strings are padded with spaces.
</li><li>
Fixed-width strings are padded with spaces.
</li><li>
MONEY data type is treated like NUMERIC(19, 2) data type.
</li><li>
MONEY data type is treated like NUMERIC(19, 2) data type.
</li><li>
Datetime value functions return the same value within a transaction.
</li><li>
Datetime value functions return the same value within a transaction.
</li><li>
ANY and SOME after comparison operators are parsed as array comparison operators.
</li></ul>
</li></ul>
<h3>
Ignite Compatibility Mode
</h3>
<h3>
Ignite Compatibility Mode
</h3>
...
...
h2/src/main/org/h2/command/Parser.java
浏览文件 @
291b541c
...
@@ -12,6 +12,9 @@ import static org.h2.util.ParserUtil.ALL;
...
@@ -12,6 +12,9 @@ import static org.h2.util.ParserUtil.ALL;
import
static
org
.
h2
.
util
.
ParserUtil
.
CHECK
;
import
static
org
.
h2
.
util
.
ParserUtil
.
CHECK
;
import
static
org
.
h2
.
util
.
ParserUtil
.
CONSTRAINT
;
import
static
org
.
h2
.
util
.
ParserUtil
.
CONSTRAINT
;
import
static
org
.
h2
.
util
.
ParserUtil
.
CROSS
;
import
static
org
.
h2
.
util
.
ParserUtil
.
CROSS
;
import
static
org
.
h2
.
util
.
ParserUtil
.
CURRENT_DATE
;
import
static
org
.
h2
.
util
.
ParserUtil
.
CURRENT_TIME
;
import
static
org
.
h2
.
util
.
ParserUtil
.
CURRENT_TIMESTAMP
;
import
static
org
.
h2
.
util
.
ParserUtil
.
DISTINCT
;
import
static
org
.
h2
.
util
.
ParserUtil
.
DISTINCT
;
import
static
org
.
h2
.
util
.
ParserUtil
.
EXCEPT
;
import
static
org
.
h2
.
util
.
ParserUtil
.
EXCEPT
;
import
static
org
.
h2
.
util
.
ParserUtil
.
EXISTS
;
import
static
org
.
h2
.
util
.
ParserUtil
.
EXISTS
;
...
@@ -30,6 +33,8 @@ import static org.h2.util.ParserUtil.IS;
...
@@ -30,6 +33,8 @@ import static org.h2.util.ParserUtil.IS;
import
static
org
.
h2
.
util
.
ParserUtil
.
JOIN
;
import
static
org
.
h2
.
util
.
ParserUtil
.
JOIN
;
import
static
org
.
h2
.
util
.
ParserUtil
.
LIKE
;
import
static
org
.
h2
.
util
.
ParserUtil
.
LIKE
;
import
static
org
.
h2
.
util
.
ParserUtil
.
LIMIT
;
import
static
org
.
h2
.
util
.
ParserUtil
.
LIMIT
;
import
static
org
.
h2
.
util
.
ParserUtil
.
LOCALTIME
;
import
static
org
.
h2
.
util
.
ParserUtil
.
LOCALTIMESTAMP
;
import
static
org
.
h2
.
util
.
ParserUtil
.
MINUS
;
import
static
org
.
h2
.
util
.
ParserUtil
.
MINUS
;
import
static
org
.
h2
.
util
.
ParserUtil
.
NATURAL
;
import
static
org
.
h2
.
util
.
ParserUtil
.
NATURAL
;
import
static
org
.
h2
.
util
.
ParserUtil
.
NOT
;
import
static
org
.
h2
.
util
.
ParserUtil
.
NOT
;
...
@@ -457,6 +462,10 @@ public class Parser {
...
@@ -457,6 +462,10 @@ public class Parser {
"LIKE"
,
"LIKE"
,
// LIMIT
// LIMIT
"LIMIT"
,
"LIMIT"
,
// LOCALTIME
"LOCALTIME"
,
// LOCALTIMESTAMP
"LOCALTIMESTAMP"
,
// MINUS
// MINUS
"MINUS"
,
"MINUS"
,
// NATURAL
// NATURAL
...
@@ -1408,7 +1417,6 @@ public class Parser {
...
@@ -1408,7 +1417,6 @@ public class Parser {
return
select
;
return
select
;
}
}
private
Prepared
parseMerge
()
{
private
Prepared
parseMerge
()
{
int
start
=
lastParseIndex
;
int
start
=
lastParseIndex
;
read
(
"INTO"
);
read
(
"INTO"
);
...
@@ -2853,23 +2861,33 @@ public class Parser {
...
@@ -2853,23 +2861,33 @@ public class Parser {
break
;
break
;
}
}
read
();
read
();
int
start
=
lastParseIndex
;
if
(
readIf
(
ALL
))
{
if
(
readIf
(
ALL
))
{
read
(
OPEN_PAREN
);
read
(
OPEN_PAREN
);
Query
query
=
parseSelect
();
if
(
isSelect
())
{
r
=
new
ConditionInSelect
(
database
,
r
,
query
,
true
,
Query
query
=
parseSelect
();
compareType
);
r
=
new
ConditionInSelect
(
database
,
r
,
query
,
true
,
compareType
);
read
(
CLOSE_PAREN
);
read
(
CLOSE_PAREN
);
}
else
if
(
database
.
getMode
().
anyAndSomeAreComparisons
&&
(
readIf
(
"ANY"
)
||
readIf
(
"SOME"
)))
{
}
else
{
parseIndex
=
start
;
read
();
r
=
new
Comparison
(
session
,
compareType
,
r
,
readConcat
());
}
}
else
if
(
readIf
(
"ANY"
)
||
readIf
(
"SOME"
))
{
read
(
OPEN_PAREN
);
read
(
OPEN_PAREN
);
if
(
currentTokenType
==
PARAMETER
&&
compareType
==
0
)
{
if
(
currentTokenType
==
PARAMETER
&&
compareType
==
0
)
{
Parameter
p
=
readParameter
();
Parameter
p
=
readParameter
();
r
=
new
ConditionInParameter
(
database
,
r
,
p
);
r
=
new
ConditionInParameter
(
database
,
r
,
p
);
}
else
{
read
(
CLOSE_PAREN
);
}
else
if
(
isSelect
())
{
Query
query
=
parseSelect
();
Query
query
=
parseSelect
();
r
=
new
ConditionInSelect
(
database
,
r
,
query
,
false
,
r
=
new
ConditionInSelect
(
database
,
r
,
query
,
false
,
compareType
);
compareType
);
read
(
CLOSE_PAREN
);
}
else
{
parseIndex
=
start
;
read
();
r
=
new
Comparison
(
session
,
compareType
,
r
,
readConcat
());
}
}
read
(
CLOSE_PAREN
);
}
else
{
}
else
{
r
=
new
Comparison
(
session
,
compareType
,
r
,
readConcat
());
r
=
new
Comparison
(
session
,
compareType
,
r
,
readConcat
());
}
}
...
@@ -3523,6 +3541,14 @@ public class Parser {
...
@@ -3523,6 +3541,14 @@ public class Parser {
}
}
}
}
private
Expression
readKeywordFunction
(
String
name
)
{
if
(
readIf
(
OPEN_PAREN
))
{
return
readFunction
(
null
,
name
);
}
else
{
return
readFunctionWithoutParameters
(
name
);
}
}
private
Expression
readFunctionWithoutParameters
(
String
name
)
{
private
Expression
readFunctionWithoutParameters
(
String
name
)
{
if
(
database
.
isAllowBuiltinAliasOverride
())
{
if
(
database
.
isAllowBuiltinAliasOverride
())
{
FunctionAlias
functionAlias
=
database
.
getSchema
(
session
.
getCurrentSchemaName
()).
findFunction
(
name
);
FunctionAlias
functionAlias
=
database
.
getSchema
(
session
.
getCurrentSchemaName
()).
findFunction
(
name
);
...
@@ -3806,6 +3832,26 @@ public class Parser {
...
@@ -3806,6 +3832,26 @@ public class Parser {
r
=
ValueExpression
.
get
(
currentValue
);
r
=
ValueExpression
.
get
(
currentValue
);
read
();
read
();
break
;
break
;
case
CURRENT_DATE:
read
();
r
=
readKeywordFunction
(
"CURRENT_DATE"
);
break
;
case
CURRENT_TIME:
read
();
r
=
readKeywordFunction
(
"CURRENT_TIME"
);
break
;
case
CURRENT_TIMESTAMP:
read
();
r
=
readKeywordFunction
(
"CURRENT_TIMESTAMP"
);
break
;
case
LOCALTIME:
read
();
r
=
readKeywordFunction
(
"LOCALTIME"
);
break
;
case
LOCALTIMESTAMP:
read
();
r
=
readKeywordFunction
(
"LOCALTIMESTAMP"
);
break
;
default
:
default
:
throw
getSyntaxError
();
throw
getSyntaxError
();
}
}
...
@@ -3844,9 +3890,15 @@ public class Parser {
...
@@ -3844,9 +3890,15 @@ public class Parser {
private
Expression
readTermWithIdentifier
(
String
name
)
{
private
Expression
readTermWithIdentifier
(
String
name
)
{
// Unquoted identifier is never empty
// Unquoted identifier is never empty
char
ch
=
name
.
charAt
(
0
);
char
ch
=
name
.
charAt
(
0
);
if
(!
identifiersToUpper
)
{
/*
* Convert a-z to A-Z. This method is safe, because only A-Z
* characters are considered below.
*/
ch
&=
0xffdf
;
}
switch
(
ch
)
{
switch
(
ch
)
{
case
'A'
:
case
'A'
:
case
'a'
:
if
(
equalsToken
(
"ARRAY"
,
name
))
{
if
(
equalsToken
(
"ARRAY"
,
name
))
{
read
(
OPEN_BRACKET
);
read
(
OPEN_BRACKET
);
ArrayList
<
Expression
>
list
=
Utils
.
newSmallArrayList
();
ArrayList
<
Expression
>
list
=
Utils
.
newSmallArrayList
();
...
@@ -3861,21 +3913,13 @@ public class Parser {
...
@@ -3861,21 +3913,13 @@ public class Parser {
}
}
break
;
break
;
case
'C'
:
case
'C'
:
case
'c'
:
if
(
equalsToken
(
"CURRENT_USER"
,
name
))
{
if
(
equalsToken
(
"CURRENT_DATE"
,
name
))
{
return
readFunctionWithoutParameters
(
"CURRENT_DATE"
);
}
else
if
(
equalsToken
(
"CURRENT_TIME"
,
name
))
{
return
readFunctionWithoutParameters
(
"CURRENT_TIME"
);
}
else
if
(
equalsToken
(
"CURRENT_TIMESTAMP"
,
name
))
{
return
readFunctionWithoutParameters
(
"CURRENT_TIMESTAMP"
);
}
else
if
(
equalsToken
(
"CURRENT_USER"
,
name
))
{
return
readFunctionWithoutParameters
(
"USER"
);
return
readFunctionWithoutParameters
(
"USER"
);
}
else
if
(
database
.
getMode
().
getEnum
()
==
ModeEnum
.
DB2
&&
equalsToken
(
"CURRENT"
,
name
))
{
}
else
if
(
database
.
getMode
().
getEnum
()
==
ModeEnum
.
DB2
&&
equalsToken
(
"CURRENT"
,
name
))
{
return
parseDB2SpecialRegisters
(
name
);
return
parseDB2SpecialRegisters
(
name
);
}
}
break
;
break
;
case
'D'
:
case
'D'
:
case
'd'
:
if
(
currentTokenType
==
VALUE
&&
currentValue
.
getType
()
==
Value
.
STRING
&&
if
(
currentTokenType
==
VALUE
&&
currentValue
.
getType
()
==
Value
.
STRING
&&
(
equalsToken
(
"DATE"
,
name
)
||
equalsToken
(
"D"
,
name
)))
{
(
equalsToken
(
"DATE"
,
name
)
||
equalsToken
(
"D"
,
name
)))
{
String
date
=
currentValue
.
getString
();
String
date
=
currentValue
.
getString
();
...
@@ -3884,7 +3928,6 @@ public class Parser {
...
@@ -3884,7 +3928,6 @@ public class Parser {
}
}
break
;
break
;
case
'E'
:
case
'E'
:
case
'e'
:
if
(
currentTokenType
==
VALUE
&&
currentValue
.
getType
()
==
Value
.
STRING
&&
equalsToken
(
"E"
,
name
))
{
if
(
currentTokenType
==
VALUE
&&
currentValue
.
getType
()
==
Value
.
STRING
&&
equalsToken
(
"E"
,
name
))
{
String
text
=
currentValue
.
getString
();
String
text
=
currentValue
.
getString
();
// the PostgreSQL ODBC driver uses
// the PostgreSQL ODBC driver uses
...
@@ -3897,21 +3940,11 @@ public class Parser {
...
@@ -3897,21 +3940,11 @@ public class Parser {
}
}
break
;
break
;
case
'I'
:
case
'I'
:
case
'i'
:
if
(
equalsToken
(
"INTERVAL"
,
name
))
{
if
(
equalsToken
(
"INTERVAL"
,
name
))
{
return
readInterval
();
return
readInterval
();
}
}
break
;
break
;
case
'L'
:
case
'l'
:
if
(
equalsToken
(
"LOCALTIME"
,
name
))
{
return
readFunctionWithoutParameters
(
"LOCALTIME"
);
}
else
if
(
equalsToken
(
"LOCALTIMESTAMP"
,
name
))
{
return
readFunctionWithoutParameters
(
"LOCALTIMESTAMP"
);
}
break
;
case
'N'
:
case
'N'
:
case
'n'
:
if
(
equalsToken
(
"NEXT"
,
name
)
&&
readIf
(
"VALUE"
))
{
if
(
equalsToken
(
"NEXT"
,
name
)
&&
readIf
(
"VALUE"
))
{
read
(
FOR
);
read
(
FOR
);
return
new
SequenceValue
(
readSequence
());
return
new
SequenceValue
(
readSequence
());
...
@@ -3923,7 +3956,6 @@ public class Parser {
...
@@ -3923,7 +3956,6 @@ public class Parser {
}
}
break
;
break
;
case
'S'
:
case
'S'
:
case
's'
:
if
(
equalsToken
(
"SYSDATE"
,
name
))
{
if
(
equalsToken
(
"SYSDATE"
,
name
))
{
return
readFunctionWithoutParameters
(
"CURRENT_TIMESTAMP"
);
return
readFunctionWithoutParameters
(
"CURRENT_TIMESTAMP"
);
}
else
if
(
equalsToken
(
"SYSTIME"
,
name
))
{
}
else
if
(
equalsToken
(
"SYSTIME"
,
name
))
{
...
@@ -3933,7 +3965,6 @@ public class Parser {
...
@@ -3933,7 +3965,6 @@ public class Parser {
}
}
break
;
break
;
case
'T'
:
case
'T'
:
case
't'
:
if
(
equalsToken
(
"TIME"
,
name
))
{
if
(
equalsToken
(
"TIME"
,
name
))
{
boolean
without
=
readIf
(
"WITHOUT"
);
boolean
without
=
readIf
(
"WITHOUT"
);
if
(
without
)
{
if
(
without
)
{
...
@@ -3986,7 +4017,6 @@ public class Parser {
...
@@ -3986,7 +4017,6 @@ public class Parser {
}
}
break
;
break
;
case
'X'
:
case
'X'
:
case
'x'
:
if
(
currentTokenType
==
VALUE
&&
currentValue
.
getType
()
==
Value
.
STRING
&&
equalsToken
(
"X"
,
name
))
{
if
(
currentTokenType
==
VALUE
&&
currentValue
.
getType
()
==
Value
.
STRING
&&
equalsToken
(
"X"
,
name
))
{
byte
[]
buffer
=
StringUtils
.
convertHexToBytes
(
currentValue
.
getString
());
byte
[]
buffer
=
StringUtils
.
convertHexToBytes
(
currentValue
.
getString
());
read
();
read
();
...
@@ -4061,10 +4091,11 @@ public class Parser {
...
@@ -4061,10 +4091,11 @@ public class Parser {
if
(
readIf
(
WITH
))
{
if
(
readIf
(
WITH
))
{
read
(
"TIME"
);
read
(
"TIME"
);
read
(
"ZONE"
);
read
(
"ZONE"
);
return
read
FunctionWithoutParameters
(
"CURRENT_TIMESTAMP"
);
return
read
KeywordFunction
(
"CURRENT_TIMESTAMP"
);
}
}
return
read
FunctionWithoutParameters
(
"LOCALTIMESTAMP"
);
return
read
KeywordFunction
(
"LOCALTIMESTAMP"
);
}
else
if
(
readIf
(
"TIME"
))
{
}
else
if
(
readIf
(
"TIME"
))
{
// Time with fractional seconds is not supported by DB2
return
readFunctionWithoutParameters
(
"CURRENT_TIME"
);
return
readFunctionWithoutParameters
(
"CURRENT_TIME"
);
}
else
if
(
readIf
(
"DATE"
))
{
}
else
if
(
readIf
(
"DATE"
))
{
return
readFunctionWithoutParameters
(
"CURRENT_DATE"
);
return
readFunctionWithoutParameters
(
"CURRENT_DATE"
);
...
@@ -5978,7 +6009,14 @@ public class Parser {
...
@@ -5978,7 +6009,14 @@ public class Parser {
private
CreateFunctionAlias
parseCreateFunctionAlias
(
boolean
force
)
{
private
CreateFunctionAlias
parseCreateFunctionAlias
(
boolean
force
)
{
boolean
ifNotExists
=
readIfNotExists
();
boolean
ifNotExists
=
readIfNotExists
();
String
aliasName
=
readIdentifierWithSchema
();
String
aliasName
;
if
(
currentTokenType
!=
IDENTIFIER
)
{
aliasName
=
currentToken
;
read
();
schemaName
=
session
.
getCurrentSchemaName
();
}
else
{
aliasName
=
readIdentifierWithSchema
();
}
final
boolean
newAliasSameNameAsBuiltin
=
Function
.
getFunction
(
database
,
aliasName
)
!=
null
;
final
boolean
newAliasSameNameAsBuiltin
=
Function
.
getFunction
(
database
,
aliasName
)
!=
null
;
if
(
database
.
isAllowBuiltinAliasOverride
()
&&
newAliasSameNameAsBuiltin
)
{
if
(
database
.
isAllowBuiltinAliasOverride
()
&&
newAliasSameNameAsBuiltin
)
{
// fine
// fine
...
...
h2/src/main/org/h2/engine/Mode.java
浏览文件 @
291b541c
...
@@ -209,12 +209,6 @@ public class Mode {
...
@@ -209,12 +209,6 @@ public class Mode {
*/
*/
public
boolean
dateTimeValueWithinTransaction
;
public
boolean
dateTimeValueWithinTransaction
;
/**
* If {@code true}, ANY and SOME after comparison operators are parsed as
* array comparison operators.
*/
public
boolean
anyAndSomeAreComparisons
;
/**
/**
* An optional Set of hidden/disallowed column types.
* An optional Set of hidden/disallowed column types.
* Certain DBMSs don't support all column types provided by H2, such as
* Certain DBMSs don't support all column types provided by H2, such as
...
@@ -360,7 +354,6 @@ public class Mode {
...
@@ -360,7 +354,6 @@ public class Mode {
dt
.
name
=
"MONEY"
;
dt
.
name
=
"MONEY"
;
mode
.
typeByNameMap
.
put
(
"MONEY"
,
dt
);
mode
.
typeByNameMap
.
put
(
"MONEY"
,
dt
);
mode
.
dateTimeValueWithinTransaction
=
true
;
mode
.
dateTimeValueWithinTransaction
=
true
;
mode
.
anyAndSomeAreComparisons
=
true
;
add
(
mode
);
add
(
mode
);
mode
=
new
Mode
(
ModeEnum
.
Ignite
);
mode
=
new
Mode
(
ModeEnum
.
Ignite
);
...
...
h2/src/main/org/h2/expression/aggregate/Aggregate.java
浏览文件 @
291b541c
...
@@ -19,6 +19,7 @@ import org.h2.engine.Session;
...
@@ -19,6 +19,7 @@ import org.h2.engine.Session;
import
org.h2.expression.Expression
;
import
org.h2.expression.Expression
;
import
org.h2.expression.ExpressionColumn
;
import
org.h2.expression.ExpressionColumn
;
import
org.h2.expression.ExpressionVisitor
;
import
org.h2.expression.ExpressionVisitor
;
import
org.h2.expression.Subquery
;
import
org.h2.expression.analysis.Window
;
import
org.h2.expression.analysis.Window
;
import
org.h2.index.Cursor
;
import
org.h2.index.Cursor
;
import
org.h2.index.Index
;
import
org.h2.index.Index
;
...
@@ -265,6 +266,15 @@ public class Aggregate extends AbstractAggregate {
...
@@ -265,6 +266,15 @@ public class Aggregate extends AbstractAggregate {
this
.
groupConcatSeparator
=
separator
;
this
.
groupConcatSeparator
=
separator
;
}
}
/**
* Returns the type of this aggregate.
*
* @return the type of this aggregate
*/
public
AggregateType
getAggregateType
()
{
return
type
;
}
private
void
sortWithOrderBy
(
Value
[]
array
)
{
private
void
sortWithOrderBy
(
Value
[]
array
)
{
final
SortOrder
sortOrder
=
orderBySort
;
final
SortOrder
sortOrder
=
orderBySort
;
if
(
sortOrder
!=
null
)
{
if
(
sortOrder
!=
null
)
{
...
@@ -812,7 +822,12 @@ public class Aggregate extends AbstractAggregate {
...
@@ -812,7 +822,12 @@ public class Aggregate extends AbstractAggregate {
on
.
getSQL
(
builder
).
append
(
')'
);
on
.
getSQL
(
builder
).
append
(
')'
);
}
else
{
}
else
{
builder
.
append
(
'('
);
builder
.
append
(
'('
);
on
.
getUnenclosedSQL
(
builder
).
append
(
')'
);
if
(
on
instanceof
Subquery
)
{
on
.
getSQL
(
builder
);
}
else
{
on
.
getUnenclosedSQL
(
builder
);
}
builder
.
append
(
')'
);
}
}
return
appendTailConditions
(
builder
);
return
appendTailConditions
(
builder
);
}
}
...
...
h2/src/main/org/h2/expression/condition/Comparison.java
浏览文件 @
291b541c
...
@@ -14,6 +14,8 @@ import org.h2.expression.ExpressionColumn;
...
@@ -14,6 +14,8 @@ import org.h2.expression.ExpressionColumn;
import
org.h2.expression.ExpressionVisitor
;
import
org.h2.expression.ExpressionVisitor
;
import
org.h2.expression.Parameter
;
import
org.h2.expression.Parameter
;
import
org.h2.expression.ValueExpression
;
import
org.h2.expression.ValueExpression
;
import
org.h2.expression.aggregate.Aggregate
;
import
org.h2.expression.aggregate.Aggregate.AggregateType
;
import
org.h2.index.IndexCondition
;
import
org.h2.index.IndexCondition
;
import
org.h2.message.DbException
;
import
org.h2.message.DbException
;
import
org.h2.table.Column
;
import
org.h2.table.Column
;
...
@@ -130,6 +132,7 @@ public class Comparison extends Condition {
...
@@ -130,6 +132,7 @@ public class Comparison extends Condition {
@Override
@Override
public
StringBuilder
getSQL
(
StringBuilder
builder
)
{
public
StringBuilder
getSQL
(
StringBuilder
builder
)
{
boolean
encloseRight
=
false
;
builder
.
append
(
'('
);
builder
.
append
(
'('
);
switch
(
compareType
)
{
switch
(
compareType
)
{
case
IS_NULL:
case
IS_NULL:
...
@@ -143,9 +146,25 @@ public class Comparison extends Condition {
...
@@ -143,9 +146,25 @@ public class Comparison extends Condition {
left
.
getSQL
(
builder
).
append
(
", "
);
left
.
getSQL
(
builder
).
append
(
", "
);
right
.
getSQL
(
builder
).
append
(
')'
);
right
.
getSQL
(
builder
).
append
(
')'
);
break
;
break
;
case
EQUAL:
case
BIGGER_EQUAL:
case
BIGGER:
case
SMALLER_EQUAL:
case
SMALLER:
case
NOT_EQUAL:
if
(
right
instanceof
Aggregate
&&
((
Aggregate
)
right
).
getAggregateType
()
==
AggregateType
.
ANY
)
{
encloseRight
=
true
;
}
//$FALL-THROUGH$
default
:
default
:
left
.
getSQL
(
builder
).
append
(
' '
).
append
(
getCompareOperator
(
compareType
)).
append
(
' '
);
left
.
getSQL
(
builder
).
append
(
' '
).
append
(
getCompareOperator
(
compareType
)).
append
(
' '
);
if
(
encloseRight
)
{
builder
.
append
(
'('
);
}
right
.
getSQL
(
builder
);
right
.
getSQL
(
builder
);
if
(
encloseRight
)
{
builder
.
append
(
')'
);
}
}
}
return
builder
.
append
(
')'
);
return
builder
.
append
(
')'
);
}
}
...
...
h2/src/main/org/h2/expression/condition/ConditionInSelect.java
浏览文件 @
291b541c
...
@@ -5,7 +5,6 @@
...
@@ -5,7 +5,6 @@
*/
*/
package
org
.
h2
.
expression
.
condition
;
package
org
.
h2
.
expression
.
condition
;
import
org.h2.api.ErrorCode
;
import
org.h2.command.dml.Query
;
import
org.h2.command.dml.Query
;
import
org.h2.engine.Database
;
import
org.h2.engine.Database
;
import
org.h2.engine.Session
;
import
org.h2.engine.Session
;
...
@@ -13,13 +12,13 @@ import org.h2.expression.Expression;
...
@@ -13,13 +12,13 @@ import org.h2.expression.Expression;
import
org.h2.expression.ExpressionColumn
;
import
org.h2.expression.ExpressionColumn
;
import
org.h2.expression.ExpressionVisitor
;
import
org.h2.expression.ExpressionVisitor
;
import
org.h2.index.IndexCondition
;
import
org.h2.index.IndexCondition
;
import
org.h2.message.DbException
;
import
org.h2.result.LocalResult
;
import
org.h2.result.LocalResult
;
import
org.h2.result.ResultInterface
;
import
org.h2.result.ResultInterface
;
import
org.h2.table.ColumnResolver
;
import
org.h2.table.ColumnResolver
;
import
org.h2.table.TableFilter
;
import
org.h2.table.TableFilter
;
import
org.h2.util.StringUtils
;
import
org.h2.util.StringUtils
;
import
org.h2.value.Value
;
import
org.h2.value.Value
;
import
org.h2.value.ValueArray
;
import
org.h2.value.ValueBoolean
;
import
org.h2.value.ValueBoolean
;
import
org.h2.value.ValueNull
;
import
org.h2.value.ValueNull
;
...
@@ -63,16 +62,25 @@ public class ConditionInSelect extends Condition {
...
@@ -63,16 +62,25 @@ public class ConditionInSelect extends Condition {
compareType
!=
Comparison
.
EQUAL_NULL_SAFE
))
{
compareType
!=
Comparison
.
EQUAL_NULL_SAFE
))
{
return
getValueSlow
(
rows
,
l
);
return
getValueSlow
(
rows
,
l
);
}
}
int
dataType
=
rows
.
getColumnType
(
0
);
int
columnCount
=
query
.
getColumnCount
();
if
(
dataType
==
Value
.
NULL
)
{
if
(
columnCount
!=
1
)
{
return
ValueBoolean
.
FALSE
;
l
=
l
.
convertTo
(
Value
.
ARRAY
);
}
Value
[]
leftValue
=
((
ValueArray
)
l
).
getList
();
l
=
l
.
convertTo
(
dataType
,
database
.
getMode
());
if
(
columnCount
==
leftValue
.
length
&&
rows
.
containsDistinct
(
leftValue
))
{
if
(
rows
.
containsDistinct
(
new
Value
[]
{
l
}))
{
return
ValueBoolean
.
TRUE
;
return
ValueBoolean
.
TRUE
;
}
}
}
else
{
if
(
rows
.
containsDistinct
(
new
Value
[]
{
ValueNull
.
INSTANCE
}))
{
int
dataType
=
rows
.
getColumnType
(
0
);
return
ValueNull
.
INSTANCE
;
if
(
dataType
==
Value
.
NULL
)
{
return
ValueBoolean
.
FALSE
;
}
l
=
l
.
convertTo
(
dataType
,
database
.
getMode
());
if
(
rows
.
containsDistinct
(
new
Value
[]
{
l
}))
{
return
ValueBoolean
.
TRUE
;
}
if
(
rows
.
containsDistinct
(
new
Value
[]
{
ValueNull
.
INSTANCE
}))
{
return
ValueNull
.
INSTANCE
;
}
}
}
return
ValueBoolean
.
FALSE
;
return
ValueBoolean
.
FALSE
;
}
}
...
@@ -84,7 +92,8 @@ public class ConditionInSelect extends Condition {
...
@@ -84,7 +92,8 @@ public class ConditionInSelect extends Condition {
boolean
result
=
all
;
boolean
result
=
all
;
while
(
rows
.
next
())
{
while
(
rows
.
next
())
{
boolean
value
;
boolean
value
;
Value
r
=
rows
.
currentRow
()[
0
];
Value
[]
currentRow
=
rows
.
currentRow
();
Value
r
=
query
.
getColumnCount
()
==
1
?
currentRow
[
0
]
:
org
.
h2
.
value
.
ValueArray
.
get
(
currentRow
);
if
(
r
==
ValueNull
.
INSTANCE
)
{
if
(
r
==
ValueNull
.
INSTANCE
)
{
value
=
false
;
value
=
false
;
hasNull
=
true
;
hasNull
=
true
;
...
@@ -116,9 +125,6 @@ public class ConditionInSelect extends Condition {
...
@@ -116,9 +125,6 @@ public class ConditionInSelect extends Condition {
left
=
left
.
optimize
(
session
);
left
=
left
.
optimize
(
session
);
query
.
setRandomAccessResult
(
true
);
query
.
setRandomAccessResult
(
true
);
session
.
optimizeQueryExpression
(
query
);
session
.
optimizeQueryExpression
(
query
);
if
(
query
.
getColumnCount
()
!=
1
)
{
throw
DbException
.
get
(
ErrorCode
.
SUBQUERY_IS_NOT_SINGLE_COLUMN
);
}
// Can not optimize: the data may change
// Can not optimize: the data may change
return
this
;
return
this
;
}
}
...
@@ -169,6 +175,12 @@ public class ConditionInSelect extends Condition {
...
@@ -169,6 +175,12 @@ public class ConditionInSelect extends Condition {
if
(!
session
.
getDatabase
().
getSettings
().
optimizeInList
)
{
if
(!
session
.
getDatabase
().
getSettings
().
optimizeInList
)
{
return
;
return
;
}
}
if
(
compareType
!=
Comparison
.
EQUAL
)
{
return
;
}
if
(
query
.
getColumnCount
()
!=
1
)
{
return
;
}
if
(!(
left
instanceof
ExpressionColumn
))
{
if
(!(
left
instanceof
ExpressionColumn
))
{
return
;
return
;
}
}
...
...
h2/src/main/org/h2/expression/function/ToChar.java
浏览文件 @
291b541c
...
@@ -6,6 +6,7 @@
...
@@ -6,6 +6,7 @@
package
org
.
h2
.
expression
.
function
;
package
org
.
h2
.
expression
.
function
;
import
java.math.BigDecimal
;
import
java.math.BigDecimal
;
import
java.math.RoundingMode
;
import
java.text.DateFormatSymbols
;
import
java.text.DateFormatSymbols
;
import
java.text.DecimalFormat
;
import
java.text.DecimalFormat
;
import
java.text.DecimalFormatSymbols
;
import
java.text.DecimalFormatSymbols
;
...
@@ -233,7 +234,7 @@ public class ToChar {
...
@@ -233,7 +234,7 @@ public class ToChar {
int
separator
=
findDecimalSeparator
(
format
);
int
separator
=
findDecimalSeparator
(
format
);
int
formatScale
=
calculateScale
(
format
,
separator
);
int
formatScale
=
calculateScale
(
format
,
separator
);
if
(
formatScale
<
number
.
scale
())
{
if
(
formatScale
<
number
.
scale
())
{
number
=
number
.
setScale
(
formatScale
,
BigDecimal
.
ROUND_
HALF_UP
);
number
=
number
.
setScale
(
formatScale
,
RoundingMode
.
HALF_UP
);
}
}
// any 9s to the left of the decimal separator but to the right of a
// any 9s to the left of the decimal separator but to the right of a
...
@@ -461,7 +462,7 @@ public class ToChar {
...
@@ -461,7 +462,7 @@ public class ToChar {
}
}
}
}
int
i
=
number
.
setScale
(
0
,
BigDecimal
.
ROUND_
HALF_UP
).
intValue
();
int
i
=
number
.
setScale
(
0
,
RoundingMode
.
HALF_UP
).
intValue
();
String
hex
=
Integer
.
toHexString
(
i
);
String
hex
=
Integer
.
toHexString
(
i
);
if
(
digits
<
hex
.
length
())
{
if
(
digits
<
hex
.
length
())
{
hex
=
StringUtils
.
pad
(
""
,
digits
+
1
,
"#"
,
true
);
hex
=
StringUtils
.
pad
(
""
,
digits
+
1
,
"#"
,
true
);
...
...
h2/src/main/org/h2/mvstore/MVMap.java
浏览文件 @
291b541c
...
@@ -1056,7 +1056,7 @@ public class MVMap<K, V> extends AbstractMap<K, V>
...
@@ -1056,7 +1056,7 @@ public class MVMap<K, V> extends AbstractMap<K, V>
return
getVersion
(
getRoot
());
return
getVersion
(
getRoot
());
}
}
private
long
getVersion
(
RootReference
rootReference
)
{
private
static
long
getVersion
(
RootReference
rootReference
)
{
RootReference
previous
=
rootReference
.
previous
;
RootReference
previous
=
rootReference
.
previous
;
return
previous
==
null
||
previous
.
root
!=
rootReference
.
root
||
return
previous
==
null
||
previous
.
root
!=
rootReference
.
root
||
previous
.
appendCounter
!=
rootReference
.
appendCounter
?
previous
.
appendCounter
!=
rootReference
.
appendCounter
?
...
...
h2/src/main/org/h2/mvstore/MVStore.java
浏览文件 @
291b541c
...
@@ -905,6 +905,7 @@ public class MVStore implements AutoCloseable {
...
@@ -905,6 +905,7 @@ public class MVStore implements AutoCloseable {
/**
/**
* Close the file and the store. Unsaved changes are written to disk first.
* Close the file and the store. Unsaved changes are written to disk first.
*/
*/
@Override
public
void
close
()
{
public
void
close
()
{
if
(
closed
)
{
if
(
closed
)
{
return
;
return
;
...
...
h2/src/main/org/h2/mvstore/Page.java
浏览文件 @
291b541c
...
@@ -1236,10 +1236,12 @@ public abstract class Page implements Cloneable
...
@@ -1236,10 +1236,12 @@ public abstract class Page implements Cloneable
}
}
}
}
@Override
public
boolean
isComplete
()
{
public
boolean
isComplete
()
{
return
complete
;
return
complete
;
}
}
@Override
public
void
setComplete
()
{
public
void
setComplete
()
{
recalculateTotalCount
();
recalculateTotalCount
();
complete
=
true
;
complete
=
true
;
...
...
h2/src/main/org/h2/table/Column.java
浏览文件 @
291b541c
...
@@ -279,10 +279,6 @@ public class Column {
...
@@ -279,10 +279,6 @@ public class Column {
return
precision
;
return
precision
;
}
}
public
void
setPrecision
(
long
p
)
{
precision
=
p
;
}
public
int
getDisplaySize
()
{
public
int
getDisplaySize
()
{
return
displaySize
;
return
displaySize
;
}
}
...
@@ -299,10 +295,6 @@ public class Column {
...
@@ -299,10 +295,6 @@ public class Column {
return
extTypeInfo
;
return
extTypeInfo
;
}
}
public
void
setExtTypeInfo
(
ExtTypeInfo
extTypeInfo
)
{
this
.
extTypeInfo
=
extTypeInfo
;
}
public
boolean
getVisible
()
{
public
boolean
getVisible
()
{
return
visible
;
return
visible
;
}
}
...
...
h2/src/main/org/h2/util/ParserUtil.java
浏览文件 @
291b541c
...
@@ -137,10 +137,20 @@ public class ParserUtil {
...
@@ -137,10 +137,20 @@ public class ParserUtil {
*/
*/
public
static
final
int
LIMIT
=
LIKE
+
1
;
public
static
final
int
LIMIT
=
LIKE
+
1
;
/**
* The token "LOCALTIME".
*/
public
static
final
int
LOCALTIME
=
LIMIT
+
1
;
/**
* The token "LOCALTIMESTAMP".
*/
public
static
final
int
LOCALTIMESTAMP
=
LOCALTIME
+
1
;
/**
/**
* The token "MINUS".
* The token "MINUS".
*/
*/
public
static
final
int
MINUS
=
L
IMIT
+
1
;
public
static
final
int
MINUS
=
L
OCALTIMESTAMP
+
1
;
/**
/**
* The token "NATURAL".
* The token "NATURAL".
...
@@ -311,12 +321,12 @@ public class ParserUtil {
...
@@ -311,12 +321,12 @@ public class ParserUtil {
return
CONSTRAINT
;
return
CONSTRAINT
;
}
else
if
(
eq
(
"CROSS"
,
s
,
ignoreCase
,
start
,
end
))
{
}
else
if
(
eq
(
"CROSS"
,
s
,
ignoreCase
,
start
,
end
))
{
return
CROSS
;
return
CROSS
;
}
}
else
if
(
eq
(
"CURRENT_DATE"
,
s
,
ignoreCase
,
start
,
end
))
{
if
(
additionalKeywords
)
{
return
CURRENT_DATE
;
if
(
eq
(
"CURRENT_DATE"
,
s
,
ignoreCase
,
start
,
end
)
||
eq
(
"CURRENT_TIME"
,
s
,
ignoreCase
,
start
,
end
)
}
else
if
(
eq
(
"CURRENT_TIME"
,
s
,
ignoreCase
,
start
,
end
))
{
||
eq
(
"CURRENT_TIMESTAMP"
,
s
,
ignoreCase
,
start
,
end
))
{
return
CURRENT_TIME
;
return
KEYWORD
;
}
else
if
(
eq
(
"CURRENT_TIMESTAMP"
,
s
,
ignoreCase
,
start
,
end
))
{
}
return
CURRENT_TIMESTAMP
;
}
}
return
IDENTIFIER
;
return
IDENTIFIER
;
case
'D'
:
case
'D'
:
...
@@ -380,11 +390,10 @@ public class ParserUtil {
...
@@ -380,11 +390,10 @@ public class ParserUtil {
return
LIMIT
;
return
LIMIT
;
}
else
if
(
eq
(
"LIKE"
,
s
,
ignoreCase
,
start
,
end
))
{
}
else
if
(
eq
(
"LIKE"
,
s
,
ignoreCase
,
start
,
end
))
{
return
LIKE
;
return
LIKE
;
}
}
else
if
(
eq
(
"LOCALTIME"
,
s
,
ignoreCase
,
start
,
end
))
{
if
(
additionalKeywords
)
{
return
LOCALTIME
;
if
(
eq
(
"LOCALTIME"
,
s
,
ignoreCase
,
start
,
end
)
||
eq
(
"LOCALTIMESTAMP"
,
s
,
ignoreCase
,
start
,
end
))
{
}
else
if
(
eq
(
"LOCALTIMESTAMP"
,
s
,
ignoreCase
,
start
,
end
))
{
return
KEYWORD
;
return
LOCALTIMESTAMP
;
}
}
}
return
IDENTIFIER
;
return
IDENTIFIER
;
case
'M'
:
case
'M'
:
...
...
h2/src/main/org/h2/value/Value.java
浏览文件 @
291b541c
...
@@ -11,6 +11,7 @@ import java.io.Reader;
...
@@ -11,6 +11,7 @@ import java.io.Reader;
import
java.io.StringReader
;
import
java.io.StringReader
;
import
java.lang.ref.SoftReference
;
import
java.lang.ref.SoftReference
;
import
java.math.BigDecimal
;
import
java.math.BigDecimal
;
import
java.math.RoundingMode
;
import
java.nio.charset.StandardCharsets
;
import
java.nio.charset.StandardCharsets
;
import
java.sql.Date
;
import
java.sql.Date
;
import
java.sql.PreparedStatement
;
import
java.sql.PreparedStatement
;
...
@@ -1431,7 +1432,7 @@ public abstract class Value {
...
@@ -1431,7 +1432,7 @@ public abstract class Value {
throw
DbException
.
get
(
throw
DbException
.
get
(
ErrorCode
.
NUMERIC_VALUE_OUT_OF_RANGE_2
,
x
.
toString
(),
getColumnName
(
column
));
ErrorCode
.
NUMERIC_VALUE_OUT_OF_RANGE_2
,
x
.
toString
(),
getColumnName
(
column
));
}
}
return
x
.
setScale
(
0
,
BigDecimal
.
ROUND_
HALF_UP
).
longValue
();
return
x
.
setScale
(
0
,
RoundingMode
.
HALF_UP
).
longValue
();
}
}
private
static
String
getColumnName
(
Object
column
)
{
private
static
String
getColumnName
(
Object
column
)
{
...
...
h2/src/main/org/h2/value/ValueDecimal.java
浏览文件 @
291b541c
...
@@ -6,6 +6,7 @@
...
@@ -6,6 +6,7 @@
package
org
.
h2
.
value
;
package
org
.
h2
.
value
;
import
java.math.BigDecimal
;
import
java.math.BigDecimal
;
import
java.math.RoundingMode
;
import
java.sql.PreparedStatement
;
import
java.sql.PreparedStatement
;
import
java.sql.SQLException
;
import
java.sql.SQLException
;
...
@@ -95,7 +96,7 @@ public class ValueDecimal extends Value {
...
@@ -95,7 +96,7 @@ public class ValueDecimal extends Value {
}
}
BigDecimal
bd
=
value
.
divide
(
dec
.
value
,
BigDecimal
bd
=
value
.
divide
(
dec
.
value
,
value
.
scale
()
+
DIVIDE_SCALE_ADD
,
value
.
scale
()
+
DIVIDE_SCALE_ADD
,
BigDecimal
.
ROUND_
HALF_DOWN
);
RoundingMode
.
HALF_DOWN
);
if
(
bd
.
signum
()
==
0
)
{
if
(
bd
.
signum
()
==
0
)
{
bd
=
BigDecimal
.
ZERO
;
bd
=
BigDecimal
.
ZERO
;
}
else
if
(
bd
.
scale
()
>
0
)
{
}
else
if
(
bd
.
scale
()
>
0
)
{
...
@@ -265,7 +266,7 @@ public class ValueDecimal extends Value {
...
@@ -265,7 +266,7 @@ public class ValueDecimal extends Value {
if
(
scale
>
BIG_DECIMAL_SCALE_MAX
||
scale
<
-
BIG_DECIMAL_SCALE_MAX
)
{
if
(
scale
>
BIG_DECIMAL_SCALE_MAX
||
scale
<
-
BIG_DECIMAL_SCALE_MAX
)
{
throw
DbException
.
getInvalidValueException
(
"scale"
,
scale
);
throw
DbException
.
getInvalidValueException
(
"scale"
,
scale
);
}
}
return
bd
.
setScale
(
scale
,
BigDecimal
.
ROUND_
HALF_UP
);
return
bd
.
setScale
(
scale
,
RoundingMode
.
HALF_UP
);
}
}
}
}
h2/src/test/org/h2/test/bench/BenchCThread.java
浏览文件 @
291b541c
...
@@ -6,6 +6,7 @@
...
@@ -6,6 +6,7 @@
package
org
.
h2
.
test
.
bench
;
package
org
.
h2
.
test
.
bench
;
import
java.math.BigDecimal
;
import
java.math.BigDecimal
;
import
java.math.RoundingMode
;
import
java.sql.PreparedStatement
;
import
java.sql.PreparedStatement
;
import
java.sql.ResultSet
;
import
java.sql.ResultSet
;
import
java.sql.SQLException
;
import
java.sql.SQLException
;
...
@@ -214,7 +215,7 @@ public class BenchCThread {
...
@@ -214,7 +215,7 @@ public class BenchCThread {
BigDecimal
olAmount
=
new
BigDecimal
(
olQuantity
).
multiply
(
BigDecimal
olAmount
=
new
BigDecimal
(
olQuantity
).
multiply
(
price
).
multiply
(
ONE
.
add
(
wTax
).
add
(
tax
)).
multiply
(
price
).
multiply
(
ONE
.
add
(
wTax
).
add
(
tax
)).
multiply
(
ONE
.
subtract
(
discount
));
ONE
.
subtract
(
discount
));
olAmount
=
olAmount
.
setScale
(
2
,
BigDecimal
.
ROUND_
HALF_UP
);
olAmount
=
olAmount
.
setScale
(
2
,
RoundingMode
.
HALF_UP
);
amt
[
number
-
1
]
=
olAmount
;
amt
[
number
-
1
]
=
olAmount
;
total
=
total
.
add
(
olAmount
);
total
=
total
.
add
(
olAmount
);
prep
=
prepare
(
"INSERT INTO ORDER_LINE (OL_O_ID, OL_D_ID, OL_W_ID, OL_NUMBER, "
prep
=
prepare
(
"INSERT INTO ORDER_LINE (OL_O_ID, OL_D_ID, OL_W_ID, OL_NUMBER, "
...
...
h2/src/test/org/h2/test/db/TestLIRSMemoryConsumption.java
浏览文件 @
291b541c
...
@@ -102,7 +102,7 @@ public class TestLIRSMemoryConsumption extends TestDb {
...
@@ -102,7 +102,7 @@ public class TestLIRSMemoryConsumption extends TestDb {
return
2560
;
return
2560
;
}
}
private
long
getMemUsedKb
()
{
private
static
long
getMemUsedKb
()
{
Runtime
rt
=
Runtime
.
getRuntime
();
Runtime
rt
=
Runtime
.
getRuntime
();
long
memory
=
Long
.
MAX_VALUE
;
long
memory
=
Long
.
MAX_VALUE
;
for
(
int
i
=
0
;
i
<
8
;
i
++)
{
for
(
int
i
=
0
;
i
<
8
;
i
++)
{
...
...
h2/src/test/org/h2/test/db/TestPersistentCommonTableExpressions.java
浏览文件 @
291b541c
...
@@ -42,7 +42,7 @@ public class TestPersistentCommonTableExpressions extends AbstractBaseForCommonT
...
@@ -42,7 +42,7 @@ public class TestPersistentCommonTableExpressions extends AbstractBaseForCommonT
String
[]
expectedRowData
=
new
String
[]{
"|meat|null"
,
"|fruit|3"
,
"|veg|2"
};
String
[]
expectedRowData
=
new
String
[]{
"|meat|null"
,
"|fruit|3"
,
"|veg|2"
};
String
[]
expectedColumnTypes
=
new
String
[]{
"VARCHAR"
,
numericName
};
String
[]
expectedColumnTypes
=
new
String
[]{
"VARCHAR"
,
numericName
};
String
[]
expectedColumnNames
=
new
String
[]{
"VAL"
,
String
[]
expectedColumnNames
=
new
String
[]{
"VAL"
,
"SUM(SELECT\n"
+
"SUM(
(
SELECT\n"
+
" X\n"
+
" X\n"
+
"FROM PUBLIC.\"\" BB\n"
+
"FROM PUBLIC.\"\" BB\n"
+
" /* SELECT\n"
+
" /* SELECT\n"
+
...
@@ -61,7 +61,7 @@ public class TestPersistentCommonTableExpressions extends AbstractBaseForCommonT
...
@@ -61,7 +61,7 @@ public class TestPersistentCommonTableExpressions extends AbstractBaseForCommonT
" GROUP BY A: A IS A.VAL\n"
+
" GROUP BY A: A IS A.VAL\n"
+
" */\n"
+
" */\n"
+
" /* scanCount: 1 */\n"
+
" /* scanCount: 1 */\n"
+
"WHERE BB.A IS A.VAL)"
};
"WHERE BB.A IS A.VAL)
)
"
};
String
setupSQL
=
String
setupSQL
=
"DROP TABLE IF EXISTS A; "
"DROP TABLE IF EXISTS A; "
...
@@ -92,7 +92,7 @@ public class TestPersistentCommonTableExpressions extends AbstractBaseForCommonT
...
@@ -92,7 +92,7 @@ public class TestPersistentCommonTableExpressions extends AbstractBaseForCommonT
"GROUP BY a) \n"
+
"GROUP BY a) \n"
+
"SELECT \n"
+
"SELECT \n"
+
"A.val, \n"
+
"A.val, \n"
+
"sum(
SELECT X FROM BB WHERE BB.a IS A.val
)\n"
+
"sum(
(SELECT X FROM BB WHERE BB.a IS A.val)
)\n"
+
"FROM A \n"
+
"GROUP BY A.val"
;
"FROM A \n"
+
"GROUP BY A.val"
;
int
maxRetries
=
3
;
int
maxRetries
=
3
;
int
expectedNumberOfRows
=
expectedRowData
.
length
;
int
expectedNumberOfRows
=
expectedRowData
.
length
;
...
...
h2/src/test/org/h2/test/jdbc/TestPreparedStatement.java
浏览文件 @
291b541c
...
@@ -1681,7 +1681,6 @@ public class TestPreparedStatement extends TestDb {
...
@@ -1681,7 +1681,6 @@ public class TestPreparedStatement extends TestDb {
anyParameterCheck
(
ps
,
300
,
new
int
[]
{
30
});
anyParameterCheck
(
ps
,
300
,
new
int
[]
{
30
});
anyParameterCheck
(
ps
,
-
5
,
new
int
[
0
]);
anyParameterCheck
(
ps
,
-
5
,
new
int
[
0
]);
// Test expression = ANY(?)
// Test expression = ANY(?)
conn
.
createStatement
().
execute
(
"SET MODE PostgreSQL"
);
ps
=
conn
.
prepareStatement
(
"SELECT ID FROM TEST WHERE VALUE = ANY(?)"
);
ps
=
conn
.
prepareStatement
(
"SELECT ID FROM TEST WHERE VALUE = ANY(?)"
);
assertThrows
(
ErrorCode
.
PARAMETER_NOT_SET_1
,
ps
).
executeQuery
();
assertThrows
(
ErrorCode
.
PARAMETER_NOT_SET_1
,
ps
).
executeQuery
();
anyParameterCheck
(
ps
,
values
,
expected
);
anyParameterCheck
(
ps
,
values
,
expected
);
...
...
h2/src/test/org/h2/test/scripts/dml/select.sql
浏览文件 @
291b541c
...
@@ -439,3 +439,27 @@ SELECT 1 FROM TEST HAVING TRUE FOR UPDATE;
...
@@ -439,3 +439,27 @@ SELECT 1 FROM TEST HAVING TRUE FOR UPDATE;
DROP
TABLE
TEST
;
DROP
TABLE
TEST
;
>
ok
>
ok
CREATE
TABLE
TEST
(
ID
INT
PRIMARY
KEY
,
V
INT
)
AS
SELECT
X
,
X
+
1
FROM
SYSTEM_RANGE
(
1
,
3
);
>
ok
SELECT
ID
FROM
TEST
WHERE
ID
!=
ALL
(
SELECT
ID
FROM
TEST
WHERE
ID
IN
(
1
,
3
));
>
ID
>
--
>
2
>
rows
:
1
SELECT
(
1
,
3
)
>
ANY
(
SELECT
ID
,
V
FROM
TEST
);
>>
TRUE
SELECT
(
1
,
2
)
>
ANY
(
SELECT
ID
,
V
FROM
TEST
);
>>
FALSE
SELECT
(
2
,
3
)
=
ANY
(
SELECT
ID
,
V
FROM
TEST
);
>>
TRUE
SELECT
(
3
,
4
)
>
ALL
(
SELECT
ID
,
V
FROM
TEST
);
>>
FALSE
DROP
TABLE
TEST
;
>
ok
h2/src/test/org/h2/test/scripts/functions/aggregate/any.sql
浏览文件 @
291b541c
...
@@ -19,3 +19,15 @@ SELECT A, ANY(B < 2), SOME(B > 3), BOOL_OR(B = 1), ANY(B = 1) FILTER (WHERE A =
...
@@ -19,3 +19,15 @@ SELECT A, ANY(B < 2), SOME(B > 3), BOOL_OR(B = 1), ANY(B = 1) FILTER (WHERE A =
DROP
TABLE
TEST
;
DROP
TABLE
TEST
;
>
ok
>
ok
SELECT
TRUE
=
(
ANY
((
SELECT
TRUE
)));
>
TRUE
=
(
ANY
((
SELECT
TRUE
FROM
SYSTEM_RANGE
(
1
,
1
)
/* PUBLIC.RANGE_INDEX */
/* scanCount: 2 */
)))
>
-----------------------------------------------------------------------------------------------
>
TRUE
>
rows
:
1
SELECT
TRUE
=
(
ANY
((
SELECT
FALSE
)));
>
TRUE
=
(
ANY
((
SELECT
FALSE
FROM
SYSTEM_RANGE
(
1
,
1
)
/* PUBLIC.RANGE_INDEX */
/* scanCount: 2 */
)))
>
------------------------------------------------------------------------------------------------
>
FALSE
>
rows
:
1
h2/src/test/org/h2/test/scripts/testScript.sql
浏览文件 @
291b541c
...
@@ -6513,9 +6513,6 @@ SELECT * FROM CUSTOMER WHERE NAME NOT IN(SELECT NAME FROM CUSTOMER);
...
@@ -6513,9 +6513,6 @@ SELECT * FROM CUSTOMER WHERE NAME NOT IN(SELECT NAME FROM CUSTOMER);
>
-- ----
>
-- ----
>
rows
:
0
>
rows
:
0
SET
MODE
PostgreSQL
;
>
ok
SELECT
*
FROM
CUSTOMER
WHERE
NAME
=
ANY
(
SELECT
NAME
FROM
CUSTOMER
);
SELECT
*
FROM
CUSTOMER
WHERE
NAME
=
ANY
(
SELECT
NAME
FROM
CUSTOMER
);
>
ID
NAME
>
ID
NAME
>
-- -------
>
-- -------
...
@@ -6548,9 +6545,6 @@ SELECT * FROM CUSTOMER WHERE NAME < ANY(SELECT NAME FROM CUSTOMER);
...
@@ -6548,9 +6545,6 @@ SELECT * FROM CUSTOMER WHERE NAME < ANY(SELECT NAME FROM CUSTOMER);
>
2
Meier
>
2
Meier
>
rows
:
2
>
rows
:
2
SET
MODE
Regular
;
>
ok
DROP
TABLE
INVOICE
;
DROP
TABLE
INVOICE
;
>
ok
>
ok
...
...
h2/src/test/org/h2/test/unit/TestSubqueryPerformanceOnLazyExecutionMode.java
浏览文件 @
291b541c
...
@@ -135,8 +135,8 @@ public class TestSubqueryPerformanceOnLazyExecutionMode extends TestDb {
...
@@ -135,8 +135,8 @@ public class TestSubqueryPerformanceOnLazyExecutionMode extends TestDb {
}
}
if
(
failCnt
>
successCnt
)
{
if
(
failCnt
>
successCnt
)
{
fail
(
"Lazy execution too slow. Avg lazy time: "
fail
(
"Lazy execution too slow. Avg lazy time: "
+
(
totalLazy
/
FAIL_REPEATS
)
+
", avg not lazy time: "
+
(
totalNotLazy
/
FAIL_REPEATS
));
+
(
totalLazy
/
FAIL_REPEATS
)
+
", avg not lazy time: "
+
(
totalNotLazy
/
FAIL_REPEATS
));
}
}
}
}
...
...
h2/src/tools/org/h2/build/doc/dictionary.txt
浏览文件 @
291b541c
...
@@ -804,4 +804,4 @@ qualification opportunity jumping exploited unacceptable vrs duplicated
...
@@ -804,4 +804,4 @@ qualification opportunity jumping exploited unacceptable vrs duplicated
queryparser tokenized freeze factorings recompilation unenclosed rfe dsync
queryparser tokenized freeze factorings recompilation unenclosed rfe dsync
econd irst bcef ordinality nord unnest
econd irst bcef ordinality nord unnest
analyst occupation distributive josaph aor engineer sajeewa isuru randil kevin doctor businessman artist ashan
analyst occupation distributive josaph aor engineer sajeewa isuru randil kevin doctor businessman artist ashan
corrupts splitted disruption unintentional octets preconditions
corrupts splitted disruption unintentional octets preconditions
predicates subq
编写
预览
Markdown
格式
0%
重试
或
添加新文件
添加附件
取消
您添加了
0
人
到此讨论。请谨慎行事。
请先完成此评论的编辑!
取消
请
注册
或者
登录
后发表评论