使用gorm.Scopes函數(shù)實現(xiàn)復(fù)用查詢邏輯示例
引言
今天要學(xué)習(xí)的是gorm.Scopes
函數(shù)的使用。該函數(shù)的作用就是復(fù)用查詢條件。
gorm Scopes是什么
在項目中,你一定會遇到過很多需要復(fù)用的查詢條件。比如常用的場景有分頁、查詢時判定數(shù)據(jù)權(quán)限等操作。
比如,我們有兩個數(shù)據(jù)資源:用戶列表和部門列表。那么,在查詢列表的時候都會涉及到分頁。當(dāng)然可以在每個列表中都增加上列表相關(guān)的查詢。同時,也可以將分頁的查詢抽取出來,做成公共的函數(shù)。
那怎么將抽取出來的分頁條件在每個列表中都能復(fù)用呢?那就是使用gorm.Scopes
函數(shù)。
我們先看一個使用gorm.Scopes函數(shù)使用的簡單例子,這個例子只是為了說明gorm.Scopes
函數(shù)的使用。如下:
func AmountGreaterThan1000(db *gorm.DB) *gorm.DB { return db.Where("amount > ?", 1000) } func PaidWithCreditCard(db *gorm.DB) *gorm.DB { return db.Where("pay_mode_sign = ?", "C") } func PaidWithCod(db *gorm.DB) *gorm.DB { return db.Where("pay_mode_sign = ?", "C") } func OrderStatus(status []string) func (db *gorm.DB) *gorm.DB { return func (db *gorm.DB) *gorm.DB { return db.Where("status IN (?)", status) } } // 查找所有金額大于 1000 的信用卡訂單 db.Scopes(AmountGreaterThan1000, PaidWithCreditCard).Find(&orders) // 查找所有金額大于 1000 的 COD 訂單 db.Scopes(AmountGreaterThan1000, PaidWithCod).Find(&orders) // 查找所有金額大于1000 的已付款或已發(fā)貨訂單 db.Scopes(AmountGreaterThan1000, OrderStatus([]string{"paid", "shipped"})).Find(&orders)
通過上述例子可以知道,前4個函數(shù)都是func (db *gorm.DB) *gorm.DB
類型的函數(shù),即輸入一個db,然后返回一個db。在該函數(shù)中的業(yè)務(wù)邏輯其實就是最常見的db.Where
、db.Offset
等常用的查詢條件語句而已。只不過是對這種公共的查詢語句進行了提取并進行復(fù)用而已。
然后將這樣的函數(shù)傳遞給Scopes
。Scopes
函數(shù)只是簡單的將func (db *gorm.DB) *gorm.DB
放到Statement.scopes
這個切片中。
最后,在最終執(zhí)行的時候,會循環(huán)遍歷Statement.scopes
切片,依次執(zhí)行該切片中的每一個func (db *gorm.DB) *gorm.DB
函數(shù)。這樣,就把提取出來的公共的查詢條件融合在一起了。
使用場景1 -- 分頁
當(dāng)然,我們在查詢時最常用的就是分頁功能。那么,如何使用gorm.Scopes
實現(xiàn)分頁查詢的復(fù)用呢。如下:
func Paginate(r *http.Request) func(db *gorm.DB) *gorm.DB { return func (db *gorm.DB) *gorm.DB { q := r.URL.Query() page, _ := strconv.Atoi(q.Get("page")) if page <= 0 { page = 1 } pageSize, _ := strconv.Atoi(q.Get("page_size")) switch { case pageSize > 100: pageSize = 100 case pageSize <= 0: pageSize = 10 } offset := (page - 1) * pageSize return db.Offset(offset).Limit(pageSize) } } db.Scopes(Paginate(r)).Find(&users) db.Scopes(Paginate(r)).Find(&articles)
你看,先定義了一個分頁的函數(shù)Paginate
函數(shù),該函數(shù)接收一個*http.Request
參數(shù),然后返回一個func(db *gorm.DB) *gorm.DB
的函數(shù)。因為gorm.Scopes
函數(shù)只接受func(db *gorm.DB) *gorm.DB
類型的函數(shù)。最后,將Paginate
函數(shù)傳遞給Scopes
函數(shù)即可。
使用場景2 -- 數(shù)據(jù)權(quán)限
在go-admin開源項目中,我們還發(fā)現(xiàn)了一個典型的應(yīng)用,就是數(shù)據(jù)權(quán)限。在我們的系統(tǒng)中,會遇到這樣的場景:一些數(shù)據(jù)只能自己查看或操作;或者你的上級也能查看或操作;或者同部門的人員能查看或操作自己部門的數(shù)據(jù),但不能查看或操作其他部門的權(quán)限;又或者只能查看同部門的數(shù)據(jù)但不能操作同部門的數(shù)據(jù)等等。
在go-admin中,就使用了gorm.Scopes
函數(shù)來統(tǒng)一使用權(quán)限查詢條件。在每個操作中,都通過Scopes
函數(shù)傳入了一個Permission
函數(shù)。Permissioin
函數(shù)是根據(jù)不同角色擁有的權(quán)限,轉(zhuǎn)換成對應(yīng)的sql語句。如下:
// DeleteAction 通用刪除動作 func DeleteAction(control dto.Control) gin.HandlerFunc { return func(c *gin.Context) { db, err := pkg.GetOrm(c) ...//省略了一些邏輯 //數(shù)據(jù)權(quán)限檢查 p := GetPermissionFromContext(c) // 將Permission函數(shù)傳入Scopes函數(shù) db = db.WithContext(c).Scopes( Permission(object.TableName(), p), ).Where(req.GetId()).Delete(object) ...//省略了一些邏輯 // 檢查操作的行數(shù),如果操作的行數(shù)是0,說明沒有權(quán)限 if db.RowsAffected == 0 { response.Error(c, http.StatusForbidden, nil, "無權(quán)刪除該數(shù)據(jù)") return } response.OK(c, object.GetId(), "刪除成功") c.Next() } } // Permission函數(shù)的邏輯 // 根據(jù)不同 的數(shù)據(jù)范圍枚舉值,轉(zhuǎn)換成不同的Where條件 func Permission(tableName string, p *DataPermission) func(db *gorm.DB) *gorm.DB { return func(db *gorm.DB) *gorm.DB { if !config.ApplicationConfig.EnableDP { return db } switch p.DataScope { case "2": return db.Where(tableName+".create_by in (select sys_user.user_id from sys_role_dept left join sys_user on sys_user.dept_id=sys_role_dept.dept_id where sys_role_dept.role_id = ?)", p.RoleId) case "3": return db.Where(tableName+".create_by in (SELECT user_id from sys_user where dept_id = ? )", p.DeptId) case "4": return db.Where(tableName+".create_by in (SELECT user_id from sys_user where sys_user.dept_id in(select dept_id from sys_dept where dept_path like ? ))", "%/"+pkg.IntToString(p.DeptId)+"/%") case "5": return db.Where(tableName+".create_by = ?", p.UserId) default: return db } } }
總結(jié)
gorm Scopes是一個非常強大的特性,它可以讓你復(fù)用你的邏輯,在查詢時實現(xiàn)更為復(fù)雜的查詢邏輯。在使用gorm Scope時,你需要定義一個Scope函數(shù),并在查詢時應(yīng)用它。Scope函數(shù)可以被鏈?zhǔn)秸{(diào)用,并且可以接收參數(shù)。學(xué)習(xí)并掌握這個特性將會使你在編寫gorm查詢時事半功倍。
以上就是使用gorm.Scopes函數(shù)實現(xiàn)復(fù)用查詢邏輯示例的詳細內(nèi)容,更多關(guān)于gorm.Scopes復(fù)用查詢邏輯的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
golang控制結(jié)構(gòu)select機制及使用示例詳解
這篇文章主要介紹了golang控制結(jié)構(gòu)select機制及使用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-10-10Go 數(shù)據(jù)結(jié)構(gòu)之堆排序示例詳解
這篇文章主要為大家介紹了Go 數(shù)據(jù)結(jié)構(gòu)之堆排序示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-08-08Go 控制協(xié)程(goroutine)的并發(fā)數(shù)量
控制協(xié)程goroutine的并發(fā)數(shù)量是一個常見的需求,本文就來介紹一下Go 控制協(xié)程的并發(fā)數(shù)量,具有一定的參考價值,感興趣的可以了解一下2025-02-02一文深入探索Go語言中的循環(huán)結(jié)構(gòu)
在編程中,循環(huán)結(jié)構(gòu)扮演著重要的角色,它使我們能夠有效地重復(fù)執(zhí)行特定的代碼塊,以實現(xiàn)各種任務(wù)和邏輯,在Go語言中,for 是 Go 中唯一的循環(huán)結(jié)構(gòu),本文將深入探討Go語言中的for循環(huán)類型以及它們的用法2023-08-08Go創(chuàng)建Grpc鏈接池實現(xiàn)過程詳解
這篇文章主要為大家介紹了Go創(chuàng)建Grpc鏈接池實現(xiàn)過程詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-03-03xorm根據(jù)數(shù)據(jù)庫生成go model文件的操作
這篇文章主要介紹了xorm根據(jù)數(shù)據(jù)庫生成go model文件的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-12-12