185
118

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 3 years have passed since last update.

【Golang】Golangのinterfaceで知っておくとお得なTips

Last updated at Posted at 2018-05-20

概要

  • 去年の8月に社会人エンジニアになり、今年の2月までAPIサーバーの開発をRuby on Railsで行なっていた新卒Rubyistです。
  • 3月にサーバーサイドを全てGolangで行う会社に転職しました。今までRailsでの業務が多かったため、interface{}という概念がどうしても身につきませんでしたが、ある程度業務に慣れ、戦い続けて学んだことをまとめようと思いました。
  • Golangのinterfaceについては知見は多くあるのですが、今年はアウトプットを大事にしていきたいという目標から、記事にまとめました。

下に読む前に

  • Golangのインタフェースは型の1つです。
  • stringint64 が関数を持っているように、インタフェースで型を宣言してその型に関数を持たせることができます。
  • 構造体がインタフェースで宣言されているGetRadius関数を持つと、この構造体の型はCircleになります。
interface1.go
type(
     Circle interface{
          GetRadius() int64
     }

     CircleImpl struct{}
)

// NewCircle が `GetRaidus`関数を持っているため、型は`Circle`になる。
func NewCircle() *CircleImpl {
     return &CircleImpl
}

func (*CircleImpl) GetRadius() int64{
     return int64(1)
}

golangに慣れている人は、当たり前だと思われるかもしれませんが、interface{} の概念を業務で使わないRubyistにとっては上の点を理解するのとしないのとでは、大きな差があったたため、記事の一番上に書かせていただきました。上のことを踏まえると、interface{}の理解度が深まると思います。

戦い続けて学んだこと

1.どんな型でも受け入れてくれるなんでも屋さん

  • Golangのinterfaceではどんな型でも受け取れることができます。
interface1.go
package main
func main() {
  intValue := int64(10) 
  strValue := "go interface"
  PrintAnyType(intValue) // => 10
  PrintAnyType(strValue) // => "go interface" 

  var interface1 interface{} = "interface1"
  fmt.Println(interface1) // => interface1
  interface1 = uint8(2)
  fmt.Println(interface1) // => 2 
}
// PrintAnyType print any type of variable.
// 引数の型がinterface{}であるために、どんな型の変数でもPrintAnyType関数は受け入れてくれる。
func PrintAnyType(variable interface{}){
  fmt.Println(variable)
}
  • なんでこれを書こうと思ったかというと、今まで動的言語でのAPIサーバーの開発が多かったため、「型が存在する言語でJSONを返すときの処理ってどうなっているだろ?」と思ったところ、気になって会社のコードを調べて、下のコードを見つけたのがきっかけでした。
  • 下のようにマップのキーは string だけど、 値はinterface{} になっているので、いちいち値の型をすべて1つの型に揃える必要がなく、JSONを作ることができるのかと納得しました。(会社では、echoのフレームワークを使っています。)
interface2.go
func JSONHTTPSuccessHandler(data interface{}, c echo.Context) error {
    return c.JSON(http.StatusOK, data)
}

2. 構造体がinterfaceの型を持つと、元々あった型の情報や構造体のメソッドやフィールドがなくなってしまう。

  • 上の1みたいに、interface{}型で宣言されている変数に値を代入して、型に沿った通りの動きをしてくれると思ったところ、下のようなコードを書いたときに、コンパイルが通らないってことがあります。
  • なぜコンパイルが通らないかというと、strSliceInterfaceは**Join関数の引数として渡せれるときに型がinterface{}となっている**からです。
  • そのため、strings.Joinが期待している []string{} とはならず、コンパイルエラーが発生してしまいます。
interface3.go
package main

import (
    "fmt"
    "strings"
)

func main() {
    var strSliceInterface interface{}
    strSliceInterface = []string{"hoge", "fuga"}
    fmt.Println(strings.Join(strSliceInterface, ", ")) 
    // => cannot use strSliceInterface (type interface {}) as type []string in argument to strings.Join: need type assertion
    // => エラー文が言っているように、strSliceInterfaceの型は`interface{}`となっているため、期待している型が渡されていない。
}
  •  interface{}がなんでも屋さんであるがゆえ、ただ代入するときはコンパイルが通ると思いますが、実際に使用するとなったときは、型がinterface{}になっているので、気をつけましょう。

