Go语言——值方法 & 指针方法
1 package main import (
"fmt"
"sort"
) type SortableStrings []string type Sortable interface {
sort.Interface
Sort()
} func (self SortableStrings) Len() int {
return len(self)
} func (self SortableStrings) Less(i, j int) bool {
return self[i] < self[j]
} func (self SortableStrings) Swap(i, j int) {
self[i], self[j] = self[j], self[i]
} func main() {
_, ok1 := interface{}(SortableStrings{}).(sort.Interface)
fmt.Println("ok1", ok1) _, ok2 := interface{}(SortableStrings{}).(Sortable)
fmt.Println("ok2", ok2)
}

自定义类型SortableStrings实现了接口sort.Interface中3个开放函数。自定义接口Sortable,除了包含sort.Interface的3个函数外,增加的Sort没有被SortableStrings实现。所以SortableStrings只实现了一个接口,即sort.Interface
package main import (
"fmt"
"sort"
) type SortableStrings []string type Sortable interface {
sort.Interface
Sort()
} func (self SortableStrings) Len() int {
return len(self)
} func (self SortableStrings) Less(i, j int) bool {
return self[i] < self[j]
} func (self SortableStrings) Swap(i, j int) {
self[i], self[j] = self[j], self[i]
} func (self SortableStrings) Sort() {
sort.Sort(self)
} func main() {
_, ok1 := interface{}(SortableStrings{}).(sort.Interface)
fmt.Println("ok1", ok1) _, ok2 := interface{}(SortableStrings{}).(Sortable)
fmt.Println("ok2", ok2)
}

对自定义类型SortableStrings增加方法Sort,其实现是调用sort.Sort函数,该函数接受一个类型为sort.Interface的参数值,并利用这个值的Len,Less,Swap方法进行排序。
package main import (
"fmt"
"sort"
) type SortableStrings []string type Sortable interface {
sort.Interface
Sort()
} func (self SortableStrings) Len() int {
return len(self)
} func (self SortableStrings) Less(i, j int) bool {
return self[i] < self[j]
} func (self SortableStrings) Swap(i, j int) {
self[i], self[j] = self[j], self[i]
} func (self *SortableStrings) Sort() {
sort.Sort(self)
} func main() {
_, ok1 := interface{}(SortableStrings{}).(sort.Interface)
fmt.Println("ok1", ok1) _, ok2 := interface{}(SortableStrings{}).(Sortable)
fmt.Println("ok2", ok2)
}

接口Sortable中有4个函数需要实现,虽然SortableStrings实现了4个函数,但是Sort版的实现,接收器是指针类型。SortableStrings{}是值类型,不能访问基本类型SortableStrings接收器是指针的方法(即Sort),相当于SortableStrings{}只实现了Len、Less、Swap 3个函数,少实现一个函数,所以不是Sortable类型
package main import (
"fmt"
"sort"
) type SortableStrings []string type Sortable interface {
sort.Interface
Sort()
} func (self SortableStrings) Len() int {
return len(self)
} func (self SortableStrings) Less(i, j int) bool {
return self[i] < self[j]
} func (self SortableStrings) Swap(i, j int) {
self[i], self[j] = self[j], self[i]
} func (self *SortableStrings) Sort() {
sort.Sort(self)
} func main() {
_, ok1 := interface{}(SortableStrings{}).(sort.Interface)
fmt.Println("ok1", ok1) _, ok2 := interface{}(&SortableStrings{}).(Sortable)
fmt.Println("ok2", ok2)
}

&SortableStrings{}是SortableStrings{}的指针类型,可以访问到基本类型SortableStrings中的4个函数。所以是Sortable的一个实现
package main import (
"fmt"
"sort"
) type SortableStrings []string type Sortable interface {
sort.Interface
Sort()
} func (self SortableStrings) Len() int {
return len(self)
} func (self SortableStrings) Less(i, j int) bool {
return self[i] < self[j]
} func (self SortableStrings) Swap(i, j int) {
self[i], self[j] = self[j], self[i]
} func (self *SortableStrings) Sort() {
sort.Sort(self)
} func main() {
_, ok1 := interface{}(&SortableStrings{}).(sort.Interface)
fmt.Println("ok1", ok1) _, ok2 := interface{}(&SortableStrings{}).(Sortable)
fmt.Println("ok2", ok2)
}

