首頁 後端開發 Golang Golang 中的物件導向程式設計 (OOP) 簡介

Golang 中的物件導向程式設計 (OOP) 簡介

Dec 23, 2024 pm 12:43 PM

當我們談論程式設計時,我們通常指的是編寫一堆修改某些資料並與某些資料互動的函數。物件導向程式設計(OOP)是一種程式設計模型,它專注於包含資料並附加一些相關功能的「物件」。物件導向程式設計有四大支柱:繼承、封裝、多態性與抽象。在本部落格中,我們將透過範例了解如何在 Golang 中實現它們中的每一個。推薦一些有關 OOP 的基本概念,但如果沒有,我將簡要介紹所有四個支柱的含義。

Introduction to Object Oriented Programming (OOP) in Golang

類別、物件和方法

物件導向程式設計的核心思想可以概括為以下幾點:

  • 您定義“類別”,它們是資料和可以呼叫該資料的函數的集合。
  • 這些特定的函數稱為該特定類別的「方法」。
  • 類別的實際實例稱為「物件」。

讓我們來看一些 Golang 中的程式碼來理解這三個概念:

package main

import "fmt"

type Batman struct {
    actor string
    year int
}

func (b Batman) SayImBatman() {
    fmt.Printf("I'm %s and I'm Batman from year %d\n", b.actor, b.year)
}

func main() {
    b1 := Batman{actor: "Michael Keaton", year: 1989}
    b2 := Batman{actor: "Christian Bale", year: 2005}

    b1.SayImBatman()
    b2.SayImBatman()
}

登入後複製
登入後複製
登入後複製

在Golang中,類別只不過是我們定義的型別。這些類型不一定必須是結構體,但通常都是結構體,因為在 OOP 中我們使用資料集合,這些資料可以是任何類型(字串、整數等)。

類別是物件的藍圖。每當實例化一個類別時,就會形成一個物件。在此範例中,b1 和 b2 是 Batman 類別的物件。

SayImBatman 函數可以在該類別的任何物件上呼叫。由於它與 Batman 類別相關,因此不稱之為常規函數,而是稱為類別的方法

我認為這應該足以讓您清楚 OOP 的基礎知識,以便您繼續進行下一部分,我們將在其中介紹 OOP 的四大支柱。

遺產

繼承引入了 OOP 中 parentchild 類別的概念。子類別是從父類別派生的類,並繼承其所有方法和屬性(資料)。讓我們來看看一些程式碼來幫助我們理解這一點:

package main

import "fmt"

type Hero struct {
    team string
}

type Batman struct {
    Hero
    name string
}

type Ironman struct {
    Hero
    power int
}

func (h Hero) SayTeam() {
    fmt.Println("My Team is", h.team)
}

func (b Batman) SayImBatman() {
    fmt.Printf("I'm %s and I'm Batman\n", b.name)
}

func (i Ironman) SayPowerLevel() {
    fmt.Printf("I'm Ironman and my powerlevel is %d\n", i.power)
}

func main() {
    b1 := Batman{Hero{team: "Justice League"}, "Christian Bale"}
    i1 := Ironman{Hero{team: "Avengers"}, 23}

    b1.SayImBatman()
    b1.SayTeam()

    i1.SayPowerLevel()
    i1.SayTeam()
}

登入後複製
登入後複製
登入後複製

在此範例中,蝙蝠俠和鋼鐵人是 Hero 父類的子類別。他們可以存取其父類別的屬性(即 team)及其方法(即 SayTeam)。正如您在聲明 b1 和 i1 實例時所看到的那樣,我們指定了父類別屬性以及它們各自類別的特定屬性。它們都能夠呼叫父類別中定義的 SayTeam 方法。但它們也有各自獨特的屬性和方法。

Golang 使用組合(在結構體中使用結構體)來實現繼承。它不像其他 OOP 語言(例如 C 或 Java)那樣具有內建的基於類別的繼承。

封裝

封裝是隱藏物件內部屬性且不允許直接修改它們的原理。相反,它依賴於提供獲取和更新這些屬性的方法。讓我們看一個例子來更好地理解這一點:

package main

import "fmt"

type Batman struct {
    actor string
    year int
}

func (b Batman) SayImBatman() {
    fmt.Printf("I'm %s and I'm Batman from year %d\n", b.actor, b.year)
}