3. interface{}に代入してしまい、下の型の情報を失った/ない変数を元の型に戻す方法

  • 2.のように、変数に代入するときはinterface{} を使ってしまったけれど、interface{}として代入する前の型を取り戻したいというときには、**変数.(型)**と書くことで型の情報を取り戻すことができます。
    • 上のinterface4.goに、 strSlice := strSliceInterface.([]string) と加えることでstrSlice[]stringの型を渡せます。
  • これを Type Assertion と言います。
  • 型変換が上手くいったかどうかをしっかり調べるために、ok(bool)true になっているかをしっかり確認しましょう。
interface4.go
package main
import (
    "fmt"
    "strings"
)
func main() {
    var strSliceInterface interface{}
    strSliceInterface = []string{"hoge", "fuga"}
    strSlice, ok := strSliceInterface.([]string) // => stringのスライスに戻している。 
    fmt.Println(ok) // => true ここが falseのときは、type assertionは失敗している。
    if ok {
        fmt.Println("type assertion succeeded.")
    }
    fmt.Println(strings.Join(strSlice, ", ")) // => hoge, fuga
}

応用

  • 上のType Assertionの応用がTour of Goにあるため、それを参考にします。
  • switch + interface{}.(type) で期待している型が渡ってきたときに特定の処理を実行することができます。
interface6.go
package main

import "fmt"

func do(i interface{}) {
    switch v := i.(type) {
    case int:
        fmt.Printf("Twice %v is %v\n", v, v*2)
    case string:
        fmt.Printf("%q is %v bytes long\n", v, len(v))
    default:
        fmt.Printf("I don't know about type %T!\n", v)
    }
}

func main() {
    do(21)
    do("hello")
    do(true)
    // => Twice 21 is 42
    // => "hello" is 5 bytes long
    // => I don't know about type bool!
}

4. 共通点がない複数の構造体に新規で型を持たせることができる。

  • interface を利用することで、Golangでダックタイピングをすることができます。
  • ダックタイピングについては知見が多くあるので、詳しくは書きませんが、interface{} を使うことで複数の構造体を同じ1つの型に変換します。
    • ads 変数に代入する前に構造体が必要な関数を持っているのかをエディタが教えてくれるため、コンパイルを実行する前の関数の追加漏れがなくなりました。
  • interface{}がないRubyだと、そのオブジェクトに必要な関数があるかどうかは実行しないとわからないため、インタフェースで先に約束が決められると、先読みしながらコードをかけるので便利だと感じました。

Before

interface3.go
package main

import "fmt"

type (
    // AdVideo ad video response struct
    AdVideo struct {
        VideoURL string `json:"video_url"`
        AdType   int64  `json:"ad_type"`
    }
    // AdPoster ad poster response struct
    AdPoster struct {
        PosterURL string `json:"poster_url"`
        AdType    int64  `json:"ad_type"`
    }
)

func main() {
    var ads []interface{}
    ads = append(ads, NewAdVideo())
    ads = append(ads, NewAdPoster())
    fmt.Println(ads) // => [{video url 1} {poster url 3}]
}

func NewAdVideo() AdVideo {
    return AdVideo{VideoURL: "video url", AdType: int64(1)}
}

func NewAdPoster() AdPoster {
    return AdPoster{PosterURL: "poster url", AdType: int64(3)}
}

After

interface4.go
package main

import "fmt"

type (
    // Adインタフェース
    Ad interface {
        GetAdType() int64
    }
    Ads []Ad
    // AdVideo ad video response struct
    AdVideo struct {
        VideoURL string `json:"video_url"`
        AdType   int64  `json:"ad_type"`
    }
    // AdPoster ad poster response struct
    AdPoster struct {
        PosterURL string `json:"poster_url"`
        AdType    int64  `json:"ad_type"`
    }
)

func main() {
    var ads Ads
    ads = append(ads, NewAdVideo())
    ads = append(ads, NewAdPoster())
    for _, ad := range ads {
        fmt.Println(ad)
    }
}

func NewAdVideo() AdVideo {
    return AdVideo{VideoURL: "video url", AdType: int64(1)}
}

// => Adのインタフェース型を持つために、`GetAdType`を実装した。
func (v AdVideo) GetAdType() int64 {
    return v.AdType
}

func NewAdPoster() AdPoster {
    return AdPoster{PosterURL: "poster url", AdType: int64(3)}
}

