From 64d21372ec688dc009c0be5f102002a74f4ffa46 Mon Sep 17 00:00:00 2001
From: icefairy <860668820@qq.com>
Date: Fri, 14 Jan 2022 12:34:10 +0800
Subject: [PATCH 1/2] =?UTF-8?q?upsert=20=E6=8E=A5=E5=8F=A3=E5=92=8Ch2?=
=?UTF-8?q?=E6=96=B9=E8=A8=80=E7=9A=84upsert=E5=AE=8C=E6=88=90=E5=B9=B6?=
=?UTF-8?q?=E6=B5=8B=E8=AF=95=E9=80=9A=E8=BF=87?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
.../main/java/cn/hutool/db/AbstractDb.java | 30 +++++++--
.../main/java/cn/hutool/db/DialectRunner.java | 31 ++++++++-
.../main/java/cn/hutool/db/SqlConnRunner.java | 1 +
.../java/cn/hutool/db/dialect/Dialect.java | 20 +++++-
.../cn/hutool/db/dialect/impl/H2Dialect.java | 27 ++++++++
.../java/cn/hutool/db/sql/SqlBuilder.java | 67 +++++++++++++++++++
.../src/test/java/cn/hutool/db/H2Test.java | 16 +++--
7 files changed, 179 insertions(+), 13 deletions(-)
diff --git a/hutool-db/src/main/java/cn/hutool/db/AbstractDb.java b/hutool-db/src/main/java/cn/hutool/db/AbstractDb.java
index c18399029..700a92027 100644
--- a/hutool-db/src/main/java/cn/hutool/db/AbstractDb.java
+++ b/hutool-db/src/main/java/cn/hutool/db/AbstractDb.java
@@ -200,9 +200,9 @@ public abstract class AbstractDb implements Serializable {
* 执行自定义的{@link PreparedStatement},结果使用{@link RsHandler}处理
* 此方法主要用于自定义场景,如游标查询等
*
- * @param 结果集需要处理的对象类型
+ * @param 结果集需要处理的对象类型
* @param statementFunc 自定义{@link PreparedStatement}创建函数
- * @param rsh 结果集处理对象
+ * @param rsh 结果集处理对象
* @return 结果对象
* @throws SQLException SQL执行异常
* @since 5.7.17
@@ -369,6 +369,26 @@ public abstract class AbstractDb implements Serializable {
}
}
+ /**
+ * 使用upsert语义插入或更新数据
+ * 根据给定的字段名查询数据,如果存在则更新这些数据,否则执行插入
+ * 如果方言未实现本方法,内部会自动调用insertOrUpdate来实现功能,由于upsert和insert使用有区别,为了兼容性保留原有insertOrUpdate不做变动
+ * @param record 记录
+ * @param keys 需要检查唯一性的字段
+ * @return 插入行数
+ * @throws SQLException SQL执行异常
+ * @since 5.7.21
+ */
+ public int upsert(Entity record, String... keys) throws SQLException {
+ Connection conn = null;
+ try {
+ conn = this.getConnection();
+ return runner.upsert(conn, record, keys);
+ } finally {
+ this.closeConnection(conn);
+ }
+ }
+
/**
* 批量插入数据
* 需要注意的是,批量插入每一条数据结构必须一致。批量插入数据时会获取第一条数据的字段结构,之后的数据会按照这个格式插入。
@@ -864,7 +884,7 @@ public abstract class AbstractDb implements Serializable {
/**
* 分页查询
*
- * @param 处理结果类型,可以将ResultSet转换为给定类型
+ * @param 处理结果类型,可以将ResultSet转换为给定类型
* @param sql SQL构建器
* @param page 分页对象
* @param rsh 结果集处理对象
@@ -884,8 +904,8 @@ public abstract class AbstractDb implements Serializable {
/**
* 分页查询
*
- * @param sql SQL语句字符串
- * @param page 分页对象
+ * @param sql SQL语句字符串
+ * @param page 分页对象
* @param params 参数列表
* @return 结果对象
* @throws SQLException SQL执行异常
diff --git a/hutool-db/src/main/java/cn/hutool/db/DialectRunner.java b/hutool-db/src/main/java/cn/hutool/db/DialectRunner.java
index 43d0574e3..b1003b987 100644
--- a/hutool-db/src/main/java/cn/hutool/db/DialectRunner.java
+++ b/hutool-db/src/main/java/cn/hutool/db/DialectRunner.java
@@ -86,6 +86,35 @@ public class DialectRunner implements Serializable {
}
}
+ /**
+ * 更新或插入数据
+ * 此方法不会关闭Connection
+ * 如果方言未实现此方法则内部自动使用insertOrUpdate来替代功能
+ *
+ * @param conn 数据库连接
+ * @param record 记录
+ * @param keys 需要检查唯一性的字段
+ * @return 插入行数
+ * @throws SQLException SQL执行异常
+ */
+ public int upsert(Connection conn, Entity record, String... keys) throws SQLException {
+ PreparedStatement ps = getDialect().psForUpsert(conn, record, keys);
+ if (null != ps) {
+ try {
+ return ps.executeUpdate();
+ } finally {
+ DbUtil.close(ps);
+ }
+ } else {
+ final Entity where = record.filter(keys);
+ if (MapUtil.isNotEmpty(where) && count(conn, where) > 0) {
+ return update(conn, record, where);
+ } else {
+ return insert(conn, record).length;
+ }
+ }
+ }
+
/**
* 插入数据
* 此方法不会关闭Connection
@@ -212,7 +241,7 @@ public class DialectRunner implements Serializable {
* 获取查询结果总数,生成类似于 SELECT count(1) from (sql) hutool_alias_count_
* 此方法会重新构建{@link SqlBuilder},并去除末尾的order by子句
*
- * @param conn 数据库连接对象
+ * @param conn 数据库连接对象
* @param sqlBuilder 查询语句
* @return 复合条件的结果数
* @throws SQLException SQL执行异常
diff --git a/hutool-db/src/main/java/cn/hutool/db/SqlConnRunner.java b/hutool-db/src/main/java/cn/hutool/db/SqlConnRunner.java
index e21d8c5af..95290d6db 100644
--- a/hutool-db/src/main/java/cn/hutool/db/SqlConnRunner.java
+++ b/hutool-db/src/main/java/cn/hutool/db/SqlConnRunner.java
@@ -15,6 +15,7 @@ import cn.hutool.db.sql.SqlUtil;
import javax.sql.DataSource;
import java.sql.Connection;
+import java.sql.PreparedStatement;
import java.sql.SQLException;
import java.util.Collection;
import java.util.List;
diff --git a/hutool-db/src/main/java/cn/hutool/db/dialect/Dialect.java b/hutool-db/src/main/java/cn/hutool/db/dialect/Dialect.java
index 7eacc8d14..16058bc4c 100644
--- a/hutool-db/src/main/java/cn/hutool/db/dialect/Dialect.java
+++ b/hutool-db/src/main/java/cn/hutool/db/dialect/Dialect.java
@@ -121,7 +121,7 @@ public interface Dialect extends Serializable {
* @return PreparedStatement
* @throws SQLException SQL执行异常
*/
- default PreparedStatement psForCount(Connection conn, Query query) throws SQLException{
+ default PreparedStatement psForCount(Connection conn, Query query) throws SQLException {
query.setFields(ListUtil.toList("count(1)"));
return psForFind(conn, query);
}
@@ -129,13 +129,13 @@ public interface Dialect extends Serializable {
/**
* 构建用于查询行数的PreparedStatement
*
- * @param conn 数据库连接对象
+ * @param conn 数据库连接对象
* @param sqlBuilder 查询语句,应该包含分页等信息
* @return PreparedStatement
* @throws SQLException SQL执行异常
* @since 5.7.2
*/
- default PreparedStatement psForCount(Connection conn, SqlBuilder sqlBuilder) throws SQLException{
+ default PreparedStatement psForCount(Connection conn, SqlBuilder sqlBuilder) throws SQLException {
sqlBuilder = sqlBuilder
.insertPreFragment("SELECT count(1) from(")
// issue#I3IJ8X@Gitee,在子查询时需设置单独别名,此处为了防止和用户的表名冲突,使用自定义的较长别名
@@ -143,6 +143,20 @@ public interface Dialect extends Serializable {
return psForPage(conn, sqlBuilder, null);
}
+ /**
+ * 构建用于upsert的PreparedStatement
+ *
+ * @param conn 数据库连接对象
+ * @param entity 数据实体类(包含表名)
+ * @param keys 查找字段
+ * @return PreparedStatement
+ * @throws SQLException SQL执行异常
+ */
+ default PreparedStatement psForUpsert(Connection conn, Entity entity, String... keys) throws SQLException {
+ return null;
+ }
+
+
/**
* 方言名
*
diff --git a/hutool-db/src/main/java/cn/hutool/db/dialect/impl/H2Dialect.java b/hutool-db/src/main/java/cn/hutool/db/dialect/impl/H2Dialect.java
index 96311bb0f..917980049 100644
--- a/hutool-db/src/main/java/cn/hutool/db/dialect/impl/H2Dialect.java
+++ b/hutool-db/src/main/java/cn/hutool/db/dialect/impl/H2Dialect.java
@@ -1,9 +1,21 @@
package cn.hutool.db.dialect.impl;
+import cn.hutool.core.lang.Assert;
+import cn.hutool.core.util.ArrayUtil;
+import cn.hutool.db.Entity;
import cn.hutool.db.Page;
+import cn.hutool.db.StatementUtil;
import cn.hutool.db.dialect.DialectName;
+import cn.hutool.db.sql.Condition;
+import cn.hutool.db.sql.Query;
import cn.hutool.db.sql.SqlBuilder;
+import java.sql.Connection;
+import java.sql.PreparedStatement;
+import java.sql.SQLException;
+import java.util.Arrays;
+import java.util.function.Function;
+
/**
* H2数据库方言
*
@@ -26,4 +38,19 @@ public class H2Dialect extends AnsiSqlDialect {
// limit A , B 表示:A就是查询的起点位置,B就是你需要多少行。
return find.append(" limit ").append(page.getStartPosition()).append(" , ").append(page.getPageSize());
}
+
+ /**
+ * 构建用于upsert的PreparedStatement
+ *
+ * @param conn 数据库连接对象
+ * @param entity 数据实体类(包含表名)
+ * @param keys 查找字段 如果不提供keys将自动使用主键
+ * @return PreparedStatement
+ * @throws SQLException SQL执行异常
+ */
+ @Override
+ public PreparedStatement psForUpsert(Connection conn, Entity entity, String... keys) throws SQLException {
+ final SqlBuilder upsert = SqlBuilder.create(wrapper).upsert(entity, this.dialectName(),keys);
+ return StatementUtil.prepareStatement(conn, upsert);
+ }
}
diff --git a/hutool-db/src/main/java/cn/hutool/db/sql/SqlBuilder.java b/hutool-db/src/main/java/cn/hutool/db/sql/SqlBuilder.java
index 95d15ed8a..cee913a2e 100644
--- a/hutool-db/src/main/java/cn/hutool/db/sql/SqlBuilder.java
+++ b/hutool-db/src/main/java/cn/hutool/db/sql/SqlBuilder.java
@@ -196,6 +196,73 @@ public class SqlBuilder implements Builder {
return this;
}
+ /**
+ * 插入
+ * 插入会忽略空的字段名及其对应值,但是对于有字段名对应值为{@code null}的情况不忽略
+ *
+ * @param entity 实体
+ * @param dialectName 方言名,用于对特殊数据库特殊处理
+ * @param keys 根据何字段来确认唯一性,不传则用主键
+ * @return 自己
+ * @since 5.7.21
+ */
+ public SqlBuilder upsert(Entity entity, String dialectName, String... keys) {
+ // 验证
+ validateEntity(entity);
+
+ if (null != wrapper) {
+ // 包装表名 entity = wrapper.wrap(entity);
+ entity.setTableName(wrapper.wrap(entity.getTableName()));
+ }
+
+ final boolean isOracle = DialectName.ORACLE.match(dialectName);// 对Oracle的特殊处理
+ final StringBuilder fieldsPart = new StringBuilder();
+ final StringBuilder placeHolder = new StringBuilder();
+
+ boolean isFirst = true;
+ String field;
+ Object value;
+ for (Entry entry : entity.entrySet()) {
+ field = entry.getKey();
+ value = entry.getValue();
+ if (StrUtil.isNotBlank(field) /* && null != value */) {
+ if (isFirst) {
+ isFirst = false;
+ } else {
+ // 非第一个参数,追加逗号
+ fieldsPart.append(", ");
+ placeHolder.append(", ");
+ }
+
+ this.fields.add(field);
+ fieldsPart.append((null != wrapper) ? wrapper.wrap(field) : field);
+ if (isOracle && value instanceof String && StrUtil.endWithIgnoreCase((String) value, ".nextval")) {
+ // Oracle的特殊自增键,通过字段名.nextval获得下一个值
+ placeHolder.append(value);
+ } else {
+ placeHolder.append("?");
+ this.paramValues.add(value);
+ }
+ }
+ }
+
+ // issue#1656@Github Phoenix兼容
+ if (DialectName.PHOENIX.match(dialectName)) {
+ sql.append("UPSERT INTO ").append(entity.getTableName());
+ } else if (DialectName.H2.match(dialectName)) {
+ sql.append("MERGE INTO ").append(entity.getTableName());
+ if (null != keys && keys.length > 0) {
+ sql.append(" KEY(").append(ArrayUtil.join(keys, ","))
+ .append(") VALUES (")
+ .append(placeHolder)
+ .append(")");
+ }
+ } else {
+ throw new RuntimeException(dialectName + " not support yet");
+ }
+ return this;
+ }
+
/**
* 删除
*
diff --git a/hutool-db/src/test/java/cn/hutool/db/H2Test.java b/hutool-db/src/test/java/cn/hutool/db/H2Test.java
index c61404527..85bd27627 100644
--- a/hutool-db/src/test/java/cn/hutool/db/H2Test.java
+++ b/hutool-db/src/test/java/cn/hutool/db/H2Test.java
@@ -1,5 +1,6 @@
package cn.hutool.db;
+import com.alibaba.druid.support.json.JSONUtils;
import org.junit.Assert;
import org.junit.BeforeClass;
import org.junit.Test;
@@ -9,14 +10,14 @@ import java.util.List;
/**
* H2数据库单元测试
- *
+ *
* @author looly
*
*/
public class H2Test {
-
+
private static final String DS_GROUP_NAME = "h2";
-
+
@BeforeClass
public static void init() throws SQLException {
Db db = Db.use(DS_GROUP_NAME);
@@ -27,7 +28,7 @@ public class H2Test {
db.insert(Entity.create("test").set("a", 3).set("b", 31));
db.insert(Entity.create("test").set("a", 4).set("b", 41));
}
-
+
@Test
public void queryTest() throws SQLException {
List query = Db.use(DS_GROUP_NAME).query("select * from test");
@@ -39,4 +40,11 @@ public class H2Test {
List query = Db.use(DS_GROUP_NAME).find(Entity.create("test"));
Assert.assertEquals(4, query.size());
}
+ @Test
+ public void upsertTest() throws SQLException {
+ Db db=Db.use(DS_GROUP_NAME);
+ db.upsert(Entity.create("test").set("a",1).set("b",111),"a");
+ Entity a1=db.get("test","a",1);
+ Assert.assertEquals(Long.valueOf(111),a1.getLong("b"));
+ }
}
From e492cf2a736f769d3937e2cf2f3999b285d690d4 Mon Sep 17 00:00:00 2001
From: icefairy <860668820@qq.com>
Date: Fri, 14 Jan 2022 17:07:33 +0800
Subject: [PATCH 2/2] =?UTF-8?q?mysql=20+=20postgres=20upsert=E5=AE=8C?=
=?UTF-8?q?=E6=88=90=E5=B9=B6=E6=B5=8B=E8=AF=95=E9=80=9A=E8=BF=87?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
.../hutool/db/dialect/impl/MysqlDialect.java | 23 +++++++++++++++-
.../db/dialect/impl/PostgresqlDialect.java | 25 ++++++++++++++++++
.../java/cn/hutool/db/sql/SqlBuilder.java | 16 +++++++++++-
.../src/test/java/cn/hutool/db/MySQLTest.java | 26 ++++++++++++++++---
.../test/java/cn/hutool/db/PostgreTest.java | 16 ++++++++++--
5 files changed, 98 insertions(+), 8 deletions(-)
diff --git a/hutool-db/src/main/java/cn/hutool/db/dialect/impl/MysqlDialect.java b/hutool-db/src/main/java/cn/hutool/db/dialect/impl/MysqlDialect.java
index d10e811fb..3ce7a199a 100644
--- a/hutool-db/src/main/java/cn/hutool/db/dialect/impl/MysqlDialect.java
+++ b/hutool-db/src/main/java/cn/hutool/db/dialect/impl/MysqlDialect.java
@@ -1,10 +1,16 @@
package cn.hutool.db.dialect.impl;
+import cn.hutool.db.Entity;
import cn.hutool.db.Page;
+import cn.hutool.db.StatementUtil;
import cn.hutool.db.dialect.DialectName;
import cn.hutool.db.sql.SqlBuilder;
import cn.hutool.db.sql.Wrapper;
+import java.sql.Connection;
+import java.sql.PreparedStatement;
+import java.sql.SQLException;
+
/**
* MySQL方言
* @author loolly
@@ -21,9 +27,24 @@ public class MysqlDialect extends AnsiSqlDialect{
protected SqlBuilder wrapPageSql(SqlBuilder find, Page page) {
return find.append(" LIMIT ").append(page.getStartPosition()).append(", ").append(page.getPageSize());
}
-
+
@Override
public String dialectName() {
return DialectName.MYSQL.toString();
}
+
+ /**
+ * 构建用于upsert的PreparedStatement
+ *
+ * @param conn 数据库连接对象
+ * @param entity 数据实体类(包含表名)
+ * @param keys 查找字段
+ * @return PreparedStatement
+ * @throws SQLException SQL执行异常
+ */
+ @Override
+ public PreparedStatement psForUpsert(Connection conn, Entity entity, String... keys) throws SQLException {
+ final SqlBuilder upsert = SqlBuilder.create(wrapper).upsert(entity, this.dialectName(),keys);
+ return StatementUtil.prepareStatement(conn, upsert);
+ }
}
diff --git a/hutool-db/src/main/java/cn/hutool/db/dialect/impl/PostgresqlDialect.java b/hutool-db/src/main/java/cn/hutool/db/dialect/impl/PostgresqlDialect.java
index d7109e3c2..82f5fe373 100644
--- a/hutool-db/src/main/java/cn/hutool/db/dialect/impl/PostgresqlDialect.java
+++ b/hutool-db/src/main/java/cn/hutool/db/dialect/impl/PostgresqlDialect.java
@@ -1,8 +1,15 @@
package cn.hutool.db.dialect.impl;
+import cn.hutool.db.Entity;
+import cn.hutool.db.StatementUtil;
import cn.hutool.db.dialect.DialectName;
+import cn.hutool.db.sql.SqlBuilder;
import cn.hutool.db.sql.Wrapper;
+import java.sql.Connection;
+import java.sql.PreparedStatement;
+import java.sql.SQLException;
+
/**
* Postgree方言
@@ -20,4 +27,22 @@ public class PostgresqlDialect extends AnsiSqlDialect{
public String dialectName() {
return DialectName.POSTGREESQL.name();
}
+
+ /**
+ * 构建用于upsert的PreparedStatement
+ *
+ * @param conn 数据库连接对象
+ * @param entity 数据实体类(包含表名)
+ * @param keys 查找字段 必须是有唯一索引的列且不能为空
+ * @return PreparedStatement
+ * @throws SQLException SQL执行异常
+ */
+ @Override
+ public PreparedStatement psForUpsert(Connection conn, Entity entity, String... keys) throws SQLException {
+ if (null==keys || keys.length==0){
+ throw new SQLException("keys不能为空");
+ }
+ final SqlBuilder upsert = SqlBuilder.create(wrapper).upsert(entity, this.dialectName(),keys);
+ return StatementUtil.prepareStatement(conn, upsert);
+ }
}
diff --git a/hutool-db/src/main/java/cn/hutool/db/sql/SqlBuilder.java b/hutool-db/src/main/java/cn/hutool/db/sql/SqlBuilder.java
index cee913a2e..ac666054b 100644
--- a/hutool-db/src/main/java/cn/hutool/db/sql/SqlBuilder.java
+++ b/hutool-db/src/main/java/cn/hutool/db/sql/SqlBuilder.java
@@ -202,7 +202,7 @@ public class SqlBuilder implements Builder {
*
* @param entity 实体
* @param dialectName 方言名,用于对特殊数据库特殊处理
- * @param keys 根据何字段来确认唯一性,不传则用主键
+ * @param keys 根据何字段来确认唯一性,不传则用主键
* @return 自己
* @since 5.7.21
*/
@@ -249,6 +249,12 @@ public class SqlBuilder implements Builder {
// issue#1656@Github Phoenix兼容
if (DialectName.PHOENIX.match(dialectName)) {
sql.append("UPSERT INTO ").append(entity.getTableName());
+ } else if (DialectName.MYSQL.match(dialectName)) {
+ sql.append("INSERT INTO ");
+ sql.append(entity.getTableName())
+ .append(" (").append(fieldsPart).append(") VALUES (")
+ .append(placeHolder).append(") on duplicate key update ")
+ .append(ArrayUtil.join(ArrayUtil.map(entity.keySet().toArray(), String.class, (k) -> k + "=values(" + k + ")"), ","));
} else if (DialectName.H2.match(dialectName)) {
sql.append("MERGE INTO ").append(entity.getTableName());
if (null != keys && keys.length > 0) {
@@ -257,6 +263,14 @@ public class SqlBuilder implements Builder {
.append(placeHolder)
.append(")");
}
+ } else if (DialectName.POSTGREESQL.match(dialectName)) {
+ sql.append("INSERT INTO ");
+ sql.append(entity.getTableName())
+ .append(" (").append(fieldsPart).append(") VALUES (")
+ .append(placeHolder).append(") on conflict (")
+ .append(ArrayUtil.join(keys,","))
+ .append(") do update set ")
+ .append(ArrayUtil.join(ArrayUtil.map(entity.keySet().toArray(), String.class, (k) -> k + "=excluded." + k ), ","));
} else {
throw new RuntimeException(dialectName + " not support yet");
}
diff --git a/hutool-db/src/test/java/cn/hutool/db/MySQLTest.java b/hutool-db/src/test/java/cn/hutool/db/MySQLTest.java
index e3e72fba1..8ecebb74e 100644
--- a/hutool-db/src/test/java/cn/hutool/db/MySQLTest.java
+++ b/hutool-db/src/test/java/cn/hutool/db/MySQLTest.java
@@ -1,6 +1,9 @@
package cn.hutool.db;
import cn.hutool.core.lang.Console;
+import cn.hutool.core.util.ArrayUtil;
+import org.junit.Assert;
+import org.junit.BeforeClass;
import org.junit.Ignore;
import org.junit.Test;
@@ -9,11 +12,16 @@ import java.util.List;
/**
* MySQL操作单元测试
- *
- * @author looly
*
+ * @author looly
*/
public class MySQLTest {
+ @BeforeClass
+ @Ignore
+ public static void createTable() throws SQLException {
+ Db db = Db.use("mysql");
+ db.executeBatch("drop table if exists testuser", "CREATE TABLE if not exists `testuser` ( `id` int(11) NOT NULL, `account` varchar(255) DEFAULT NULL, `pass` varchar(255) DEFAULT NULL, PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8");
+ }
@Test
@Ignore
@@ -34,13 +42,13 @@ public class MySQLTest {
*
* @throws SQLException SQL异常
*/
- @Test(expected=SQLException.class)
+ @Test(expected = SQLException.class)
@Ignore
public void txTest() throws SQLException {
Db.use("mysql").tx(db -> {
int update = db.update(Entity.create("user").set("text", "描述100"), Entity.create().set("id", 100));
db.update(Entity.create("user").set("text", "描述101"), Entity.create().set("id", 101));
- if(1 == update) {
+ if (1 == update) {
// 手动指定异常,然后测试回滚触发
throw new RuntimeException("Error");
}
@@ -64,4 +72,14 @@ public class MySQLTest {
Console.log(all);
}
+ @Test
+ @Ignore
+ public void upsertTest() throws SQLException {
+ Db db = Db.use("mysql");
+ db.insert(Entity.create("testuser").set("id", 1).set("account", "ice").set("pass", "123456"));
+ db.upsert(Entity.create("testuser").set("id", 1).set("account", "icefairy").set("pass", "a123456"));
+ Entity user = db.get(Entity.create("testuser").set("id", 1));
+ System.out.println("user======="+user.getStr("account")+"___"+user.getStr("pass"));
+ Assert.assertEquals(user.getStr("account"), new String("icefairy"));
+ }
}
diff --git a/hutool-db/src/test/java/cn/hutool/db/PostgreTest.java b/hutool-db/src/test/java/cn/hutool/db/PostgreTest.java
index a19559a7e..250930efe 100644
--- a/hutool-db/src/test/java/cn/hutool/db/PostgreTest.java
+++ b/hutool-db/src/test/java/cn/hutool/db/PostgreTest.java
@@ -2,6 +2,7 @@ package cn.hutool.db;
import java.sql.SQLException;
+import org.junit.Assert;
import org.junit.Ignore;
import org.junit.Test;
@@ -9,9 +10,8 @@ import cn.hutool.core.lang.Console;
/**
* PostgreSQL 单元测试
- *
- * @author looly
*
+ * @author looly
*/
public class PostgreTest {
@@ -34,4 +34,16 @@ public class PostgreTest {
Console.log(entity.get("id"));
}
}
+
+ @Test
+ @Ignore
+ public void upsertTest() throws SQLException {
+ Db db = Db.use("postgre");
+ db.executeBatch("drop table if exists ctest",
+ "create table if not exists \"ctest\" ( \"id\" serial4, \"t1\" varchar(255) COLLATE \"pg_catalog\".\"default\", \"t2\" varchar(255) COLLATE \"pg_catalog\".\"default\", \"t3\" varchar(255) COLLATE \"pg_catalog\".\"default\", CONSTRAINT \"ctest_pkey\" PRIMARY KEY (\"id\") ) ");
+ db.insert(Entity.create("ctest").set("id", 1).set("t1", "111").set("t2", "222").set("t3", "333"));
+ db.upsert(Entity.create("ctest").set("id", 1).set("t1", "new111").set("t2", "new222").set("t3", "bew333"),"id");
+ Entity et=db.get(Entity.create("ctest").set("id", 1));
+ Assert.assertEquals("new111",et.getStr("t1"));
+ }
}