自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

如何使用 Go 語言寫出面向?qū)ο箫L格的代碼

運維 數(shù)據(jù)庫運維
Go語言本身就不是一個面向?qū)ο蟮木幊陶Z言,所以Go語言中沒有類的概念,但是他是支持類型的,因此我們可以使用struct類型來提供類似于java中的類的服務(wù),可以定義屬性、方法、還能定義構(gòu)造器。

[[433767]]

文末本文轉(zhuǎn)載自微信公眾號「Golang夢工廠」,作者Golang夢工廠。轉(zhuǎn)載本文請聯(lián)系Golang夢工廠公眾號。

前言

!! 哈嘍,大家好,我是asong。在上一篇文章:小白也能看懂的context包詳解:從入門到精通 分析context的源碼時,我們看到了一種編程方法,在結(jié)構(gòu)體里內(nèi)嵌匿名接口,這種寫法對于大多數(shù)初學(xué)Go語言的朋友看起來是懵逼的,其實在結(jié)構(gòu)體里內(nèi)嵌匿名接口、匿名結(jié)構(gòu)體都是在面向?qū)ο缶幊讨欣^承和重寫的一種實現(xiàn)方式,之前寫過java、python對面向?qū)ο缶幊讨械睦^承和重寫應(yīng)該很熟悉,但是轉(zhuǎn)Go語言后寫出的代碼都是面向過程式的代碼,所以本文就一起來分析一下如何在Go語言中寫出面向?qū)ο蟮拇a。

面向?qū)ο蟪绦蛟O(shè)計是一種計算機編程架構(gòu),英文全稱:Object Oriented Programming,簡稱OOP。OOP的一條基本原則是計算機程序由單個能夠起到子程序作用的單元或?qū)ο蠼M合而成,OOP達到了軟件工程的三個主要目標:重用性、靈活性和擴展性。OOP=對象+類+繼承+多態(tài)+消息,其中核心概念就是類和對象。

這一段話在網(wǎng)上介紹什么是面向?qū)ο缶幊虝r經(jīng)常出現(xiàn),大多數(shù)學(xué)習Go語言的朋友應(yīng)該也都是從C++、python、java轉(zhuǎn)過來的,所以對面向?qū)ο缶幊痰睦斫鈶?yīng)該很深了,所以本文就沒必要介紹概念了,重點來看一下如何使用Go語言來實現(xiàn)面向?qū)ο缶幊痰木幊田L格。

Go語言本身就不是一個面向?qū)ο蟮木幊陶Z言,所以Go語言中沒有類的概念,但是他是支持類型的,因此我們可以使用struct類型來提供類似于java中的類的服務(wù),可以定義屬性、方法、還能定義構(gòu)造器。來看個例子:

  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就是定義的構(gòu)造器。因為Go語言的特性問題,構(gòu)造器只能夠依靠我們手動來實現(xiàn)。

這里方法的實現(xiàn)是依賴于結(jié)構(gòu)體的值接收者、指針接收者的特性來實現(xiàn)的。

封裝

封裝是把一個對象的屬性私有化,同時提供一些可以被外界訪問的屬性和方法,如果不想被外界訪問,我們大可不必提供方法給外界訪問。在Go語言中實現(xiàn)封裝我們可以采用兩種方式:

Go語言支持包級別的封裝,小寫字母開頭的名稱只能在該包內(nèi)程序中可見,所以我們?nèi)绻幌氡┞兑恍┓椒?,可以通過這種方式私有包中的內(nèi)容,這個理解比較簡單,就不舉例子了。

Go語言可以通過 type 關(guān)鍵字創(chuàng)建新的類型,所以我們?yōu)榱瞬槐┞兑恍傩院头椒ǎ梢圆捎脛?chuàng)建一個新類型的方式,自己手寫構(gòu)造器的方式實現(xiàn)封裝,舉個例子:

  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,本質(zhì)是一個string類型,NewIdCard用來構(gòu)造對象,

GetPlaceOfBirth、GetBirthDay就是封裝的方法。

繼承

