2017-02-19 7 views
2

私はnodejsの開発者です。私は、頻繁に使用されるlibsおよび設定オプションへの参照を保持する設定パッケージ/オブジェクトを保持する私のアプリケーションの構造体を使用します。一般的に、この設定オブジェクトはデータベース接続も保持しており、私のアプリケーションに到達することができます。実行時エラー:パブリックポインタの無効なメモリアドレスまたはnilポインタの逆参照

私はこれと同じようにビルドしようとしましたが、ひどく失敗しました。

私の計画は、私のconfig構造体の参照を保持するパブリック変数を構築することでした。私は私のConfig.Databaseを呼び出すしようとすると、しかし、私はこのパニックを得る:ここで

2017/02/19 14:05:44 http: panic serving 127.0.0.1:53554: runtime error: invalid memory address or nil pointer dereference 
    goroutine 50 [running]: 
    net/http.(*conn).serve.func1(0xc42027c000) 
     /usr/local/go/src/net/http/server.go:1491 +0x12a 
    panic(0x9f45c0, 0xc42000c100) 
     /usr/local/go/src/runtime/panic.go:458 +0x243 
    main.SignUp(0xc4202780e0) 
     /home/attila/dev/gopath/src/github.com/attilasatan/helloiris/handlers.go:31 +0x258 
    github.com/kataras/iris.HandlerFunc.Serve(0xafaf00, 0xc4202780e0) 
     /home/attila/dev/gopath/src/github.com/kataras/iris/http.go:211 +0x30 
    github.com/kataras/iris.(*Context).Do(0xc4202780e0) 
     /home/attila/dev/gopath/src/github.com/kataras/iris/context.go:152 +0x4d 
    github.com/kataras/iris.(*serveMux).BuildHandler.func1(0xc4202780e0) 
     /home/attila/dev/gopath/src/github.com/kataras/iris/http.go:1059 +0x6ea 
    github.com/kataras/iris.(*Framework).Build.func1.1(0xd87e20, 0xc4202701a0, 0xc420284000) 
     /home/attila/dev/gopath/src/github.com/kataras/iris/iris.go:411 +0x72 
    net/http.HandlerFunc.ServeHTTP(0xc420235080, 0xd87e20, 0xc4202701a0, 0xc420284000) 
     /usr/local/go/src/net/http/server.go:1726 +0x44 
    net/http.serverHandler.ServeHTTP(0xc420089f80, 0xd87e20, 0xc4202701a0, 0xc420284000) 
     /usr/local/go/src/net/http/server.go:2202 +0x7d 
    net/http.(*conn).serve(0xc42027c000, 0xd88820, 0xc42015c200) 
     /usr/local/go/src/net/http/server.go:1579 +0x4b7 
    created by net/http.(*Server).Serve 
     /usr/local/go/src/net/http/server.go:2293 +0x44d 
    2017/02/19 14:05:44 http: panic serving 127.0.0.1:53560: runtime error: invalid memory address or nil pointer dereference 
    goroutine 51 [running]: 
    net/http.(*conn).serve.func1(0xc42027c180) 
     /usr/local/go/src/net/http/server.go:1491 +0x12a 
    panic(0x9f45c0, 0xc42000c100) 
     /usr/local/go/src/runtime/panic.go:458 +0x243 
    main.SignUp(0xc4202ac070) 
     /home/attila/dev/gopath/src/github.com/attilasatan/helloiris/handlers.go:31 +0x258 
    github.com/kataras/iris.HandlerFunc.Serve(0xafaf00, 0xc4202ac070) 
     /home/attila/dev/gopath/src/github.com/kataras/iris/http.go:211 +0x30 
    github.com/kataras/iris.(*Context).Do(0xc4202ac070) 
     /home/attila/dev/gopath/src/github.com/kataras/iris/context.go:152 +0x4d 
    github.com/kataras/iris.(*serveMux).BuildHandler.func1(0xc4202ac070) 
     /home/attila/dev/gopath/src/github.com/kataras/iris/http.go:1059 +0x6ea 
    github.com/kataras/iris.(*Framework).Build.func1.1(0xd87e20, 0xc4202a60d0, 0xc4202840f0) 
     /home/attila/dev/gopath/src/github.com/kataras/iris/iris.go:411 +0x72 
    net/http.HandlerFunc.ServeHTTP(0xc420235080, 0xd87e20, 0xc4202a60d0, 0xc4202840f0) 
     /usr/local/go/src/net/http/server.go:1726 +0x44 
    net/http.serverHandler.ServeHTTP(0xc420089f80, 0xd87e20, 0xc4202a60d0, 0xc4202840f0) 
     /usr/local/go/src/net/http/server.go:2202 +0x7d 
    net/http.(*conn).serve(0xc42027c180, 0xd88820, 0xc42015c480) 
     /usr/local/go/src/net/http/server.go:1579 +0x4b7 
    created by net/http.(*Server).Serve 

