浅谈Go语言中的结构体struct & 接口Interface & 反射

结构体struct

struct 用来自定义复杂数据结构,可以包含多个字段(属性),可以嵌套;

go中的struct类型理解为类,可以定义方法,和函数定义有些许区别;

struct类型是值类型。

struct定义

type User struct {
 Name string
 Age int32
 mess string
}
var user User
var user1 *User = &User{}
var user2 *User = new(User)

struct使用

下面示例中user1和user2为指针类型,访问的时候编译器会自动把 user1.Name 转为 (*user1).Name

func main() {
 var user User
 user.Name = "nick"
 user.Age = 18
 user.mess = "lover"

 var user1 *User = &User{
  Name: "dawn",
  Age: 21,
 }
 fmt.Println(*user1)     //{dawn 21 }
 fmt.Println(user1.Name, (*user1).Name) //dawn dawn

 var user2 *User = new(User)
 user2.Name = "suoning"
 user2.Age = 18
 fmt.Println(user2)      //fmt.Println(user2.Name, (*user2).Name) //suoning suoning
}

构造函数

golang中的struct没有构造函数,可以伪造一个

type User struct {
 Name string
 Age int32
 mess string
}

func NewUser(name string, age int32, mess string) *User {
 return &User{Name:name,Age:age,mess:mess}
}

func main() {
 //user := new(User)
 user := NewUser("suoning", 18, "lover")
 fmt.Println(user, user.mess, user.Name, user.Age)
}

内存布局

struct中的所有字段在内存是连续的

var user User
 user.Name = "nick"
 user.Age = 18
 user.mess = "lover"

 fmt.Println(user)     //{nick 18 lover}
 fmt.Printf("Name:%p\n", &user.Name) //Name:0xc420016180
 fmt.Printf("Age: %p\n", &user.Age) //Age: 0xc420016190
 fmt.Printf("mess:%p\n", &user.mess) //mess:0xc420016198 8字节为内存对齐

方法

方法是作用在特定类型的变量上,因此自定义类型,都可以有方法,而不仅仅是struct。

方法的访问控制也是通过大小写控制。

init函数是通过传入指针实现,这样改变struct字段值,因为是值类型。

type User struct {
 Name string
 Age int
 sex string
}

func (this *User) init(name string, age int, sex string) {
 this.Name = name
 this.Age = age
 this.sex = sex
}

func (this User) GetName() string {
 return this.Name
}

func main() {
 var user User
 user.init("nick", 18, "man")
 //(&user).init("nick", 18, "man")
 name := user.GetName()
 fmt.Println(name)
}

匿名字段

如果有冲突的, 则最外的优先

type User struct {
 Name stirng
 Age int
}

type Lover struct {
  User
  sex time.Time
  int
  Age int
}

继承 & 多重继承

一个结构体继承多个结构体,访问通过点。继承字段以及方法。

可以起别名,如下面 u1(user1),访问 user.u1.Age。

如果继承的结构体都拥有同一个字段,通过user.name访问就会报错,必须通过user.user1.name来访问。

type user1 struct {
 name string
 Age int
}

type user2 struct {
 name string
 age int
 sex time.Time
}

type User struct {
 u1 user1 //别名
 user2
 Name string
 Age int
}

func main() {
 var user User
 user.Name = "nick"
 user.u1.Age = 18
 fmt.Println(user) //{{ 18} { 0 {0 0 <nil>}} nick 0}
}

tag

在go中,首字母大小写有特殊的语法含义,小写包外无法引用。由于需要和其它的系统进行数据交互,例如转成json格式。这个时候如果用属性名来作为键值可能不一定会符合项目要求。tag在转换成其它数据格式的时候,会使用其中特定的字段作为键值。

import "encoding/json"

type User struct {
 Name string `json:"userName"`
 Age int `json:"userAge"`
}

func main() {
 var user User
 user.Name = "nick"
 user.Age = 18

 conJson, _ := json.Marshal(user)
 fmt.Println(string(conJson)) //{"userName":"nick","userAge":0}
}

String()

如果实现了String()这个方法,那么fmt默认会调用String()。

type name1 struct {
 int
 string
}

func (this *name1) String() string {
 return fmt.Sprintf("This is String(%s).", this.string)
}

func main() {
 n := new(name1)
 fmt.Println(n) //This is String().
 n.string = "suoning"
 d := fmt.Sprintf("%s", n) //This is String(suoning).
 fmt.Println(d)
}

接口Interface

Interface类型可以定义一组方法,但是这些不需要实现。并且interface不能包含任何变量。

