欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

詳解Go語言如何利用高階函數(shù)寫出優(yōu)雅的代碼

 更新時(shí)間:2023年01月05日 09:38:00   作者:nil  
高階函數(shù)(Hiher-order?Function)定義為:滿足下列條件之一的函數(shù):接收一個(gè)或多個(gè)函數(shù)作為參數(shù);返回值是一個(gè)函數(shù)。本文為大家介紹了如何利用高階函數(shù)寫出優(yōu)雅的代碼,希望對(duì)大家有所幫助

前言

go項(xiàng)目中經(jīng)常需要查詢db,按照以前java開發(fā)經(jīng)驗(yàn),會(huì)根據(jù)查詢條件寫很多方法,如:

  • GetUserByUserID
  • GetUsersByName
  • GetUsersByAge

每一種查詢條件寫一個(gè)方法,這種方式對(duì)外是挺好的,對(duì)外遵循嚴(yán)格原則,讓每個(gè)對(duì)外的方法接口是明確的。但是對(duì)內(nèi)的話,應(yīng)該盡可能的通用,做到代碼復(fù)用,少寫代碼,讓代碼看起來更優(yōu)雅、整潔。

問題

在review代碼的時(shí)候,針對(duì)上面3個(gè)方法,一般寫法是

func GetUserByUserID(ctx context.Context, userID int64) (*User, error){
    db := GetDB(ctx)
    var user User
    if userID > 0 {
        db = db.Where(`userID = ?`, userID)
    }
    if err := db.Model(&User{}).Find(&user).Err; err != nil {
        return nil, err
    }
    
    return user, nil
}

func GetUsersByName(ctx context.Context, name string) (*User, error){
    db := GetDB(ctx)
    var users []User
    if name != "" {
        db = db.Where(`name like '%%'`, name)
    }
    if err := db.Model(&User{}).Find(&users).Err; err != nil {
        return nil, err
    }
    
    return users, nil
}

func GetUsersByAge(ctx context.Context, age int64) (*User, error){
    db := GetDB(ctx)
    var user User
    if age > 0 {
        db = db.Where(`age = ?`, age)
    }
    if err := db.Model(&User{}).Find(&user).Err; err != nil {
        return nil, err
    }
    
    return user, nil
}

當(dāng)User表上字段有幾十個(gè)的時(shí)候,上面類似的方法會(huì)越來越多,代碼沒有做到復(fù)用。當(dāng)有Teacher表、Class表等其他表的時(shí)候,上面的查詢方法又要翻倍。

調(diào)用方也會(huì)寫的很死,參數(shù)固定。當(dāng)要增加一個(gè)查詢條件的時(shí)候,要么改原來的函數(shù),增加一個(gè)參數(shù),這樣其他調(diào)用的地方也都要改;要么新寫一個(gè)函數(shù),這樣函數(shù)越來越多,難以維護(hù)和閱讀。

上面是青銅寫法,針對(duì)這種情況,下面介紹幾種白銀、黃金、王者寫法

白銀

將入?yún)⒍x成一個(gè)結(jié)構(gòu)體

type UserParam struct {
    ID int64
    Name string
    Age int64
}

將入?yún)⒍挤旁赨serParam結(jié)構(gòu)體中

func GetUserInfo(ctx context.Context, info *UserParam) ([]*User, error) {    
        db := GetDB(ctx)    
        db = db.Model(&User{})
        var infos []*User
        if info.ID > 0 {
            db = db.Where("user_id = ?", info.ID)    
        }
        if info.Name != "" {       
            db = db.Where("user_name = ?", info.Name)    
        }    
        if info.Age > 0 {       
            db = db.Where("age = ?", info.Age)    
        } 
        if err := db.Find(&infos).Err; err != nil {
            return nil, err
        }
        
        return infos, nil
}

這個(gè)代碼寫到這里,相比最開始的方法其實(shí)已經(jīng)好了不少,至少 dao 層的方法從很多個(gè)入?yún)⒆兂闪艘粋€(gè),調(diào)用方的代碼也可以根據(jù)自己的需要構(gòu)建參數(shù),不需要很多空占位符。但是存在的問題也比較明顯:仍然有很多判空不說,還引入了一個(gè)多余的結(jié)構(gòu)體。如果我們就到此結(jié)束的話,多少有點(diǎn)遺憾。

