美文网首页
go面向对象-结构体

go面向对象-结构体

作者: 呦丶耍脾气 | 来源:发表于2023-03-09 19:44 被阅读0次

    Go没有沿袭传统面向对象编程中的诸多概念,也没有提供类(class),但是它提供了结构体(struct),方法(method)可以在结构体上添加。与类相似,结构体提供了捆绑数据和方法的行为。

    1. 介绍

    1.1 概念

    单一的数据类型已经满足不了现实开发需求,于是 Go 语言提供了结构体来定义复杂的数据类型。结构体是由一系列相同类型或不同类型的数据构成的数据集合。结构体的定义只是一种内存布局的描述,只有当结构体实例化时,才会真正地分配内存。因此必须在定义结构体并实例化后才能使用结构体的字段。

    1.2 语法

    type 类型名称 struct {
       field type
       field type 
       field1,field2,field3 type // 同类型变量可以写在一行
    }
    

    1.3 注意事项

    • 类型名是标识结构体的名称,在同一个包内不能重复。
    • 结构体的属性,也叫字段(field),必须唯一。
    • 同类型的成员属性可以写在一行。
    • 结构体是值类型。
    • 只有当结构体实例化时,才能使用结构体的字段。

    2. 实例化

    2.1 使用var

    package main
    import "fmt"
    // 定义结构体
    type student struct {
        name string
        age int
        like []string
    }
    func main() {
        // 使用var 实例化结构体
        var s student
        fmt.Printf("变量s--> 类型: %T 值: %v \n",s,s)
        // 给属性赋值
        s.name = "张三"
        s.age = 20
        s.like = []string{"打游戏","看动漫"}
        fmt.Printf("给属性赋值: 变量s--> 类型: %T 值: %v \n",s,s)
    }
    /*输出
    变量s--> 类型: main.student 值: { 0 []} 
    给属性赋值: 变量s--> 类型: main.student 值: {张三 20 [打游戏 看动漫]} 
    */
    

    3.2 使用简短声明 (:=)

    package main
    import "fmt"
    // 定义结构体
    type student struct {
        name string
        age int
        like []string
    }
    func main() {
        // 方式一: 先实例化结构体,后赋值
        s := student{}
        // 给属性赋值
        s.name = "张三"
        s.age = 20
        s.like = []string{"打游戏","看动漫"}
        fmt.Printf(" 变量s--> 类型: %T 值: %v \n",s,s)
    
        // 方式二: 声明时初始化
        s1 := student{
            name: "李四",
            age:  23,
            like: []string{"旅游","运动"},
        }
        fmt.Printf(" 变量s1--> 类型: %T 值: %v \n",s1,s1)
    
        // 方式三: 声明时初始化,省略属性
        s2 := student{"王麻子",30,[]string{"睡觉","吃饭"}}
        fmt.Printf(" 变量s2--> 类型: %T 值: %v \n",s2,s2)
    }
    /**输出
     变量s--> 类型: main.student 值: {张三 20 [打游戏 看动漫]} 
     变量s1--> 类型: main.student 值: {李四 23 [旅游 运动]} 
     变量s2--> 类型: main.student 值: {王麻子 30 [睡觉 吃饭]} 
    */
    

    3.3 使用new

    使用内置函数new()对结构体进行实例化,结构体实例化后形成指针类型的结构体,new()内置函数会分配内存。第一个参数是类型,而不是值,返回的值是指向该类型新分配的零值的指针。

    package main
    import "fmt"
    // 定义结构体
    type student struct {
        name string
        age int
        like []string
    }
    func main() {
        // 使用new实例化
        s := new(student)
        fmt.Printf(" 变量s--> 类型: %T 值: %v \n",s,s)
        // 给属性赋值
        (*s).name = "包青天"
        (*s).age = 55
        (*s).like = []string{"判案"}
        fmt.Printf(" 变量s--> 类型: %T 值: %v \n",s,s)
        // 语法糖写法(省略*)
        s.name = "包大人"
        s.age = 99
        s.like = []string{"判案","元芳你怎么看"}
        fmt.Printf(" 变量s--> 类型: %T 值: %v \n",s,s)
    }
    /**输出
     变量s--> 类型: *main.student 值: &{ 0 []} 
     变量s--> 类型: *main.student 值: &{包青天 55 [判案]} 
     变量s--> 类型: *main.student 值: &{包大人 99 [判案 元芳你怎么看]} 
    */
    

    3. 结构体在函数中使用

    结构体作为函数参数,若复制一份传递到函数中,在函数中对参数进行修改,不会影响到实际参数,证明结构体是值类型。

    3.1 传结构体值作为参数

    package main
    import "fmt"
    // 定义结构体
    type student struct {
        name string
        age int
        like []string
    }
    func main() {
        s := student{name:"张三",age:17}
        fmt.Printf(" 变量s--> 值: %v \n",s)
        grownUp(s)
        fmt.Printf("调用函数后,变量s--> 值: %v \n",s)
    }
    // 传结构体值作为参数
    func grownUp( s student)  {
        s.age = 80
        s.name = "长大的 "+s.name
    }
    /**输出
     变量s--> 值: {张三 17 []} 
     调用函数后,变量s--> 值: {张三 17 []} 
    */
    

    3.2 传结构体指针作为参数

    package main
    import "fmt"
    // 定义结构体
    type student struct {
        name string
        age int
        like []string
    }
    func main() {
        s := student{name:"张三",age:17}
        fmt.Printf(" 变量s--> 值: %v \n",s)
      // 取址
        grownUp(&s)
        fmt.Printf("调用函数后,变量s--> 值: %v \n",s)
    }
    // 传结构体指针作为参数
    func grownUp( s *student)  {
        s.age = 80
        s.name = "长大的 "+s.name
    }
    /** 输出:
     变量s--> 值: {张三 17 []} 
     调用函数后,变量s--> 值: {长大的 张三 80 []} 
    */
    

    3.3 返回对象

    package main
    import "fmt"
    // 定义结构体
    type student struct {
        name string
        age int
        like []string
    }
    func main() {
        s := getStudent("杨过",40,[]string{"骑大雕"})
        fmt.Printf("函数返回值 s--> 值: %v  类型: %T \n",s,s)
    }
    // 作为值类型传递
    func getStudent( name string, age int,likes []string) student  {
        return student{name,age,likes}
    }
    // 函数返回值 s--> 值: {杨过 40 [骑大雕]}  类型: main.student 
    

    3.4 返回指针

    package main
    import "fmt"
    // 定义结构体
    type student struct {
        name string
        age int
        like []string
    }
    func main() {
        s := getStudent("杨过",40,[]string{"骑大雕"})
        fmt.Printf("函数返回值 s--> 值: %v  类型: %T \n",s,s)
    }
    
    // 返回指针
    func getStudent( name string, age int,likes []string) *student  {
        return &student{name,age,likes}
    }
    // 输出: 函数返回值 s--> 值: &{杨过 40 [骑大雕]}  类型: *main.student 
    

    4.匿名结构体

    4.1 语法

    变量名 := struct {
      // 定义成员属性
    } { /*初始化成员属性*/ }
    

    4.2 使用

    package main
    import "fmt"
    func main() {
        // 声明初始化匿名结构体
        s := struct {
            name, home, phone string
            age               int
        }{
            name:  "张二十",
            phone: "17600111111",
            age:   18,
        }
      // 打印
        fmt.Printf("变量 s--> 值: %v  类型: %T \n", s, s)
    }
    // 输出: 变量 s--> 值: {张二十  17600111111 18}  类型: struct { name string; home string; phone string; age int } 
    

    5.匿名字段

    5.1 定义

    匿名字段就是在结构体中的字段没有名字,只包含一个没有字段名的类型。这些字段被称为匿名字段。在同一个结构体中,同一个类型只能有一个匿名字段。

    5.2 使用

    package main
    import "fmt"
    type people struct {
        name, home string
        int        // 匿名字段
        float32    // 匿名字段
    }
    func main() {
        // 声明初始化匿名结构体
        s := people{name: "张三", home: "北京", int: 18, float32: 1.73}
        fmt.Printf("变量 s--> 值: %v \n", s)
        // 声明初始化匿名结构体(省略属性名)
        s2 := people{"李四", "南京", 22, 1.80}
        fmt.Printf("变量 s2--> 值: %v \n", s2)
    }
    /** 输出
      变量 s--> 值: {张三 北京 18 1.73} 
      变量 s2--> 值: {李四 南京 22 1.8} 
    */
    

    6. 结构体嵌套

    6.1 定义

    将一个结构体作为另一个结构体的属性(字段),这种结构就是结构体嵌套。

    结构体嵌套可以模拟面向对象编程中的以下两种关系。

    • 聚合关系: 一个类作为另一个类的属性。
    • 继承关系: 一个类作为另一个类的子类。子类和父类的关系。

    6.2 聚合场景

    模拟聚合关系时一定要采用有名字的结构体作为字段。

    package main
    import "fmt"
    // 定义学生结构体
    type student struct {
        name       string
        height      float32
        schoolInfo  school
    }
    // 定义学习结构体
    type school struct {
        schoolName, schoolAddress string
    }
    func main() {
        // 简短声明嵌套结构体
        s := student{"小张",1.72,school{"北京大学","北京"}}
        fmt.Printf("变量 s--> 值: %v 类型: %T \n", s,s)
        // 使用var
        var ss student
        ss.name = "小龙"
        ss.height = 1.67
        ss.schoolInfo = school{"南京大学","南京"}
        fmt.Printf("变量 ss--> 值: %v 类型: %T \n", ss,ss)
        // 使用new
        s2 := new(student)
        s2.name = "小虎"
        s2.height = 1.77
        s2.schoolInfo.schoolName = "武汉大学"
        s2.schoolInfo.schoolAddress = "武汉"
        fmt.Printf("变量 s2--> 值: %v 类型: %T \n", s2,s2)
    }
    /** 输出:
      变量 s--> 值: {小张 1.72 {北京大学 北京}} 类型: main.student 
      变量 ss--> 值: {小龙 1.67 {南京大学 南京}} 类型: main.student 
      变量 ss--> 值: &{小虎 1.77 {武汉大学 武汉}} 类型: *main.student 
    */
    

    6.3 模拟继承

    在结构体中,属于匿名结构体的字段称为提升字段,它们可以被访问,匿名结构体就像是该结构体的父类。

    package main
    import "fmt"
    
    // 定义父类结构体
    type people struct {
        name  string
        age   int
    }
    type student struct {
        people // 集成父类结构体
        class string
    }
    
    func main() {
        // 方式1.使用new声明结构体
        var s = new(student)
        // 集成父类成员
        s.name = "张三"
        s.age = 12
        // 自己成员
        s.class = "三年级"
        fmt.Printf("变量s -> %v \n",s)
        // 方式2.使用简短声明
        s2 := student{people{"李四",13},"四年级"}
        fmt.Printf("变量s2 -> %v \n",s2)
    }
    /** 输出
    变量s -> &{{张三 12} 三年级} 
    变量s2 -> {{李四 13} 四年级} 
    */
    

    6.4 成员冲突

    package main
    import "fmt"
    type A struct {
        name string
        age  int
    }
    type B struct {
        name   string
        height float32
    }
    // 在C结构体中嵌套A和B
    type C struct {
        A
        B
    }
    func main() {
        // 定义结构体C
        c := C{}
        // 不冲突的成员赋值
        c.age = 12
        c.height = 1.88
        // 冲突的成员赋值
        c.A.name = "这是A的成员"
        c.B.name = "这是B的成员"
        fmt.Printf("变量c -> %v \n", c)
    }
    // 输出: 变量c -> {{这是A的成员 12} {这是B的成员 1.88}} 
    

    7 结构体的遍历

    可以先了解,后面章节会有reflect的使用

    package main
    
    import (
        "fmt"
        "reflect"
    )
    
    type Student2 struct {
        Name string
        Age  int
    }
    
    type Girl struct {
        Like string
        Student2
    }
    
    func reflectFor() {
        s := Student2{"zhangsna", 11}
        valueOf := reflect.ValueOf(s)
        typeOf := reflect.TypeOf(s)
        for i := 0; i < typeOf.NumField(); i++ {
            fmt.Println("field:", typeOf.Field(i).Name, "value:", valueOf.Field(i))
        }
    }
    
    func reflectFor2() {
        s := Girl{"zhangsna", Student2{"lisi", 15}}
        valueOf := reflect.ValueOf(s)
        typeOf := reflect.TypeOf(s)
        for i := 0; i < typeOf.NumField(); i++ {
            if valueOf.Field(i).Type().Kind() == reflect.Struct {
                structFiled := valueOf.Field(i).Type()
                for j := 0; j < structFiled.NumField(); j++ {
                    fmt.Println("嵌套结构==>field:", structFiled.Field(j).Name, "value:", valueOf.Field(i).Field(j).Interface())
                }
            } else {
                fmt.Println("field:", typeOf.Field(i).Name, "value:", valueOf.Field(i))
            }
        }
    }
    
    func main() {
        reflectFor()
        reflectFor2()
    }
    
    /*
    field: Name value: zhangsna
    field: Age value: 11              
    field: Like value: zhangsna       
    嵌套结构==>field: Name value: lisi
    嵌套结构==>field: Age value: 15   
    */
    

    相关文章

      网友评论

          本文标题:go面向对象-结构体

          本文链接:https://www.haomeiwen.com/subject/uyqnldtx.html