interface类型默认是一个指针。

Interface定义

type Car interface {
 NameGet() string
 Run(n int)
 Stop()
}

Interface实现

Golang中的接口,不需要显示的实现。只要一个变量,含有接口类型中的所有方法,那么这个变量就实现这个接口。因此,golang中没有implement类似的关键字;

如果一个变量含有了多个interface类型的方法,那么这个变量就实现了多个接口;如果一个变量只含有了1个interface的方部分方法,那么这个变量没有实现这个接口。

空接口 Interface{}:空接口没有任何方法,所以所有类型都实现了空接口。

var a int
var b interface{} //空接口
b = a

多态

一种事物的多种形态,都可以按照统一的接口进行操作。

栗子:

type Car interface {
 NameGet() string
 Run(n int)
 Stop()
}

type BMW struct {
 Name string
}
func (this *BMW) NameGet() string {
 return this.Name
}
func (this *BMW) Run(n int) {
 fmt.Printf("BMW is running of num is %d \n", n)
}
func (this *BMW) Stop() {
 fmt.Printf("BMW is stop \n")
}

type Benz struct {
 Name string
}
func (this *Benz) NameGet() string {
 return this.Name
}
func (this *Benz) Run(n int) {
 fmt.Printf("Benz is running of num is %d \n", n)
}
func (this *Benz) Stop() {
 fmt.Printf("Benz is stop \n")
}
func (this *Benz) ChatUp() {
 fmt.Printf("ChatUp \n")
}

func main() {
 var car Car
 fmt.Println(car) // <nil>

 var bmw BMW = BMW{Name: "宝马"}
 car = &bmw
 fmt.Println(car.NameGet()) //宝马
 car.Run(1)     //BMW is running of num is 1
 car.Stop()     //BMW is stop

 benz := &Benz{Name: "大奔"}
 car = benz
 fmt.Println(car.NameGet()) //大奔
 car.Run(2)     //Benz is running of num is 2
 car.Stop()     //Benz is stop
 //car.ChatUp() //ERROR: car.ChatUp undefined (type Car has no field or method ChatUp)
}

Interface嵌套

一个接口可以嵌套在另外的接口。

即需要实现2个接口的方法。

type Car interface {
 NameGet() string
 Run(n int)
 Stop()
}

type Used interface {
 Car
 Cheap()
}

类型断言

类型断言,由于接口是一般类型,不知道具体类型,

如果要转成具体类型,可以采用以下方法进行转换:

var t int
var x interface{}
x = t

y = x.(int)  //转成int
y, ok = x.(int) //转成int,不报错

栗子一:

func test(i interface{}) {
 // n := i.(int)
 n, ok := i.(int)
 if !ok {
  fmt.Println("error")
  return
 }
 n += 10
 fmt.Println(n)
}

func main() {
 var t1 int
 test(t1)
}

栗子二:

switch & type

type Student struct {
 Name string
}

func judgmentType(items ...interface{}) {
 for k, v := range items {
  switch v.(type) {
  case string:
   fmt.Printf("string, %d[%v]\n", k, v)
  case bool:
   fmt.Printf("bool, %d[%v]\n", k, v)
  case int, int32, int64:
   fmt.Printf("int, %d[%v]\n", k, v)
  case float32, float64:
   fmt.Printf("float, %d[%v]\n", k, v)
  case Student:
   fmt.Printf("Student, %d[%v]\n", k, v)
  case *Student:
   fmt.Printf("Student, %d[%p]\n", k, v)
  }
 }
}

func main() {
 stu1 := &Student{Name: "nick"}
 judgmentType(1, 2.2, "learing", stu1)
}

栗子三:

判断一个变量是否实现了指定接口

type Stringer interface {
 String() string
}

type Mystruct interface {

}
type Mystruct2 struct {

}
func (this *Mystruct2) String() string {
 return ""
}

func main() {
 var v Mystruct
 var v2 Mystruct2
 v = &v2

 if sv, ok := v.(Stringer); ok {
  fmt.Printf("%v implements String(): %s\n", sv.String());
 }
}

反射 reflect

reflect包实现了运行时反射,允许程序操作任意类型的对象。

典型用法是用静态类型interface{}保存一个值,

  通过调用TypeOf获取其动态类型信息,该函数返回一个Type类型值。

  调用ValueOf函数返回一个Value类型值,该值代表运行时的数据。

func TypeOf(i interface{}) Type

TypeOf返回接口中保存的值的类型,TypeOf(nil)会返回nil。

func ValueOf(i interface{}) Value