Go并沒有原生級別的繼承支持,不過我們可以使用組合的方式來實現(xiàn)繼承,通過結(jié)構(gòu)體內(nèi)嵌類型的方式實現(xiàn)繼承,典型的應(yīng)用是內(nèi)嵌匿名結(jié)構(gòu)體類型和內(nèi)嵌匿名接口類型,這兩種方式還有點細微差別:

  • 內(nèi)嵌匿名結(jié)構(gòu)體類型:將父結(jié)構(gòu)體嵌入到子結(jié)構(gòu)體中,子結(jié)構(gòu)體擁有父結(jié)構(gòu)體的屬性和方法,但是這種方式不能支持參數(shù)多態(tài)。
  • 內(nèi)嵌匿名接口類型:將接口類型嵌入到結(jié)構(gòu)體中,該結(jié)構(gòu)體默認實現(xiàn)了該接口的所有方法,該結(jié)構(gòu)體也可以對這些方法進行重寫,這種方式可以支持參數(shù)多態(tài),這里要注意一個點是如果嵌入類型沒有實現(xiàn)所有接口方法,會引起編譯時未被發(fā)現(xiàn)的運行錯誤。

內(nèi)嵌匿名結(jié)構(gòu)體類型實現(xiàn)繼承的例子

  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) 

上面注釋掉的方法就證明了不能進行參數(shù)多態(tài)。

內(nèi)嵌匿名接口類型實現(xiàn)繼承的例子

直接拿一個業(yè)務(wù)場景舉例子,假設(shè)現(xiàn)在我們現(xiàn)在要給用戶發(fā)一個通知,web、app端發(fā)送的通知內(nèi)容都是一樣的,但是點擊后的動作是不一樣的,所以我們可以進行抽象一個接口OrderChangeNotificationHandler來聲明出三個公共方法:GenerateMessage、GeneratePhotos、generateUrl,所有類都會實現(xiàn)這三個方法,因為web、app端發(fā)送的內(nèi)容是一樣的,所以我們可以抽相出一個父類OrderChangeNotificationHandlerImpl來實現(xiàn)一個默認的方法,然后在寫兩個子類WebOrderChangeNotificationHandler、AppOrderChangeNotificationHandler去繼承父類重寫generateUrl方法即可,后面如果不同端的內(nèi)容有做修改,直接重寫父類方法就可以了,來看例子:

  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) 

因為所有組合都實現(xiàn)了OrderChangeNotificationHandler類型,所以可以處理任何特定類型以及是該特定類型的派生類的通配符。

多態(tài)

多態(tài)是面向?qū)ο缶幊痰谋举|(zhì),多態(tài)是支代碼可以根據(jù)類型的具體實現(xiàn)采取不同行為的能力,在Go語言中任何用戶定義的類型都可以實現(xiàn)任何接口,所以通過不同實體類型對接口值方法的調(diào)用就是多態(tài),舉個例子:

  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) 

總結(jié)

歸根結(jié)底面向?qū)ο缶幊叹褪且环N編程思想,只不過有些語言在語法特性方面更好的為這種思想提供了支持,寫出面向?qū)ο蟮拇a更容易,但是寫代碼的還是我們自己,并不是我們用了java就一定會寫出更抽象的代碼,在工作中我看到用java寫出面向過程式的代碼不勝其數(shù),所以無論用什么語言,我們都應(yīng)該思考如何寫好一份代碼,大量的抽象接口幫助我們精簡代碼,代碼是優(yōu)雅了,但也會面臨著可讀性的問題,什么事都是有兩面性的,寫出好代碼的路還很長,還需要不斷探索............。

文中示例代碼已經(jīng)上傳github:https://github.com/asong2020/Golang_Dream/tree/master/code_demo/oop

 

責任編輯:武曉燕 來源: Golang夢工廠
相關(guān)推薦

2009-08-07 10:20:38

WebSphere A

2024-04-02 07:32:58

Go語言接口

2021-05-28 05:34:06

Golang語言編程

2024-01-08 07:02:48

數(shù)據(jù)設(shè)計模式

2022-11-28 08:15:14

Go語言代碼

2022-12-05 09:32:29

Go 語言風格規(guī)范

2013-03-11 09:23:22

Go語言面向?qū)ο?/a>

2022-10-21 09:01:41

StudentC++類型

2010-03-01 17:47:53

Python語言

2021-12-27 08:27:18

RepoGo 代碼

2021-05-20 08:54:16

Go面向對象

2009-07-14 16:51:50

Jython中的對象

2020-05-14 09:15:52

設(shè)計模式SOLID 原則JS

2019-11-18 17:05:02

JavaScript面向?qū)ο蟪绦蚓幊?/a>Java

2020-07-15 08:17:16

代碼

2010-08-24 16:00:43

C語言

2011-06-02 09:47:11

C語言重構(gòu)

2010-01-22 10:26:40

C++語言

2013-08-20 18:50:46

JS模板引擎模塊化Web App

2020-05-11 15:23:58

CQRS代碼命令
點贊
收藏

51CTO技術(shù)棧公眾號