@


1. 结构体别名定义

变量别名定义

package main

import "fmt"

type integer int

func main() {
//类型别名定义
var i integer = 1000
fmt.Printf("值: %d, 类型: %T\n", i, i) var j int = 100
j = int(i) //j和i不属于同一类型,需要转换
fmt.Println(j)
} //输出结果如下
值: 1000, 类型: main.integer
1000

结构体别名定义

package main

import "fmt"

//创建结构体Student
type Student struct {
Number int
} //结构体定义别名
type Stu Student func main() {
//声明Student类型结构体
var a Student
a = Student{30} //声明Stu类型结构体
var b Stu
b = Stu{20} //强转类型后才能进行赋值
a = Student(b)
fmt.Printf("a = %d,类型: %T\n", a, a)
b = Stu(a)
fmt.Printf("b = %d,类型: %T\n", b, b)
} //输出结果如下
a = {20},类型: main.Student
b = {20},类型: main.Stu

2. 工厂模式

  • Go 中所谓的工厂模式其实就是:

    包内一个不可直接实例的结构体(结构体名称首字母小写),包外不可直接实例,那么为了解决这个问题,就写一个包外可调用的函数,通过这个函数实现返回结构体对象。
package main

import "fmt"

type Student struct {
Name string
Age int
} func main() {
//初始化
stu1 := new(Student)
fmt.Println(stu1) //工厂模式处理
stu2 := NewStudent("张三", 18)
fmt.Println(stu2)
} //工厂模式
func NewStudent(name string, age int) *Student {
return &Student{
Name: name,
Age: age,
}
} //输出结果如下
&{ 0}
&{张三 18}

总结:

make 用来创建mapslicechannel

new 用来创建值类型


3. Tag 原信息

  • 在和其他语言进行对接交互时使用JSON格式,有些语言格式大小写规范比较严格,为了使Go语言和其他语言对接数据传输,所以使用Tag原信息进行解决

  • 通俗的来说就相当于是一个充电的转接口

示例

package main

import (
"encoding/json"
"fmt"
) type Student struct {
Name string
Age int
Score float32
} func main() {
//初始化
var stu = new(Student)
stu.Name = "stu"
stu.Age = 20
stu.Score = 88 //使用Json处理结构体,转换成字节数组
data, err := json.Marshal(stu)
if err != nil {
fmt.Println("错误提示:", err)
return
}
fmt.Println(data) //字节数组形式输出
fmt.Println(string(data)) //转换成字符串输出
} //输出结果如下
[123 34 78 97 109 101 34 58 34 115 116 117 34 44 34 65 103 101 34 58 50 48 44 34 83 99 111 114 101 34 58 56 56 125]
{"Name":"stu","Age":20,"Score":88}

JSON格式化字段名

package main

import (
"encoding/json"
"fmt"
) type Student struct {
//json打包时字段名
Name string `json:"stu_name"`
Age int `json:"stu_age"`
Score float32 `json:"stu_score"`
} func main() {
//初始化
var stu = new(Student)
stu.Name = "stu"
stu.Age = 20
stu.Score = 88 //使用Json处理结构体,转换成字节数组
data, err := json.Marshal(stu)
if err != nil {
fmt.Println("错误提示:", err)
return
}
fmt.Println(data)
fmt.Println(string(data))
} //输出结果如下
[123 34 115 116 117 95 110 97 109 101 34 58 34 115 116 117 34 44 34 115 116 117 95 97 103 101 34 58 50 48 44 34 115 116 117 95 115 99 111 114 101 34 58 56 56 125]
{"stu_name":"stu","stu_age":20,"stu_score":88}

4. 匿名字段

结构体中的字段(属性)没有名称,称之为匿名字段

示例

package main

import "fmt"

type Cart struct {
name string
color string
} type Train struct {
//匿名字段
Cart //实现继承
int //数据类型定义,仅能存在一次,两个int则会冲突
} func main() {
//初始化赋值
var t Train
t.name = "train"
t.color = "red"
t.int = 10 //直接调用数据类型赋值
fmt.Println(t)
} //输出结果如下
{{train red} 10}

双引用结构体,多继承(继承的两个结构体中定义相同属性)

package main

import "fmt"

//父结构体
type Cart struct {
name string
color string
} //父结构体
type Box struct {
color string
} //子结构体
type Train struct {
//匿名字段
Cart //实现继承
Box
int //数据类型定义,仅能存在一次,两个int则会冲突
} func main() {
//初始化赋值
var t Train
t.name = "train"
t.Cart.color = "red"
t.Box.color = "blue"
t.int = 10 //直接调用数据类型赋值
fmt.Println(t)
} //输出结果如下
{{train red} {blue} 10}
package main

import "fmt"

