Skip to content
项目
群组
代码片段
帮助
正在加载...
帮助
为 GitLab 提交贡献
登录/注册
切换导航
H
h2database
项目
项目
详情
活动
周期分析
仓库
仓库
文件
提交
分支
标签
贡献者
分枝图
比较
统计图
议题
0
议题
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
CI / CD
CI / CD
流水线
作业
计划
统计图
Wiki
Wiki
代码片段
代码片段
成员
成员
折叠边栏
关闭边栏
活动
分枝图
统计图
创建新议题
作业
提交
议题看板
打开侧边栏
Administrator
h2database
Commits
1e462004
Unverified
提交
1e462004
authored
6 年前
作者:
Evgenij Ryazanov
提交者:
GitHub
6 年前
浏览文件
操作
浏览文件
下载
差异文件
Merge pull request #1287 from katzyn/distinct
DISTINCT does not work with ORDER BY in some queries
上级
e8cf7005
2f55c6cf
隐藏空白字符变更
内嵌
并排
正在显示
8 个修改的文件
包含
232 行增加
和
18 行删除
+232
-18
Query.java
h2/src/main/org/h2/command/dml/Query.java
+69
-2
Comparison.java
h2/src/main/org/h2/expression/Comparison.java
+13
-6
ConditionAndOr.java
h2/src/main/org/h2/expression/ConditionAndOr.java
+13
-6
ConditionNot.java
h2/src/main/org/h2/expression/ConditionNot.java
+9
-0
Operation.java
h2/src/main/org/h2/expression/Operation.java
+18
-0
FullText.java
h2/src/main/org/h2/fulltext/FullText.java
+4
-4
TestScript.java
h2/src/test/org/h2/test/scripts/TestScript.java
+1
-0
distinct.sql
h2/src/test/org/h2/test/scripts/distinct.sql
+105
-0
没有找到文件。
h2/src/main/org/h2/command/dml/Query.java
浏览文件 @
1e462004
...
...
@@ -12,10 +12,16 @@ import org.h2.api.ErrorCode;
import
org.h2.command.Prepared
;
import
org.h2.engine.Database
;
import
org.h2.engine.Session
;
import
org.h2.engine.Mode.ModeEnum
;
import
org.h2.expression.Alias
;
import
org.h2.expression.Comparison
;
import
org.h2.expression.ConditionAndOr
;
import
org.h2.expression.ConditionNot
;
import
org.h2.expression.Expression
;
import
org.h2.expression.ExpressionColumn
;
import
org.h2.expression.ExpressionVisitor
;
import
org.h2.expression.Function
;
import
org.h2.expression.Operation
;
import
org.h2.expression.Parameter
;
import
org.h2.expression.ValueExpression
;
import
org.h2.message.DbException
;
...
...
@@ -478,8 +484,11 @@ public abstract class Query extends Prepared {
}
if
(!
isAlias
)
{
if
(
mustBeInResult
)
{
throw
DbException
.
get
(
ErrorCode
.
ORDER_BY_NOT_IN_RESULT
,
e
.
getSQL
());
if
(
session
.
getDatabase
().
getMode
().
getEnum
()
!=
ModeEnum
.
MySQL
)
{
if
(!
checkOrderOther
(
session
,
e
,
expressionSQL
))
{
throw
DbException
.
get
(
ErrorCode
.
ORDER_BY_NOT_IN_RESULT
,
e
.
getSQL
());
}
}
}
expressions
.
add
(
e
);
String
sql
=
e
.
getSQL
();
...
...
@@ -490,6 +499,64 @@ public abstract class Query extends Prepared {
}
}
/**
* An additional check for expression in ORDER BY list for DISTINCT selects
* that was not matched with selected expressions in regular way. This
* method allows expressions based only on selected expressions in different
* complicated ways with functions, comparisons, or operators.
*
* @param session session
* @param expr expression to check
* @param expressionSQL SQL of allowed expressions
* @return whether the specified expression should be allowed in ORDER BY
* list of DISTINCT select
*/
private
static
boolean
checkOrderOther
(
Session
session
,
Expression
expr
,
ArrayList
<
String
>
expressionSQL
)
{
if
(
expr
.
isConstant
())
{
return
true
;
}
if
(
expressionSQL
!=
null
)
{
String
exprSQL
=
expr
.
getSQL
();
for
(
String
sql:
expressionSQL
)
{
if
(
session
.
getDatabase
().
equalsIdentifiers
(
exprSQL
,
sql
))
{
return
true
;
}
}
}
if
(
expr
instanceof
Function
)
{
Function
function
=
(
Function
)
expr
;
if
(!
function
.
isDeterministic
())
{
return
false
;
}
for
(
Expression
e
:
function
.
getArgs
())
{
if
(!
checkOrderOther
(
session
,
e
,
expressionSQL
))
{
return
false
;
}
}
return
true
;
}
if
(
expr
instanceof
Operation
)
{
Operation
operation
=
(
Operation
)
expr
;
Expression
right
=
operation
.
getRightSubExpression
();
return
checkOrderOther
(
session
,
operation
.
getLeftSubExpression
(),
expressionSQL
)
&&
(
right
==
null
||
checkOrderOther
(
session
,
right
,
expressionSQL
));
}
if
(
expr
instanceof
ConditionAndOr
)
{
ConditionAndOr
condition
=
(
ConditionAndOr
)
expr
;
return
checkOrderOther
(
session
,
condition
.
getLeftSubExpression
(),
expressionSQL
)
&&
checkOrderOther
(
session
,
condition
.
getRightSubExpression
(),
expressionSQL
);
}
if
(
expr
instanceof
ConditionNot
)
{
return
checkOrderOther
(
session
,
((
ConditionNot
)
expr
).
getSubCondition
(),
expressionSQL
);
}
if
(
expr
instanceof
Comparison
)
{
Comparison
condition
=
(
Comparison
)
expr
;
return
checkOrderOther
(
session
,
condition
.
getLeftSubExpression
(),
expressionSQL
)
&&
checkOrderOther
(
session
,
condition
.
getRightSubExpression
(),
expressionSQL
);
}
return
false
;
}
/**
* Create a {@link SortOrder} object given the list of {@link SelectOrderBy}
* objects. The expression list is extended if necessary.
...
...
This diff is collapsed.
Click to expand it.
h2/src/main/org/h2/expression/Comparison.java
浏览文件 @
1e462004
...
...
@@ -597,14 +597,21 @@ public class Comparison extends Condition {
}
/**
* Get the left
or the right
sub-expression of this condition.
* Get the left sub-expression of this condition.
*
* @param getLeft true to get the left sub-expression, false to get the
* right sub-expression.
* @return the sub-expression
* @return the left sub-expression
*/
public
Expression
getExpression
(
boolean
getLeft
)
{
return
getLeft
?
this
.
left
:
right
;
public
Expression
getLeftSubExpression
()
{
return
left
;
}
/**
* Get the right sub-expression of this condition.
*
* @return the right sub-expression
*/
public
Expression
getRightSubExpression
()
{
return
right
;
}
}
This diff is collapsed.
Click to expand it.
h2/src/main/org/h2/expression/ConditionAndOr.java
浏览文件 @
1e462004
...
...
@@ -284,14 +284,21 @@ public class ConditionAndOr extends Condition {
}
/**
* Get the left
or the right
sub-expression of this condition.
* Get the left sub-expression of this condition.
*
* @param getLeft true to get the left sub-expression, false to get the
* right sub-expression.
* @return the sub-expression
* @return the left sub-expression
*/
public
Expression
getExpression
(
boolean
getLeft
)
{
return
getLeft
?
this
.
left
:
right
;
public
Expression
getLeftSubExpression
()
{
return
left
;
}
/**
* Get the right sub-expression of this condition.
*
* @return the right sub-expression
*/
public
Expression
getRightSubExpression
()
{
return
right
;
}
}
This diff is collapsed.
Click to expand it.
h2/src/main/org/h2/expression/ConditionNot.java
浏览文件 @
1e462004
...
...
@@ -98,4 +98,13 @@ public class ConditionNot extends Condition {
return
condition
.
getCost
();
}
/**
* Get the sub-expression of this condition.
*
* @return the sub-expression
*/
public
Expression
getSubCondition
()
{
return
condition
;
}
}
This diff is collapsed.
Click to expand it.
h2/src/main/org/h2/expression/Operation.java
浏览文件 @
1e462004
...
...
@@ -407,4 +407,22 @@ public class Operation extends Expression {
return
left
.
getCost
()
+
1
+
(
right
==
null
?
0
:
right
.
getCost
());
}
/**
* Get the left sub-expression of this operation.
*
* @return the left sub-expression
*/
public
Expression
getLeftSubExpression
()
{
return
left
;
}
/**
* Get the right sub-expression of this operation.
*
* @return the right sub-expression
*/
public
Expression
getRightSubExpression
()
{
return
right
;
}
}
This diff is collapsed.
Click to expand it.
h2/src/main/org/h2/fulltext/FullText.java
浏览文件 @
1e462004
...
...
@@ -669,14 +669,14 @@ public class FullText {
ArrayList
<
String
>
data
,
Expression
expr
)
{
if
(
expr
instanceof
ConditionAndOr
)
{
ConditionAndOr
and
=
(
ConditionAndOr
)
expr
;
Expression
left
=
and
.
get
Expression
(
true
);
Expression
right
=
and
.
get
Expression
(
false
);
Expression
left
=
and
.
get
LeftSubExpression
(
);
Expression
right
=
and
.
get
RightSubExpression
(
);
addColumnData
(
columns
,
data
,
left
);
addColumnData
(
columns
,
data
,
right
);
}
else
{
Comparison
comp
=
(
Comparison
)
expr
;
ExpressionColumn
ec
=
(
ExpressionColumn
)
comp
.
get
Expression
(
true
);
ValueExpression
ev
=
(
ValueExpression
)
comp
.
get
Expression
(
false
);
ExpressionColumn
ec
=
(
ExpressionColumn
)
comp
.
get
LeftSubExpression
(
);
ValueExpression
ev
=
(
ValueExpression
)
comp
.
get
RightSubExpression
(
);
String
columnName
=
ec
.
getColumnName
();
columns
.
add
(
columnName
);
if
(
ev
==
null
)
{
...
...
This diff is collapsed.
Click to expand it.
h2/src/test/org/h2/test/scripts/TestScript.java
浏览文件 @
1e462004
...
...
@@ -99,6 +99,7 @@ public class TestScript extends TestDb {
testScript
(
"testScript.sql"
);
testScript
(
"comments.sql"
);
testScript
(
"derived-column-names.sql"
);
testScript
(
"distinct.sql"
);
testScript
(
"dual.sql"
);
testScript
(
"indexes.sql"
);
testScript
(
"information_schema.sql"
);
...
...
This diff is collapsed.
Click to expand it.
h2/src/test/org/h2/test/scripts/distinct.sql
0 → 100644
浏览文件 @
1e462004
-- Copyright 2004-2018 H2 Group. Multiple-Licensed under the MPL 2.0,
-- and the EPL 1.0 (http://h2database.com/html/license.html).
-- Initial Developer: H2 Group
--
CREATE
TABLE
TEST
(
ID
BIGINT
,
NAME
VARCHAR
);
>
ok
INSERT
INTO
TEST
VALUES
(
1
,
'a'
),
(
2
,
'B'
),
(
3
,
'c'
),
(
1
,
'a'
);
>
update
count
:
4
CREATE
TABLE
TEST2
(
ID2
BIGINT
);
>
ok
INSERT
INTO
TEST2
VALUES
(
1
),
(
2
);
>
update
count
:
2
SELECT
DISTINCT
NAME
FROM
TEST
ORDER
BY
NAME
;
>
NAME
>
----
>
B
>
a
>
c
>
rows
(
ordered
):
3
SELECT
DISTINCT
NAME
FROM
TEST
ORDER
BY
LOWER
(
NAME
);
>
NAME
>
----
>
a
>
B
>
c
>
rows
(
ordered
):
3
SELECT
DISTINCT
ID
FROM
TEST
ORDER
BY
ID
;
>
ID
>
--
>
1
>
2
>
3
>
rows
(
ordered
):
3
SELECT
DISTINCT
ID
FROM
TEST
ORDER
BY
-
ID
-
1
;
>
ID
>
--
>
3
>
2
>
1
>
rows
(
ordered
):
3
SELECT
DISTINCT
ID
FROM
TEST
ORDER
BY
(
-
ID
+
10
)
>
0
AND
NOT
(
ID
=
0
),
ID
;
>
ID
>
--
>
1
>
2
>
3
>
rows
(
ordered
):
3
SELECT
DISTINCT
NAME
,
ID
+
1
FROM
TEST
ORDER
BY
UPPER
(
NAME
)
||
(
ID
+
1
);
>
NAME
ID
+
1
>
---- ------
>
a
2
>
B
3
>
c
4
>
rows
(
ordered
):
3
SELECT
DISTINCT
ID
FROM
TEST
ORDER
BY
NAME
;
>
exception
ORDER_BY_NOT_IN_RESULT
SELECT
DISTINCT
ID
FROM
TEST
ORDER
BY
CURRENT_TIMESTAMP
;
>
exception
ORDER_BY_NOT_IN_RESULT
SET
MODE
MySQL
;
>
ok
SELECT
DISTINCT
ID
FROM
TEST
ORDER
BY
NAME
;
>
ID
>
--
>
2
>
1
>
3
>
rows
(
ordered
):
3
SELECT
DISTINCT
ID
FROM
TEST
ORDER
BY
LOWER
(
NAME
);
>
ID
>
--
>
1
>
2
>
3
>
rows
(
ordered
):
3
SELECT
DISTINCT
ID
FROM
TEST
JOIN
TEST2
ON
ID
=
ID2
ORDER
BY
LOWER
(
NAME
);
>
ID
>
--
>
1
>
2
>
rows
(
ordered
):
2
SET
MODE
Regular
;
>
ok
DROP
TABLE
TEST
;
>
ok
DROP
TABLE
TEST2
;
>
ok
This diff is collapsed.
Click to expand it.
编写
预览
Markdown
格式
0%
重试
或
添加新文件
添加附件
取消
您添加了
0
人
到此讨论。请谨慎行事。
请先完成此评论的编辑!
取消
请
注册
或者
登录
后发表评论