Go函数、包和错误处理

Go函数、包和错误处理

函数基本语法
package main
import(
	"fmt"
)

/*
	func 函数名 (形参列表) (返回值列表) {
		执行语句
		return 返回值列表
	}
*/

func cal (n1 float64, n2 float64, operator byte) float64 {
	var res float64
	switch operator {
	case '+':
		res = n1 + n2
	case '-':
		res = n1 - n2
	case '*':
		res = n1 * n2
	case '/':
		res = n1 / n2
	default:
		fmt.Println("输入有误")
	}
	return res
}

func main(){
	var operator byte
	var n1 float64
	var n2 float64
	fmt.Scanln(&n1)
	fmt.Scanln(&operator)
	fmt.Scanln(&n2)
	result := cal(n1,n2,operator)
	fmt.Println(result)
}

//Go函数不支持函数重载
//支持函数返回值命名
package main
import(
	"fmt"
)

func getSumAndSub (n1 int, n2 int) (sum int, sub int) {
	sub = n1 - n2
	sum = n1 + n2
	return
}

func main(){
	a1,b1 := getSumAndSub(10,20)
	fmt.Println(a1,b1)
}
  • 包的本质是创建不同的文件夹存放程序文件

  • go的每一个文件都属于一个包,即go使以包的形式管理文件和项目目录结构

  • 包的三大作用

    • 区分相同名字的函数、变量等标识符
    • 当程序文件很多时,可以更好的管理项目
    • 控制函数、变量等访问范围,即作用域
  • 打包基本语法

    • package 包名
      
  • 引入包的基本语法

    • import "包的路径"
      
  • [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-ySc6FJqu-1622206087527)(C:\Users\51936\AppData\Roaming\Typora\typora-user-images\image-20210325213313872.png)]

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-v4YHzIWm-1622206087528)(C:\Users\51936\AppData\Roaming\Typora\typora-user-images\image-20210325213344720.png)]

  • 包使用细节

    • 在给一个文件打包时,该包对应一个文件夹,比如此处的utils文件夹对应的包名就是utils,文件的包名通常和文件所在的文件夹名一致,一般为小写字母

    • 当一个文件要使用其他函数或变量时,需要先引入对应的包

    • package指令在文件第一行,其次是import指令

    • 如果包名较长,Go支持给包取别名,取别名后原来的包名不能使用

      • util"project01/utils"
        
    • 若果要编译成一个可执行程序文件,需要将此包声明为main,即package main,如果是写一个库,包名可以自定义

函数调用机制底层
  • 栈区:基本数据类型一般分配到栈区,编译器存在一个逃逸分析
  • 堆区:引用数据类型一般分配到堆区,编译器存在一个逃逸分析
  • 代码区:存放代码的地方
  • 在调用一个函数时,会给该函数分配一个新的空间,编译器会通过自身的处理让这个新的空间和其他的栈的空间区分开
  • 在每个函数对应的栈中,数据空间时独立的
  • 当一个函数调用完毕之后,程序会销毁这个函数对应的栈空间
  • 基本数据类型和数组默认都是值传递的
return语句
//Go函数支持返回多个值
/*基本语法
func 函数名 (形参列表) (返回值类型列表) {
	语句
	return 返回值列表
}
*/

//如果返回多个值在接收时希望忽略某个返回值,可以使用_符号表示占位忽略
//如果返回值只有一个,返回值列表可以不写()

func a (n1 int,n2 int) (int,int) {
	sum := n1 + n2
	sub := n1 - n2
	return sum,sub
}
自定义数据类型
package main
import(
	"fmt"
)

//基本语法
/*
type 自定义数据类型名 数据类型
*/

func main(){
	type myInt int

	var num1 myInt
	var num2 int
	num1 = 40
	num2 = int(num1)//go认为myInt与int是两个类型,需要显示转换
	fmt.Println(num1,num2)
	
}
init函数
package main
import(
	"fmt"
)

//每个源文件都可以包含一个init函数,该函数会在main函数执行前被Go运行框架调用
//即init会在main函数前被调用
//init函数可以用于作初始化
//如果一个文件中包含全局变量定义,init函数和main函数
//执行的流程是 全局变量定义->init函数->main函数

var age = test()
//为了看到全局变量是先被初始化的,可以使用函数
func test() int {
	fmt.Println("test()")
	return 90
}

func init(){
	fmt.Println("init()...")
}
func main(){
	fmt.Println("main()...")
}
匿名函数
package main

import(
	"fmt"
)
//Go支持匿名函数,匿名函数就是没有名字的函数
//如果希望某个函数只使用一次,可以使用匿名函数
//匿名函数也可以实现多次调用