は私のconfigureファイルです。ご覧のとおり、私はをredis接続に使用しています。私はConfig.Databaseを使用します。ここで、

configure.go

package main 

import (
    "fmt" 
    "strconv" 
    "time" 

    "github.com/tideland/golib/redis" 
) 

/*Configuration is the main type of app configuration */ 
type Configuration struct { 
    Database *redis.Connection 
} 

/*Config is app configuration holder */ 
var Config *Configuration 

/*Configure handles database connection */ 
func Configure() (*Configuration, error) { 

    db, err := redis.Open(redis.TcpConnection("127.0.0.1:6379", 30*time.Second)) 

    if err != nil { 
     fmt.Printf("Database connection error") 
     return nil, err 
    } 
    conn, err := db.Connection() 

    n, _ := conn.DoInt("INCR", "IDIDID") 

    fmt.Printf(strconv.Itoa(n)) 

    if err != nil { 
     fmt.Printf("Database connection error") 
     return nil, err 
    } 

    /*Config is the main configuration object*/ 

    Config := &Configuration{conn} 

    return Config, err 
} 

そしてここです。

handlers.go

func SignUp(ctx *iris.Context) { 
    mail := ctx.FormValue("email") 
    password := ctx.FormValue("password") 
    passwordConfirm := ctx.FormValue("password-confirm") 

    if password != passwordConfirm { 
     ctx.RenderWithStatus(iris.StatusBadRequest, "400.html", ErrorPageData{"passwords dont match"}) 
    } else { 

     user := User{mail, password, 0} 
     db := Config.Database 

     userID, err := db.DoInt("INCR", "HELLOIRIS:ID") 

     if err != nil { 
      ctx.RenderWithStatus(iris.StatusBadRequest, "400.html", ErrorPageData{"passwords dont match"}) 
     } else { 

      user.ID = userID 

      fmt.Printf("SAVED") 

      ctx.Render("signup-success.html", nil) 
     } 
     ctx.JSON(200, user) 
    } 
} 

私はこのようなConfigure機能を変更し、これが失敗した後:

configure.go

func Configure() (Config *Configuration, err error) { 

    if Config != nil { 
    return 
    } 
} 

と私は、ハンドラの使用状況を変え

handlers.go

config, err := Configure() 

if err != nil { 
    ctx.RenderWithStatus(iris.StatusBadRequest, "400.html", ErrorPageData{"try again later"}) 
    return 
} 

user := User{mail, password, 0} 
db := config.Database 

...とすべてが完璧に動作し始めます。

なぜ私は単純になぜ理解していないのですか...パブリックポインタを使用すると、なぜこのパニックが発生し、関数から同じポインタを返したのですか?

答えて

4
Config := &Configuration{conn} 

この行は、作成する新しいローカル変数Configので、グローバルConfigが何に設定されることはありません。 :を削除して新しい変数を作成せず、代わりにグローバルConfigを使用してください。

1

Configグローバル変数が設定されていないか、またはnil値を保持しているため、この問題が発生しています。エラーが言ったように:

runtime error: invalid memory address or nil pointer dereference

を確認してくださいあなたのmain機能

に呼び出されるハンドラの前に func Configure()と呼ばれ、あなたはこのようになり、グローバル変数にconnのRedisを割り当てるときtkausl答えとして:を削除しますに設定してください。goファイル:

Config = &Configuration{conn} 
関連する問題