自定义类型

两种用户定义的类型

  • 结构类型
  • 基于一个已有的类型,将其作为新类型的类型说明

结构体变量定义和初始化

type struct_variable_type struct {
member definition
member definition
...
member definition
}

一旦定义了结构体类型,它就能用于变量的声明,语法格式如下:

variable_name := structure_variable_type {value1, value2...valuen}
// 或
variable_name := structure_variable_type { key1: value1, key2: value2..., keyn: valuen}

实例

package main

import "fmt"

type Person struct {
name string
sex byte
age int
} func main() { // 顺序初始化,必须全部初始化完整
var person Person = Person{"wang", 'f', }
fmt.Println(person) // 部分初始化
person1 := Person{name: "zhao", sex: 'm'}
fmt.Println(person1.name)
}

赋值、比较和传参

使用.索引成员变量

    // 部分初始化
person1 := Person{name: "zhao", sex: 'm'}
fmt.Println(person1.name) person1.name = "li"
fmt.Println(person1.name)

比较

只能使用 == 和 !=,不能使用 > < >= <=

    // 结构体比较
person2 := Person{name: "zhao", sex: 'm', age: }
person3 := Person{name: "zhao", sex: 'm', age: }
person4 := Person{name: "zhao", sex: 'm', age: }
fmt.Println(person2 == person3) // true
fmt.Println(person3 == person4) // false

相同类型的结构体可以进行赋值

相同类型结构体:成员变量的类型、个数、顺序一致

var tmp Person
tmp = person1
fmt.Println(tmp.name)

因此函数体内部可以使用结构体传参,因为实参可以赋值给相同结构体类型的形参,值传递。几乎不用,内存消耗大,效率低。

func test(person Person)  {
println(person.name)
} func main() { var temp Person
test(temp) }

结构体指针

结构体指针变量定义和初始化

1.顺序初始化和部分初始化

    var man *Person  = &Person{"zhao", 'm', }
fmt.Println(man.name)
var man1 *Person = &Person{name: "zhao", sex: 'm'}
fmt.Println(man1.name)

2.使用new

var  man2 *Person = new(Person);
man2.name = "zhao"
man2.sex =
man2.age =

使用.指针索引成员变量

结构体变量的地址就是结构体首个元素的地址

    fmt.Printf("&tmp = %p \n", &tmp)
fmt.Printf("&tmp.name = %p \n", &tmp.name)

结果

&tmp = 0xc000004500
&tmp.name = 0xc000004500

结构体指针的值就是结构体首个元素的地址

fmt.Printf("man2 = %p\n", man2)
fmt.Printf("&man2.name = %p\n", &man2.name)

结果

man2 = 0xc00008e080
&man2.name = 0xc00008e080

结构体指针作为函数参数

import "fmt"

type Person struct {
name string
sex byte
age int
} func test1(person * Person) {
person.name = "wang"
} func main() { fmt.Println("man2:", man2)
test1(man2)
fmt.Println("man2", man2) }

结果

man2: &{zhao  }
man2 &{wang }

发现通过指针变量可以修改结构体,传引用,使用频率高

也是实参拷贝值给形参,不过这次拷贝的是地址值

fmt.Println(person1.name)
test1(&person1)
fmt.Println(person1.name)

结果

li
wang

函数是一等公民

与其他主要编程语言的差异

1.可以有多个返回值

2.所有参数都是值传递

slice、map、channel会有传引用是错觉,如切片背后是数组,是一个数据结构,里面包含了指向对应数组的指针,数据结构被复制,指针操作的仍是同一块空间,感觉像是传引用

3.函数可以作为变量的值

4.函数可以作为参数和返回值

package fun_test

import (
"fmt"
"math/rand"
"testing"
"time"
) // 多个返回值
func returnMultiValues()(int, int) {
return rand.Intn(), rand.Intn()
} // 函数可以作为参数和返回值
// 计算inner函数运行的时间
func timeSpent(inner func(op int)int) func(opt int) int {
return func(n int) int {
start := time.Now()
ret := inner(n)
fmt.Println("time spent:", time.Since(start).Seconds())
return ret
}
} func slowFun(op int)int{
time.Sleep(time.Second*)
return op
} func TestFn(t *testing.T){
a, _ := returnMultiValues()
t.Log(a)
tsSF := timeSpent(slowFun)
t.Log(tsSF())
}

结果:

=== RUN TestFn
time spent: 1.0000582
--- PASS: TestFn (1.00s)
func_test.go:32: 1
func_test.go:34: 10
PASS

可变参数及defer

可变参数会被转换为数组,通过数组变量来完成

func sum(ops ...int) int  {
s :=
for _, op := range ops{
s += op
}
return s
} func TestParam(t *testing.T) {
s := sum(, , , , )
fmt.Println(s)
}

=== RUN TestParam
15
--- PASS: TestParam (0.00s)
PASS

defer

func clear()  {
fmt.Println("clear resources")
} func TestDefer(t *testing.T) {
defer clear()
fmt.Println("Start")
panic("Fatal error") //defer仍会执行
println("Stop")
}

=== RUN TestDefer
Start
clear resources
--- FAIL: TestDefer (0.00s)
panic: Fatal error [recovered]
panic: Fatal error

会发现出现了异常,clear()依然会执行,