func main() {
    b1 := Batman{actor: "Michael Keaton", year: 1989}
    b2 := Batman{actor: "Christian Bale", year: 2005}

    b1.SayImBatman()
    b2.SayImBatman()
}

登入後複製
登入後複製
登入後複製
package main

import "fmt"

type Hero struct {
    team string
}

type Batman struct {
    Hero
    name string
}

type Ironman struct {
    Hero
    power int
}

func (h Hero) SayTeam() {
    fmt.Println("My Team is", h.team)
}

func (b Batman) SayImBatman() {
    fmt.Printf("I'm %s and I'm Batman\n", b.name)
}

func (i Ironman) SayPowerLevel() {
    fmt.Printf("I'm Ironman and my powerlevel is %d\n", i.power)
}

func main() {
    b1 := Batman{Hero{team: "Justice League"}, "Christian Bale"}
    i1 := Ironman{Hero{team: "Avengers"}, 23}

    b1.SayImBatman()
    b1.SayTeam()

    i1.SayPowerLevel()
    i1.SayTeam()
}

登入後複製
登入後複製
登入後複製

在Golang中,從套件中導出的屬性和方法以大寫字母開頭。當我們在utils包中定義小寫的actor和year時,我們確保它們不能被直接修改。相反,就像您在 main.go 檔案中看到的那樣,您需要使用匯出的方法(以大寫字母開頭) - GetActor、SetActor 等來獲取和修改它們。

這就是封裝的全部內容 - 確保防止資料意外更改,並提供與資料安全互動的方法。

您會注意到不同的一件事是,在 Batman 類別的所有方法中,我們使用指標接收器 *Batman 而不是像前面範例中那樣使用值接收器 Batman。這是因為我們希望能夠在 Set 方法中修改原始結構。在 Golang 中,最佳實踐是,如果某些方法需要指標接收器,則讓所有方法都使用指標接收器以保持一致性。這就是為什麼 Get 方法也使用指標接收器,即使它們沒有修改原始結構。

此外,還要注意的一件事是,僅僅因為我們使用的是指標接收器,我們就不必這樣做:(&b1).GetActor。在 Golang 中,帶有指針參數的函數必須採用指針,但帶有指針接收器的方法可以採用值或指針作為接收器。

TL;DR:Golang 自動將b1.GetActor 轉換為(&b1).GetActor,因為GetActor 方法有一個指標接收器,但如果GetActor 是一個普通函數,它不會將GetActor(b1) 轉換為GetActor( &b1)一個指標參數。

多態性與抽象

OOP 的接下來兩個支柱可以合併在一起,因為它們的程式碼範例看起來非常相似。多態性是指兩個不同類別的兩個不同物件可以被視為同一公共超類別的物件的程式設計實踐。這意味著您可以在兩個不同的物件上呼叫相同的函數,就好像它們是同一類別的物件一樣。這應該開始讓您感受到所涉及的介面的味道:)

讓我們看一些程式碼來更好地理解這一點:

package main

import "fmt"

type Batman struct {
    actor string
    year int
}

func (b Batman) SayImBatman() {
    fmt.Printf("I'm %s and I'm Batman from year %d\n", b.actor, b.year)
}

func main() {
    b1 := Batman{actor: "Michael Keaton", year: 1989}
    b2 := Batman{actor: "Christian Bale", year: 2005}

    b1.SayImBatman()
    b2.SayImBatman()
}

登入後複製
登入後複製
登入後複製

在此範例中,StartFight 函數可以傳遞 b1 和 i1 對象,即使它們彼此沒有任何關係。試著理解這與繼承有何不同,在繼承中,子類別可以存取父類別的方法。在此範例中,沒有子類別和父類別(也沒有共用方法)。相反,兩個不同的物件被函數視為相同:這稱為多態性。

現在,這也可以被視為抽象的例子。顧名思義,抽像是一種隱藏實作細節,而只是提供為您處理事務的函數的程式設計實踐。在這個例子中,你不需要關心單一英雄的方法是如何被配置的。當您想使用任何英雄的戰鬥功能時,您可以隨時繼續使用開始戰鬥功能。這樣,實作細節對使用者來說是隱藏的,只暴露基本的細節。

