文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

如何使用 Go 语言写出面向对象风格的代码

2024-12-02 17:54

关注

前言

!! 哈喽,大家好,我是asong。在上一篇文章:小白也能看懂的context包详解:从入门到精通 分析context的源码时,我们看到了一种编程方法,在结构体里内嵌匿名接口,这种写法对于大多数初学Go语言的朋友看起来是懵逼的,其实在结构体里内嵌匿名接口、匿名结构体都是在面向对象编程中继承和重写的一种实现方式,之前写过java、python对面向对象编程中的继承和重写应该很熟悉,但是转Go语言后写出的代码都是面向过程式的代码,所以本文就一起来分析一下如何在Go语言中写出面向对象的代码。

面向对象程序设计是一种计算机编程架构,英文全称:Object Oriented Programming,简称OOP。OOP的一条基本原则是计算机程序由单个能够起到子程序作用的单元或对象组合而成,OOP达到了软件工程的三个主要目标:重用性、灵活性和扩展性。OOP=对象+类+继承+多态+消息,其中核心概念就是类和对象。

这一段话在网上介绍什么是面向对象编程时经常出现,大多数学习Go语言的朋友应该也都是从C++、python、java转过来的,所以对面向对象编程的理解应该很深了,所以本文就没必要介绍概念了,重点来看一下如何使用Go语言来实现面向对象编程的编程风格。

