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

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

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

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

Q&A

解決済

2回答

5348閲覧

Goで引数にinterfaceを指定したとき、そのinterfaceの中身がsliceであることを知らせたい

shogiOtakku

総合スコア123

Go

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

0グッド

1クリップ

投稿2018/04/17 18:39

例えば、

go

1package main 2 3import ( 4 "fmt" 5) 6 7func main() { 8 fmt.Println("Hello, playground") 9 a := []int{1, 2, 3} 10 b := []string{"a", "b", "c"} 11 targets := []interface{}{a, b} 12 printSlices(targets) 13} 14 15func printSlices(targets []interface{}) { 16 c := len(targets) 17 18 for i := 0; i < c; i++ { 19 fmt.Println(targets[i]) 20 } 21}

とした時、[]interface{}というスライスの中には[]intや[]string
といったスライスが入っています。

ただこの実装だとprintSlicesにわたす[]interface{}には単なるintやstringも入れることができます。
printSlicesの中でreflectを使ってスライスか確認することは可能ですが、その前にスライスでないものが[]interface{}に入る時点でエラーを通知されるようなそんな仕組みはできないでしょうか。

例えば

func printSlices(targets []sliceableInterface{}) {

というのを理想としています。

よろしくお願いします。

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

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

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

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

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

guest

回答2

0

仮にlibパッケージとします

go

1package lib 2type Sliceable interface { 3 isSlice() 4} 5type SliceBase struct{} 6func (s SliceBase) isSlice(){}

上記のように、小文字始まりのメソッドをSliceableインターフェースに定義し、
別途そのインターフェースを満たすSliceBaseクラスを公開しておきます。

上記のパッケージとはべつから

go

1import "####/####/lib" 2type MyStrings struct { 3 lib.SliceBase 4 []string 5}

というようなユーザー定義型を構築するようにすると、
libとは異なるパッケージからはSliceableインターフェースを満たすものは作れなくなります。

スライス型そのものをこのように扱うのはできませんが、
構造体でラップしてもアプリの要件を満たすメソッドを生やすことはできると思います。

投稿2018/04/25 01:00

nobonobo

総合スコア3367

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

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

shogiOtakku

2018/04/26 14:33

別解、ありがとうございます。 パッケージ単位で縛るという手法、目から鱗です。 納得いたしました!
nobonobo

2018/04/27 03:57

また、SliceBaseの定義をinternalパッケージにいれるか小文字始まりにして非公開にすると libパッケージにしかSliceableを満たす実装は書けなくなります。
guest

0

ベストアンサー

sliceableInterfaceに期待する機能を明確に列挙しましょう。
例えばsliceableInterfaceにはGetにて要素にアクセスするのみを
期待する場合、以下のように書けます。

go

1 2import ( 3 "fmt" 4) 5 6type sliceableInterface interface { 7 Get(n int) interface{} 8} 9 10type IntSlice []int 11 12func (s IntSlice) Get(n int) interface{} { return s[n] } 13 14type StringSlice []string 15 16func (s StringSlice) Get(n int) interface{} { return s[n] } 17 18func main() { 19 fmt.Println("Hello, playground") 20 a := []int{1, 2, 3} 21 b := []string{"a", "b", "c"} 22 targets := []Sliceable{} 23 targets = append(targets, IntSlice(a)) 24 targets = append(targets, StringSlice(b)) 25 printSlices(targets...) 26} 27 28func printSlices(targets []sliceableInterface) { 29 c := len(targets) 30 31 for i := 0; i < c; i++ { 32 fmt.Println(targets[i]) 33 } 34}

投稿2018/04/18 00:55

nobonobo

総合スコア3367

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

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

shogiOtakku

2018/04/20 10:17

nobonobo様 返事が遅くなり申し訳ございません。ご回答ありがとうございます。 interfaceを定義して対象となるslicableInterfaceにそのinterfaceの関数を定義する方法について質問ですが、 上記の例であればGetはダミーとして働いているのでしょうか。 または配列の要素を取得するtargets[i]で使われるものになるのでしょうか。
nobonobo

2018/04/23 02:01

できれば、「targetsをどのように利用するのか」を明確にして 必要な機能をslicableInterfaceに列挙すべきところです。 targetsの利用予定が「文字列化して表示する」だけであれば 以下の定義で良いでしょう。 type slicableInterface interface { String() string } 「targetsをどのように利用するのか」はアプリ作成者が定義することです。 これを明確にすれば自ずと必要なインターフェースは決まってくるはずです。 (インターフェース定義の中身が空になることが無くなるはず)
shogiOtakku

2018/04/23 13:56

ご返事ありがとうございます。 関数(またはメソッド)の中でslicableInterfaceの使われ方を明確化し、 それを機能にするということ、理解しました。 実はinterfaceで限定させるという方法は思いついていました。 ただ、interfaceにしてもsliceでないtypeにそのinterfaceの機能を実装 してしまえばtargetsに入れられると思ってより良い方法がないか模索していました。(開発はグループで行うことを想定しています) 現状だとやはりinterfaceで限定し、コメントで補完するのがよさそうですね。
nobonobo

2018/04/23 23:18

そうして差し込まれたたまたまインターフェースを満たしてしまった実装はちゃんと利用でき期待した挙動を得ることができるはずですので問題にはならないと思います。 また、もっとインターフェースを満たすのが困難にする手法もあります。別解します。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問