站長資訊網(wǎng)
        最全最豐富的資訊網(wǎng)站

        golang中有沒有類

        golang中沒有類。golang不是一門純面向?qū)ο缶幊陶Z言,它沒有class(類)的概念,也就沒有繼承的說法,但Go也可以模擬面向?qū)ο蟮木幊谭绞健T贕o中,可以將struct比作其它語言中的class;通過struct定義結(jié)構(gòu)體,表征一類對象,例“type person struct {…}”。

        golang中有沒有類

        本教程操作環(huán)境:windows7系統(tǒng)、GO 1.18版本、Dell G3電腦。

        面向?qū)ο笕筇卣鳎悍庋b,繼承,多態(tài)。

        Go不是一門純面向?qū)ο缶幊陶Z言,它沒有class(類)的概念,也就沒有繼承的說法。但Go也可以模擬面向?qū)ο蟮木幊谭绞剑纯梢詫truct比作其它語言中的class。

        對象

        Go沒有class的概念,通過struct定義結(jié)構(gòu)體,表征一類對象。

        type person struct { 	Age  int 	Name string }
        登錄后復(fù)制

        對象是狀態(tài)與行為的有機體。例如下面的java代碼:

        public class Person {      int age;      String name;      public int getAge() {         return age;     }      public void setAge(int age) {         this.age = age;     }      public String getName() {         return name;     }      public void setName(String name) {         this.name = name;     } }
        登錄后復(fù)制

        不同于Java,Go的方法不需要跟類的數(shù)據(jù)綁定在一個class的定義里面,只需要定義在同一個包內(nèi)。這一點可能初學(xué)Go的同學(xué),會感覺很奇怪。

        type person struct { 	Age  int 	Name string }  func (p *person) GetAge() int { 	return p.Age }  func (p *person) SetAge(age int)  { 	p.Age = age }  func (p *person) GetName() string { 	return p.Name }  func (p *person) SetName(name string) { 	p.Name = name }
        登錄后復(fù)制

        構(gòu)造函數(shù)

        Go沒有構(gòu)造函數(shù),對象的數(shù)據(jù)載體就是一個struct。Java支持構(gòu)造函數(shù),構(gòu)造函數(shù)名字就跟類名字一樣,多個構(gòu)造函數(shù)通過函數(shù)重載實現(xiàn)。

        而Go構(gòu)造函數(shù)則通過工廠函數(shù)進行模擬。實例如下:

        type person struct { 	Age  int 	Name string }  /** 	構(gòu)造函數(shù)1--通過名字初始化  */ func newPersonByName(name string) *person { 	return &person{ 		Name: name, 	} }  /** 	構(gòu)造函數(shù)2--通過年齡初始化  */ func newPersonByAge(age int) *person { 	return &person{ 		Age: age, 	} }
        登錄后復(fù)制

        需要注意的是,person結(jié)構(gòu)體的名稱首字母要小寫,避免外部直接越過模擬的構(gòu)造函數(shù)

        訪問權(quán)限

        Java有四種訪問權(quán)限,如下所示:

        java訪問控制符
        public protected

        friendly

        (default)

        private
        同一個類 yes yes yes yes
        同一個包 yes yes yes no
        不同包子類 yes yes no no
        不同包非子類 yes no no no

        Go則做了簡化,可見性的最小粒度是包。也就是說,Go保留兩種,friendly和public。Go的變量名如果首字母是小寫,則代表包內(nèi)可見;如果首字母是大寫,則代表任何地方都可見。

        封裝

        封裝,把抽象出來的結(jié)構(gòu)體跟操作結(jié)構(gòu)體內(nèi)部數(shù)據(jù)的函數(shù)綁定在一起。外部程序只能根據(jù)導(dǎo)出的函數(shù)API(public方法)修改結(jié)構(gòu)體的內(nèi)部狀態(tài)。

        封裝有兩個好處:

        隱藏實現(xiàn):我們只希望使用者直接使用API操作結(jié)構(gòu)體內(nèi)部狀態(tài),而無需了解內(nèi)部邏輯。就好像一座冰山,我們只看到它露出水面的那一部分。

        保護數(shù)據(jù):我們可以對數(shù)據(jù)的修改和訪問施加安全措施,調(diào)用setter方法的時候,我們可以對參數(shù)進行校驗;調(diào)用getter方法,我們可以增加訪問日志等等。

        一個簡單的bean定義如下所示:

        type person struct { 	Age  int 	Name string }  func NewPerson(age int, name string) *person{ 	return &person{age, name} }  func (p *person) SetAge(age int)  { 	p.Age = age }  func (p *person) SetName(name string) { 	p.Name = name }  func main() { 	p:= NewPerson(10, "Lily") 	p.SetName("Lucy") 	p.SetAge(18) }
        登錄后復(fù)制

        需要注意的是,Go的方法是一種特殊的函數(shù),只是編譯器的一種語法糖,編譯器瞧瞧幫我們把對象的引用作為函數(shù)的第一個參數(shù)。例如,下面的代碼是等價的

        func main() { 	p:= NewPerson(10, "Lily")  	p.SetName("Lily1") 	// 等價于下面的寫法 	// p是一個引用,函數(shù)引用 	setNameFunc := (*person).SetName 	setNameFunc(p, "Lily2") 	fmt.Println(p.Name) }
        登錄后復(fù)制

        繼承

        繼承,子類繼承父類,則獲得父類的特征和行為。繼承的主要目的是為了重用代碼。Java實現(xiàn)代碼重用的兩大利器,就是繼承和組合。

        Go沒有class的概念,談不上繼承。但Go可以通過匿名組合來模擬繼承。

        如下所示,Cat通過匿名聚合了Animal結(jié)構(gòu)體,就自動獲得了Animal的move()和Shout()方法:

        type Animal struct { 	Name string }  func (Animal) move()  { 	fmt.Println("我會走") }  func (Animal) shout()  { 	fmt.Println("我會叫") }  type Cat struct { 	Animal // 匿名聚合 }  func main() { 	cat := &Cat{Animal{"貓"}}  	cat.move() 	cat.shout() }
        登錄后復(fù)制

        多態(tài)

        多態(tài),申明為基類的變量,可以在運行期指向不同的子類,并調(diào)用不同子類的方法。多態(tài)的目的是為了統(tǒng)一實現(xiàn)。

        我們通過接口來實現(xiàn)多態(tài)。在java里,我們通過interface來定義接口,通過implements來實現(xiàn)接口。

        interface Animal {      void move();      void shout(); }  class Dog implements Animal {      @Override     public void move() {         System.out.println("我會走");     }      @Override     public void shout() {         System.out.println("我會叫");     } }
        登錄后復(fù)制

        而Go則是通過鴨子類型推斷,只要某個對象長得想鴨子,叫起來像鴨子,那么它就是鴨子。也就是說,Go的接口是比較隱匿的,只要某個對象實現(xiàn)來接口申明的所有方法,那么就認為它屬于該接口。

        type Animal interface {  	move() 	shout() }  type Cat struct { 	Animal // 匿名聚合 }  func (Cat)move()  { 	fmt.Println("貓會走") }  func (Cat)shout()  { 	fmt.Println("貓會叫") }  type Dog struct { 	Animal  // 匿名聚合 }   func (Dog)move()  { 	fmt.Println("狗會走") }  func (Dog)shout()  { 	fmt.Println("狗會叫") }  func main() { 	cat := Cat{} 	dog := Dog{}     // 申明接口數(shù)組  	animals := []Animal{cat, dog} 	for _,ele := range animals {         // 統(tǒng)一訪問 		ele.move() 		ele.shout() 	} }
        登錄后復(fù)制

        贊(0)
        分享到: 更多 (0)
        網(wǎng)站地圖   滬ICP備18035694號-2    滬公網(wǎng)安備31011702889846號
        主站蜘蛛池模板: 国产精品成人99久久久久| 国产一区二区三区欧美精品| 精品久久久久久无码中文字幕| 久久精品www人人爽人人| 久久亚洲精品无码播放| 久久精品这里热有精品| 国内精品久久久久影院一蜜桃| 四虎国产精品永久在线看| 国产精品日韩欧美在线第3页| 99久久国产热无码精品免费| 香港aa三级久久三级老师2021国产三级精品三级在 | 亚洲精品狼友在线播放| 人妻少妇精品久久| 韩国三级中文字幕hd久久精品 | 久久精品国产精品亜洲毛片| 91久久精品国产成人久久| 久久亚洲国产精品一区二区| 久久青草国产精品一区| 成人午夜精品网站在线观看| 精品亚洲麻豆1区2区3区| 中文字幕精品一区二区三区视频| 久久亚洲av无码精品浪潮| 精品多毛少妇人妻AV免费久久 | 99re6这里有精品热视频| 孩交VIDEOS精品乱子| 欧美成人精品高清在线观看| 日韩国产精品无码一区二区三区| 亚洲精品无码久久久久去q | 国产精品v欧美精品v日韩| 奇米精品一区二区三区在线观看 | 国产cosplay精品视频| 国产成人精品无码片区在线观看 | 国内精品久久久久久野外| 精品999在线| 日韩精品一区二区三区四区 | 久久这里只精品国产99热| 欧美精品一区二区三区视频| 麻豆精品不卡国产免费看| 日本一区精品久久久久影院 | 国产精品1024香蕉在线观看| 精品国产日产一区二区三区|