我跟着this tutorial了解了如何使用PostgreSQL,Gorilla Mux和GORM设置基本API。
这是我的应用:
package main
import (
"encoding/json"
"net/http"
"github.com/gorilla/mux"
"github.com/jinzhu/gorm"
_ "github.com/jinzhu/gorm/dialects/postgres"
)
var db *gorm.DB
type Ticket struct {
gorm.Model
InfoHash string
Status *int `gorm:"default:'0'"`
}
func main() {
router := mux.NewRouter()
db, err := gorm.Open("postgres", "host=localhost user=user dbname=db_development password=password sslmode=disable")
db.LogMode(true)
if err != nil {
panic(err)
}
defer db.Close()
db.AutoMigrate(&Ticket{})
router.HandleFunc("/", TicketsIndex).Methods("GET")
http.ListenAndServe(":3000", router)
}
func TicketsIndex(w http.ResponseWriter, r *http.Request) {
tickets := []Ticket{}
db.Find(&tickets)
json.NewEncoder(w).Encode(&tickets)
}
问题是当我访问localhost:3000
时,服务器只是停止而没有错误日志,只是停止并退出应用程序。它应该将存储在数据库中的票证作为JSON返回。
如果我在TicketsIndex函数中打开数据库并关闭它,我得到了它,如下所示:
func TicketsIndex(w http.ResponseWriter, r *http.Request) {
db, err := gorm.Open("postgres", "host=localhost user=user dbname=db_development password=password sslmode=disable")
tickets := []Ticket{}
db.Find(&tickets)
json.NewEncoder(w).Encode(&tickets)
defer db.Close()
}
但我认为这不是正确的方法。我也尝试将此代码移动到main函数中并且也可以工作:
tickets := []Ticket{}
db.Find(&tickets)
所以我假设它可能是未正确分配的全局变量var db *gormDB
。我做错了什么?
答案 0 :(得分:4)
当您键入db, err := ...
时,您实际上使用名为db的函数局部变量来遮蔽全局var db
,而不是分配给全局变量。您需要使用等号(=
vs :=
)来分配已定义的var。这也意味着您需要在作业之前在主范围内写var err error
,因为您不再自动从:=
获取声明。