ValueOf返回一个初始化为i接口保管的具体值的Value,ValueOf(nil)返回Value零值。

reflect.Value.Kind

获取变量的类别,返回一个常量

const (
 Invalid Kind = iota
 Bool
 Int
 Int8
 Int16
 Int32
 Int64
 Uint
 Uint8
 Uint16
 Uint32
 Uint64
 Uintptr
 Float32
 Float64
 Complex64
 Complex128
 Array
 Chan
 Func
 Interface
 Map
 Ptr
 Slice
 String
 Struct
 UnsafePointer
)

reflect.Value.Kind()方法返回的常量

reflect.Value.Interface()

转换成interface{}类型

【变量<-->Interface{}<-->Reflect.Value】

获取变量的值:

reflect.ValueOf(x).Int()
reflect.ValueOf(x).Float()
reflect.ValueOf(x).String()
reflect.ValueOf(x).Bool()

通过反射的来改变变量的值

reflect.Value.SetXX相关方法,比如:
reflect.Value.SetInt(),设置整数
reflect.Value.SetFloat(),设置浮点数
reflect.Value.SetString(),设置字符串

栗子一

import "reflect"

func main() {
 var x float64 = 5.21
 fmt.Println("type:", reflect.TypeOf(x)) //type: float64

 v := reflect.ValueOf(x)
 fmt.Println("value:", v)   //value: 5.21
 fmt.Println("type:", v.Type()) //type: float64
 fmt.Println("kind:", v.Kind()) //kind: float64
 fmt.Println("value:", v.Float()) //value: 5.21

 fmt.Println(v.Interface())     //5.21
 fmt.Printf("value is %1.1e\n", v.Interface()) //value is 5.2e+00
 y := v.Interface().(float64)
 fmt.Println(y) //5.21
}

栗子二(修改值)

SetXX(x) 因为传递的是 x 的值的副本,所以SetXX不能够改 x,改动 x 必须向函数传递 x 的指针,SetXX(&x) 。

//错误代码!!!
//panic: reflect: reflect.Value.SetFloat using unaddressable value
func main() {
 var a float64
 fv := reflect.ValueOf(&a)
 fv.SetFloat(520.00)
 fmt.Printf("%v\n", a)
}
//正确的,传指针
func main() {
 var a2 float64
 fv2 := reflect.ValueOf(&a2)
 fv2.Elem().SetFloat(520.00)
 fmt.Printf("%v\n", a2) //520
}

反射操作结构体

reflect.Value.NumField()获取结构体中字段的个数

reflect.Value.Method(n).Call(nil)来调用结构体中的方法

栗子一(通过反射操作结构体)

import "reflect"

type NotknownType struct {
 S1 string
 S2 string
 S3 string
}

func (n NotknownType) String() string {
 return n.S1 + " & " + n.S2 + " & " + n.S3
}

var secret interface{} = NotknownType{"Go", "C", "Python"}

func main() {
 value := reflect.ValueOf(secret)
 fmt.Println(value) //Go & C & Python
 typ := reflect.TypeOf(secret)
 fmt.Println(typ) //main.NotknownType

 knd := value.Kind()
 fmt.Println(knd) // struct

 for i := 0; i < value.NumField(); i++ {
  fmt.Printf("Field %d: %v\n", i, value.Field(i))
 }

 results := value.Method(0).Call(nil)
 fmt.Println(results) // [Go & C & Python]
}

栗子二(通过反射修改结构体)

import "reflect"

type T struct {
 A int
 B string
}

func main() {
 t := T{18, "nick"}
 s := reflect.ValueOf(&t).Elem()
 typeOfT := s.Type()

 for i := 0; i < s.NumField(); i++ {
  f := s.Field(i)
  fmt.Printf("%d: %s %s = %v\n", i,
   typeOfT.Field(i).Name, f.Type(), f.Interface())
 }

 s.Field(0).SetInt(25)
 s.Field(1).SetString("nicky")
 fmt.Println(t)
}

/*
输出:
0: A int = 18
1: B string = nick
{25 nicky}
*/
import "reflect"

type test struct {
 S1 string
 s2 string
 s3 string
}

var s interface{} = &test{
 S1: "s1",
 s2: "s2",
 s3: "s3",
}

func main() {
 val := reflect.ValueOf(s)
 fmt.Println(val)      //fmt.Println(val.Elem())    //{s1 s2 s3}
 fmt.Println(val.Elem().Field(0))  //s1
 val.Elem().Field(0).SetString("hehe") //S1大写
}

栗子三(struct tag 内部实现)

