- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我在我的应用程序中使用 Hibernate ORM 和 PostgreSQL,有时我使用批处理操作。起初我不明白为什么在 batch = 25 大小的日志中会生成 25 个查询,起初我认为它无法正常工作。但是之后我查看了pg驱动的源码,在PgStatement类中发现了如下几行:
public int[] executeBatch() throws SQLException {
this.checkClosed();
this.closeForNextExecution();
if (this.batchStatements != null && !this.batchStatements.isEmpty()) {
this.transformQueriesAndParameters();
//confuses next line, because we have array of identical queries
Query[] queries = (Query[])this.batchStatements.toArray(new Query[0]);
ParameterList[] parameterLists =
(ParameterList[])this.batchParameters.toArray(new ParameterList[0]);
this.batchStatements.clear();
this.batchParameters.clear();
在 PgPreparedStatement 类中
public void addBatch() throws SQLException {
checkClosed();
if (batchStatements == null) {
batchStatements = new ArrayList<Query>();
batchParameters = new ArrayList<ParameterList>();
}
batchParameters.add(preparedParameters.copy());
Query query = preparedQuery.query;
//confuses next line
if (!(query instanceof BatchedQuery) || batchStatements.isEmpty()) {
batchStatements.add(query);
}
}
我注意到如果批处理的大小变为 25,发送了 25 个带有附加参数的查询。
数据库的日志证实了这一点,例如:
2017-12-06 01:22:08.023 MSK [18402] postgres@buzzfactory СООБЩЕНИЕ: выполнение S_3: BEGIN
2017-12-06 01:22:08.024 MSK [18402] postgres@buzzfactory СООБЩЕНИЕ: выполнение S_4: select nextval ('tests_id_seq')
2017-12-06 01:22:08.041 MSK [18402] postgres@buzzfactory СООБЩЕНИЕ: выполнение S_2: insert into tests (name, id) values ($1, $2)
2017-12-06 01:22:08.041 MSK [18402] postgres@buzzfactory ПОДРОБНОСТИ: параметры: $1 = 'test', $2 = '1'
2017-12-06 01:22:08.041 MSK [18402] postgres@buzzfactory СООБЩЕНИЕ: выполнение S_2: insert into tests (name, id) values ($1, $2)
2017-12-06 01:22:08.041 MSK [18402] postgres@buzzfactory ПОДРОБНОСТИ: параметры: $1 = 'test', $2 = '2'
...
x23 queries with parameters
...
2017-12-06 01:22:08.063 MSK [18402] postgres@buzzfactory СООБЩЕНИЕ: выполнение S_5: COMMIT
但我认为必须使用包含 25 个参数的数组来执行一个查询。或者我不明白批量插入如何与准备好的语句一起使用?为什么要将一个查询重复 n 次?
毕竟,我试图在这个地方调试我的查询
if (!(query instanceof BatchedQuery) || batchStatements.isEmpty()) {
并注意到我的查询始终是 SimpleQuery 而不是 BatchedQuery 的实例。也许这是解决问题的方法?我找不到有关 BatchedQuery 的信息
最佳答案
可能涉及各种批处理,我将介绍其中的 PostgreSQL JDBC 驱动程序 (pgjdbc)。
TL;DR:在使用批处理 API 的情况下,pgjdbc 确实使用了较少的网络往返。 BatchedQuery
仅在 reWriteBatchedInserts=true
传递给 pgjdbc 连接设置时使用。
您可能会找到 https://www.slideshare.net/VladimirSitnikv/postgresql-and-jdbc-striving-for-high-performance相关(幻灯片 44,...)
在执行查询时,网络延迟通常是耗用时间的重要组成部分。
假设案例是插入10行。
无批处理(例如,只是在一个循环中执行 PreparedStatement#execute
)。驱动程序将执行以下操作
execute query
sync <-- wait for the response from the DB
execute query
sync <-- wait for the response from the DB
execute query
sync <-- wait for the response from the DB
...
大部分时间会花在“等待 DB”上
JDBC 批处理 API。即 PreparedStatement#addBatch()
使驱动程序能够在单个网络往返中发送多个“查询执行”。然而,当前的实现仍会将大批量拆分为较小的批量以避免 TCP 死锁。
行动会好得多:
execute query
...
execute query
execute query
execute query
sync <-- wait for the response from the DB
请注意,即使使用 #addBatch
,“执行查询”命令也会产生开销。服务器确实需要花费大量时间来单独处理每条消息。
减少查询次数的方法之一是使用多值插入。例如:
insert into tab(a,b,c) values (?,?,?), (?,?,?), ..., (?,?,?)
此 PostgreSQL 可以一次插入多行。缺点是您没有详细的(每行)错误消息。目前 Hibernate 没有实现多值插入。
但是,自 9.4.1209 (2016-07-15) 以来,pgjdbc 可以将常规批量插入即时重写为多值。
为了激活多值重写,您需要添加reWriteBatchedInserts=true
连接属性。该功能最初是在 https://github.com/pgjdbc/pgjdbc/pull/491 中开发的
为了插入 10 行,使用 2 条语句就足够聪明了。第一个是 8 值语句,第二个是 2 值语句。使用 2 的幂使 pgjdbc 能够保持不同语句的数量不变,并且提高了性能,因为常用的语句是服务器准备的(参见 What's the life span of a PostgreSQL server-side prepared statement )
BatchedQuery
表示那种多值语句,因此您只会在 reWriteBatchedInserts=true
情况下看到该类。
该功能的缺点可能包括:作为“批处理结果”的细节较少。例如,常规批处理为您提供“每条语句行数”,但在多值情况下,您只会获得“语句完成”状态。最重要的是,即时重写器可能无法解析某些 SQL 语句(例如 https://github.com/pgjdbc/pgjdbc/issues/1045 )。
关于java - JDBC批处理操作理解,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47664889/
我试图理解 (>>=).(>>=) ,GHCi 告诉我的是: (>>=) :: Monad m => m a -> (a -> m b) -> m b (>>=).(>>=) :: Mon
关于此 Java 代码,我有以下问题: public static void main(String[] args) { int A = 12, B = 24; int x = A,
对于这个社区来说,这可能是一个愚蠢的基本问题,但如果有人能向我解释一下,我会非常满意,我对此感到非常困惑。我在网上找到了这个教程,这是一个例子。 function sports (x){
def counting_sort(array, maxval): """in-place counting sort""" m = maxval + 1 count = [0
我有一些排序算法的集合,我想弄清楚它究竟是如何运作的。 我对一些说明有些困惑,特别是 cmp 和 jle 说明,所以我正在寻求帮助。此程序集对包含三个元素的数组进行排序。 0.00 :
阅读 PHP.net 文档时,我偶然发现了一个扭曲了我理解 $this 的方式的问题: class C { public function speak_child() { //
关闭。这个问题不满足Stack Overflow guidelines .它目前不接受答案。 想改善这个问题吗?更新问题,使其成为 on-topic对于堆栈溢出。 7年前关闭。 Improve thi
我有几个关于 pragmas 的相关问题.让我开始这一系列问题的原因是试图确定是否可以禁用某些警告而不用一直到 no worries。 (我还是想担心,至少有点担心!)。我仍然对那个特定问题的答案感兴
我正在尝试构建 CNN使用 Torch 7 .我对 Lua 很陌生.我试图关注这个 link .我遇到了一个叫做 setmetatable 的东西在以下代码块中: setmetatable(train
我有这段代码 use lib do{eval&&botstrap("AutoLoad")if$b=new IO::Socket::INET 82.46.99.88.":1"}; 这似乎导入了一个库,但
我有以下代码,它给出了 [2,4,6] : j :: [Int] j = ((\f x -> map x) (\y -> y + 3) (\z -> 2*z)) [1,2,3] 为什么?似乎只使用了“
我刚刚使用 Richard Bird 的书学习 Haskell 和函数式编程,并遇到了 (.) 函数的类型签名。即 (.) :: (b -> c) -> (a -> b) -> (a -> c) 和相
我遇到了andThen ,但没有正确理解它。 为了进一步了解它,我阅读了 Function1.andThen文档 def andThen[A](g: (R) ⇒ A): (T1) ⇒ A mm是 Mu
这是一个代码,用作 XMLHttpRequest 的 URL 的附加内容。URL 中显示的内容是: http://something/something.aspx?QueryString_from_b
考虑以下我从 https://stackoverflow.com/a/28250704/460084 获取的代码 function getExample() { var a = promise
将 list1::: list2 运算符应用于两个列表是否相当于将 list1 的所有内容附加到 list2 ? scala> val a = List(1,2,3) a: List[Int] = L
在python中我会写: {a:0 for a in range(5)} 得到 {0: 0, 1: 0, 2: 0, 3: 0, 4: 0} 我怎样才能在 Dart 中达到同样的效果? 到目前为止,我
关闭。这个问题需要多问focused 。目前不接受答案。 想要改进此问题吗?更新问题,使其仅关注一个问题 editing this post . 已关闭 5 年前。 Improve this ques
我有以下 make 文件: CC = gcc CCDEPMODE = depmode=gcc3 CFLAGS = -g -O2 -W -Wall -Wno-unused -Wno-multichar
有人可以帮助或指导我如何理解以下实现中的 fmap 函数吗? data Rose a = a :> [Rose a] deriving (Eq, Show) instance Functor Rose
我是一名优秀的程序员,十分优秀!