zoukankan      html  css  js  c++  java
  • Go--struct

    Go语言中,也和C或者其它语言一样,也可以声明新的类型,作为其它类型的属性或者是字段的容器。例如我们可以创建一个自定义类型person代表一个人的实体。这个实体拥有属性:姓名和年龄。这样的类型我们称之为struct

    type person struct {
    	name string
    	age int
    }
    

      上面的person中包含了两个字段:

        一个string类型的字段name,用来保存用户名称这个属性

        一个int类型的字段age,用来保存用户年龄这个属性

    type person struct {
    	name string
    	age int
    }
    
    var P person  // P现在就是person类型的变量了
    
    P.name = "Astaxie"  // 赋值"Astaxie"给P的name属性.
    P.age = 25  // 赋值"25"给变量P的age属性
    fmt.Printf("The person's name is %s", P.name)  // 访问P的name属性.
    

      除了上面P的声明使用之外,还有另外集中声明使用方式:

        1、按照顺序提供初始化值

          P := person{"Tom", 25}

        2、通过field:value的方式初始化,这种方式的顺序可以任意调换

          P := person{age:24, name:"Tom"}

        3、可以通过new函数分配一个指针      

          P := new(person)

    package main
    
    import "fmt"
    
    // 声明一个新的类型
    type person struct {
    	name string
    	age int
    }
    
    // 比较两个人的年龄,返回年龄大的那个人,并且返回年龄差
    // struct也是传值的
    func Older(p1, p2 person) (person, int) {
    	if p1.age>p2.age {  // 比较p1和p2这两个人的年龄
    		return p1, p1.age-p2.age
    	}
    	return p2, p2.age-p1.age
    }
    
    func main() {
    	var tom person
    
    	// 赋值初始化
    	tom.name, tom.age = "Tom", 18
    
    	// 两个字段都写清楚的初始化
    	bob := person{age:25, name:"Bob"}
    
    	// 按照struct定义顺序初始化值
    	paul := person{"Paul", 43}
    
    	tb_Older, tb_diff := Older(tom, bob)
    	tp_Older, tp_diff := Older(tom, paul)
    	bp_Older, bp_diff := Older(bob, paul)
    
    	fmt.Printf("Of %s and %s, %s is older by %d years
    ",
    		tom.name, bob.name, tb_Older.name, tb_diff)
    
    	fmt.Printf("Of %s and %s, %s is older by %d years
    ",
    		tom.name, paul.name, tp_Older.name, tp_diff)
    
    	fmt.Printf("Of %s and %s, %s is older by %d years
    ",
    		bob.name, paul.name, bp_Older.name, bp_diff)
    }
    

      

    struct的匿名字段

      上面只是写了如何定义一个struct,定义的时候是字段名与其类型一一对应,实际上Go支持只提供类型,而不写字段名的方式,也就是匿名字段,也可以称为嵌入字段。

      当匿名字段是一个struct的时候,那么这个struct所拥有的全部字段都被隐式的引入了当前定义的struct

    package main
    
    import "fmt"
    
    type Human struct {
    	name string
    	age int
    	weight int
    }
    
    type Student struct {
    	Human  // 匿名字段,那么默认Student就包含了Human的所有字段
    	speciality string
    }
    
    func main() {
    	// 我们初始化一个学生
    	mark := Student{Human{"Mark", 25, 120}, "Computer Science"}
    
    	// 我们访问相应的字段
    	fmt.Println("His name is ", mark.name)
    	fmt.Println("His age is ", mark.age)
    	fmt.Println("His weight is ", mark.weight)
    	fmt.Println("His speciality is ", mark.speciality)
    	// 修改对应的备注信息
    	mark.speciality = "AI"
    	fmt.Println("Mark changed his speciality")
    	fmt.Println("His speciality is ", mark.speciality)
    	// 修改他的年龄信息
    	fmt.Println("Mark become old")
    	mark.age = 46
    	fmt.Println("His age is", mark.age)
    	// 修改他的体重信息
    	fmt.Println("Mark is not an athlet anymore")
    	mark.weight += 60
    	fmt.Println("His weight is", mark.weight)
    }
    

      

      在这里面我们可以看到Student访问age和name的时候就像访问自己的字段一样,其实匿名字段就是这样的,能够实现字段的继承

      通过匿名访问和修改字段相当有用,但是不仅仅是struct字段,所有的内置类型和自定义类型都是可以做为匿名字段的

    package main
    
    import "fmt"
    
    type Skills []string
    
    type Human struct {
    	name string
    	age int
    	weight int
    }
    
    type Student struct {
    	Human  // 匿名字段,struct
    	Skills // 匿名字段,自定义的类型string slice
    	int    // 内置类型作为匿名字段
    	speciality string
    }
    
    func main() {
    	// 初始化学生Jane
    	jane := Student{Human:Human{"Jane", 35, 100}, speciality:"Biology"}
    	// 现在我们来访问相应的字段
    	fmt.Println("Her name is ", jane.name)
    	fmt.Println("Her age is ", jane.age)
    	fmt.Println("Her weight is ", jane.weight)
    	fmt.Println("Her speciality is ", jane.speciality)
    	// 我们来修改他的skill技能字段
    	jane.Skills = []string{"anatomy"}
    	fmt.Println("Her skills are ", jane.Skills)
    	fmt.Println("She acquired two new ones ")
    	jane.Skills = append(jane.Skills, "physics", "golang")
    	fmt.Println("Her skills now are ", jane.Skills)
    	// 修改匿名内置类型字段
    	jane.int = 3
    	fmt.Println("Her preferred number is", jane.int)
    }
    

      struct不仅能够将struct作为匿名字段,自定义类型、内置类型都可以作为匿名字段,而且可以在相应的字段上面进行函数操作

      如果要是碰到两个struct重名的时候,Go中最外层的优先访问,这样就允许我们去重载通过匿名字段继承的一些字段,当然如果我们像访问重载后对应匿名类型里面的字段,可以通过匿名字段名来进行访问,

    package main
    
    import "fmt"
    
    type Human struct {
    	name string
    	age int
    	phone string  // Human类型拥有的字段
    }
    
    type Employee struct {
    	Human  // 匿名字段Human
    	speciality string
    	phone string  // 雇员的phone字段
    }
    
    func main() {
    	Bob := Employee{Human{"Bob", 34, "777-444-XXXX"}, "Designer", "333-222"}
    	fmt.Println("Bob's work phone is:", Bob.phone)
    	// 如果我们要访问Human的phone字段
    	fmt.Println("Bob's personal phone is:", Bob.Human.phone)
    }
    

      

  • 相关阅读:
    ASP.NET MVC5 :Attribute路由使用详解
    C# 常用字符串加密解密方法
    C#获取当前主机硬件信息
    用C#调用Windows API向指定窗口发送按键消息
    Win32 编程消息常量(C#)
    C#程序员开发WinForm必须知道的 Window 消息大全
    C#通过SendMessage发送消息,改变其他程序的下拉框控件(ComboBox)的值
    C#常用 API函数大全
    SendKeys发送组合键
    webapi get请求 FromUri list参数传递
  • 原文地址:https://www.cnblogs.com/tulintao/p/11825670.html
Copyright © 2011-2022 走看看