//父结构体
type Cart struct {
name string
color string
} //父结构体
type Box struct {
color string
} //子结构体
type Train struct {
//匿名字段
Cart //实现继承
Box
int //数据类型定义,仅能存在一次,两个int则会冲突
color string
} func main() {
//初始化赋值
var t Train
t.name = "train"
t.Cart.color = "red" //Cart的属性
t.Box.color = "blue" //Box的属性
t.color = "yellow" //train自身属性
t.int = 10 //直接调用数据类型赋值
fmt.Println(t)
}

5. 方法

  • Go 中的方法是作用在特定类型的变量上,因此自定义类型,都可以有方法,而不仅仅是 struct
  • 语法格式如下
func (recevier type) methodName(参数列表)(返回值){}

recevier type     特定类型,如指针、别名,结构体
methodName 方法名
  • 示例
package main

import "fmt"

//定义结构体
type Student struct {
Name string
Age int
} //定义方法
func (s Student) init(name string, age int) Student {
s.Name = name
s.Age = age
return s
} func main() {
var stu Student
s := stu.init("zhangsan", 18)
fmt.Printf("s: %v\n", s)
} //输出结果
s: {zhangsan 18}

定义返回方法是否会把初始化的值给返回?

package main

import "fmt"

//定义结构体
type Student struct {
Name string
Age int
Score float32
} //初始化方法
func (s *Student) init(name string, age int, score float32) {
s.Name = name
s.Age = age
s.Score = score
fmt.Println("初始化完成")
} //返回结构体
func (s *Student) get() Student {
return *s
} func main() {
var stu Student
//定义值
stu.init("zhangsan", 18, 90)
//返回值
stu1 := stu.get()
fmt.Println(stu1)
} //输出结果如下
初始化完成
{zhangsan 18 90}

传统数据类型自定义方法,做数据类型转换

package main

import "fmt"

//别名类型
type integer int //传统数据类型自定义方法
func (p integer) convert() string {
return fmt.Sprintf("%d", p)
} func main() {
var i integer
i = 100
s := i.convert()
fmt.Printf("类型:%T,值:%s\n", s, s)
} //输出结果如下
类型:string,值:100

指针传入和值传入的区别

值传入不会对数值进行改变,指针传入才可以改变数值

package main

import "fmt"

type integer int

//传统数据类型自定义方法
func (p integer) convert() string {
return fmt.Sprintf("%d", p)
} //方法传指针进行数据同步修改
func (p *integer) set(b integer) {
*p = b
} func main() {
var i integer
i = 100
s := i.convert()
fmt.Printf("类型: %T ,值: %s\n", s, s)
fmt.Printf("类型: %T ,值: %d\n", i, i)
i.set(200)
fmt.Printf("i: %v\n", i)
} //输出结果如下
类型: string ,值: 100
类型: main.integer ,值: 100
i: 200

方法继承,组合(匿名字段是组合的特殊形式)

package main

import "fmt"

//父结构体
type Car struct {
weight int
name string
} //父方法
func (c *Car) Run() {
fmt.Println("Running")
} //子结构体Bike
type Bike struct {
//组合(有名字)
c Car
wheel int
} //子结构体Train
type Train struct {
//匿名
Car
wheel int
} func main() {
var bike Bike
bike.c.name = "bike"
bike.c.weight = 500
bike.wheel = 2 var train Train
train.name = "train"
train.weight = 140000
train.wheel = 8 fmt.Println(bike)
//方法继承,调用父结构体方法
bike.c.Run() fmt.Println(train)
//方法继承
train.Run()
} //输出结果如下
{{500 bike} 2}
Running
{{140000 train} 8}
Running
package main

import "fmt"

//父结构体
type Cart struct {
weight int
Color string
} //父方法
func (c Cart) Run() {
fmt.Println("Running")
} //子结构体train
type Train struct {
Cart
wheel int
} //子结构体方法
func (t Train) String() string {
str := fmt.Sprintf("color:[%s],weight:[%d],wheel:[%d]\n", t.Color, t.weight, t.wheel)
return str
} func main() {
var train Train
train.Color = "red"
train.weight = 14000
train.wheel = 8
fmt.Println(train)
train.Run()
fmt.Printf("%s\n", train)
} //输出结果如下
color:[red],weight:[14000],wheel:[8] Running
color:[red],weight:[14000],wheel:[8]

