Record
是jOOQ定义的用于储存数据库结果记录的一个接口,其主要是将一个表字段的列表和值的列表使用相同的顺序储存在一起,可以看做是一个用于储存列/值的映射的对象。通常有以下几种形式
与数据库表一一对应,如果包含主键,会继承UpdatableRecordImpl
类,该类提供了使用 update
, delete
API进行数据操作。进行查询操作时,jOOQ会将结果集包装为一个TableRecord
对象。 在使用代码生成器的时候,会生成更详细的表记录类,包含表的每个字段操作等,通常以表名为开头 XxxxRecord
此类 Record
对象一般都有对应字段的getter/setter
方法,但其都只是去调用get/set
方法。其储存的方式还是通过两个数组来储存对应列/值的数据的,所以Record
对象是不能被JSON
直接序列化和反序列化的
通常用于 Oracle 等支持用户自定义数据类型的数据库记录,这里接触较少,不作讲解
通用记录类型的一种,当你的字段不超过22个时,会根据字段个数反射成 Record1
, Record2
... Record22
类的对象。这些对象需要的泛型个数和后面的数字一致,类型和字段类型一致。jOOQ自动生成的Record
对象里,如果字段个数不超过 22 个,会同时实现 Record[N]
接口
举个例子
s1_user
这张表有6个字段,其生成的 Record
对象,继承了 UpdatableRecordImpl
并且实现了 Record6
接口。Record6
接口的泛型参数和表字段类型一一对应,类型顺序和数据库内字段顺序一致class S1UserRecord extends UpdatableRecordImpl
implements Record6
s4_columen_gt22
这张表来用于演示,该表的字段因为超过了22个,所以不会去实现Record[N]
接口,只继承了UpdatableRecordImpl
class S4ColumenGt22Record extends UpdatableRecordImpl
再看看 Record[N]
的接口定义,此接口主要是提供了获取字段,获取值,设置值的方法。由泛型决定字段/值类型和顺序,N 决定字段/值的个数。此接口的目的其实也很简单,就是为了更快速的操作指定位置的字段/值
interface Record[N] {
// 获取字段
Field field1();
...
Field field[N]();
// 获取值
T1 value1();
...
T[N] valueN();
// 设置值
Record[1] value1(T1 value)
...
Record[N] value1(T[N] value)
}
这里说的创建Record
对象,指的是在jOOQ已经生成了对应表的Record
类的情况下,创建Record进行。使用 Record
提供的方法,我们可以方便的做一些针对表数据的操作。创建对象的方式有以下几种
new
由jOOQ生成的Record
对象,可以通过 new
的方式直接创建一个实例。 通过直接 new
的方式创建对象,由于没有连接相关信息,无法直接进行 insert
, update
, delete
方法的调用。但是可以通过 DSLContext
的API进行操作数据操作,通过这种方式创建的Record
对象可以理解为一个单纯的数据储存对象
S1UserRecord s1UserRecord = new S1UserRecord();
s1UserRecord.setUsername("new1");
s1UserRecord.setEmail("[email protected]");
// insert会报错,因为没有连接配置
s1UserRecord.insert();
// 不需要获取主键,直接执行insert
int row = dslContext.insertInto(S1_USER).set(s1UserRecord)
.execute()
// 执行insert并返回相应字段
Integer id = dslContext.insertInto(S1_USER).set(s1UserRecord)
.returning(S1_USER.ID)
.fetchOne().getId();
newRecord
在获取到 DSLContext
实例后,可以使用 dslContext.newRecord(Table> table)
方法来创建一个指定表的Record
对象,这也是比较常用的方法。通过此方法创建的对象包含了dslContext
内的数据连接配置,可以直接进行 insert
, update
, delete
等操作
S1UserRecord s1UserRecord = dslContext.newRecord(S1_USER);
s1UserRecord.setUsername("newRecord1");
s1UserRecord.setEmail("[email protected]");
s1UserRecord.insert();
fetch
通过fetch*
方法读取到结果 Record
对象,同样带有数据库连接相关配置,并且带有查询结果的数据。可以直接进行数据操作
S1UserRecord s1UserRecord = dslContext.selectFrom(S1_USER).where(S1_USER.ID.eq(1))
.fetchOne();
s1UserRecord.setEmail("hello email");
int row = s1UserRecord.update();
//
S1UserRecord fetchIntoUserRecord = dslContext.select().from(S1_USER)
.where(S1_USER.ID.eq(1))
.fetchOneInto(S1UserRecord.class);
fetchIntoUserRecord.setEmail("hello email2");
int row2 = fetchIntoUserRecord.update();
insert
此方法进行数据插入操作,几个重载可以指定插入的数据字段
insert()
插入所有set
后的字段insert(Field> ... feilds)
插入指定set
后的字段insert(Collection extends Field>> fields)
插入指定set
后的字段需要注意的是,插入的字段必须显式的进行set
操作,才会在最终的SQL语句中体现出来
// 常规用法
S1UserRecord userRecord = dslContext.newRecord(S1_USER);
userRecord.setUsername("insertUsername");
userRecord.setEmail("email");
userRecord.insert();
// insert into `learn-jooq`.`s1_user` (`username`, `email`)
// values ('insertUsername', 'email')
// 指定字段插入
userRecord = dslContext.newRecord(S1_USER);
userRecord.setUsername("insertUsername");
userRecord.setEmail("email");
userRecord.insert(S1_USER.USERNAME, S1_USER.ADDRESS);
// insert into `learn-jooq`.`s1_user` (`username`)
// values ('insertUsername')
update
此方法进行更新操作,和 insert
方法类似,重载的也一样
update()
更新所有set
后的字段update(Field> ... feilds)
更新指定set
后的字段update(Collection extends Field>> fields)
更新指定set
后的字段重载参数的目的是为了约束更新的字段,同样的,只有经过set
的字段,才会被更新处理
S1UserRecord userRecord = dslContext.selectFrom(S1_USER)
.where(S1_USER.ID.eq(1))
.fetchSingle();
userRecord.setEmail(null);
userRecord.setUsername("hello");
userRecord.update(S1_USER.USERNAME, S1_USER.ADDRESS);
// update `learn-jooq`.`s1_user`
// set `learn-jooq`.`s1_user`.`username` = 'hello'
// where `learn-jooq`.`s1_user`.`id` = 1
delete
delete
方法根据主键进行数据删除操作
delete()
根据主键删除数据// 有主键值
S1UserRecord userRecord = dslContext.newRecord(S1_USER);
userRecord.setId(1);
userRecord.delete();
// delete from `learn-jooq`.`s1_user` where `learn-jooq`.`s1_user`.`id` = 1
// 无主键值
userRecord = dslContext.newRecord(S1_USER);
userRecord.delete();
// delete from `learn-jooq`.`s1_user` where `learn-jooq`.`s1_user`.`id` is null
get
get
系列方法主要是用于获取字段值
get(..)
可以取任意字段的值,非常通用,有很多重载,基本涵盖了所有业务场景,包括取值,转换。 这里列举几个常见用法Record record = dslContext.select().from(S1_USER).limit(1).fetchOne();
// 直接取出指定字段数据
Integer id = record.get(S1_USER.ID);
// 取出数据并指定转换的数据类型
Long createTimeLong = record.get(S1_USER.CREATE_TIME, Long.class);
get[FieldName]()
jOOQ生成和表字段一一对应的getter
方法,可以通过此方法直接获取指定字段,实际调用的是 get(fieldIndex)
方法S1UserRecord userRecord = dslContext.select().from(S1_USER)
.fetchAny().into(S1_USER);
Integer id = userRecord.getId();
Timestamp createTime = userRecord.getCreateTime();
value[N]
方法为Record[N]
接口提供的类,查询明确个数的字段值时,可以快速的获取对应位置下标的值Record3 record3 = dslContext
.select(S1_USER.ID,
S1_USER.USERNAME,
S1_USER.CREATE_TIME)
.from(S1_USER)
.fetchAny();
Integer id = record3.value1();
String username = record3.value2();
Timestamp createTime = record3.get(S1_USER.CREATE_TIME);
set
set
系列方法主要是用于设置字段值
set(..)
可以设置字段的值,通常在设置值后用于 insert
/update
操作S1UserRecord userRecord = dslContext.newRecord(S1_USER);
userRecord.set(S1_USER.ID, 1);
userRecord.set(S1_USER.USERNAME, "username");
userRecord.update();
get[FieldName]()
jOOQ生成和表字段一一对应的setter
方法,可以通过此方法直接设置指定字段值,实际调用的是 set(fieldIndex, object)
方法S1UserRecord userRecord = dslContext.newRecord(S1_USER);
userRecord.setId(1);
userRecord.setUsername("username");
userRecord.update();
changed
此方法用于修改字段更新标识,一般 update
/insert
方法配合使用,可以设置指定字段是否 更新/储存
S1UserRecord userRecord = dslContext.newRecord(S1_USER);
userRecord.setId(1);
userRecord.setUsername("username");
userRecord.setEmail(null);
userRecord.changed(S1_USER.EMAIL, false);
userRecord.update();
reset
此方法用于重置字段更新标识,效果和 changed(Field field, false)
相同
S1UserRecord userRecord = dslContext.newRecord(S1_USER);
userRecord.setId(1);
userRecord.setUsername("username");
userRecord.setEmail(null);
userRecord.reset(S1_USER.EMAIL);
userRecord.update();
Record
提供了转换类API,可以方便快捷的将Record
转换为其他任意类型,也可以将任意类型填充至Record
对象中。其核心主要是 from
/into
系列方法
from
此系列方法包含 from(...)
, fromMap(...)
, fromArray(...)
三个方法,主要是可以将任意对象填充至Record
中
from(Object object)
S1UserPojo userPojo = new S1UserPojo();
userPojo.setUsername("username");
userPojo.setAddress("address");
S1UserRecord userRecord = dslContext.newRecord(S1_USER);
userRecord.from(userPojo);
fromMap(Map map)
Map userDataMap = new HashMap<>();
userDataMap.put("username", "username1");
userDataMap.put("address", "user-address-1");
S1UserRecord userRecord1 = dslContext.newRecord(S1_USER);
userRecord1.fromMap(userDataMap);
fromArray(Object[] array, Field> ... fields)
Object[] userDataArray = new Object[]{"username2", "user-address-2"};
S1UserRecord userRecord2 = dslContext.newRecord(S1_USER);
userRecord2.fromArray(userDataArray, S1_USER.USERNAME, S1_USER.ADDRESS);
into
此系列方法是将Record
转换为其他任意指定类型,常用的方法有
into(Class> type)
此方法主要是通过反射的方式,创建目标类实例,并将对应字段值设置到目标实例中,也是在业务代码中用的最多的一个重载S1UserRecord userRecord = dslContext.selectFrom(S1_USER)
.where(S1_USER.ID.eq(id))
.fetchOne();
S1UserPojo userPojo = userRecord.into(S1UserPojo.class);
into(Field> ... fields)
Record2 record2 = userRecord
.into(S1_USER.ID, S1_USER.USERNAME);
Integer id = record2.value1();
intoArray()
Object[] userArray = userRecord.intoArray();
Integer fromArrayId = (Integer) userArray[0];
intoMap()
Map userMap = userRecord.intoMap();
Integer mapId = userMap.get("id");
intoResultSet()
ResultSet resultSet = userRecord.intoResultSet();