Go语言本身就不是一个面向对象的编程语言,所以Go语言中没有类的概念,但是他是支持类型的,因此我们可以使用struct类型来提供类似于java中的类的服务,可以定义属性、方法、还能定义构造器。来看个例子:

  1. type Hero struct { 
  2.  Name string 
  3.  Age uint64 
  4.  
  5. func NewHero() *Hero { 
  6.  return &Hero{ 
  7.   Name"盖伦"
  8.   Age: 18, 
  9.  } 
  10.  
  11. func (h *Hero) GetName() string { 
  12.  return h.Name 
  13.  
  14. func (h *Hero) GetAge() uint64 { 
  15.  return h.Age 
  16.  
  17.  
  18. func main()  { 
  19.  h := NewHero() 
  20.  print(h.GetName()) 
  21.  print(h.GetAge()) 

这就一个简单的 "类"的使用,这个类名就是Hero,其中Name、Age就是我们定义的属性,GetName、GetAge这两个就是我们定义的类的方法,NewHero就是定义的构造器。因为Go语言的特性问题,构造器只能够依靠我们手动来实现。

这里方法的实现是依赖于结构体的值接收者、指针接收者的特性来实现的。

封装

封装是把一个对象的属性私有化,同时提供一些可以被外界访问的属性和方法,如果不想被外界访问,我们大可不必提供方法给外界访问。在Go语言中实现封装我们可以采用两种方式:

Go语言支持包级别的封装,小写字母开头的名称只能在该包内程序中可见,所以我们如果不想暴露一些方法,可以通过这种方式私有包中的内容,这个理解比较简单,就不举例子了。

Go语言可以通过 type 关键字创建新的类型,所以我们为了不暴露一些属性和方法,可以采用创建一个新类型的方式,自己手写构造器的方式实现封装,举个例子:

  1. type IdCard string 
  2.  
  3. func NewIdCard(card string) IdCard { 
  4.  return IdCard(card) 
  5.  
  6. func (i IdCard) GetPlaceOfBirth() string { 
  7.  return string(i[:6]) 
  8.  
  9. func (i IdCard) GetBirthDay() string { 
  10.  return string(i[6:14]) 

声明一个新类型IdCard,本质是一个string类型,NewIdCard用来构造对象,

GetPlaceOfBirth、GetBirthDay就是封装的方法。

继承

Go并没有原生级别的继承支持,不过我们可以使用组合的方式来实现继承,通过结构体内嵌类型的方式实现继承,典型的应用是内嵌匿名结构体类型和内嵌匿名接口类型,这两种方式还有点细微差别:

内嵌匿名结构体类型实现继承的例子

  1. type Base struct { 
  2.  Value string 
  3.  
  4. func (b *Base) GetMsg() string { 
  5.  return b.Value 
  6.  
  7.  
  8. type Person struct { 
  9.  Base 
  10.  Name string 
  11.  Age uint64 
  12.  
  13. func (p *Person) GetName() string { 
  14.  return p.Name 
  15.  
  16. func (p *Person) GetAge() uint64 { 
  17.  return p.Age 
  18.  
  19. func check(b *Base)  { 
  20.  b.GetMsg() 
  21.  
  22. func main()  { 
  23.  m := Base{Value: "I Love You"
  24.  p := &Person{ 
  25.   Base: m, 
  26.   Name"asong"
  27.   Age: 18, 
  28.  } 
  29.  fmt.Print(p.GetName(), "  ", p.GetAge(), " and say ",p.GetMsg()) 
  30.  //check(p) 

上面注释掉的方法就证明了不能进行参数多态。

内嵌匿名接口类型实现继承的例子

直接拿一个业务场景举例子,假设现在我们现在要给用户发一个通知,web、app端发送的通知内容都是一样的,但是点击后的动作是不一样的,所以我们可以进行抽象一个接口OrderChangeNotificationHandler来声明出三个公共方法:GenerateMessage、GeneratePhotos、generateUrl,所有类都会实现这三个方法,因为web、app端发送的内容是一样的,所以我们可以抽相出一个父类OrderChangeNotificationHandlerImpl来实现一个默认的方法,然后在写两个子类WebOrderChangeNotificationHandler、AppOrderChangeNotificationHandler去继承父类重写generateUrl方法即可,后面如果不同端的内容有做修改,直接重写父类方法就可以了,来看例子:

  1. type Photos struct { 
  2.  width uint64 
  3.  height uint64 
  4.  value string 
  5.  
  6. type OrderChangeNotificationHandler interface { 
  7.  GenerateMessage() string 
  8.  GeneratePhotos() Photos 
  9.  generateUrl() string 
  10.  
  11.  
  12. type OrderChangeNotificationHandlerImpl struct { 
  13.  url string 
  14.  
  15. func NewOrderChangeNotificationHandlerImpl() OrderChangeNotificationHandler { 
  16.  return OrderChangeNotificationHandlerImpl{ 
  17.   url: "https://base.test.com"
  18.  } 
  19.  
  20. func (o OrderChangeNotificationHandlerImpl) GenerateMessage() string { 
  21.  return "OrderChangeNotificationHandlerImpl GenerateMessage" 
  22.  
  23. func (o OrderChangeNotificationHandlerImpl) GeneratePhotos() Photos { 
  24.  return Photos{ 
  25.   width: 1, 
  26.   height: 1, 
  27.   value: "https://www.baidu.com"
  28.  } 
  29.  
  30. func (w OrderChangeNotificationHandlerImpl) generateUrl() string { 
  31.  return w.url 
  32.  
  33. type WebOrderChangeNotificationHandler struct { 
  34.  OrderChangeNotificationHandler 
  35.  url string 
  36.  
  37. func (w WebOrderChangeNotificationHandler) generateUrl() string { 
  38.  return w.url 
  39.  
  40. type AppOrderChangeNotificationHandler struct { 
  41.  OrderChangeNotificationHandler 
  42.  url string 
  43.  
  44. func (a AppOrderChangeNotificationHandler) generateUrl() string { 
  45.  return a.url 
  46.  
  47. func check(handler OrderChangeNotificationHandler)  { 
  48.  fmt.Println(handler.GenerateMessage()) 
  49.  
  50. func main()  { 
  51.  base := NewOrderChangeNotificationHandlerImpl() 
  52.  web := WebOrderChangeNotificationHandler{ 
  53.   OrderChangeNotificationHandler: base, 
  54.   url: "http://web.test.com"
  55.  } 
  56.  fmt.Println(web.GenerateMessage()) 
  57.  fmt.Println(web.generateUrl()) 
  58.  
  59.  check(web) 

因为所有组合都实现了OrderChangeNotificationHandler类型,所以可以处理任何特定类型以及是该特定类型的派生类的通配符。

多态

多态是面向对象编程的本质,多态是支代码可以根据类型的具体实现采取不同行为的能力,在Go语言中任何用户定义的类型都可以实现任何接口,所以通过不同实体类型对接口值方法的调用就是多态,举个例子:

  1. type SendEmail interface { 
  2.  send() 
  3.  
  4. func Send(s SendEmail)  { 
  5.  s.send() 
  6.  
  7. type user struct { 
  8.  name string 
  9.  email string 
  10.  
  11. func (u *user) send()  { 
  12.  fmt.Println(u.name + " email is " + u.email + "already send"
  13.  
  14. type admin struct { 
  15.  name string 
  16.  email string 
  17.  
  18. func (a *admin) send()  { 
  19.  fmt.Println(a.name + " email is " + a.email + "already send"
  20.  
  21. func main()  { 
  22.  u := &user
  23.   name"asong"
  24.   email: "你猜"
  25.  } 
  26.  a := &admin{ 
  27.   name"asong1"
  28.   email: "就不告诉你"
  29.  } 
  30.  Send(u) 
  31.  Send(a) 

总结

归根结底面向对象编程就是一种编程思想,只不过有些语言在语法特性方面更好的为这种思想提供了支持,写出面向对象的代码更容易,但是写代码的还是我们自己,并不是我们用了java就一定会写出更抽象的代码,在工作中我看到用java写出面向过程式的代码不胜其数,所以无论用什么语言,我们都应该思考如何写好一份代码,大量的抽象接口帮助我们精简代码,代码是优雅了,但也会面临着可读性的问题,什么事都是有两面性的,写出好代码的路还很长,还需要不断探索............。

文中示例代码已经上传github:https://github.com/asong2020/Golang_Dream/tree/master/code_demo/oop

 

来源:Golang梦工厂内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