Вернуть ноль для структуры в Go - PullRequest
0 голосов
/ 05 июня 2018

Я подключаюсь к базе данных, получаю строку и затем отправляю ее обратно пользователю.Однако я хочу написать return, если не могу найти эту строку или у меня ошибка.

Поскольку я возвращаю структуру, я не могу вернуть ноль и получаюэта ошибка cannot use nil as type Item in return argument (Item - это моя структура)

Я прочитал в Интернете, что если я использую указатель в операторе возврата и возвращаю * Item вместо Item, то я смогу передать nil, когда я попытаюсьдля создания item := *Item{} я получаю следующую ошибку invalid indirect of Item literal (type Item)

Я думаю, что есть несколько решений для этого, но я не могу найти ни одного, то, что я действительно хочу знать, это:

  • Как вернуть указатель * Item вместо Item
  • Есть ли другой способ вернуть nil для структуры?

Вот мой код:

package main

import (
    "fmt"
    "github.com/aws/aws-lambda-go/lambda"

  "github.com/aws/aws-sdk-go/aws"
  "github.com/aws/aws-sdk-go/aws/session"
  "github.com/aws/aws-sdk-go/service/dynamodb"
  "github.com/aws/aws-sdk-go/service/dynamodb/dynamodbattribute"
)

type Request struct {
    Name string `json:"name"`
}

type Item struct {
  Name string `json:"name"`
  Stock int `json:"stock"`
  Price float64 `json:"price"`
}

func Handler(request Request) (Item, error) {
  sess, err := session.NewSession(&aws.Config{
    Region: aws.String("us-west-2")},
  )

  // Create DynamoDB client
  svc := dynamodb.New(sess)

  result, err := svc.GetItem(&dynamodb.GetItemInput{
    TableName: aws.String("Inventory"),
    Key: map[string]*dynamodb.AttributeValue{
      "name": {
          S: aws.String(request.Name),
      },
    },
  })

  if err != nil {
    fmt.Println(err.Error())
//     return nil, err
  }

  item := Item{}

  err = dynamodbattribute.UnmarshalMap(result.Item, &item)

  if err != nil {
    panic(fmt.Sprintf("Failed to unmarshal Record, %v", err))
//     return nil, err
  }

  if item.Name == "" {
      fmt.Println("Could not find item")
//       return nil, nil
  }

  fmt.Println("Found item:")
  fmt.Println("Name:  ", item.Name)
  fmt.Println("Stock: ", item.Stock)
  fmt.Println("Price:  ", item.Price)

    return item, nil
}

func main() {
    lambda.Start(Handler)
}

1 Ответ

0 голосов
/ 05 июня 2018

Вы неверно назначаете переменную item.Вы сказали, что пытались item := *Item{}, тогда как способ создания указателя - либо через встроенный new, либо для создания литерала и оператора address-of (&).Последний подход наиболее часто встречается на Голанге.В некоторых случаях можно использовать new, но в этом случае я бы выбрал второй подход:

Так что либо:

item := &Item{}
// or
item := new(Item)

Наконец, вы можете оставитькод, как есть, и просто вернуть указатель в конце:

item := Item{}
// some code here
return &item, nil

В случае, если вам нужно вернуть ошибку, вы все равно можете иметь return nil, err

Так что все вместе:

// return *Item instead of Item
func Handler(request Request) (*Item, error) {
   // your code here, eg:
   item := Item{}
   if err := dynamodbattribute.UnmarshalMap(result.Item, &item); err != nil {
        return nil, err
    }
    return &item, nil
}

Также можно назначить item указателем с начала

func Handler(request Request) (*Item, error) {
   // your code here, eg:
   item := &Item{}
   if err := dynamodbattribute.UnmarshalMap(result.Item, item); err != nil {
        return nil, err
    }
    return item, nil
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...