&SortableStrings{}是SortableStrings{}的指针类型。指针类型可以访问基本类型中接收器为值(或者指针)的类型。
package main import (
"fmt"
"sort"
) type SortableStrings []string type Sortable interface {
sort.Interface
Sort()
} func (self SortableStrings) Len() int {
return len(self)
} func (self SortableStrings) Less(i, j int) bool {
return self[i] < self[j]
} func (self SortableStrings) Swap(i, j int) {
self[i], self[j] = self[j], self[i]
} func (self *SortableStrings) Sort() {
sort.Sort(self)
} func main() {
ss := SortableStrings{"", "", ""}
ss.Sort()
fmt.Println("Sortable Strings", ss)
_, ok1 := interface{}(SortableStrings{}).(sort.Interface)
fmt.Println("ok1", ok1) _, ok2 := interface{}(SortableStrings{}).(Sortable)
fmt.Println("ok2", ok2) _, ok3 := interface{}(&SortableStrings{}).(sort.Interface)
fmt.Println("ok3", ok3) _, ok4 := interface{}(&SortableStrings{}).(Sortable)
fmt.Println("ok4", ok4)
}

package main import (
"fmt"
"sort"
) type SortableStrings []string type Sortable interface {
sort.Interface
Sort()
} func (self *SortableStrings) Len() int {
return len(self)
} func (self *SortableStrings) Less(i, j int) bool {
return self[i] < self[j]
} func (self *SortableStrings) Swap(i, j int) {
self[i], self[j] = self[j], self[i]
} func (self *SortableStrings) Sort() {
sort.Sort(self)
} func main() {
ss := SortableStrings{"", "", ""}
ss.Sort()
fmt.Println("Sortable Strings", ss)
_, ok1 := interface{}(SortableStrings{}).(sort.Interface)
fmt.Println("ok1", ok1) _, ok2 := interface{}(SortableStrings{}).(Sortable)
fmt.Println("ok2", ok2) _, ok3 := interface{}(&SortableStrings{}).(sort.Interface)
fmt.Println("ok3", ok3) _, ok4 := interface{}(&SortableStrings{}).(Sortable)
fmt.Println("ok4", ok4)
}

上面关于“XX可以访问OO”的描述中,“访问“一词用的不准确。值变量可以访问接收器是指针类型的方法,指针变量也可以访问接收器是值类型的方法
package main import (
"fmt"
"sort"
) type SortableStrings []string type Sortable interface {
sort.Interface
Sort()
} func (self SortableStrings) Len() int {
return len(self)
} func (self SortableStrings) Less(i, j int) bool {
return self[i] < self[j]
} func (self SortableStrings) Swap(i, j int) {
self[i], self[j] = self[j], self[i]
} func (self *SortableStrings) Sort() {
sort.Sort(self)
} func main() {
ss := SortableStrings{"", "", ""}
ss.Sort() //值变量可以访问接收器是值类型的方法
fmt.Println("Sortable Strings", ss) fmt.Println("ss : self[0] < self[1]", ss.Less(, ))
fmt.Println("&ss : self[0] < self[1]", (&ss).Less(, )) //指针变量可以访问接收器是值类型的方法 _, ok1 := interface{}(ss).(sort.Interface)
fmt.Println("ok1", ok1) _, ok2 := interface{}(ss).(Sortable)
fmt.Println("ok2", ok2) _, ok3 := interface{}(&ss).(sort.Interface)
fmt.Println("ok3", ok3) _, ok4 := interface{}(&ss).(Sortable)
fmt.Println("ok4", ok4) _, ok5 := interface{}(SortableStrings{}).(sort.Interface)
fmt.Println("ok5", ok5) _, ok6 := interface{}(SortableStrings{}).(Sortable)
fmt.Println("ok6", ok6) _, ok7 := interface{}(&SortableStrings{}).(sort.Interface)
fmt.Println("ok7", ok7) _, ok8 := interface{}(&SortableStrings{}).(Sortable)
fmt.Println("ok8", ok8)
}

