我的在线博客:http://golang.iswbm.com
我的 Github:github.com/iswbm/GolangCodingTime
1. 对方法的调用限制
接口是一组固定的方法集,由于静态类型的限制,接口变量有时仅能调用其中特定的一些方法。
请看下面这段代码
package main import "fmt" type Phone interface { call() } type iPhone struct { name string } func (phone iPhone)call() { fmt.Println("Hello, iPhone.") } func (phone iPhone)send_wechat() { fmt.Println("Hello, Wechat.") } func main() { var phone Phone phone = iPhone{name:"ming's iphone"} phone.call() phone.send_wechat() }
我定义了一个 Phone 的接口,只要求实现 call 方法即可,也就是只要能打电话的设备就是一个电话(好像是一句没用的废话)。
然后再定义了一个 iPhone 的结构体,该结构体接收两个方法,一个是打电话( call 函数),一个是发微信(send_wechat 函数)。
最后一步是关键,我们定义了一个 Phone 接口类型的 phone 对象,该对象的内容是 iPhone 结构体。然后我们调用该对象的 call 方法,一切正常。
但是当你调用 phone.send_wechat
方法,程序会报错,提示我们 Phone 类型的方法没有 send_wechat 的字段或方法。
# command-line-arguments ./demo.go:30:10: phone.send_wechat undefined (type Phone has no field or method send_wechat)
原因也很明显,因为我们的phone对象显示声明为 Phone 接口类型,因此 phone调用的方法会受到此接口的限制。
那么如何让 phone 可以调用 send_wechat 方法呢?
答案是可以不显示的声明为 Phone接口类型 ,但要清楚 phone 对象实际上是隐式的实现了 Phone 接口,如此一来,方法的调用就不会受到接口类型的约束。
修改 main 方法成如下
func main() { phone := iPhone{name:"ming's iphone"} phone.call() phone.send_wechat() }
运行后,一切正常,没有报错。
Hello, iPhone.
Hello, Wechat.
2. 调用函数时的隐式转换
Go 语言中的函数调用都是值传递的,变量会在方法调用前进行类型转换。
比如下面这段代码
import ( "fmt" ) func printType(i interface{}) { switch i.(type) { case int: fmt.Println("参数的类型是 int") case string: fmt.Println("参数的类型是 string") } } func main() { a := 10 printType(a) }
如果你运行后,会发现一切都很正常
参数的类型是 int
但是如果你把函数内的内容搬到到外面来
package main import "fmt" func main() { a := 10 switch a.(type) { case int: fmt.Println("参数的类型是 int") case string: fmt.Println("参数的类型是 string") } }
就会有意想不到的结果,居然报错了。
# command-line-arguments ./demo.go:9:5: cannot type switch on non-interface value a (type int)
这个操作会让一个新人摸不着头脑,代码逻辑都是一样的,为什么一个不会报错,一个会报错呢?
原因其实很简单。
当一个函数接口 interface{} 空接口类型时,我们说它可以接收什么任意类型的参数(江湖上称之为无招胜有招)。
当你使用这种写法时,Go 会默默地为我们做一件事,就是把传入函数的参数值(注意:Go 语言中的函数调用都是值传递的)的类型隐式的转换成 interface{} 类型。
如何进行接口类型的显示转换
上面了解了函数中 接口类型的隐式转换后,你的心里可能开始有了疑问了,难道我使用类型断言,只能通过一个接收空接口类型的函数才能实现吗?
答案当然是 No.
如果你想手动对其进行类型转换,可以像下面这样子,就可以将变量 a 的静态类型转换为 interface{} 类型然后赋值给 b (此时 a 的静态类型还是 int,而 b 的静态类型为 interface{})
var a int = 25 b := interface{}(a)
知道了方法后,将代码修改成如下:
package main import "fmt" func main() { a := 10 switch interface{}(a).(type) { case int: fmt.Println("参数的类型是 int") case string: fmt.Println("参数的类型是 string") } }
运行后,一切正常。
参数的类型是 int
3. 类型断言中的隐式转换
上面我们知道了,只有静态类型为接口类型的对象才可以进行类型断言。
而当类型断言完成后,会返回一个静态类型为你断言的类型的对象,也就是说,当我们使用了类型断言,Go 实际上又会默认为我们进行了一次隐式的类型转换。
验证方法也很简单,使用完一次类型断言后,对返回的对象再一次使用类型断言,Goland 立马就会提示我们新对象 b 不是一个接口类型的对象,不允许进行类型断言。
总结
免责声明:本站资源来自互联网收集,仅供用于学习和交流,请遵循相关法律法规,本站一切资源不代表本站立场,如有侵权、后门、不妥请联系本站删除!
《魔兽世界》大逃杀!60人新游玩模式《强袭风暴》3月21日上线
暴雪近日发布了《魔兽世界》10.2.6 更新内容,新游玩模式《强袭风暴》即将于3月21 日在亚服上线,届时玩家将前往阿拉希高地展开一场 60 人大逃杀对战。
艾泽拉斯的冒险者已经征服了艾泽拉斯的大地及遥远的彼岸。他们在对抗世界上最致命的敌人时展现出过人的手腕,并且成功阻止终结宇宙等级的威胁。当他们在为即将于《魔兽世界》资料片《地心之战》中来袭的萨拉塔斯势力做战斗准备时,他们还需要在熟悉的阿拉希高地面对一个全新的敌人──那就是彼此。在《巨龙崛起》10.2.6 更新的《强袭风暴》中,玩家将会进入一个全新的海盗主题大逃杀式限时活动,其中包含极高的风险和史诗级的奖励。
《强袭风暴》不是普通的战场,作为一个独立于主游戏之外的活动,玩家可以用大逃杀的风格来体验《魔兽世界》,不分职业、不分装备(除了你在赛局中捡到的),光是技巧和战略的强弱之分就能决定出谁才是能坚持到最后的赢家。本次活动将会开放单人和双人模式,玩家在加入海盗主题的预赛大厅区域前,可以从强袭风暴角色画面新增好友。游玩游戏将可以累计名望轨迹,《巨龙崛起》和《魔兽世界:巫妖王之怒 经典版》的玩家都可以获得奖励。