//方式三
//全局匿名函数
var (
	//Fun1是一个全局匿名函数
	Fun1 = func (n1 int,n2 int) int {
		return n1 * n2
	}
)

func main(){
	//使用方式一
	//在定义匿名函数的时候就直接调用,这种方式匿名函数只能调用一次
	res1 := func (n1 int,n2 int) int {
		return n1 + n2
	} (10,20)
	fmt.Println(" res1=",res1)

	//使用方式二
	//将匿名函数赋给变量
	//该变量的数据类型就是函数类型,该变量不是函数名
	//可以通过a实现多次调用
	a := func (n1 int, n2 int) int {
		return n1 + n2
	}
	res2 := a(10,20)
	res3 := a(20,30)
	fmt.Println(" res2=",res2)
	fmt.Println(" res3=",res3)

	//全局匿名函数的使用
	res4 := Fun1(4,9)
	fmt.Println("res4=",res4)
}
闭包
package main
import(
	"fmt"
)

//闭包就是一个函数和与其相关的引用环境组合的一个整体(实体)
//即定义在函数内的函数
//累加器
//AddUpper是一个函数,返回的数据类型是fun (int) int
//闭包的说明
/*var n int = 10
return func (x int) int {
		n = n + x
		return n
	}
*/
//返回的是一个匿名函数,这个匿名函数引用到函数外的n
//因此该匿名函数和n形成一个整体,构成闭包
//当反复调用函数时,因为n时初始化一次,所以每调用一次就进行累计

//n相当于闭包中的全局函数
func AddUpper() func (int) int {
	var n int = 10
	return func (x int) int {
		n = n + x
		return n
	}
}

func main(){
	
    f := AddUpper()//f类型为func (int)int
	fmt.Println(f(1))//11
	fmt.Println(f(2))//13
	fmt.Println(f(2))//16
}
//编写一个函数makeSuffix(suffix string)
//可以接收一个文件后缀名(比如.jpg),并返回一个闭包
//调用闭包,可以传入一个文件名,如果该文件名没有指定的后缀(比如.jpg)
//则返回文件名.jpg,如果有则返回原文件名
//要求使用闭包的方式完成
package main
import(
	"fmt"
	"strings"
)

func makeSuffix (suffix string) func (string) string {
	return func  (name string) string {
		//如果name没有指定后缀,则加上,否则就返回原来名字
		if !strings.HasSuffix(name,suffix) {
			return name + suffix
		}

		return name
	}
}

func main(){
	f2 := makeSuffix(".jpg")
	fmt.Println("文件名处理后=",f2("winter"))
	fmt.Println("文件名处理后=",f2("bird,jpg"))
}
defer
package main
import(
	"fmt"
)

//在函数中,经常需要创建资源(如数据库连接、文件句柄、锁等)
//为了在函数执行后及时释放资源
//Go提供defer(延时机制)

func sum (n1 int, n2 int) int {
	//当执行到defer时,暂时不执行,会将defer后面的语句压入到独立的栈(defer栈)
	//当函数执行完毕后,再从defer栈,按照先进后出的方式出栈,执行
	//在defer将语句放入到栈时,也会把相关的值拷贝同时入栈
	defer fmt.Println("ok1 n1=",n1)//3 ok1 n1=10
	defer fmt.Println("ok2 n2=",n2)//2 ok2 n2=20
	n1++
	n2++
	res := n1 + n2
	fmt.Println("ok3 res=",res)//1 ok3 res=32
	return res
}

func main(){
	res := sum (10,20) 
	fmt.Println("res=",res)//4 res=32
}

//defer在golang中通常做法
//创建资源后,比如(打开了文件,获取了数据库的链接或者是锁链接)
//可以执行defer file.Close()defer connect.Close()
//在defer后,可以继续使用创建资源
//当函数完毕后,系统会依次从defer栈中取出语句,关闭资源
字符串常用的系统函数
package main
import (
	"fmt"
	"strconv"
	"strings"
)

func main(){

	//统计字符串的长度,按字节 len(str)
	golang的编码统一为utf-8 (ascii的字符(字母和数字) 占一个字节,汉字占用3个字节)
	str := "hello北" 
	fmt.Println("str len=", len(str)) // 8


	str2 := "hello北京"
	//字符串遍历,同时处理有中文的问题 r := []rune(str)
	r := []rune(str2)
	for i := 0; i < len(r); i++ {
		fmt.Printf("字符=%c\n", r[i])
	}


	//字符串转整数:	 n, err := strconv.Atoi("12")
	n, err := strconv.Atoi("123")
	if err != nil {
		fmt.Println("转换错误", err)
	}else {
		fmt.Println("转成的结果是", n)
	}

	//4)整数转字符串  str = strconv.Itoa(12345)
	str = strconv.Itoa(12345)
	fmt.Printf("str=%v, str=%T\n", str, str)

	//5)字符串 转 []byte:  var bytes = []byte("hello go")
	var bytes = []byte("hello go")
	fmt.Printf("bytes=%v\n", bytes)

	//6)[]byte 转 字符串: str = string([]byte{97, 98, 99})
	str = string([]byte{97, 98, 99}) 
	fmt.Printf("str=%v\n", str)

	//10进制转 2, 8, 16进制:  str = strconv.FormatInt(123, 2),返回对应的字符串
	str = strconv.FormatInt(123, 2)
	fmt.Printf("123对应的二进制是=%v\n", str)
	str = strconv.FormatInt(123, 16)
	fmt.Printf("123对应的16进制是=%v\n", str)

	//查找子串是否在指定的字符串中: strings.Contains("seafood", "foo") //true
	b := strings.Contains("seafood", "mary")
	fmt.Printf("b=%v\n", b) 

	//统计一个字符串有几个指定的子串 : strings.Count("ceheese", "e") //4
	num := strings.Count("ceheese", "e")
	fmt.Printf("num=%v\n", num)

	//10)不区分大小写的字符串比较(==是区分字母大小写的): fmt.Println(strings.EqualFold("abc", "Abc")) // true

	b = strings.EqualFold("abc", "Abc")
	fmt.Printf("b=%v\n", b) //true

	fmt.Println("结果","abc" == "Abc") // false //区分字母大小写

	//11)返回子串在字符串第一次出现的index值,如果没有返回-1 : 
	//strings.Index("NLT_abc", "abc") // 4

	index := strings.Index("NLT_abcabcabc", "abc") // 4
	fmt.Printf("index=%v\n",index)

	//12)返回子串在字符串最后一次出现的index,
	//如没有返回-1 : strings.LastIndex("go golang", "go")

	index = strings.LastIndex("go golang", "go") //3
	fmt.Printf("index=%v\n",index)

	//将指定的子串替换成 另外一个子串: strings.Replace("go go hello", "go", "go语言", n) 
	//n可以指定你希望替换几个,如果n=-1表示全部替换

	str2 = "go go hello"
	str = strings.Replace(str2, "go", "北京", -1)
	fmt.Printf("str=%v str2=%v\n", str, str2)

	//按照指定的某个字符,为分割标识,将一个字符串拆分成字符串数组: 
	//strings.Split("hello,wrold,ok", ",")
	strArr := strings.Split("hello,wrold,ok", ",")
	for i := 0; i < len(strArr); i++ {
		fmt.Printf("str[%v]=%v\n", i, strArr[i])
	} 
	fmt.Printf("strArr=%v\n", strArr)

	//15)将字符串的字母进行大小写的转换: 
	//strings.ToLower("Go") // go strings.ToUpper("Go") // GO

	str = "goLang Hello"
	str = strings.ToLower(str) 
	str = strings.ToUpper(str) 
	fmt.Printf("str=%v\n", str) //golang hello

	//将字符串左右两边的空格去掉: strings.TrimSpace(" tn a lone gopher ntrn   ")
	str = strings.TrimSpace(" tn a lone gopher ntrn   ")
	fmt.Printf("str=%q\n", str)

	//17)将字符串左右两边指定的字符去掉 : 
	//strings.Trim("! hello! ", " !")  // ["hello"] //将左右两边 ! 和 " "去掉
	str = strings.Trim("! he!llo! ", " !")
	fmt.Printf("str=%q\n", str)

	//20)判断字符串是否以指定的字符串开头: 
	//strings.HasPrefix("ftp://192.168.10.1", "ftp") // true

	b = strings.HasPrefix("ftp://192.168.10.1", "hsp") //true
	fmt.Printf("b=%v\n", b)
}
Go时间和日期函数
package main

//时间和日期相关函数需要导入time包
import(
	"fmt"
	"time"
)