所以上面”访问“一词用的不准确。值变量可以访问接收器是值类型的方法 还是 指针变量可以访问接收器是值类型的方法,都是编译器帮我们做了部分转换工作。
http://play.golang.org/p/KG8-Qb7gqM
package main import (
"log"
) type User struct {
Name string
Email string
} func (u *User) Notify() error {
log.Printf("User: Sending User Email To %s<%s>\n",
u.Name,
u.Email) return nil
} type Notifier interface {
Notify() error
} func SendNotification(notify Notifier) error {
return notify.Notify()
} func main() {
user := User{
Name: "AriesDevil",
Email: "ariesdevil@xxoo.com",
} SendNotification(user)
}

“值类型(或 指针类型)是否是该接口类型的实现呢?”
User的值类型变量不是 接口 Notifer的实现。也就是
_, ok1 := interface{}(user).(Notifier)
fmt.Println("ok1", ok1) //false
_, ok2 := interface{}(&user).(Notifier)
fmt.Println("ok2", ok2) //true
SendNotification(user)
上文报错,是因为user不是Notifier的实现,也就不能赋值给 SendNotification(notify Notifier)的notify.调用函数,参数就涉及到传值,只有相同类型或者其实现才可能相互赋值。如果不是调用函数,某一类型的值或者指针直接调用自己的方法(接收器是值或者指针)都是可以的
package main import (
"fmt"
"log"
) type User struct {
Name string
Email string
} func (u *User) Notify() error {
log.Printf("User: Sending User Email To %s<%s>\n",
u.Name,
u.Email) return nil
} func (u User) NameString() {
u.Name = "Bob"
fmt.Println("Name", u.Name)
} func (u *User) EmailString() {
u.Email = "Bob@hotmail.com"
fmt.Println("Email", u.Email)
} type Notifier interface {
Notify() error
NameString()
EmailString()
} func SendNotification(notify Notifier) error {
return notify.Notify()
} func main() {
user := User{
Name: "AriesDevil",
Email: "ariesdevil@xxoo.com",
} user.NameString()
user.EmailString()
fmt.Println("Name", user.Name)
fmt.Println("Email", user.Email) (&user).NameString()
(&user).EmailString()
fmt.Println("Name", (&user).Name)
fmt.Println("Email", (&user).Email) _, ok1 := interface{}(user).(Notifier)
fmt.Println("ok1", ok1) //false _, ok2 := interface{}(&user).(Notifier)
fmt.Println("ok2", ok2) //true //SendNotification(user) }

Go语言——值方法 & 指针方法的更多相关文章
- Go语言值,指针,引用类型
原文:https://www.jianshu.com/p/af42cb368cef ---------------------------------------------------- Go语言的 ...
- python 调用dll中c或c++语言的带指针方法,
在项目开发中遇到了,python需要去调用一个动态链接库dll中的c++方法.这个方法的参数为一个指针类型的参数,一个bool类型参数, 在python中并未对数字类型进行区分. int LP_Agc ...
- C语言函数返回指针方法
1.将函数内部定义的变量用static修饰 由于static修饰的变量,分配在静态内存区(类似于全局变量区),函数返回时,并不会释放内存,因此可以将要返回的变量加static修饰. int *test ...
- 表达式求值(二叉树方法/C++语言描述)(二)
表达式二叉树节点的数据可能是运算数或运算符,可以使用一个联合体进行存储:同时还需要一个变量来指示存储的是运算数还是运算符,可以采用和栈方法求值中一样的枚举类型TokenType: typedef en ...
- Golang 方法接收者是值还是指针问题
对于普通结构体作为接收者,值和指针并没有区别. (以下代码摘抄自Go In Action 中文版) type defaultMatcher struct{} // 方法声明为使用 defaultMat ...
- 用c语言产生随机数的方法
用c语言产生随机数的方法 在C语言中,rand()函数可以用来产生随机数,但是这不是真正意义上的随机数,是一个伪随机数,是根据一个数,我们可以称它为种子,为基准以某个递推公式推算出来的一系数,当这系列 ...
- C语言程序设计:现代方法阅读笔记
第二十六章 atexit函数允许用户“注册”在程序终止时要调用的函数:atexit(func); 在程序终止后,func函数会被自动调用 clock()函数可以计算程序运行时间 time函数返回当前的 ...
- C++ 全面刨析使用指针方法 _new _delete
指针 #include<iostream> using namespace std; int main() { ; int* pn;//声明 int* pn = &avr;//初始 ...
- Go 语言 结构体和方法
@ 目录 1. 结构体别名定义 2. 工厂模式 3. Tag 原信息 4. 匿名字段 5. 方法 1. 结构体别名定义 变量别名定义 package main import "fmt&quo ...
随机推荐
- DLT(Direct Linear Transform)算法
1.DLT定义 DLT是一个 用于解决包含尺度问题的最小二乘问题 的算法. DLT解决问题的标准形式为: ...
- Egret入门学习日记 --- 第一篇 (引擎的选择)
第一篇 (引擎的选择) 我人比较笨,得慢慢学,我就一点一点来好了. 首先,我个人喜欢游戏.网页开发相对游戏开发来说,网页开发实在太枯燥了,没劲.所以打算转游戏开发了. 游戏开发要选择游戏引擎,我去看了 ...
- JavaSE基础(一)--初识Java
Java 简介 Java是由Sun Microsystems公司于1995年5月推出的Java面向对象程序设计语言和Java平台的总称.由James Gosling和同事们共同研发,并在1995年正式 ...
- 解决无/var/log/messages 问题
转载于:https://blog.csdn.net/C_Major/article/details/51321684 1 内核编程insmod后,Ubuntu查看日志无/var/log/message ...
- [转帖]Nginx服务器的六种负载均衡策略详解
Nginx服务器的六种负载均衡策略详解 咔咔侃技术 2019-09-11 17:40:12 一.关于Nginx的负载均衡 在服务器集群中,Nginx起到一个代理服务器的角色(即反向代理),为了避免单独 ...
- vmware 安装XP 32位Professional版本
VMware14 安装XP 专业版,总部报错Dicrectory Boot not found 下载了N多个XP的版本发现,原来是XP版本的问题,现将正确版本写在这里,以备后用 百度网盘共享位置 ...
- Python笔记day20-面向对象
目录 面向对象 1 装饰器 1.1 装饰器是什么? 1.2 装饰器 2 面向对象 (Object Oriented) 简称OO 2.1 面向对象相关术语 2.2 类和对象 2.3 类和对象的实现和书写 ...
- java之集合那些事
集合概述: 集合和数组都可以保存多个对象,但是数组的长度不可变,集合可以保存数量变化的数据.java中的集合类主要由两个接口派生出,Collection和Map Collection接口和Iterat ...
- golang跨平台编译
// 目标平台linux 64 SET CGO_ENABLED=0 SET GOOS=linux SET GOARCH=amd64 go build //目标平台windows SET CGO_ENA ...
- Luogu5327 ZJOI2019语言(树上差分+线段树合并)
暴力树剖做法显然,即使做到两个log也不那么优美. 考虑避免树剖做到一个log.那么容易想到树上差分,也即要对每个点统计所有经过他的路径产生的总贡献(显然就是所有这些路径端点所构成的斯坦纳树大小),并 ...