package main

import (
 "fmt"
 "reflect"
)

type User struct {
 Name string `json:"user_name"`
}

func main() {
 var user User
 userType := reflect.TypeOf(user)
 jsonString := userType.Field(0).Tag.Get("json")
 fmt.Println(jsonString)  //user_name
}

以上这篇浅谈Go语言中的结构体struct & 接口Interface & 反射就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • Go语言中反射的正确使用

    介绍 反射是元数据编程的一种形式,指的是程序获得本身结构的一种能力.不同语言的反射模型实现不一样,本文中的反射,仅仅指的是Go语言中的反射模型. 反射有两个问题,在使用前需要三思: 大量的使用反射会损失一定性能 Clear is better than clever. Reflection is never clear. Go的类型设计上有一些基本原则,理解这些基本原则会有助于你理解反射的本质: 变量包括 <type, value> 两部分.理解这一点你就知道为什么nil != nil了. t

  • Go语言interface详解

    interface Go语言里面设计最精妙的应该算interface,它让面向对象,内容组织实现非常的方便,当你看完这一章,你就会被interface的巧妙设计所折服. 什么是interface 简单的说,interface是一组method的组合,我们通过interface来定义对象的一组行为. 我们前面一章最后一个例子中Student和Employee都能SayHi,虽然他们的内部实现不一样,但是那不重要,重要的是他们都能say hi 让我们来继续做更多的扩展,Student和Employe

  • Go语言结构体定义和使用方法

    本文实例讲述了Go语言结构体定义和使用方法.分享给大家供大家参考.具体分析如下: 一个结构体(struct)就是一个字段的集合. (而 type 定义跟其字面意思相符.) 复制代码 代码如下: package main import "fmt" type Vertex struct {     X int     Y int } func main() {     fmt.Println(Vertex{1, 2}) } 结构体字段使用点号来访问. 复制代码 代码如下: package

  • 谈谈Go语言的反射三定律

    简介 Reflection(反射)在计算机中表示 程序能够检查自身结构的能力,尤其是类型.它是元编程的一种形式,也是最容易让人迷惑的一部分. 虽然Go语言没有继承的概念,但为了便于理解,如果一个struct A 实现了 interface B的所有方法时,我们称之为"继承". 类型和接口 反射建立在类型系统之上,因此我们从类型基础知识说起. Go是静态类型语言.每个变量都有且只有一个静态类型,在编译时就已经确定.比如 int.float32.*MyType.[]byte. 如果我们做出

  • go语言中的interface使用实例

    go语言中的interface是一组未实现的方法的集合,如果某个对象实现了接口中的所有方法,那么此对象就实现了此接口.与其它面向对象语言不同的是,go中无需显示声明调用了哪个接口. 复制代码 代码如下: package main   import (  "fmt" )   type I interface {  Get() int  Put(int) }   type S struct{ i int }   func (p *S) Get() int  { return p.i } f

  • Go语言struct类型详解

    struct Go语言中,也和C或者其他语言一样,我们可以声明新的类型,作为其它类型的属性或字段的容器.例如,我们可以创建一个自定义类型person代表一个人的实体.这个实体拥有属性:姓名和年龄.这样的类型我们称之struct.如下代码所示: 复制代码 代码如下: type person struct {     name string     age int } 看到了吗?声明一个struct如此简单,上面的类型包含有两个字段. 1.一个string类型的字段name,用来保存用户名称这个属性

  • 浅谈Go语言中的结构体struct & 接口Interface & 反射

    结构体struct struct 用来自定义复杂数据结构,可以包含多个字段(属性),可以嵌套: go中的struct类型理解为类,可以定义方法,和函数定义有些许区别: struct类型是值类型. struct定义 type User struct { Name string Age int32 mess string } var user User var user1 *User = &User{} var user2 *User = new(User) struct使用 下面示例中user1和

  • 浅谈c语言中一种典型的排列组合算法

    c语言中的全排列算法和组合数算法在实际问题中应用非常之广,但算法有许许多多,而我个人认为方法不必记太多,最好只记熟一种即可,一招鲜亦可吃遍天 全排列: #include<stdio.h> void swap(int *p1,int *p2) { int t=*p1; *p1=*p2; *p2=t; } void permutation(int a[],int index,int size) { if(index==size) { for(int i=0;i<size;i++) print

  • 浅谈c语言中类型隐性转换的坑

    谨记:在C语言中,当两种不同类型之间运算时,低字节长度类型会向高自己长度类型转换,有符号会向无符号类型转换. 举例子如下: #include <stdio.h> void func(void) { int i = 1; unsigned char c1 = 1; signed char c2 = -1; if (c2 > i){ printf("\r\n -1 > 1"); } else{ printf("\r\n -1 <= 1");

  • 浅谈c语言中转义字符的用法及注意事项

    c语言中的转义字符: \a 响铃符 \b 退格 \f 换页符 \n 换行符 \r 回车符(回到该行的首位置) \v 纵向制表符 \\ 反斜杠 \? 问号(?经vs10测试可以直接打印) \"(\') 双引号(单引号) \ooo 八进制数(ooo表示一个用8进制数表示出来的对应ANSII代码对应出字符,用此方法可以表示出所有ASCII字符.不过测试发现打不出%号,存疑!) \xhh 十六进制数(功能同八进制数,用hh表示一个十六进制数,如\x20表示空格) 注:使用转义字符的退格符,换行符,回车符

  • 浅谈C语言中的注释风格小结

    C语言中常用的注释风格有两种,一种是通过如下模式进行一段代码的注释: /* comment*/ 另一种是单行注释符号: // comment 学生时代的注释我一般是选用后者,那时候编码量十分有限,即使是简单的小段落注释使用的IDE也支持批量添加单行注释符.而在编码之中,简单的单行注释进行注释的时候键盘的操作更为简单一点. 不过,工作之后接触了相应的编码规范之后,C语言的注释我基本上放弃了单行注释的方法,最多仅仅在调试的时候做简单的使用. 其实,单行注释是从C++中借鉴来的,算是C++风格的注释方

  • C语言中的结构体在Python中实现转换

    目录 struct介绍 struct中的常用接口 pack() unpack() fmt 示例 struct介绍 Python中提供了struct接口,用来处理类似C语言中的结构体. 处理的方式是将结构体表现位字符串,这个字符串其实就是结构体的一个个字节. struct中的常用接口 主要就是两个,pack()和unpack(). pack()就是将结构体转换成字符串(或者说字节序),unpack()则相反. pack() pack()函数的说明如下(来自Python 2.7.15 documen

  • Go语言学习函数+结构体+方法+接口

    目录 1. 函数 1.1 函数返回值 同一种类型返回值 带变量名的返回值 函数中的参数传递 函数变量 1.2 匿名函数——没有函数名字的函数 在定义时调用匿名函数 将匿名函数赋值给变量 匿名函数用作回调函数 可变参数——参数数量不固定的函数形式 1.3 闭包 1.4 defer语句 处理运行时发生的错误 1.5 宕机恢复(recover)——防止程序崩溃 2. 结构体 2.1 定义与给结构体赋值 3. 方法 3.1 结构体方法 3.2 接收器 指针接收器 非指针类型接收器 4. 接口 4.1 声

  • C语言中隐藏结构体的细节

    我们都知道,在C语言中,结构体中的字段都是可以访问的.或者说,在C++ 中,类和结构体的主要区别就是类中成员变量默认为private,而结构体中默认为public.结构体的这一个特性,导致结构体中封装的数据,实际上并没有封装,外界都可以访问结构体重的字段. C++中我们尚可用类来替代结构体,但是,C语言中是没有类的,只能用结构体,但很多时候,我们需要隐藏结构体的字段,不让外界直接访问,而是通过我们写的函数进行间接访问,这样就提高了程序的封装性. 实现方法,简单来说,就是,结构体定义时,要定义在.

  • 浅谈C语言中结构体的初始化

    <代码大全>建议在变量定义的时候进行初始化,但是很多人,特别是新人对结构体或者结构体数组定义是一般不会初始化,或者不知道怎么初始化.1.初始化 复制代码 代码如下: typedef struct _TEST_T {        int i;        char c[10];}TEST_T;TEST_T gst  = {1, "12345"};//可以初始化,设置i为1,s为一个字符串.TEST_T gst  = {1};//初始化个数少于实际个数时,只初始化前面的成员

  • C语言中的结构体的入门学习教程

    C语言中数组允许定义类型的变量,可容纳相同类型的多个数据项,但结构体在C语言编程中,它允许定义不同种类的数据项可供其他用户定义的数据类型. 结构是用来代表一个记录,假设要跟踪图书馆的书籍.可能要跟踪有关每本书以下属性: Title - 标题 Author - 作者 Subject - 科目 Book ID - 编号 定义结构体 定义一个结构体,必须使用结构体的struct语句.该struct语句定义了一个新的数据类型,程序不止一个成员.struct语句的格式是这样的: struct [struc

随机推荐