質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.35%
Go

Go(golang)は、Googleで開発されたオープンソースのプログラミング言語です。

Q&A

解決済

1回答

627閲覧

GoでのObjectの操作について

kagepedia

総合スコア16

Go

Go(golang)は、Googleで開発されたオープンソースのプログラミング言語です。

0グッド

0クリップ

投稿2020/03/08 14:48

編集2020/03/08 14:51

以下のようなデータ操作について

model

1package model 2 3import "time" 4 5type Task struct { 6 ID int64 `json:"id"` 7 Task string `json:"task"` 8 CreatedAt time.Time `json:"createdAt"` 9 UpdatedAt time.Time `json:"updatedAt"` 10} 11

datastore

1type TaskRepository struct { 2 DB *sql.DB 3} 4 5func (tr *TaskRepository) All() (tasks []*model.Task, err error) { 6 taskList := []*model.Task 7 taskList, err = tr.DB.Query("SELECT * FROM t_task") 8 9 if err != nil { 10 log.Fatal(err) 11 } 12 defer taskList.Close() 13 14 for taskList.Next() { 15 var task model.Task 16 17 err := taskList.Scan(&task) 18 19 if err != nil { 20 log.Fatal(err) 21 } 22 23 taskList = append(taskList, &task) 24 } 25 return taskList, nil 26}

error

1type []*model.Task is not an expression

taskList := []*model.Task
の部分の記述っぽいですが、lintは出ないものでよくわかりません。

代入や宣言の方法はあっていると思うのですが、、、
分かる方いましたら教えてください。

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

ベストアンサー

まず変数の宣言方法が間違っています。

go

1taskList := []*model.Task

のエラーに関してはシンタックスが間違っています。上記の方法にならうならば

go

1taskList := []*model.Task{}

いうように宣言する必要があります。ちなみにスライスの宣言は以下のようにするのがおすすめです。

go

1var taskList []*model.Task

次に Query の戻り値は *sql.Rows 型です。これは []*model.Task 型ではないので以下の実装はコンパイルエラーになります。

go

1taskList, err = tr.DB.Query("SELECT * FROM t_task")

おそらくやりたいこと(SQL の結果を []*model.Task 型のスライスに格納して return する)は以下のようになるのではないでしょうか?

go

1func (tr *TaskRepository) All() (tasks []*model.Task, err error) { 2 var taskList []*model.Task 3 rows, err := tr.DB.Query("SELECT * FROM t_task") 4 5 if err != nil { 6 log.Fatal(err) 7 } 8 defer rows.Close() 9 10 for rows.Next() { 11 var task model.Task 12 13 err := rows.Scan(&task) 14 15 if err != nil { 16 log.Fatal(err) 17 } 18 19 taskList = append(taskList, &task) 20 } 21 return taskList, nil 22}

余談ですが、rows.Next() での呼び出し時にエラーが発生する場合があるので、以下の追加した部分のようにエラーをチェックするのを推奨します。

diff

1func (tr *TaskRepository) All() (tasks []*model.Task, err error) { 2 var taskList []*model.Task 3 rows, err := tr.DB.Query("SELECT * FROM t_task") 4 5 if err != nil { 6 log.Fatal(err) 7 } 8 defer rows.Close() 9 10 for rows.Next() { 11 var task model.Task 12 13 err := rows.Scan(&task) 14 15 if err != nil { 16 log.Fatal(err) 17 } 18 19 taskList = append(taskList, &task) 20 } 21+ if err := rows.Err(); err != nil { 22+ return nil, err 23+ } 24 return taskList, nil 25}

少し古い資料にはなりますが Go database/sql tutorial Handling Errors の Errors From Iterating Resultsets にも同様の内容が記載されています。こちらは公式の Wiki のリンクに記載されている資料になります。

投稿2020/03/08 15:24

編集2020/03/08 16:18
d_tutuz

総合スコア730

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

kagepedia

2020/03/09 14:16

ありがとうございます。 エラーはなく動作することができました。 戻り値を処理すると、 &{0 0xc000010030 0 {0 0} [] map[] 0 0 0xc0000840c0 0xc00006c2a0 false map[] map[] 0 0 0 <nil> 0 0 0 0x1131ab0} このような表示が出てきます。 アドレスが表示されているのでしょうか? 別ファイルでAll()関数を呼んでいますでいます。
d_tutuz

2020/03/09 21:39

All() で tasks []*model.Task のスライスが帰ってくると思いますが、それを fmt.Println(tasks) などとしているのでしょうか??
kagepedia

2020/03/10 11:23

インターフェースを作成し type TaskRepository interface { All() ([]*model.Task, error) } アプリケーション層 type TaskApplication struct { Repo repository.TaskRepository } func (ta *TaskApplication) All() (tasks []*model.Task, err error) { tasks, err = ta.Repo.All() return } コントローラー type TaskController struct { Ta application.TaskApplication // Logger application.Logger } func (tc *TaskController) ServeHTTP(w http.ResponseWriter, r *http.Request) { fmt.Fprint(w, "Hello World from TaskGo.\n") // tc.Logger.LogAccess("%s %s %s\n", r.RemoteAddr, r.Method, r.URL) taskList, _ := tc.Ta.All() res, err := json.Marshal(&taskList) fmt.Printf("%T", res) if err != nil { fmt.Println("n = nil, Invalid Error") // log.Fatal("Error: %s", err) // tc.Logger.LogError("%s", err) } w.Header().Set("Content-Type", "application/json") w.WriteHeader(500) fmt.Fprint(w, string(res)) } ルーティング処理 var ( Tc = &controller.TaskController{} ) func InitRouting() { http.Handle("/task", Tc) } こんな感じの作りにしています。
d_tutuz

2020/03/11 00:08 編集

fmt.Printf("%T", res) の内容であれば、res の型([]byte)が表示されると思いますよ。 ちなみに上記の変数 res はバイトのスライスです。
kagepedia

2020/03/11 10:39

ここでstring型にキャストしてるんですがダメなんですかね? fmt.Fprint(w, string(res))
d_tutuz

2020/03/11 12:15

特に問題ないかと思います。 ひとつひとつの挙動をデバッグ(printデバッグなど)で確認するのが早いと思いますよ
kagepedia

2020/03/11 14:18

ありがとうございます!
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.35%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問