关于如何在 Golang 中实现抽象类

2022-06-13 01:48:25 +08:00
 LoremIpSum

假设现在我们定义了一个接口I,有两个方法Foo,Bar, 结构体Parent实现了 Foo,而Bar作为抽象方法由子类实现, 示例代码如下:

package main

type I interface {
	Foo()
	Bar() 
}

type Parent struct {
	I
}

func (p *Parent) Foo() {
	p.Bar()
}



type Child struct {
	Parent
}

func (c *Child) Bar() {
	fmt.Println("Child.Bar()")
}

func main() {
	c := &Child{
		Parent: Parent{},
	}
	c.I = c // 如果注释掉这几句会提示空指针异常
	c.Foo()
}

输出如下:

Child.Bar()

问题: 1.上面的代码实现了类似Java风格的抽象类,但是在main方法中的这句代码没看懂c.I = c,这句代码的本质是做了什么事情?为什么c.I = c就会报空指针异常

3088 次点击
所在节点    Go 编程语言
11 条回复
doomfirst
2022-06-13 02:18:19 +08:00
你不觉得你为了在 go 实现 java 的抽象类 写的很别扭么?
akaHenry
2022-06-13 02:57:18 +08:00
不要用 Java 的思维来写 Go.

理解好 Go Interface 接口使用. 就够了.

之所以只有 Java 语言, 特别爱强调设计模式, 是 Java 的设计缺陷, 才需要显式打补丁(不是优点, 不要搞反了).

其他语言, 设计模式, 早就融入语言本身的设计.

Go Interface, 是面向接口编程的典范. 非常漂亮. 兼顾优雅 /简洁 /易用.

理解 Go 很简单. 核心就 2 点. 面向接口(interface) + 面向队列(channel) 编程. (沙雕泛型 generic, 先跳过)

面向接口(interface) = 面向动作(action), 数据和操作分离. 聚焦对操作(action)作抽象. 以操作为主, 数据来适配操作.

面向队列(channel) = MQ(kafka) = 生产者 /消费者模型. 你可以把 go channel, 当成内置的 kafka 来理解, 使用方式也一样.

so. go 的精华, 只有这么多. 剩下的部分, 都是裁剪 c 语言. (垃圾泛型除外)
rrfeng
2022-06-13 09:03:13 +08:00
你把 I 放到 parent 里是个什么意思。。。。
Hyvi
2022-06-13 09:22:28 +08:00
你把内嵌结构体理解成继承 /接口实现了。看看 embeded struct
HiShan
2022-06-13 09:54:34 +08:00
本质上是让 Parent 继承 I 接口的具体实现,从而实现调用到子类的实现。。不过这样写实在是太奇怪了。。。
PS:go 设计的只有简单与漂亮\简单不沾边
HiShan
2022-06-13 09:55:28 +08:00
@HiShan 漂亮\优雅
hzjseasea
2022-06-13 09:57:24 +08:00
近几年新出来的一些语言,不要参考 java 来学习
LoremIpSum
2022-06-13 09:58:34 +08:00
@rrfeng 这样写可以实现这功能,但是底层原理不知道为什么
akaHenry
2022-06-13 10:47:55 +08:00
多说一句. Go 和 Rust 等更现代的语言设计, 都是重视 组合 /嵌套 > 类继承.

Go 的 接口(interface) 设计, 都是 `组合` 的方式来扩展功能.

不要被 `类继承` 束缚思维. `类继承`, 是糟糕的东西.

不要总是想定义很大的类 struct (数据+方法过多).

尽量切分小类. 标准化操作 = 抽象接口方法(interface).

另外, go struct{}, 可以作为命名空间 namespace 使用 = 裸类(没数据, 只有方法).

小模块, 搭积木, 自由组合. 可以写出非常干净 /清晰的代码.
akaHenry
2022-06-13 10:50:35 +08:00
理性状态: 接口设计(interface) 和 数据(data), 应该是正交的.

推荐读 go-micro 这个项目源码来学习如何写 go + 设计大型框架系统.

go-micro 的代码写的非常非常漂亮.
mengdodo
2022-06-14 09:38:42 +08:00
不要用设计模式去包裹 go, 那不是它

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/859173

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX