欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

Go语言-为什么返回值为接口类型,却返回结构体

程序员文章站 2023-10-28 13:16:10
最近由于项目需求,阅读一些go语言编写的项目的源代码,在某一个函数中发现了一个奇怪的现象:一个函数的返回值类型声明的是一个接口的类型,但是实际在函数体内返回的却是一个结构体类型的对象。这个现象对于新手...

最近由于项目需求,阅读一些go语言编写的项目的源代码,在某一个函数中发现了一个奇怪的现象:一个函数的返回值类型声明的是一个接口的类型,但是实际在函数体内返回的却是一个结构体类型的对象。

这个现象对于新手的我来说很是费解。在经过一些资料的查阅之后,自己得到了如下的解释:

一个结构体实现了一个接口,那么函数中返回值类型为接口时,就应该返回这个结构体。

下面举一个例子来说明:

package main
import (
	"fmt"
)
/**
shape接口定义两个函数:
area() :计算面积
circumference() :计算周长
*/
type shape interface {
	area() float64
	circumference() float64
}
//结构体正方形,属性边长
type square struct {
	length float64
}
//方法area,由正方形结构体实现
func (s square) area()  float64 {
	sarea := s.length * s.length
	return sarea
}
//方法circumference,由正方形结构体实现
func (s square) circumference()  float64 {
	scircumference := s.length * 4
	return scircumference
}
func getarea(len float64) shape {
	s := square{
		length:4,
	}
	fmt.println("正方形的面积为:",s.area())
	fmt.println("正方形的周长为:",s.circumference())
	return s
}
func main() {
	getarea(4)
}

或者另一个版本:

package main
import (
	"fmt"
)
/**
shape接口定义两个函数:
area() :计算面积
circumference() :计算周长
*/
type shape interface {
	area() float64
	circumference() float64
}
//结构体正方形,属性边长
type square struct {
	length float64
}
//方法area,由正方形结构体实现
func (s *square) area()  float64 {
	sarea := s.length * s.length
	return sarea
}
//方法circumference,由正方形结构体实现
func (s *square) circumference()  float64 {
	scircumference := s.length * 4
	return scircumference
}
func getarea(len float64) shape {
	s := &square{
		length:4,
	}
	fmt.println("正方形的面积为:",s.area())
	fmt.println("正方形的周长为:",s.circumference())
	return s
}
func main() {
	getarea(4)
}

这两个代码的区别就是前者使用了值传递,后者使用了指针传递。由于这里没有改变结构体中的属性值,所以两种方法在这样的应用场景下,没有什么区别,下面来解释一下这些简单的demo:

首先我定义了一个shape接口,里面有两个待实现的方法area() :计算面积 和 circumference() :计算周长

然后定义了一个正方形结构体,里面只有一个边长属性。

然后使用正方形结构体实现这个shape接口

接着我们就可以进入正题,试验我们标题的问题了,使用shape接口类型作为返回值,但是在函数体内实际的返回值是正方形结构体。

这是go的一种语法,但实际的作用或者为是什么这样写,我还没有弄清楚,但是通过以上这个实实在在的例子,关于为什么返回值类型和实际返回的不一样有了一定的理解。

补充:go语言-结构体和接口

结构体和接口

接口嵌套

接口中允许嵌套其他接口,效果等同于复制被嵌套的接口中的方法

当前的接口中不允许有与嵌入的接口相同的方法

方法相同的接口相等同

接口不能为空,否则等同于空接口

结构体嵌套

结构体中的匿名成员内的成员和方法会被嵌套到当前结构体中

当前结构体中允许有与被嵌套结构体相同的成员和方法,且会覆盖被嵌套的结构体的成员和方法

两个被嵌套的结构体有相同的成员或方法,会发生冲突

有时候编辑器不会提示,但会产生运行时错误

成员名称和类型完全相同的结构体

如果其中一个是匿名的,可以直接赋值或判断相等

类型名不同可以进行类型转换,不可以直接赋值或判断相等

方法的接收器只能是在当前包中指定名称的类型,不能是原生类型、复合类型、其他包中的类型

重新命名的结构体与原结构体成员完全相同(包括tag),但是没有原来的方法

接口实现

一个类型实现了接口的所有方法,就是实现了接口,不管类型和接口之间是否有关联

方法的接收器可以是这个类型或者这个类型的指针类型

指针类型的接收器可以被修改成员

非指针类型的实例直接调用指针类型接收器的方法,会遇到无法调用指针方法,无法获取地址的问题

指针类型的实例调用非指针类型接收器的方法不会出现问题

将实例赋值给变量再调用不会出现问题

以上为个人经验,希望能给大家一个参考,也希望大家多多支持。如有错误或未考虑完全的地方,望不吝赐教。