現在回到多態,有兩個更常見的例子,那就是方法重寫和重載。

方法重寫

方法重寫是指子類別定義自己的父類別中定義的方法的實作。現在使用此實作來代替原始父類別的實作。讓我們看看之前用於繼承的程式碼,看看它在方法重寫後的樣子:

package main

import "fmt"

type Hero struct {
    team string
}

type Batman struct {
    Hero
    name string
}

type Ironman struct {
    Hero
    power int
}

func (h Hero) SayTeam() {
    fmt.Println("My Team is", h.team)
}

func (b Batman) SayImBatman() {
    fmt.Printf("I'm %s and I'm Batman\n", b.name)
}

func (i Ironman) SayPowerLevel() {
    fmt.Printf("I'm Ironman and my powerlevel is %d\n", i.power)
}

func main() {
    b1 := Batman{Hero{team: "Justice League"}, "Christian Bale"}
    i1 := Ironman{Hero{team: "Avengers"}, 23}

    b1.SayImBatman()
    b1.SayTeam()

    i1.SayPowerLevel()
    i1.SayTeam()
}

登入後複製
登入後複製
登入後複製

程式的輸出是:

//oops-in-go/utils/utils.go

package utils

type Batman struct {
    actor string
    year int
}

func (b *Batman) GetActor() string {
    return b.actor
}

func (b *Batman) GetYear() int {
    return b.year
}

func (b *Batman) SetActor(actor string) {
    b.actor = actor
}

func (b *Batman) SetYear(year int) {
    b.year = year
}
登入後複製

Batman 類別的物件現在使用自己的 SayTeam 方法,而不是父 Hero 類別的方法。由於 Ironman 類別沒有自己的 SayTeam 方法,因此它的物件仍然使用其父類別的方法。這就是方法重寫的意思,子類別「重寫」父類別中定義的方法。

方法重載

這是指同一個函數能夠接受多個不同的參數。這些參數的數量或類型可能不同。 Golang 提供了兩種方法來實現這一點:透過可變參數函數,另一種透過介面。

讓我們來看看兩者的程式碼,這將幫助您更好地理解:

使用可變參數函數

// oops-in-go/main.go

package main

import (
    "fmt"
    "oops-in-go/utils"
)

func main() {
    b1 := utils.Batman{}
    b1.SetActor("Michael Keaton")
    b1.SetYear(1989)
    fmt.Printf("I'm %s and I'm Batman from year %d\n", b1.GetActor(), b1.GetYear())

    b1.SetActor("Christian Bale")
    b1.SetYear(2005)
    fmt.Printf("I'm %s and I'm Batman from year %d\n", b1.GetActor(), b1.GetYear())
}

登入後複製

在這裡,您可以使用任意數量個參數「重載」listMembers函數。

使用介面

package main

import "fmt"

type Hero interface {
    Fight()
}

type Batman struct {
    weapon string
}

type Ironman struct {
    weapon string
}

func (b Batman) Fight() {
    fmt.Printf("Batman hits with a %s\n", b.weapon)
}

func (i Ironman) Fight() {
    fmt.Printf("Ironman hits with a %s\n", i.weapon)
}

func StartFight(h Hero) {
    fmt.Println("Fight has started.")
    h.Fight()
}

func main() {
    b1 := Batman{"Batarang"}
    i1 := Ironman{"Repulsor rays"}

    StartFight(b1)
    StartFight(i1)
}

登入後複製

程式的輸出是:

package main

import "fmt"

type Hero struct {
    team string
}

type Batman struct {
    Hero
    name string
}

type Ironman struct {
    Hero
    power int
}

func (h Hero) SayTeam() {
    fmt.Println("My Team is", h.team)
}

func (b Batman) SayImBatman() {
    fmt.Printf("I'm %s and I'm Batman\n", b.name)
}

func (i Ironman) SayPowerLevel() {
    fmt.Printf("I'm Ironman and my powerlevel is %d\n", i.power)
}

func (b Batman) SayTeam() {
    fmt.Printf("I'm Batman and my team is %s\n", b.team)
}

func main() {
    b1 := Batman{Hero{team: "Justice League"}, "Christian Bale"}
    i1 := Ironman{Hero{team: "Avengers"}, 23}

    b1.SayImBatman()
    b1.SayTeam()

    i1.SayPowerLevel()
    i1.SayTeam()
}

