2016-12-04 59 views
1

有每行對應於一個結構具有以下字段我如何循環struct的字段查詢過濾

type item struct { 
    ItemId *string `json:"item_id"` 
    OwnerId *string `json:"owner_id"` 
    Status *string `json:"status"` 
    ... // many more 
} 

數據庫在數據庫中,填寫了所有字段的所有行。現在我想要一個函數,它需要一個item對象,它的字段可能不會被填充爲輸入並返回一個SQL查詢字符串。例如,

func FindItems(filter item) string 

輸入item充當過濾器。邏輯如下(以蟒蛇風格的種類)

query = `select * from item_table` 
condition = "" 
for field, value in filter: 
    if value != nil: 
     condition = " and " if condition else " where " 
     condition += " field=value" 
query += condition 

我該怎麼辦呢?或者是否有更好的方法來進行過濾?

+0

[Go:從接口獲取所有字段]的可能重複(http://stackoverflow.com/questions/39866503/go-get-all-fields-from-an-interface/39866671#39866671)。 – icza

回答

2

您可以使用reflect包枚舉您的結構域和值:

package main 

import (
    "fmt" 
    "reflect" 
) 

type item struct { 
    ItemID *string `json:"item_id"` 
    OwnerID *string `json:"owner_id"` 
    Status *string `json:"status"` 
} 

func FindItemsQuery(filter item) string { 
    query := `select * from item_table` 
    condition := "" 
    val := reflect.ValueOf(filter) 
    for i := 0; i < val.NumField(); i++ { 
     valField := val.Field(i) 
     if !valField.IsNil() { 
      if condition != "" { 
       condition += " and " 
      } else { 
       condition += " where " 
      } 
      condition += fmt.Sprintf("%s=%v", val.Type().Field(i).Tag.Get("json"), valField.Elem()) 
     } 
    } 
    return query + condition 
} 

func main() { 
    itemID := "123" 
    item := item{ItemID: &itemID} 
    fmt.Println(FindItemsQuery(item)) // select * from item_table where item_id=123 
} 

請記住,通過另外的JSON標籤,如json:"item_id,omitempty"將打破您的查詢。您應該考慮使用自定義結構標記來定義SQL字段的名稱。

+0

爲什麼你需要'valField.Elem().interface()'而不是'valField.Elem()'? – nos

+0

@nos我沒有意識到'fmt.Sprintf'能夠處理'reflect.Value'類型,你是對的。 – KAdot