另外,如果我們?cè)贁U(kuò)展一下業(yè)務(wù)場景,我們使用的不是等值查詢,而是多值查詢或者區(qū)間查詢,比如查詢 status in (a, b),那上面的代碼又怎么擴(kuò)展呢?是不是又要引入一個(gè)方法,方法繁瑣暫且不說,方法名叫啥都會(huì)讓我們糾結(jié)很久;或許可以嘗試把每個(gè)參數(shù)都從單值擴(kuò)展成數(shù)組,然后賦值的地方從 = 改為 in()的方式,所有參數(shù)查詢都使用 in 顯然對(duì)性能不是那么友好。

黃金

更高級(jí)的優(yōu)化方法,是使用高階函數(shù)。

type Option func(*gorm.DB)

定義 Option 是一個(gè)函數(shù),這個(gè)函數(shù)的入?yún)㈩愋褪?gorm.DB,返回值為空。

然后針對(duì)每一個(gè)需要查詢的字段,定義一個(gè)高階函數(shù)

func UserID(ID int64) Option {    
    return func(db *gorm.DB) {       
        db.Where("`id` = ?", ID)    
    } 
}

func Name(name int64) Option {    
    return func(db *gorm.DB) {       
        db.Where("`name` like %?%", name)    
    } 
}

func Age(age int64) Option {    
    return func(db *gorm.DB) {       
        db.Where("`age` = ?", age)    
    } 
}

返回值是Option類型。

這樣上面3個(gè)方法就可以合并成一個(gè)方法了

func GetUsersByCondition(ctx context.Context, opts ...Option)([]*User, error) {
    db := GetDB(ctx)
    for i:=range opts {
        opts[i](db)
    }
    var users []User
    if err := db.Model(&User{}).Find(&users).Err; err != nil {
        return nil, err
    }
    return users, nil
}

沒有對(duì)比就沒有傷害,通過和最開始的方法比較,可以看到方法的入?yún)?strong>由多個(gè)不同類型的參數(shù)變成了一組相同類型的函數(shù),因此在處理這些參數(shù)的時(shí)候,也無需一個(gè)一個(gè)的判空,而是直接使用一個(gè) for 循環(huán)就搞定,相比之前已經(jīng)簡潔了很多。

還可以擴(kuò)展其他查詢條件,比如IN,大于等

func UserIDs(IDs int64) Option {    
    return func(db *gorm.DB) {       
        db.Where("`id` in (?)", IDs)    
    } 
}

func AgeGT(age int64) Option {    
    return func(db *gorm.DB) {       
        db.Where("`age` > ?", age)    
    } 
}

而且這個(gè)查詢條件最終是轉(zhuǎn)換成Where條件,跟具體的表無關(guān),也就是說這些定義是可以被其他表復(fù)用的。

王者

優(yōu)化到上述方法已經(jīng)可以了,但是王者一般會(huì)繼續(xù)優(yōu)化。

上述方法GetUsersByCondition只能查User表,能不能更通用一些,查任意表呢?分享GetUsersByCondition方法,發(fā)現(xiàn)如果要做到查任意表,有2個(gè)阻礙:

  • 表明是在方法中寫死的
  • 返回值定義的是[]*User,不能通用

針對(duì)第一個(gè)問題,我們可以定義一個(gè)Option來實(shí)現(xiàn)

func TableName(tableName string) Option {
    return func(db *grom.DB) {
        db.Table(tableName)
    }
}

針對(duì)第二個(gè)問題,可以將返回參數(shù)作為入?yún)?,通過引用的方式傳進(jìn)來

func GetRecords(ctx context.Context, in any, opts ...Option) {
    db := GetDB(ctx)
    for i:=range opts {
        opts[i](db)
    }
    
    return db.Find(in).Err
}

// 調(diào)用:根據(jù)user name 和age 查詢users
var users []User
if err := GetRecords(ctx, &users, TableName("user"), Name("張三"), Age(18)); err != nil {
    // TODO
}