登入後複製

這裡我們「重載」了 saySomething 方法來接受不同類型的參數。我們採用一個空介面作為參數,它可以是任何類型,然後使用 switch case 檢查其類型並相應地列印輸出。

結論

我很清楚這是一篇很長的文章,如果您堅持讀到最後,我想讓您知道我真的很高興:) 我真誠地希望您學到很多有關面向對象編程的新知識以及如何在Golang 中實現它。我在我的網站上撰寫有關不同技術概念的博客,如果您有興趣學習新事物,我建議您訂閱我的時事通訊。

以上是Golang 中的物件導向程式設計 (OOP) 簡介的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

<🎜>:泡泡膠模擬器無窮大 - 如何獲取和使用皇家鑰匙
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
北端:融合系統,解釋
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
Mandragora:巫婆樹的耳語 - 如何解鎖抓鉤
3 週前 By 尊渡假赌尊渡假赌尊渡假赌

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

熱門話題

Java教學
1666
14
CakePHP 教程
1425
52
Laravel 教程
1324
25
PHP教程
1272
29
C# 教程
1251
24
Golang vs. Python:性能和可伸縮性 Golang vs. Python:性能和可伸縮性 Apr 19, 2025 am 12:18 AM

Golang在性能和可擴展性方面優於Python。 1)Golang的編譯型特性和高效並發模型使其在高並發場景下表現出色。 2)Python作為解釋型語言,執行速度較慢,但通過工具如Cython可優化性能。

Golang和C:並發與原始速度 Golang和C:並發與原始速度 Apr 21, 2025 am 12:16 AM

Golang在並發性上優於C ,而C 在原始速度上優於Golang。 1)Golang通過goroutine和channel實現高效並發,適合處理大量並發任務。 2)C 通過編譯器優化和標準庫,提供接近硬件的高性能,適合需要極致優化的應用。

Golang的影響:速度,效率和簡單性 Golang的影響:速度,效率和簡單性 Apr 14, 2025 am 12:11 AM

goimpactsdevelopmentpositationality throughspeed,效率和模擬性。 1)速度:gocompilesquicklyandrunseff,IdealforlargeProjects.2)效率:效率:ITScomprehenSevestAndardArdardArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdEcceSteral Depentencies,增強的Depleflovelmentimency.3)簡單性。

開始GO:初學者指南 開始GO:初學者指南 Apr 26, 2025 am 12:21 AM

goisidealforbeginnersandsubableforforcloudnetworkservicesduetoitssimplicity,效率和concurrencyFeatures.1)installgromtheofficialwebsitealwebsiteandverifywith'.2)

Golang vs.C:性能和速度比較 Golang vs.C:性能和速度比較 Apr 21, 2025 am 12:13 AM

Golang適合快速開發和並發場景,C 適用於需要極致性能和低級控制的場景。 1)Golang通過垃圾回收和並發機制提升性能,適合高並發Web服務開發。 2)C 通過手動內存管理和編譯器優化達到極致性能,適用於嵌入式系統開發。

Golang vs. Python:主要差異和相似之處 Golang vs. Python:主要差異和相似之處 Apr 17, 2025 am 12:15 AM

Golang和Python各有优势:Golang适合高性能和并发编程,Python适用于数据科学和Web开发。Golang以其并发模型和高效性能著称,Python则以简洁语法和丰富库生态系统著称。

C和Golang:表演至關重要時 C和Golang:表演至關重要時 Apr 13, 2025 am 12:11 AM

C 更適合需要直接控制硬件資源和高性能優化的場景,而Golang更適合需要快速開發和高並發處理的場景。 1.C 的優勢在於其接近硬件的特性和高度的優化能力,適合遊戲開發等高性能需求。 2.Golang的優勢在於其簡潔的語法和天然的並發支持,適合高並發服務開發。

Golang和C:性能的權衡 Golang和C:性能的權衡 Apr 17, 2025 am 12:18 AM

Golang和C 在性能上的差異主要體現在內存管理、編譯優化和運行時效率等方面。 1)Golang的垃圾回收機制方便但可能影響性能,2)C 的手動內存管理和編譯器優化在遞歸計算中表現更為高效。

See all articles