func main () {
	//获取当前时间
	//now := time.Now()//now的数据类型为time.Time
	now := time.Now()
	fmt.Printf("now=%v now type=%T\n",now,now)

	//通过now获取年月日,时分秒
	fmt.Printf("年=%v\n",now.Year())
	fmt.Printf("月=%v\n",now.Month())
	fmt.Printf("月=%v\n",int(now.Month()))
	fmt.Printf("日=%v\n",now.Day())
	fmt.Printf("时=%v\n",now.Hour())
	fmt.Printf("分=%v\n",now.Minute())
	fmt.Printf("秒=%v\n",now.Second())

	//格式化日期时间
	
	//方式一 使用Printf或Sprintf
	fmt.Printf("当前年月日 %d-%d-%d %d:%d:%d \n",now.Year(),
now.Month(),now.Day(),now.Hour(),now.Minute(),now.Second())

	dateStr := fmt.Sprintf("当前年月日 %d-%d-%d %d:%d:%d \n",now.Year(),
	now.Month(),now.Day(),now.Hour(),now.Minute(),now.Second())
	fmt.Printf("dateStr=%v\n",dateStr)

	//方式二
	//使用now.Fowmat() 参数的数字时固定的
	fmt.Printf(now.Format("2006/01/02 15:04:05"))
	fmt.Println()
	fmt.Printf(now.Format("2006-01-02"))
	fmt.Println()
	fmt.Printf(now.Format("15:04:05"))
	fmt.Println()

	//时间常量
	/*
	const (
		Nanosecond Duration = 1//纳秒
		Microsecond = 1000*Nanosecond//微秒
		Millisecond = 1000*Microsecond//毫秒
		Second = 1000*Millisecond//秒
		Minute = 60 * Second//分钟
		Hour = 60 * Minute//小时
	)
	*/
	//常量的作用:在程序中可用于获取指定时间单位

	//需求,每隔1秒打印一个数字,打到100就退出
	i := 0
	for {
		i++
		fmt.Println(i)
		//休眠
		//Sleep中参数必须为整数
		time.Sleep(time.Second)
		if i == 100 {
			break
		}
	}

	//获取当前unix时间戳和unixnano时间戳
	//可以获取随机数字
	fmt.Printf("unix的时间戳=%v unixnano的时间戳=%v\n",now.Unix(),now.UnixNano())

}
内置函数
  • len:求长度

  • new:用来分配内存,主要用来分配值类型

    • package main
      import(
      	"fmt"
      )
      
      func main(){
      	num1 := 100
      	fmt.Printf("num1的类型%T,num1的值%v,num1的地址%v\n",num1,num1,&num1)
      
      	num2 := new(int)
      	//num2的类型为int*
      	fmt.Printf("num2的类型%T,num2的值%v,num2的地址%v,num2指向的值%v\n",num2,num2,&num2,*num2)
      }
      
  • make:用来分配内存,主要用来分配引用类型

Go的错误处理机制
package main
import(
	"fmt"
)

//Go不支持传统的try...catch...finall等处理方式
//Go中引入的处理方式为defer,panic,recover
//Go中可以抛出一个panic异常,然后再defer中通过recover捕获这个异常,然后正常处理

func test () {
	//使用defer + recover处理异常
	defer func() {
		err := recover()//recover()内置函数,可以捕获到异常
		if err != nil {//说明捕获到异常
			fmt.Println("err=",err)
		}
	}()
	num1 : = 10
	num2 := 0
	res := num1 / num2
	fmt.Println("res=",res)
}

func main(){

	test()
	fmt.Println("main()下面的代码")
}
Go的自定义错误
package main
import(
	"fmt"
	"errors"
)
//在Go中,支持自定义错误,使用error.New和panic内置函数
//error.New("错误说明"),会返回一个error类型的值,表示一个错误
//panic内置函数,接收一个interface{}类型的值(即任何值)作为参数
//可以接收error类型的变量,输出错误信息,并退出程序

func readConf (name string) (err error) {
	if name == "config.ini" {
		//读取...
		return nil
	} else {
		//返回一个自定义错误
		return errors.New("读取文件错误...")
	}
}

func test () {
	err := readConf("config.ini")
	if err != nil {
		//读取文件发生错误,就输出这个错误,并终止程序
		panic(err)
	}

	fmt.Println("test()继续执行")
}

func main() {
	test()
	fmt.Println("main()下面的代码")
}

est()
fmt.Println(“main()下面的代码”)
}


#### Go的自定义错误

```go
package main
import(
	"fmt"
	"errors"
)
//在Go中,支持自定义错误,使用error.New和panic内置函数
//error.New("错误说明"),会返回一个error类型的值,表示一个错误
//panic内置函数,接收一个interface{}类型的值(即任何值)作为参数
//可以接收error类型的变量,输出错误信息,并退出程序

func readConf (name string) (err error) {
	if name == "config.ini" {
		//读取...
		return nil
	} else {
		//返回一个自定义错误
		return errors.New("读取文件错误...")
	}
}

func test () {
	err := readConf("config.ini")
	if err != nil {
		//读取文件发生错误,就输出这个错误,并终止程序
		panic(err)
	}

	fmt.Println("test()继续执行")
}

func main() {
	test()
	fmt.Println("main()下面的代码")
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值