异常之前的fmt.Println("Start")被执行

异常之后的fmt.Println("Stop")没有被执行

Go(二)函数的更多相关文章

  1. Python进阶(二)----函数参数,作用域

    Python进阶(二)----函数参数,作用域 一丶形参角度:*args,动态位置传参,**kwargs,动态关键字传参 *args: ​ 动态位置参数. 在函数定义时, * 将实参角度的位置参数聚合 ...

  2. 深入理解PHP内核(十二)函数-函数的定义、传参及返回值

    原文链接:http://www.orlion.ga/344/ 一.函数的定义 用户函数的定义从function 关键字开始,如下 function foo($var) {    echo $var; ...

  3. Python学习笔记总结(二)函数和模块

    一.函数 函数的作用:可以计算出一个返回值,最大化代码重用,最小化代码冗余,流程的分解. 1.函数相关的语句和表达式 语句        例子 Calls        myfunc(‘diege', ...

  4. C++实验二——函数重载、函数模板、简单类的定义和实现

    一.实验过程 函数重载编程练习 实验要求:编写重载函数add(),实现对int型,double型,complex型数据的加法.在main函数中定义不同类型的数据,调用测试. 代码实现: 先是简单的体验 ...

  5. Mysql(二)函数与连接

    一.函数 1.if函数 if(exp1, exp2, exp3)     判断exp1是否为true(不为0,并且不为nlll),如果为true,返回 exp2的值,否则返回exp3的值. selec ...

  6. python 二——函数、装饰器、生成器、面向对象编程(初级)

    本节内容 1.函数 2.装饰器 3.生成器 4.类 一.函数 函数式:将某功能代码封装到函数中,日后便无需重复编写,仅调用函数即可 面向对象:对函数进行分类和封装,让开发“更快更好更强...” 函数式 ...

  7. iOS学习笔记42-Swift(二)函数和闭包

    上一节我们讲了Swift的基础部分,例如数据类型.运算符和控制流等,现在我们来看下Swift的函数和闭包 一.Swift函数 函数是一个完成独立任务的代码块,Swift中的函数不仅可以像C语言中的函数 ...

  8. 【C++】模板简述(二):函数模板

    我们上文讲了,模板的引入,我们发现在某种特殊的情况下,必须得通过模板才能完美的解决问题. 本文就来简述一下函数模板的基本使用. 一.函数模板格式 template<typename Param1 ...

  9. 《前端之路》- TypeScript(二) 函数篇

    目录 一.定义函数方法 二.定义函数传参 三.可选传参 四.默认传参 五.传递剩余参数 六.函数重载 七.箭头函数 八.总结 一.定义函数方法 在 es5 中定时函数的方法有 命名函数和函数表达式(匿 ...

  10. php基础之二 函数

    一.语句:分支语句,循环语句 1.分支语句: 1.1 if $a = 7;if($a == 5){ echo "相等";}else{ echo "不相等";} ...

随机推荐

  1. Spring---条件注解@Conditional

    1.概述 1.1.Spring4  提供了一个更通用的  基于条件的Bean的创建,即使用@Conditional注解: 1.2.案例 package com.an.config; import co ...

  2. Web核心之Request对象

    HTTP协议中Request请求部分格式 //请求行(这种是POST类型的请求) POST /HttpServleLogin.html HTTP/1.1 //请求头(User-Agent里有Firef ...

  3. 【leetcode】1022. Smallest Integer Divisible by K

    题目如下: Given a positive integer K, you need find the smallest positive integer N such that N is divis ...

  4. 命令查看当前运行APP的包名和Activity

    先用usb链接手机 启动要查看的程序,命令查看当前运行的包名和Activity更清晰. 使用adb shell dumpsys window | findstr mCurrentFocus  命令查看 ...

  5. Web项目改名的带来的404not found问题

    为了保留上一次编辑的billsys web项目,把项目复制一份到同一个工作空间后,对原来项目名进行了重命名,如右图: 结果再去访问,一直报404错误 解决思路如下: 其实仔细观察,会在项目部署界面发现 ...

  6. 如何隐藏一个让人很难发现的bug?

    程序员的日常三件事:写bug.改bug.背锅.连程序员都自我调侃道,为什么每天都在加班?因为我的眼里常含bug. 那么如何写出一个让(坑)人(王)很(之)难(王)发现的bug呢? - 1 - 新手开发 ...

  7. nofollow标签的作用 nofollow标签添加方法

    这篇文章主要介绍了nofollow标签的作用 nofollow标签添加方法,需要的朋友可以参考下   nofollow标签的作用 nofollow标签添加方法  模拟搜狗蜘蛛   nofollow标签 ...

  8. k-近邻算法(kNN)完整代码

    from numpy import *#科学计算包 from numpy import tile from numpy import zeros import operator #运算符模块 impo ...

  9. LOJ 2721 「NOI2018」屠龙勇士——扩展中国剩余定理

    题目:https://loj.ac/problem/2721 1.注意别一输入 p[ i ] 就 a[ i ] %= p[ i ] ,因为在 multiset 里找的时候还需要真实值. 2.注意用 m ...

  10. window使用

    运行命令 regedit #注册表编译器 firewall.cpl #打开防火墙配置 powershell #打开powershell control #打开控制面版 cnpa.cpl #打开网络设置 ...