我正在使用http://github.com/Go-SQL-Driver/MySQL
我想从数据库中获取类似“0000”的值votes
并将其更新为“1000”。在db.Prepare()
之前它正常工作。但在此之后,votes
的值发生了变化。除了db.Prepare()
之外,我没有对它做任何事情。我的代码是
func Vote(_type, did int, username string) (isSucceed bool) {
db := lib.OpenDb()
defer db.Close()
stmt, err := db.Prepare(
`SELECT votes
FROM users
WHERE username = ?`)
lib.CheckErr(err)
res := stmt.QueryRow(username)
stmt.Close()
var votes Votes
res.Scan(&votes)
fmt.Println(votes)//output: [48 48 48 48]
fmt.Println(string(votes))//output: 0000
isSucceed = votes.add(VoteType(_type), 1)
fmt.Println(votes)//output: [49 48 48 48]
fmt.Println(string(votes))//output: 1000
//v := []byte{[]byte(votes)[0], []byte(votes)[1], []byte(votes)[2], []byte(votes)[3]}
if isSucceed {
//Update user votes
stmt, err := db.Prepare(
`UPDATE users
SET votes = ?
WHERE username = ?`)
lib.CheckErr(err)
fmt.Println(votes)//output: [4 254 0 0]
fmt.Println(string(votes))//output: [EOT]□[NUL][NUL]
//_, _ = stmt.Exec(v, username)
_, _ = stmt.Exec(votes, username)
stmt.Close()
//Insert the vote data
stmt, err = db.Prepare(
`INSERT votes
SET did = ?, username = ?, date = ?`)
lib.CheckErr(err)
today := time.Now()
_, _ = stmt.Exec(did, username, today)
stmt.Close()
}
return
}
Votes
类型为:
type Votes []byte
type VoteType int
func (this *Votes) add(_type VoteType, num int) (isSucceed bool) {
if []byte(*this)[_type] > VOTE_MAX-1 { //beyond
isSucceed = false
} else {
[]byte(*this)[_type]++
isSucceed = true
}
return
}
最后,我将votes
的值复制到v
,效果很好。我无法理解为什么votes
的值会发生变化。我的代码有什么问题吗?任何帮助,将不胜感激。
答案 0 :(得分:0)
我认为问题出在声明中:
res.Scan(&votes)
应该是:
res.Scan((*[]byte)(&votes))
除非你做出断言,否则你传递给扫描的*投票将不会被识别为* []字节。
以下示例清楚地显示了识别问题:
package main
import "fmt"
type BYTES []byte
func test(v interface{}) {
b, ok := v.(*[]byte)
fmt.Println(b, ok)
}
func main() {
p := BYTES("hello")
test(&p)
test((*[]byte)(&p))
}
上面的代码打印:
<nil> false
&[104 101 108 108 111] true