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 ...
随机推荐
- 文件被sourceTree忽略了怎么办
- vs2012编译的程序不能在XP和2003下执行问题的解决方法
问题如题,通过无数次百度和谷歌后,发现,微软已经确认这是一个缺陷,安装Vs2012的update 3的升级包就可以解决问题.同时,在分发包的地方,vcredist_x86.exe 随程序分发一份就可以 ...
- Pyhthon3之使用__slots__
正常情况下,我们定义了一个class,创建了一个class实例后,我们可以给该实例绑定任何属性和方法,这就是动态语言的灵活性.先定义class: >>> class Student( ...
- 十六章 综合实例——《跟我学Shiro》
目录贴:跟我学Shiro目录贴 简单的实体关系图 简单数据字典 用户(sys_user) 名称 类型 长度 描述 id bigint 编号 主键 username varchar 100 用户名 pa ...
- 第四章 INI配置——《跟我学Shiro》
转发地址:https://www.iteye.com/blog/jinnianshilongnian-2020820 第四章 INI配置——<跟我学Shiro> 博客分类: 跟我学Shir ...
- Docker 容器的资源限制 cgroup(九)
目录 一.cgroup简介 二.CPU资源配额控制 1.CPU份额控制 2.CPU周期控制 3.CPU core控制 4.CPU配额控制参数的混合使用 二.对内存的限额 三.对 Block IO 的限 ...
- 第一篇博客==>Hello_World
1,为什么写博客? 大佬都说程序员需要写博客的说,被无数到的大佬帮我洗脑之后,慢慢也发现了写博客的好处,写博客我认为主要有以下几个作用: 1.打开博客,记录世界记录你.emmm 2.可以把自己的一些经 ...
- VMware15安装Centos7超详细过程
本篇文章主要介绍了VMware安装Centos7超详细过程(图文),具有一定的参考价值,感兴趣的小伙伴们可以参考一下 1.软硬件准备 软件:推荐使用VMwear15,我用的是VMwear 15 镜像: ...
- Windows 10 下使用 MinGW-w64 编译 OpenCV-4.1.1
1. 下载安装 CMake 2. 下载安装 MinGW-w64,假设安装于 X:\path\to\MinGW-w64 此处注意选择 POSIX 版本而非 Win32 版本,因为 OpenCV 调用了标 ...
- Mac配置虚拟环境Virtualenv,安装Python科学计算包详解
参考: https://www.jianshu.com/p/51140800e8b4 安装 virtualenvwrapper Virtaulenvwrapper是virtualenv的扩展包,可以更 ...