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语言——值方法 & 指针方法的更多相关文章

  1. Go语言值,指针,引用类型

    原文:https://www.jianshu.com/p/af42cb368cef ---------------------------------------------------- Go语言的 ...

  2. python 调用dll中c或c++语言的带指针方法,

    在项目开发中遇到了,python需要去调用一个动态链接库dll中的c++方法.这个方法的参数为一个指针类型的参数,一个bool类型参数, 在python中并未对数字类型进行区分. int LP_Agc ...

  3. C语言函数返回指针方法

    1.将函数内部定义的变量用static修饰 由于static修饰的变量,分配在静态内存区(类似于全局变量区),函数返回时,并不会释放内存,因此可以将要返回的变量加static修饰. int *test ...

  4. 表达式求值(二叉树方法/C++语言描述)(二)

    表达式二叉树节点的数据可能是运算数或运算符,可以使用一个联合体进行存储:同时还需要一个变量来指示存储的是运算数还是运算符,可以采用和栈方法求值中一样的枚举类型TokenType: typedef en ...

  5. Golang 方法接收者是值还是指针问题

    对于普通结构体作为接收者,值和指针并没有区别. (以下代码摘抄自Go In Action 中文版) type defaultMatcher struct{} // 方法声明为使用 defaultMat ...

  6. 用c语言产生随机数的方法

    用c语言产生随机数的方法 在C语言中,rand()函数可以用来产生随机数,但是这不是真正意义上的随机数,是一个伪随机数,是根据一个数,我们可以称它为种子,为基准以某个递推公式推算出来的一系数,当这系列 ...

  7. C语言程序设计:现代方法阅读笔记

    第二十六章 atexit函数允许用户“注册”在程序终止时要调用的函数:atexit(func); 在程序终止后,func函数会被自动调用 clock()函数可以计算程序运行时间 time函数返回当前的 ...

  8. C++ 全面刨析使用指针方法 _new _delete

    指针 #include<iostream> using namespace std; int main() { ; int* pn;//声明 int* pn = &avr;//初始 ...

  9. Go 语言 结构体和方法

    @ 目录 1. 结构体别名定义 2. 工厂模式 3. Tag 原信息 4. 匿名字段 5. 方法 1. 结构体别名定义 变量别名定义 package main import "fmt&quo ...

随机推荐

  1. PHP重建数组的索引

    sort() array_merge()跟一个空数组合并都可以重建索引数组的键(key)

  2. jQuery动态创建html元素的常用方法汇总

    在使用jQuery进行WEB程序设计的时候非常有用.分享给大家供大家参考.具体方法如下: 一般来说,可以通过以下几种方式动态创建html元素: 1.使用jQuery创建元素的语法 2.把动态内容存放到 ...

  3. 车道线检测github集锦

    re1. github_lane_detection; end

  4. Ehcache开启JMX支持

    Ehcache提供了基于JMX的监控支持,支持对以下几类信息的监控. CacheManager Cache CacheConfiguration CacheStatistics 按照JMX的规范,为了 ...

  5. node更改默认npm阿里地址

    npm config set registry https://registry.npm.taobao.orgsudo npm install cnpm -g --registry=https://r ...

  6. 遵循统一的机器学习框架理解SVM

    遵循统一的机器学习框架理解SVM 一.前言 我的博客仅记录我的观点和思考过程.欢迎大家指出我思考的盲点,更希望大家能有自己的理解. 本文参考了李宏毅教授讲解SVM的课程和李航大大的统计学习方法. 二. ...

  7. 微信H5支付开发全过程(除内置微信以外的浏览器)

    前言:网上都是~ 呵呵 自己搞 只要花时间 多问客服总会有的 只说程序不说准备工作 自己ID 或者秘钥都准备好了  写的有点儿乱 可以把所有的方法 放在一个文件中调用 public function ...

  8. 使用Apache服务部署网站(基于IP,域名,端口)

    本篇主要学习Apache网站服务程序的基本部署,基于IP地址.主机名(域名).端口号的虚拟主机功能. 1.基于IP地址 首先我们需要在虚拟机中线安装Apache服务程序,Apache服务程序的软件包名 ...

  9. windows批量导出文件名到txt

    做图像处理时,经常会遇到训练数据样本,这个时候一般就会要一个保存了大量文件名txt文件作为数据的输入 windows系统中可以直接使用dir,具体使用可以直接打开windows的终端 输入 dir/? ...

  10. 【leetcode算法-简单】66. 加一

    [题目描述] 给定一个由整数组成的非空数组所表示的非负整数,在该数的基础上加一. 最高位数字存放在数组的首位, 数组中每个元素只存储单个数字. 你可以假设除了整数 0 之外,这个整数不会以零开头. 示 ...