總結(jié)

這里通過對(duì) grom 查詢條件的抽象,大大簡化了對(duì) DB 組合查詢的寫法,提升了代碼的簡潔。

以上就是詳解Go語言如何利用高階函數(shù)寫出優(yōu)雅的代碼的詳細(xì)內(nèi)容,更多關(guān)于Go語言高階函數(shù)的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Go?錯(cuò)誤處理實(shí)踐總結(jié)示例

    Go?錯(cuò)誤處理實(shí)踐總結(jié)示例

    這篇文章主要為大家介紹了Go錯(cuò)誤處理實(shí)踐的總結(jié)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-07-07
  • Golang錯(cuò)誤處理:異常捕捉和恢復(fù)機(jī)制

    Golang錯(cuò)誤處理:異常捕捉和恢復(fù)機(jī)制

    Golang中,異常處理是通過 defer + panic + recover 的方式來實(shí)現(xiàn)的,使用 defer 可以將清理操作注冊(cè)到函數(shù)執(zhí)行完畢后執(zhí)行,而 panic 和 recover 可以用于處理異常,通過組合使用這些功能,可以實(shí)現(xiàn)更加健壯的程序
    2024-01-01
  • Go語言項(xiàng)目中使用Viper獲取配置信息詳解

    Go語言項(xiàng)目中使用Viper獲取配置信息詳解

    Viper是Go應(yīng)用的完整配置解決方案,它能處理所有類型的配置需求和配置格式,這篇文章主要介紹了Go項(xiàng)目中使用Viper獲取配置信息,需要的可以參考下
    2024-04-04
  • go for range遍歷二維數(shù)組的示例

    go for range遍歷二維數(shù)組的示例

    今天小編就為大家分享一篇關(guān)于go for range遍歷二維數(shù)組的示例,小編覺得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來看看吧
    2019-04-04
  • go語言如何導(dǎo)入和使用包示例詳解

    go語言如何導(dǎo)入和使用包示例詳解

    這篇文章主要為大家介紹了go語言如何導(dǎo)入和使用包示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-08-08
  • golang struct 實(shí)現(xiàn) interface的方法

    golang struct 實(shí)現(xiàn) interface的方法

    這篇文章主要介紹了golang struct 實(shí)現(xiàn) interface的方法,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2018-07-07
  • GO語言實(shí)現(xiàn)的http抓包分析工具pproxy介紹

    GO語言實(shí)現(xiàn)的http抓包分析工具pproxy介紹

    這篇文章主要介紹了GO語言實(shí)現(xiàn)的http抓包分析工具pproxy介紹,本文同時(shí)對(duì)比了Fiddler、Charles等抓包軟件,需要的朋友可以參考下
    2015-03-03
  • Go并發(fā)讀寫文件、分片寫、分片下載文件的實(shí)現(xiàn)示例

    Go并發(fā)讀寫文件、分片寫、分片下載文件的實(shí)現(xiàn)示例

    讀寫文件在很多項(xiàng)目中都可以用到,本文主要介紹了Go并發(fā)讀寫文件、分片寫、分片下載文件的實(shí)現(xiàn)示例,具有一定的參考價(jià)值,感興趣的可以了解一下
    2024-01-01
  • 淺談JWT在GO中的使用方法及原理

    淺談JWT在GO中的使用方法及原理

    JWT是一種基于?JSON?的開放標(biāo)準(zhǔn),用于在網(wǎng)絡(luò)應(yīng)用間傳遞聲明,JWT被設(shè)計(jì)為可安全地將用戶身份驗(yàn)證和授權(quán)數(shù)據(jù)作為?JSON?對(duì)象在各個(gè)應(yīng)用程序之間傳遞,本文將詳細(xì)給大家介紹JWT原理及在Go中的用法,需要的朋友可以參考下
    2023-05-05
  • Go 字符串比較的實(shí)現(xiàn)示例

    Go 字符串比較的實(shí)現(xiàn)示例

    本文主要介紹了Go 字符串比較的實(shí)現(xiàn)示例,主要包括三種比較方式,具有一定的參考價(jià)值,感興趣的可以了解一下
    2022-01-01

最新評(píng)論