// => Adのインタフェース型を持つために、`GetAdType`を実装した。
func (p AdPoster) GetAdType() int64 {
    return p.AdType
}

5. interface{}を構造体に埋め込み

  • こちらのinterface{}の使い方も記事を書く際に多くの知見があると知ったので特にここでは細かく書くことはしません。
  • PHPやJavaでは、implements を使ってinterface を取り込みますが、Golangではフィールドがinterface{}の型を持つようにして、interface{}を取り組むという言語仕様になっています。
  • この言語仕様を理解したときに便利だなと感じました。
    • 例えば、新しいAPI実装で新たに構造体を作成するときに、必要なメソッドだけ揃っていたら正常に動くため、とりあえずinterface{}で宣言された関数を新しい構造体に実装すればいいだけと感じました。
    • 上の4と似ていますが、やることリストが最初から決まっていて、リストにある関数を定義することでやることが完了していく感覚になったので、わかりやすいなと感じました。
interface5.go
package main

import "fmt"

type (
    // TagClient tag clientインタフェース
    TagClient interface {
        GetTag()
    }

    // TagClientImpl TagClientの実際の処理
    TagClientImpl struct {
    }

    // TagClientImplV2 TagClientの実際の処理
    TagClientImplV2 struct {
    }

    // ArticleClientImpl Tagという名前で、TagClientインタフェースを持つ。
    ArticleClientImpl struct {
        Tag TagClient
    }
)

func main() {
    articleClientImpl := NewArticleClientImpl()
    // 記事のタグを取得するには、`Tag`のインタフェースを経由して、`GetTag`を呼び出す。
    articleClientImpl.Tag.GetTag()
}

func NewArticleClientImpl() *ArticleClientImpl {
    // Tagフィールドに渡すものをNewTag()からNewTagV2()に変更するだけ。簡単...!!
    return &ArticleClientImpl{Tag: NewTagV2()}
}

func NewTag() TagClient {
    return &TagClientImpl{}
}

func (c *TagClientImpl) GetTag() {
    fmt.Println("Here is article tag")
}

func NewTagV2() TagClient {
    return &TagClientImplV2{}
}

func (c *TagClientImplV2) GetTag() {
    fmt.Println("Here is article tag V2")
}

おまけ - カスタムエラーの作り方

  • golangのerrorってどんな中身なのかを調べようと思い、ドキュメンテーションを読んだところ、なんとinterface{}でした。
error.go
type error interface {
    Error() string
}
  • ということは、構造体にError() string関数を持たせれば、その構造体をerror型として扱ってくれるのか...?
  • 実験で書いてみたら、なんと自前で用意した構造体が見事error型となりました...!
  • Error() stringさえあれば、error型になるので、いろいろな場面のエラー構造体が作成できると思います。
interface6.go
package main

import (
    "fmt"
    "time"
)

type (
    Article struct {
        postedAt time.Time
    }
    // カスタムエラー構造体
    ArticleNotFoundError struct {
        Message string
    }
)

func main() {
    article, err := GetArticle()
    if err != nil {
        fmt.Println(err.Error()) // => "article not found"
    }
}

func GetArticle() (*Article, error) {
    return nil, NewArticleNotFoundError("article not found")
}

func NewArticleNotFoundError(message string) *ArticleNotFoundError {
    return &ArticleNotFoundError{Message: message}
}

// このメソッドがあるおかげで、ArticleNotFoundError構造体の型はerrorになり、GetArticleの第2引数の型はerrorになりました。
func (e *ArticleNotFoundError) Error() string {
    return e.Message
}

まとめ

  • 長くなりましたが、この記事を短くまとめると、golangでの interface{} は型である になります。
    • 転職する前にgolangのinterface{}のことを勉強しておきましたが、実際に業務に入って書いてみると理解できていないということもあり、多く戦わないといけなくなりました。
    • (PHPやJavaには軽く触れていて、インターフェースという概念を全く知らないわけではありませんでしたが、いまいち「これを使う必要はあるのか...?」と腑に落ちていないことが多かったです...。)
  • 業務で触っているみると、便利なものだと感じたので、戦って生き延びたかいがあったと思いました。これからも適材適所で使っていきたいと思います。
  • GWで時間ができたため学んだことをまとめましたが、今後新たに学ぶことがあったら、追記していきます。
185
118
1

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
185
118

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?