本文汇总一些工程中使用到的查询数据表的代码示例。由于是代码片段,不一定保证完整。但其思想可以参考。
问题提出
工程中经常要使用到数据表(工作多年,不可避免沦为了CRUD工具人),打交道最多的是查询,对于插入更新接触比较少,由于所涉及的数据表是生产环境的,所以不敢越雷池半步。
golangsql
解决方案
方案一:指定字段变量
func GetIntervalDetail_inner(sqldb *sql.DB, version string) (rets []conf.TestInfo) {
sqlstr := fmt.Sprintf("select * from TestTable a where a.version = '%v' order by a.ID", version)
// klog.Println("sqlstr: ", sqlstr)
results, err := sqldb.Query(sqlstr)
if err != nil {
klog.Printf("Query error: %s\n", err.Error())
return
}
for results.Next() {
var item1, item2, item3, item4, item5, item7, item9, item10 sql.NullString
var item6, item8 int
err := results.Scan(&item1, &item2, &item3, &item4, &item5,
&item6, &item7, &item8, &item9, &item10)
if err != nil {
klog.Println("scan error: ", err)
break
}
// item7 有空格,先去掉
item7.String = strings.TrimSpace(item7.String)
if item7.String == "" {
item7.String = "none"
}
var tmp conf.TestInfo
tmp.Version = item1.String
tmp.Code = item2.String
tmp.ID = item3.String
tmp.Code1 = item4.String
tmp.Code2 = item5.String
tmp.Length = item6
tmp.F1_CODE = item7.String
tmp.F1_length = item8
tmp.F2_CODE = item9.String
tmp.F2_length = item10.String
rets = append(rets, tmp)
}
return
}
即使用到结构体,其本质也没有发生变化
func GetSpecialFee_inner(sqldb *sql.DB, Version string) (rets []conf.MySpeedRate) {
sqlstr := fmt.Sprintf("select * from FoobarTable where Version=%v", Version)
// klog.Println("sqlstr: ", sqlstr)
results, err := sqldb.Query(sqlstr)
if err != nil {
klog.Printf("Query error: %s\n", err.Error())
return
}
for results.Next() {
var data conf.MySpeedRate
err := results.Scan(&data.Version, &data.Rate[0], &data.Rate[1],
&data.Rate[2], &data.Rate[3], &data.Rate[4], &data.Rate[5], &data.Rate[6], &data.Rate[7],
&data.Rate[8], &data.Rate[9], &data.Rate[10], &data.Rate[11], &data.Rate[12], &data.Rate[13],
&data.Rate[14], &data.Rate[15])
if err != nil {
klog.Println("scan error: ", err)
break
}
rets = append(rets, data)
}
return
}
方案二:泛化:结构体
func GetAllRowsPtr(sqldb *sql.DB, query string, strname interface{}) (*[]interface{}, error) {
result := make([]interface{}, 0)
rows, err := sqldb.Query(query)
if err != nil {
return &result, err
}
defer rows.Close()
s := reflect.ValueOf(strname).Elem()
leng := s.NumField()
scans := make([]interface{}, leng)
for i := 0; i < leng; i++ {
scans[i] = s.Field(i).Addr().Interface()
}
for rows.Next() {
err = rows.Scan(scans...)
if err != nil {
panic(err)
}
result = append(result, s.Interface())
}
return &result, nil
}
方案三:泛化:字符串数组
func GetAllRowsString(sqldb *sql.DB, query string) ([]string, error) {
rows, err := sqldb.Query(query)
if err != nil {
return []string{}, err
}
defer rows.Close()
var results []string
var tmp string
// 获取字段名称
tmp = ""
cols, _ := rows.Columns()
for i := range cols {
tmp += cols[i] + ","
}
results = append(results, tmp)
// 根据字段数量,指定查询scan的参数
values := make([]sql.RawBytes, len(cols))
scans := make([]interface{}, len(cols))
for i := range values {
scans[i] = &values[i]
}
for rows.Next() {
if err := rows.Scan(scans...); err != nil {
fmt.Println("Error")
return []string{}, err
}
// 组装,暂定以逗号隔开
tmp = ""
for j := range values {
tmp += string(values[j]) + ","
}
results = append(results, tmp)
}
return results, nil
}
小结
方案一是将数据表映射到结构体中,因此需要定义对应的结构体,但可以与数据表不一一对应,即根据sql语句只查询个别字段并赋值,如果不嫌麻烦,或者结构体随处需要使用,则可用此法。
panic: sql: expected 19 destination arguments in Scan, not 4
方案三在方案二基础上进行修改。会自动匹配查询sql的字段。最终组装成csv格式字符串数组。
不同方案均有优劣,按需使用即可。