Go 语言 结构体和方法的更多相关文章

  1. C语言结构体初始化方法

    早上苏凯童鞋问我这个问题来着,写在这里. 我了解到的C中结构体初始化的方法大概有三种. 如这里我定义了一个结构体: typedef struct node { int x, y; }Node; 第一种 ...

  2. c语言结构体使用方法

      结构(struct)      结构是由基本数据类型构成的.并用一个标识符来命名的各种变量的组合.  结构中能够使用不同的数据类型.      1. 结构说明和结构变量定义      在Turbo ...

  3. C语言结构体定义的几种方法

    什么是结构体? 在C语言中,结构体(struct)指的是一种数据结构,是C语言中聚合数据类型(aggregate data type)的一类.结构体可以被声明为变量.指针或数组等,用以实现较复杂的数据 ...

  4. Go语言 - 结构体 | 方法

    自定义类型和类型别名 自定义类型 在Go语言中有一些基本的数据类型,如string.整型.浮点型.布尔等数据类型, Go语言中可以使用type关键字来定义自定义类型. 自定义类型是定义了一个全新的类型 ...

  5. Go语言结构体(struct)

    Go 语言结构体 Go 语言中数组可以存储同一类型的数据,但在结构体中我们可以为不同项定义不同的数据类型. 结构体是由一系列具有相同类型或不同类型的数据构成的数据集合. 结构体表示一项记录,比如保存图 ...

  6. 对嵌入式开发C语言结构体的一点总结

    今天冬至居然不上班,公司的良心啊!这回有心情写博客和日志了,好了,废话不多说.直接看下文: 鉴于嵌入式开发过程中,C语言结构体的使用当然是必不可少.话说,基础什么的比你会更牛逼的算法更重要,基础不牢, ...

  7. 在C语言结构体中添加成员函数

    我们在使用C语言的结构体时,经常都是只定义几个成员变量,而学过面向对象的人应该知道,我们定义类时,不只是定义了成员变量,还定义了成员方法,而类的结构和结构体非常的相似,所以,为什么不想想如何在C语言结 ...

  8. 将c语言的结构体定义变成对应的golang语言的结构体定义,并将golang语言结构体变量的指针传递给c语言,cast C struct to Go struct

    https://groups.google.com/forum/#!topic/golang-nuts/JkvR4dQy9t4 https://golang.org/misc/cgo/gmp/gmp. ...

  9. Linux C语言结构体-学习笔记

    Linux C语言结构体简介 前面学习了c语言的基本语法特性,本节进行更深入的学习. 预处理程序. 编译指令: 预处理, 宏定义, 建立自己的数据类型:结构体,联合体,动态数据结构 c语言表达式工具 ...

随机推荐

  1. 玩转SpringBoot之定时任务@Scheduled线程池配置

    序言 对于定时任务,在SpringBoot中只需要使用@Scheduled 这个注解就能够满足需求,它的出现也给我们带了很大的方便,我们只要加上该注解,并且根据需求设置好就可以使用定时任务了. 但是, ...

  2. 搭建nuget服务器(二):制作nuget包

    生成nuget包可以使用nuget.exe或者下载nuget package explorer工具 nuget package explorer 下载地址:https://github.com/NuG ...

  3. ShellExecuteA加载exe文件指定工作目录找不到文件的问题

    使用ShellExecuteA调用exe文件时,指定工作目录需要注意 函数原型为: HINSTANCE ShellExecuteA( HWND hwnd, LPCTSTR lpOperation, L ...

  4. struts2学习一:hello struts2及struts2环境配置中遇到的问题

    17年下半年的时候简单学了下strus2,好吧,现在已经全忘了,idea也是刚开始用,本来想按教程写个hello struts2,结果,出了以下系列问题. pre:step1-5是我按照百度的教程搭的 ...

  5. Kafka 的设计架构你知道吗?

    Producer :消息生产者,就是向 kafka broker 发消息的客户端. Consumer :消息消费者,向 kafka broker 取消息的客户端. Topic :可以理解为一个队列,一 ...

  6. redis 为什么是单线程的?

    一.Redis为什么是单线程的? 因为Redis是基于内存的操作,CPU不是Redis的瓶颈,Redis的瓶颈最有可能是机器内存的大小或者网络带宽.既然单线程容易实现,而且CPU不会成为瓶颈,那就顺理 ...

  7. Spring Cloud 解决了哪些问题?

    在使用 Spring Boot 开发分布式微服务时,我们面临的问题很少由 Spring Cloud解决.与分布式系统相关的复杂性 – 包括网络问题,延迟开销,带宽问题,安 全问题.处理服务发现的能力 ...

  8. Redis 集群的主从复制模型是怎样的?

    为了使在部分节点失败或者大部分节点无法通信的情况下集群仍然可用,所 以集群使用了主从复制模型,每个节点都会有 N-1 个复制品.

  9. SQLAlchemy 使用教程

    前戏: ​ 不用怀疑,你肯定用过Django中的orm,这个orm框架是django框架中自己封装的,在Django中配置和使用较为简单,但是并不适用于其他web框架,而今天说的sqlalchemy是 ...

  10. springboot项目中的日志输出

    #修改默认输出级别,trace < debug < info < warn < errorlogging.level.com.lagou=trace#控制台输出logging. ...