- Java锁的逻辑(结合对象头和ObjectMonitor)
- 还在用饼状图?来瞧瞧这些炫酷的百分比可视化新图形(附代码实现)⛵
- 自动注册实体类到EntityFrameworkCore上下文,并适配ABP及ABPVNext
- 基于Sklearn机器学习代码实战
Gorm源码学习系列 。
此文是Gorm源码学习系列的第二篇,主要梳理下通过Gorm创建表的流程.
。
gorm提供了以下几个接口来创建行记录 。
func (db *DB) Create(value interface{}) (tx *DB)
func (db *DB) CreateInBatches(value interface{}, batchSize int) (tx *DB)
func (db *DB) Save(value interface{}) (tx *DB)
详细请看 教程 及源码 finisher_api.go ,这里使用 func (db *DB) Create(value interface{}) (tx *DB) 来说明创建行记录等大致流程.
。
type Stu struct {
ID int64 `gorm:"column:id; primary_key" json:"id"`
Age int64 `gorm:"column:age;"`
Height int64 `gorm:"column:height;"`
Weight int64 `gorm:"column:weight;"`
}
// 设置表名
func (Stu) TableName() string {
return "t_student"
}
模型代码的主要用途如下, 。
gorm
标签指定每个字断对于的表的列名 Tabler
接口指定了固定的表名,接口定义如下
type Tabler interface {
TableName() string
}
关于模型定义中更多的约定和约束等,请看 教程 .
出于分表等业务场景,我们并不希望固定模型等表名,gorm提供了 func (db *DB) Table(name string, args ...interface{}) (tx *DB) 等方法 。
来动态指定表名,详情请看 教程 .
。
func main() {
// 数据库连接, 具体查看https://www.cnblogs.com/amos01/p/16890747.html 连接数据库代码示例
db, _ := dbOpen()
// 打开调试模式、会打印DML
db = db.Debug()
stu := &Stu{
Age: 18,
Height: 185,
Weight: 70,
}
db = db.Create(stu)
fmt.Printf("Error:%v ID:%v RowsAffected:%v\n", db.Error, stu.ID, db.RowsAffected)
}
代码输出如下 。
$ go run main.go
2022/12/11 14:59:59 /Users/zbw/workspace/test/main.go:33
[1.910ms] [rows:1] INSERT INTO `t_student` (`age`,`height`,`weight`) VALUES (18,185,70)
Error:<nil> ID:1027 RowsAffected:1
从代码输出可以看,行记录的ID为1027,连接数据库查询,结果如下.
mysql> select * from t_student where id = 1027\G
*************************** 1. row ***************************
id: 1027
age: 18
height: 185
weight: 70
1 row in set (0.01 sec)
因此,我们带着以下问题来梳理下Gorm创建行记录的流程 。
。
func (db *DB) Create(value interface{}) (tx *DB) 的实现如下 。
// Create inserts value, returning the inserted data's primary key in value's id
func (db *DB) Create(value interface{}) (tx *DB) {
if db.CreateBatchSize > 0 {
return db.CreateInBatches(value, db.CreateBatchSize)
}
tx = db.getInstance()
tx.Statement.Dest = value
return tx.callbacks.Create().Execute(tx)
}
func (p *processor) Execute(db *DB) *DB 的实现比较长,具体代码见 github 。
总结下来,做了两件主要的事情, 。
X 。
gorm.Statement
// Statement statement
type Statement struct {
*DB
TableExpr *clause.Expr
Table string // 表名
Model interface{} // model定义
Unscoped bool
Dest interface{} // model的另外一种表达,如map
ReflectValue reflect.Value
Clauses map[string]clause.Clause
BuildClauses []string
Distinct bool
Selects []string // selected columns
Omits []string // omit columns
Joins []join
Preloads map[string][]interface{}
Settings sync.Map
ConnPool ConnPool // 数据库连接
Schema *schema.Schema // 表结构化信息
Context context.Context
RaiseErrorOnNotFound bool
SkipHooks bool
SQL strings.Builder // 最终的DML语句
Vars []interface{} // DML语句的参数值
CurDestIndex int // 批量创建/更新时,gorm当前操作的数组/slice的下标
attrs []interface{}
assigns []interface{}
scopes []func(*DB) *DB
}
schema.Schem
type Schema struct {
Name string
ModelType reflect.Type
Table string // 表名
PrioritizedPrimaryField *Field
DBNames []string // 表每列的名字
PrimaryFields []*Field
PrimaryFieldDBNames []string // 表的主键列明
Fields []*Field // gorm自定义的model每个字短
FieldsByName map[string]*Field
FieldsByDBName map[string]*Field
FieldsWithDefaultDBValue []*Field // fields with default value assigned by database
Relationships Relationships
CreateClauses []clause.Interface // 创建行的子句
QueryClauses []clause.Interface
UpdateClauses []clause.Interface
DeleteClauses []clause.Interface
BeforeCreate, AfterCreate bool
BeforeUpdate, AfterUpdate bool
BeforeDelete, AfterDelete bool
BeforeSave, AfterSave bool
AfterFind bool
err error
initialized chan struct{}
namer Namer
cacheStore *sync.Map
}
schema.Field
// Field is the representation of model schema's field
type Field struct {
Name string // model的字段名
DBName string // 对应表的列名
BindNames []string
DataType DataType
GORMDataType DataType
PrimaryKey bool
AutoIncrement bool
AutoIncrementIncrement int64
Creatable bool
Updatable bool
Readable bool
AutoCreateTime TimeType
AutoUpdateTime TimeType
HasDefaultValue bool
DefaultValue string
DefaultValueInterface interface{}
NotNull bool
Unique bool
Comment string
Size int
Precision int
Scale int
IgnoreMigration bool
FieldType reflect.Type // 反射类型
IndirectFieldType reflect.Type // 反射类型
StructField reflect.StructField // model字段信息
Tag reflect.StructTag // tag
TagSettings map[string]string
Schema *Schema
EmbeddedSchema *Schema
OwnerSchema *Schema
ReflectValueOf func(context.Context, reflect.Value) reflect.Value // 通过反射获取该字段的反射对象
ValueOf func(context.Context, reflect.Value) (value interface{}, zero bool) // 通过反射获取该字段的值 get方法
Set func(context.Context, reflect.Value, interface{}) error // 通过反射设置该字段的值 set方法
Serializer SerializerInterface
NewValuePool FieldNewValuePool
}
clause.Interface
及 clause.Clause
gorm定义了多种clause,包括 。
// Interface clause interface
type Interface interface {
Name() string
Build(Builder)
MergeClause(*Clause)
}
// Clause
type Clause struct {
Name string // WHERE
BeforeExpression Expression
AfterNameExpression Expression
AfterExpression Expression
Expression Expression
Builder ClauseBuilder
}
。
通过调用 stmt.Parse(stmt.Model) 进行model解析 。
stmt.Parse(stmt.Model) 会调用到函数 func ParseWithSpecialTableName(dest interface{}, cacheStore *sync.Map, namer Namer, specialTableName string) (*Schema, error) 进行解析.
详细代码见 schema.go ,下面列举重要的几个点.
dest interface{}
是否为 reflect.Struct
Tabler
接口
// 获取表名
modelValue := reflect.New(modelType)
tableName := namer.TableName(modelType.Name())
if tabler, ok := modelValue.Interface().(Tabler); ok {
tableName = tabler.TableName()
}
if tabler, ok := modelValue.Interface().(TablerWithNamer); ok {
tableName = tabler.TableName(namer)
}
if en, ok := namer.(embeddedNamer); ok {
tableName = en.Table
}
if specialTableName != "" && specialTableName != tableName {
tableName = specialTableName
}
// 通过反射获取每个字段
for i := 0; i < modelType.NumField(); i++ {
if fieldStruct := modelType.Field(i); ast.IsExported(fieldStruct.Name) {
// 解析每个字段
if field := schema.ParseField(fieldStruct); field.EmbeddedSchema != nil {
schema.Fields = append(schema.Fields, field.EmbeddedSchema.Fields...)
} else {
schema.Fields = append(schema.Fields, field)
}
}
}
func (field *Field) setupValuerAndSetter()
初始化每个Field的 ReflectValueOf
、 ValueOf
、 Set
方法。
for _, field := range schema.Fields {
if field.DBName == "" && field.DataType != "" {
field.DBName = namer.ColumnName(schema.Table, field.Name)
}
if field.DBName != "" {
// nonexistence or shortest path or first appear prioritized if has permission
if v, ok := schema.FieldsByDBName[field.DBName]; !ok || ((field.Creatable || field.Updatable || field.Readable) && len(field.BindNames) < len(v.BindNames)) {
if _, ok := schema.FieldsByDBName[field.DBName]; !ok {
schema.DBNames = append(schema.DBNames, field.DBName)
}
// gorm tag字段到field的映射
schema.FieldsByDBName[field.DBName] = field
// model 字段到field的映射
schema.FieldsByName[field.Name] = field
if v != nil && v.PrimaryKey {
for idx, f := range schema.PrimaryFields {
if f == v {
schema.PrimaryFields = append(schema.PrimaryFields[0:idx], schema.PrimaryFields[idx+1:]...)
}
}
}
// 主键
if field.PrimaryKey {
schema.PrimaryFields = append(schema.PrimaryFields, field)
}
}
}
if of, ok := schema.FieldsByName[field.Name]; !ok || of.TagSettings["-"] == "-" {
schema.FieldsByName[field.Name] = field
}
// 挂载字段的set方法和get方法
field.setupValuerAndSetter()
}
。
值得一提的是,每个model解析后的结果是一致,可以将结果解析的结构缓存下来,并且通过 chan 来解决并发的问题.
解析model之后,通过 process 获取到钩子函数及创建行的函数,具体代码见 Github 。
for _, f := range p.fns {
f(db)
}
。
创建行的函数及对应的钩子函数位于 create.go 。
if db.Statement.SQL.Len() == 0 {
db.Statement.SQL.Grow(180)
db.Statement.AddClauseIfNotExists(clause.Insert{})
db.Statement.AddClause(ConvertToCreateValues(db.Statement))
db.Statement.Build(db.Statement.BuildClauses...)
}
这里插入两个 clause.Clause ,分别为 clause.Insert 以及 clause.Values ,然后调用这两种 clause.Clause 的 build 方法生成 SQL 语句.
首先,看下 ConvertToCreateValues 的实现,这里只截取部分代码 。
values = clause.Values{Columns: make([]clause.Column, 0, len(stmt.Schema.DBNames))}
// 获取每一列的名字
for _, db := range stmt.Schema.DBNames {
if field := stmt.Schema.FieldsByDBName[db]; !field.HasDefaultValue || field.DefaultValueInterface != nil {
if v, ok := selectColumns[db]; (ok && v) || (!ok && (!restricted || field.AutoCreateTime > 0 || field.AutoUpdateTime > 0)) {
values.Columns = append(values.Columns, clause.Column{Name: db})
}
}
}
// 获取每一列对应的值
switch stmt.ReflectValue.Kind() {
case reflect.Slice, reflect.Array:
case reflect.Struct:
values.Values = [][]interface{}{make([]interface{}, len(values.Columns))}
for idx, column := range values.Columns {
field := stmt.Schema.FieldsByDBName[column.Name]
// func (field *Field) setupValuerAndSetter() 挂载的方法
if values.Values[0][idx], isZero = field.ValueOf(stmt.Context, stmt.ReflectValue); isZero {
if field.DefaultValueInterface != nil {
values.Values[0][idx] = field.DefaultValueInterface
stmt.AddError(field.Set(stmt.Context, stmt.ReflectValue, field.DefaultValueInterface))
} else if field.AutoCreateTime > 0 || field.AutoUpdateTime > 0 {
stmt.AddError(field.Set(stmt.Context, stmt.ReflectValue, curTime))
values.Values[0][idx], _ = field.ValueOf(stmt.Context, stmt.ReflectValue)
}
} else if field.AutoUpdateTime > 0 && updateTrackTime {
stmt.AddError(field.Set(stmt.Context, stmt.ReflectValue, curTime))
values.Values[0][idx], _ = field.ValueOf(stmt.Context, stmt.ReflectValue)
}
}
通过 ConvertToCreateValues 获取了每一列的名称及对应的值.
接下来,看 clause.Clause 到 SQL 语句的过程.
遍历加入 clause ,此时分别为 clause.Insert 以及 clause.Values 。
// Build build sql with clauses names
func (stmt *Statement) Build(clauses ...string) {
var firstClauseWritten bool
for _, name := range clauses {
if c, ok := stmt.Clauses[name]; ok {
// 代码有删减
c.Build(stmt)
}
}
}
接着调用 func (c Clause) Build(builder Builder) 。
// Build build clause
func (c Clause) Build(builder Builder) {
// 有删减
// c为clause.Insert以及clause.Values
if c.Name != "" {
// builder写入 INSERT 或者 VALUES
builder.WriteString(c.Name)
builder.WriteByte(' ')
}
// 通过clause.Insert以及clause.Values的MergeClause函数,c.Expression为clause.Insert以及clause.Values
// 因此,这里调用clause.Insert或者clause.Values的Build的方法
c.Expression.Build(builder)
}
接下来分别看 clause.Insert 以及 clause.Values 。
// Build build insert clause
func (insert Insert) Build(builder Builder) {
// builder写入INTO,此时builder为INSERT INTO
builder.WriteString("INTO ")
// builder写入表名
builder.WriteQuoted(currentTable)
}
从调用的链路可以得出,这里 builder 为 stmt *Statement ,并且 currentTable 类型为 clause.Table ,因此 。
// WriteQuoted write quoted value
func (stmt *Statement) WriteQuoted(value interface{}) {
stmt.QuoteTo(&stmt.SQL, value)
}
// QuoteTo write quoted value to writer 有删减
func (stmt *Statement) QuoteTo(writer clause.Writer, field interface{}) {
write := func(raw bool, str string) {
// mysql驱动Dialector
stmt.DB.Dialector.QuoteTo(writer, str)
}
switch v := field.(type) {
case clause.Table:
write(v.Raw, stmt.Table)
}
}
至此, builder 已经拼装出 INSERT INTO `t_student` ,解析来再看 clause.Values 的 build 方法 。
// Build build from clause
func (values Values) Build(builder Builder) {
if len(values.Columns) > 0 {
builder.WriteByte('(')
for idx, column := range values.Columns {
if idx > 0 {
builder.WriteByte(',')
}
builder.WriteQuoted(column)
}
builder.WriteByte(')')
builder.WriteString(" VALUES ")
for idx, value := range values.Values {
if idx > 0 {
builder.WriteByte(',')
}
builder.WriteByte('(')
builder.AddVar(builder, value...)
builder.WriteByte(')')
}
} else {
builder.WriteString("DEFAULT VALUES")
}
}
func (values Values) Build(builder Builder) 取出所有列名和列对应的值 。
最终 builder 拼装成例子的完整SQL语句 INSERT INTO `t_student` (`age`,`height`,`weight`) VALUES (18,185,70) 。
有了SQL语句,就可以执行了 。
result, err := db.Statement.ConnPool.ExecContext(
db.Statement.Context, db.Statement.SQL.String(), db.Statement.Vars...,
)
通过前一面学习, db.Statement.ConnPool 的值为 sql.DB ,实际执行的函数为 func (db *DB) ExecContext(ctx context.Context, query string, args ...any) (Result, error) 。
至此,从Model到DML到流程已经完成.
。
看返回参数 sql.Result ,因此通过 LastInsertId() (int64, error) 可以获取到插入行的ID值.
// A Result summarizes an executed SQL command.
type Result interface {
// LastInsertId returns the integer generated by the database
// in response to a command. Typically this will be from an
// "auto increment" column when inserting a new row. Not all
// databases support this feature, and the syntax of such
// statements varies.
LastInsertId() (int64, error)
// RowsAffected returns the number of rows affected by an
// update, insert, or delete. Not every database or database
// driver may support this.
RowsAffected() (int64, error)
}
获取到刚插入的行ID值,再通过反射写入model的ID字段即可.
db.RowsAffected, _ = result.RowsAffected()
if db.RowsAffected != 0 && db.Statement.Schema != nil &&
db.Statement.Schema.PrioritizedPrimaryField != nil &&
db.Statement.Schema.PrioritizedPrimaryField.HasDefaultValue {
insertID, err := result.LastInsertId()
switch db.Statement.ReflectValue.Kind() {
case reflect.Struct:
_, isZero := db.Statement.Schema.PrioritizedPrimaryField.ValueOf(db.Statement.Context, db.Statement.ReflectValue)
if isZero {
// 通过反射更新ID
db.AddError(db.Statement.Schema.PrioritizedPrimaryField.Set(db.Statement.Context, db.Statement.ReflectValue, insertID))
}
}
}
。
使用反射解析Model,获得每个成员对应的表的列名、值等信息.
定义SQL各个关键词如 INSERT 、 VALUES 、 FROM 、 DELETE 的结构体,并实现 clause.Interface 接口 。
进而对SQL语句的构造进行抽象封装.
。
最后此篇关于Gorm源码学习-创建行记录的文章就讲到这里了,如果你想了解更多关于Gorm源码学习-创建行记录的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我有一个网站。 必须登录才能看到里面的内容。 但是,我使用此代码登录。 doc = Jsoup.connect("http://46.137.207.181/Account/Login.aspx")
我正在尝试为我的域创建一个 SPF 记录并使我的邮件服务器能够对其进行评估。我在邮件服务器上使用 Postfix 并使用 policyd-spf (Python) 来评估记录。目前,我通过我的私有(p
我需要为负载平衡的 AWS 站点 mywebsite.com 添加 CName 记录。记录应该是: @ CNAME mywebsite.us-east-1.elb.amazon
我目前正在开发一个相当大的多层应用程序,该应用程序将部署在海外。虽然我希望它在解聚后不会折叠或爆炸,但我不能 100% 确定这一点。因此,如果我知道我可以请求日志文件,以准确找出问题所在以及原因,那就
我使用以下命令从我的网络摄像头录制音频和视频 gst-launch-0.10 v4l2src ! video/x-raw-yuv,width=640,height=480,framerate=30/1
我刚刚开始使用 ffmpeg 将视频分割成图像。我想知道是否可以将控制台输出信息保存到日志文件中。我试过“-v 10”参数,也试过“-loglevel”参数。我在另一个 SO 帖子上看到使用 ffmp
我想针对两个日期查询我的表并检索其中的记录。 我这样声明我的变量; DECLARE @StartDate datetime; DECLARE @EndDate datetime; 并像这样设置我的变量
在 javascript 中,我可以使用简单的 for 循环访问对象的每个属性,如下所示 var myObj = {x:1, y:2}; var i, sum=0; for(i in myObj) s
最近加入了一个需要处理大量代码的项目,我想开始记录和可视化调用图的一些流程,让我更好地理解一切是如何组合在一起的。这是我希望在我的理想工具中看到的: 每个节点都是一个函数/方法 如果一个函数可以调用另
如何使用反射在F#中创建记录类型?谢谢 最佳答案 您可以使用 FSharpValue.MakeRecord [MSDN]创建一个记录实例,但是我认为F#中没有任何定义记录类型的东西。但是,记录会编译为
关闭。这个问题不满足Stack Overflow guidelines .它目前不接受答案。 想改善这个问题吗?更新问题,使其成为 on-topic对于堆栈溢出。 3年前关闭。 Improve thi
我是 Sequelize 的新手并且遇到了一些语法问题。我制作了以下模型: // User sequelize.define('user', { name: { type: DataTyp
${student.name} Notify 这是我的output.jsp。请注意,我已经放置了一个链接“Notify”以将其转发到 display.jsp 上。但我不确定如何将 Stud
例如,这是我要做的查询: server:"xxx.xxx.com" AND request_url:"/xxx/xxx/xxx" AND http_X_Forwarded_Proto:(https O
我一直在开发大量 Java、PHP 和 Python。所有这些都提供了很棒的日志记录包(分别是 Log4J、Log 或logging)。这在调试应用程序时有很大帮助。特别是当应用程序 headless
在我的Grails应用程序中,我异步运行一些批处理过程,并希望该过程记录各种状态消息,以便管理员以后可以检查它们。 我考虑过将log4j JDBC附加程序用作最简单的解决方案,但是据我所知,它不使用D
我想将进入 MQ 队列的消息记录到数据库/文件或其他日志队列,并且我无法修改现有代码。是否有任何方法可以实现某种类似于 HTTP 嗅探器的消息记录实用程序?或者也许 MQ 有一些内置的功能来记录消息?
如果我有一条包含通用字段的记录,在更改通用字段时是否有任何方法可以模仿方便的 with 语法? 即如果我有 type User = // 'photo can be Bitmap or Url {
假设我有一个名为 Car 的自定义对象。其中的所有字段都是私有(private)的。 public class Car { private String mName; private
当记录具有特定字段时,我需要返回 true 的函数,反之亦然。示例: -record(robot, {name, type=industrial, ho
我是一名优秀的程序员,十分优秀!