golang добавить проблему внутри цикла for
Проблема:
append
внутри Users()
цикл for ниже добавляет последний элемент в users
3 раза в userRxs []*UserResolver
Ожидание:
append
следует добавить каждый элемент внутри users
в userRxs []*UserResolver
// Users return all users from Db
func (r *RootResolver) Users() ([]*UserResolver, error) {
var userRxs []*UserResolver
users := r.Db.Users()
for _, u := range users {
log.Printf("userID: %s, username: %s, email: %s, password: %s", u.UserID, u.Username, u.Email, u.Password)
userRxs = append(userRxs, &UserResolver{&u})
}
log.Printf("%v", userRxs)
return userRxs, nil
}
внутри цикла for, log.Printf
печатает это
userID: 0374402a-3dc4-48da-86c4-949905ccc26c, username: sunnysan, email: sunnysan@gmail.com, password: 12345678
userID: 53f21c4f-2cd8-4e67-b3e9-5ef344806230, username: sunnysan2, email: sunnysan2@gmail.com, password: 12345678
userID: 0a47d3af-03dc-4050-a028-7a41599af497, username: sunnysan3, email: sunnysan3@gmail.com, password: 12345678
после цикла for, log.Printf("%v", userRxs)
печатает это
[
User {
userID: 0a47d3af-03dc-4050-a028-7a41599af497,
username: sunnysan3,
email: sunnysan3@gmail.com,
password: 12345678
}
User {
userID: 0a47d3af-03dc-4050-a028-7a41599af497,
username: sunnysan3,
email: sunnysan3@gmail.com,
password: 12345678
}
User {
userID: 0a47d3af-03dc-4050-a028-7a41599af497,
username: sunnysan3,
email: sunnysan3@gmail.com,
password: 12345678
}
]
вот весь файл для большего контекста
package main
import (
"fmt"
"log"
graphql "github.com/graph-gophers/graphql-go"
)
/*
* User GQL type
type User {
userID: ID!
username: String!
email: String!
password: String!
}
*/
// User type should match the exact shape of the schema commented above
type User struct {
UserID graphql.ID
Username string
Email string
Password string
}
// RootResolver ingests Db to run queries (getters) against it
type RootResolver struct {
*Db
}
// String prints pretty structs ie. fmt.Printf can't do it on its own
func (r *UserResolver) String() string {
return fmt.Sprintf("User{userID: %s, username: %s, email: %s, password: %+v}",
r.u.UserID, r.u.Username, r.u.Email, r.u.Password,
)
}
// Users return all users from Db
func (r *RootResolver) Users() ([]*UserResolver, error) {
var userRxs []*UserResolver
users := r.Db.Users()
for _, u := range users {
log.Printf("userID: %s, username: %s, email: %s, password: %s", u.UserID, u.Username, u.Email, u.Password)
userRxs = append(userRxs, &UserResolver{&u})
}
log.Printf("%v+", userRxs)
return userRxs, nil
}
// User returns a single user from Db
func (r *RootResolver) User(args struct{ UserID graphql.ID }) (*UserResolver, error) {
// Find user:
u, err := r.Db.User(args.UserID)
if err != nil {
// Didn’t find user:
return nil, nil
}
return &UserResolver{&u}, nil
}
// UserResolver ingests properties from User
type UserResolver struct{ u *User }
// UserID returns the userId of the user
func (r *UserResolver) UserID() graphql.ID {
return r.u.UserID
}
// Username returns the username of the user
func (r *UserResolver) Username() string {
return r.u.Username
}
// Email returns the email of the user
func (r *UserResolver) Email() string {
return r.u.Email
}
// Password returns the password of the user
func (r *UserResolver) Password() string {
return r.u.Password
}
r.Db.Users()
метод
// Users correspond to Users query from graphql
func (d *Db) Users() []User {
rows, err := d.Query("SELECT * FROM users")
Check(err, "stmt.Query")
// Create User struct for holding each row's data
var r User
// Create slice of Users for our response
users := []User{}
// Copy the columns from row into the values pointed at by r (User)
// order matters here from db table
defer rows.Close()
for rows.Next() {
err = rows.Scan(
&r.UserID,
&r.Username,
&r.Email,
&r.Password,
)
Check(err, "rows.Next")
users = append(users, r)
}
return users
}
1 ответ
Решение
Переменная диапазона перезаписывается на каждой итерации, и &u
та же. Итак, вы добавляетеUserResolver
содержащие один и тот же адрес несколько раз. Вам необходимо использовать локальную копию этой переменной. Попробуй это:
for _, u := range users {
u:=u // Make a copy of the variable and redeclare it
userRxs = append(userRxs, &UserResolver{&u})
}