golang面向对象分析
说道面向对象(OOP)编程, 就不得不提到下面几个概念:
- 抽象
- 封装
- 继承
- 多态
其实有个问题Is Go An Object Oriented Language?, 随便谷歌了一下, 你就发现讨论这个的文章有很多:
那么问题来了
- Golang是OOP吗?
- 使用Golang如何实现OOP?
一. 抽象和封装
抽象和封装就放在一块说了. 这个其实挺简单. 看一个例子就行了.
type rect struct {
width int
height int
}
func (r *rect) area() int {
return r.width * r.height
}
func main() {
r := rect{width: 10, height: 5}
fmt.Println("area: ", r.area())
}
要说明的几个地方:
1、Golang中的struct和其他语言的class是一样的.
2、可见性. 这个遵循Go语法的大小写的特性
3、上面例子中, 称*rect为receiver. 关于receiver 可以有两种方式的写法:
func (r *rect) area() int {
return r.width * r.height
}
func (r rect) area() int {
return r.width * r.height
}
这其中有什么区别和联系呢?
简单来说, Receiver可以是值传递, 还是可以是指针, 两者的差别在于, 指针作为Receiver会对实例对象的内容发生操作,而普通类型作为Receiver仅仅是以副本作为操作对象,并不对原实例对象发生操作。
4、当Receiver为*rect指针的时候, 使用的是r.width, 而不是(*r).width, 是由于Go自动帮我转了,两种方式都是正确的.
5、任何类型都可以声明成新的类型, 因为任何类型都可以有方法.
type Interger int
func (i Interger) Add(interger Interger) Interger {
return i + interger
}
6、虽然Interger是从int声明而来, 但是这样用是错误的.
var i Interger = 1
var a int = i //cannot use i (type Interger) as type int in assignment
这是因为Go中没有隐式转换(写C++的同学都会特别讨厌这个, 因为编译器背着我们干的事情太多了). Golang中类型之间的相互赋值都必须显式声明.
上面的例子改成下面的方式就可以了.
var i Interger = 1
var a int = int(i)
二. 继承(Composition)
说道继承,其实在Golang中是没有继承(Extend)这个概念. 因为Golang舍弃掉了像C++, Java的这种传统的、类型驱动的子类。
Go Effictive says:
Go does not provide the typical, type-driven notion of subclassing, but it does have the ability to “borrow” pieces of an implementation by embedding types within a struct or interface.
换句话说, Golang中没有继承, 只有Composition.
Golang中的Compostion有两种形式, 匿名组合(Pseudo is-a)和非匿名组合(has-a)
注: 如果不了解OOP的is-a和has-a关系的话, 请自行google.
1. has-a
package main
import (
"fmt"
)
type Human struct {
name string
age int
phone string
}
type Student struct {
h Human //非匿名字段
school string
}
func (h *Human) SayHi() {
fmt.Printf("Hi, I am %s you can call me on %s\n", h.name, h.phone)
}
func (s *Student) SayHi() {
fmt.Printf("Hi student, I am %s you can call me on %s", s.h.name, s.h.phone)
}
func main() {
mark := Student{Human{"Mark", 25, "222-222-YYYY"}, "MIT"}
fmt.Println(mark.h.name, mark.h.age, mark.h.phone, mark.school)
mark.h.SayHi()
mark.SayHi()
}
Output
Mark 25 222-222-YYYY MIT
Hi, I am Mark you can call me on 222-222-YYYY
Hi student, I am Mark you can call me on 222-222-YYYY
这种组合方式, 其实对于了解传统OOP的话, 很好理解, 就是把一个struct作为另一个struct的字段.
从上面例子可以, Human完全作为Student的一个字段使用. 所以也就谈不上继承的相关问题了.我们也不去重点讨论.
2. is-a(Pseudo)----Embedding
type Human struct {
name string
age int
phone string
}
type Student struct {
Human //匿名字段
school string
}
func (h *Human) SayHi() {
fmt.Printf("Hi, I am %s you can call me on %s\n", h.name, h.phone)
}
func main() {
mark := Student{Human{"Mark", 25, "222-222-YYYY"}, "MIT"}
fmt.Println(mark.name, mark.age, mark.phone, mark.school)
mark.SayHi()
}
Output
Mark 25 222-222-YYYY MIT
Hi, I am Mark you can call me on 222-222-YYYY
这里要说的有几点:
1、字段
现在Student访问Human的字符, 就可以直接访问了, 感觉就是在访问自己的属性一样. 这样就实现了OOP的继承.
fmt.Println("Student age:", mark.age) //输出: Student age: 25
但是, 我们也可以间接访问:
fmt.Println("Student age:", mark.Human.age) //输出: Student age: 25
这有个问题, 如果在Student也有个字段name, 那么当使用mark.name会以Student的name为准.
fmt.Println("Student name:", mark.name) //输出:Student Name: student name
2、方法
Student也继承了Human的SayHi()方法
mark.SayHi() // 输出: Hi, I am Mark you can call me on 222-222-YYYY
当然, 我们也可以重写SayHi()方法:
type Human struct {
name string
age int
phone string
}
type Student struct {
Human //匿名字段
school string
name string
}
func (h *Human) SayHi() {
fmt.Printf("Hi, I am %s you can call me on %s\n", h.name, h.phone)
}
func (h *Student) SayHi() {
fmt.Println("Student Sayhi")
}
func main() {
mark := Student{Human{"Mark", 25, "222-222-YYYY"}, "MIT", "student name"}
mark.SayHi()
}
Output
Student Sayhi
3、为什么称其为Pseudo is-a呢?
因为匿名组合不提供多态的特性. 如下面的代码:
package main
type A struct{
}
type B struct {
A //B is-a A
}
func save(A) {
//do something
}
func main() {
b := new(B)
save(*b);
}
Output
cannot use *b (type B) as type A in argument to save
还有一个面试题的例子
type People struct{}
func (p *People) ShowA() {
fmt.Println("showA")
p.ShowB()
}
func (p *People) ShowB() {
fmt.Println("showB")
}
type Teacher struct {
People
}
func (t *Teacher) ShowB() {
fmt.Println("teacher showB")
}
func main() {
t := Teacher{}
t.ShowA()
}
输出结果是什么呢?
Output
ShowA
ShowB
Effective Go Says:
There's an important way in which embedding differs from subclassing. When we embed a type, the methods of that type become methods of the outer type, but when they are invoked the receiver of the method is the inner type, not the outer one
也就是说, Teacher由于组合了People, 所以Teacher也有了ShowA()方法, 但是在ShowA()方法里执行到ShowB时, 这个时候的receiver是*People而不是*Teacher, 主要原因还是因为embedding是一个Pseudo is-a, 没有多态的功能.
4、 "多继承"的问题
package main
import "fmt"
type School struct {
address string
}
func (s *School) Address() {
fmt.Println("School Address:", s.address)
}
type Home struct {
address string
}
func (h *Home) Address() {
fmt.Println("Home Address:", h.address)
}
type Student struct {
School
Home
name string
}
func main() {
mark := Student{School{"aaa"}, Home{"bbbb"}, "cccc"}
fmt.Println(mark)
mark.Address()
fmt.Println(mark.address)
mark.Home.Address()
fmt.Println(mark.Home.address)
}
输出结果:
30: ambiguous selector mark.Address
31: ambiguous selector mark.address
由此可以看出, Golang中不管是方法还是属性都不存在类似C++那样的多继承的问题. 要访问Embedding相关的属性和方法, 需要在加那个相应的匿名字段, 如:
mark.Home.Address()
5、Embedding value和 Embedding pointer的区别
package main
import (
"fmt"
)
type Person struct {
name string
}
type Student struct {
*Person
age int
}
type Teacher struct {
Person
age int
}
func main() {
s := Student{&Person{"student"}, 10}
t := Teacher{Person{"teacher"}, 40}
fmt.Println(s, s.name)
fmt.Println(t, t.name)
}
Output
{0x1040c108 10} student
{{teacher} 40} teacher
I. 两者对于结果来说, 没有啥区别, 只是对传参的时候有影响
II. Embedding value是比较常规的写法
III. Embedding pointer比较有优势一点, 不需要关注指针是什么时间被初始化的.
三. Interface
Golang中Composite不提供多态的功能, 那是否Golang不提供多态呢? 答案肯定是否定. Golang依靠Interface实现多态的功能.
下面是我工程里面一段代码的简化:
package main
import (
"fmt"
)
type Check interface {
CheckOss()
}
type CheckAudio struct {
//something
}
func (c *CheckAudio) CheckOss() {
fmt.Println("CheckAudio do CheckOss")
}
func main() {
checkAudio := CheckAudio{}
var i Check
i = &checkAudio //想一下这里为啥需要&
i.CheckOss()
}
1、Interface 如何Composite ?
type Reader interface {
Read(p []byte) (n int, err error)
}
type Writer interface {
Write(p []byte) (n int, err error)
}
type ReadWriter interface {
Reader
Writer
}
其实很简单, 就是把Reader, Writer嵌入到ReadWriter中, 这样ReadWriter就拥有了Reader和Writer的方法.
尾声
至此, 基本说完了Golang的面向对象. 有哪里我理解的不对的地方, 请给我留言.
参考资料
- Effective Go: Embedding
- Go面试题
- Is Go An Object Oriented Language?
- go web编程
- object-oriented-programming-in-go
golang面向对象分析的更多相关文章
- 面向对象分析设计-------02UML+UML各种图形及作用
一.UML是什么?UML有什么用? 二.UML的历史 三.UML的上层结构(Superstructure) 四.UML建模工具 五.UML的图(重点) 1.用例图(use case diagram) ...
- 面向对象分析方法(I)
找出最关键的一些业务场景:一般通过动词来寻找,比如招聘系统中,一个应聘人投递一个职位就是一次应聘,应聘就是一个业务场景:一个学生参加某门课的考试,那么考试就是一个业务场景:一个学生去图书馆借书,那么借 ...
- <二>面向对象分析之几个关键的概念
一:建模 --->建模,是指通过对[客观事物]建立一种抽象的方法用以表征事物并获得对事物本身的理解.同时把这种理解概念化,将这些逻辑概念组织起来,构成一种对所观察对象的内部结构和工 ...
- <一>面向对象分析之面向对象和面向过程
面向对象 ---->注重的是拆分,组装. ---->封装,继承,多态,复用(只是现象) ---->面向对象变成的目标从来就不是复用.相反,对 ...
- 面向对象的软件project——面向对象分析
为了解决软件危机.一些IT前辈国产软件project这个词汇,软件project它被引入到整个软件开发过程的维护. 软件project从程序的设计角度能够分为两类.一类是面向结构的软件project. ...
- UML和模式应用学习笔记-1(面向对象分析和设计)
UML和模式应用学习笔记-1(面向对象分析和设计) 而只是对情节的记录:此处的用例场景为:游戏者请求掷骰子.系统展示结果:如果骰子的总点数是7,则游戏者赢得游戏,否则为输 (2)定义领域模型:在领域模 ...
- golang面向对象和interface接口
一. golang面向对象介绍 1.golang也支持面向对象编程,但是和传统的面向对象编程有区别,并不是纯粹的面向对象语言.2.golang没有类(class),golang语言的结合体(struc ...
- Golang面向对象编程-struct(结构体)
Golang面向对象编程-struct(结构体) 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.什么是面向对象编程 面向对象编程(Object Oriented Program ...
- golang 面向对象编程
概述 Golang语言的面向对象与c++,py等语言有所不同,是由于Golang不支持继承:与上述支持聚合和继承的面向对象的语言不同,Golang只支持聚合(也叫做组合)和嵌入.聚合和嵌入的区别: t ...
随机推荐
- Ping 的工作原理你懂了,那 ICMP 你懂不懂?
计算机网络我也连载了很多篇了,大家可以在我的公众号「程序员cxuan」 或者我的 github 系统学习. 计算机网络第一篇,聊一聊网络基础 :计算机网络基础知识总结 计算机网络第二篇,聊一聊 TCP ...
- Java基础语法:注释
书写注释是一个非常好的习惯. 注释并不会被执行,是给我们写代码的人看的. Java中的注释有三种: 单行注释(Line comment) 多行注释(Block comment) 文档注释(JavaDo ...
- 正则表达式匹配${key}并在Java中使用
1.正则表达式匹配${key} \$\{([a-z]+)\} 能够匹配字符串中以${key}形式的文本(其中key为小写应为字母) .*\$\{([a-z]+)\}.* 可以用来检测文本中是否有${k ...
- WPF -- 一种添加静态资源的方式
本文介绍使用独立的xaml文件添加静态资源的方式. 步骤 创建XAML文件,如ImageButton.xaml,添加ResourceDictionary标签,并添加静态资源: 在App.xaml的Ap ...
- 后端程序员之路 8、一种内存kv数据库的实现
键值(Key-Value)存储数据库,这是一种NoSQL(非关系型数据库)模型,其数据按照键值对的形式进行组织.索引和存储.KV存储非常适合不涉及过多数据关系业务关系的业务数据,同时能有效减少读写磁盘 ...
- .NET Core中的Worker Service
当你想到ASP.NET Core时,可能会想到Web应用程序后端代码,包括MVC和WebAPI.MVC视图和Razor页面还允许使用后端代码生成带有HTML元素的前端UI.全新的Blazor更进一步, ...
- 虚拟机测试cobbler,网络安装加载最后出现 dracut:/#
1.cobbler的几个重要概念: distro:发行版系统容,我理解为镜像来源,提供了kernel 和 initrd 文件以及repo源 profile:kickstart文件,用于定制系统,定制安 ...
- 创建一个scrapy爬虫框架的项目
第一步:打开pycharm,选择"terminal",如图所示: 第二步:在命令中端输入创建scrapy项目的命令:scrapy startproject demo (demo指的 ...
- 2020年12月-第02阶段-前端基础-CSS基础选择器
CSS选择器(重点) 理解 能说出选择器的作用 id选择器和类选择器的区别 1. CSS选择器作用(重点) 如上图所以,要把里面的小黄人分为2组,最快的方法怎办? 很多, 比如 一只眼睛的一组,剩下的 ...
- celery 与 flask 实现异步任务调度
Flask 定了2中上下文,来实现机遇线程\协程的,wsgi服务的请求(request.session)和存储(g,current_app )过程,通过栈来完成不同线程和协程的上下文切换,在与cele ...