每个package可以有多个源码文件,但是必须位于同一层目录。package内可以有子目录,只不过子目录下就是另一个package的源码了。package内可以使用同一个package的所有资源,不需要像c一样先声明再使用。当import一个package后就可以使用其中的大写字母开头的函数和变量了。

在工程化的Go语言开发项目中,Go语言的源码复用是建立在包(package)基础之上的

包(package)介绍

包(package)是多个Go源码的集合,是一种高级的代码复用方案,Go语言为我们提供了很多内置包,如fmtosio等。

fmt 我们是从哪里导过来的?我们可以自己去自己的GOROOT/src 下去查看,你们可以看到大致如下的包

golang docker打包 golang package_开发语言

包(package)的使用

使用示例1:包导入的常规语法

package main

import (
	"fmt"
)

func main() {
	//然后在代码里面可以通过如下的方式调用
	fmt.Println("go mod test!")
}

----输出结果----
go mod test!

fmt是Go语言的标准库,这个引用其实是去GOROOT下去加载该模块

使用示例2:相对路径

这个在使用GO111MODULE的时候是无法使用的,会有下面的报错

main.go:6:2: "./mymod" is relative, but relative import paths are not supported in module mode

golang docker打包 golang package_后端_02

//main.go
package main

import (
	"fmt"

	"./mymod"
)

func main() {
	//然后在代码里面可以通过如下的方式调用
	mymod.SaiHi()
	fmt.Println("hello world")
}
//mymod.go
package mymod

import (
	"fmt"
)

func SaiHi() {
	//然后在代码里面可以通过如下的方式调用
	fmt.Println("sai hi!")
}
//go.mod
module mymod

go 1.19

使用示例3:绝对路径

由于我们默认使用的都是mod,所以我这里的示例是go.mod的利用方式、

golang docker打包 golang package_后端_03

//main.go
package main

import (
	"fmt"

	"mymod"
)

func main() {
	//然后在代码里面可以通过如下的方式调用
	mymod.SaiHi()
	fmt.Println("hello world")
}

----运行结果----
sai hi!
hello world
//go.mod
module testPackage

go 1.19
require mymod v0.0.0
replace mymod => C:\Users\kaikai\Desktop\golang\testPackage\mymod\
//mymod/mymod.go
package mymod

import (
	"fmt"
)

func SaiHi() {
	//然后在代码里面可以通过如下的方式调用
	fmt.Println("sai hi!")
}
//mymod/go.mod
module mymod

go 1.19

包(package)可见性

如果想在一个包中引用另外一个包里的标识符(如变量、常量、类型、函数等)时,该标识符必须是对外可见的(public)。在Go语言中只需要将标识符的首字母大写就可以让标识符对外可见了。

main.go代码如下

package mymod

import "fmt"

// 包变量可见性

var a = 100 // 首字母小写,外部包不可见,只能在当前包内使用

// 首字母大写外部包可见,可在其他包中使用
const Mode = 1

type person struct { // 首字母小写,外部包不可见,只能在当前包内使用
	name string
}


// 首字母大写,外部包可见,可在其他包中使用
func Add(x, y int) int {
	return x + y
}

func age() { // 首字母小写,外部包不可见,只能在当前包内使用
	var Age = 18 // 函数局部变量,外部包不可见,只能在当前函数内使用
	fmt.Println(Age)
}

结构体中的字段名和接口中的方法名如果首字母都是大写,外部包可以访问这些字段和方法。例如:

type Student struct {
	Name  string //可在包外访问的方法
	class string //仅限包内访问的字段
}

type Payer interface {
	init() //仅限包内访问的方法
	Pay()  //可在包外访问的方法
}

包(package)的导入

导包顺序是: 标准 - 本地 - golang - github

假如go的安装目录为/usr/local/go,也就是说GOROOT=/usr/local/go,而GOPATH环境变量GOPATH=~/mycode:~/mylib,那么要搜索net/http包的时候,将按照如下顺序进行搜索

//可以使用go env查看goroot与gopath的真实路径
usr/local/go/srcnet/http
~/mycode/src/net/http
~/mylib/src/net/http

编译器会根据上面指定的相对路径去搜索包然后导入,这个相对路径是从GOROOT或GOPATH(workspace)下的src下开始搜索的。

注意事项:

  • 一个文件夹下面直接包含的文件只能归属一个package,同样一个package的文件不能在多个文件夹下。
  • 包名可以不和文件夹的名字一样,包名不能包含 - 符号。
  • 包名为main的包为应用程序的入口包,这种包编译后会得到一个可执行文件,而编译不包含main包的源代码则不会得到可执行文件。

单行导入

单行导入的格式如下:

import "包1"

多行导入

多行导入的格式如下:

import (
    "包1"
    "包2"
)

自定义包名

在导入包名的时候,我们还可以为导入的包设置别名。通常用于导入的包名太长或者导入的包名冲突的情况。具体语法格式如下:

import 别名 "包的路径"

定义别名

import f "fmt"

func main() {
	f.Println("hello world")
}
----运行结果----
hello world

匿名导入包

如果只希望导入包,而不使用包内部的数据时,可以使用匿名导入包。具体的格式如下:

import _ "包的路径"

匿名导入的包与其他方式导入的包一样都会被编译到可执行文件中。

init()初始化函数

init()函数介绍

在Go语言程序执行时导入包语句会自动触发包内部init()函数的调用。需要注意的是: init()函数没有参数也没有返回值。 init()函数在程序运行时自动被调用执行,不能在代码中主动调用它。

package main
import "fmt"

func init() {
	fmt.Println("test -- init")
}

func main() {
	fmt.Println("this is test.test")
}

----运行结果----
test -- init
this is test.test

包初始化执行的顺序如下图所示

golang docker打包 golang package_golang docker打包_04

init()函数执行顺序

Go语言包会从main包开始检查其导入的所有包,每个包中又可能导入了其他的包。Go编译器由此构建出一个树状的包引用关系,再根据引用顺序决定编译顺序,依次编译这些包的代码。

在运行时,被最后导入的包会最先初始化并调用其init()函数, 如下图示:

golang docker打包 golang package_后端_05

程序加载过程

golang docker打包 golang package_golang docker打包_06

结构体之包的使用

golang docker打包 golang package_golang docker打包_07

//main.go
package main

import (
	"baozi"
	"fmt"
)

func main() {
	//baozi := baozi.baozi{"rou"}  // 无法使用,因为baozi结构体 小写开头
	baozi := baozi.NewBaozi("rou")
	fmt.Println("包子的种类", baozi.Kind)
	baozi.Product()
}

----运行结果----
包子的种类 rou
生产了一个肉包
//go.mod
module baozitest

go 1.19
require baozi v0.0.0
replace baozi => C:\Users\kaikai\Desktop\golang\Packages\baozi
//baozi/go.mod
module baozi

go 1.19
//baozi/factory.go
package baozi

import "fmt"

// 创建工厂结构体
type baozi struct {
	Kind string // 包子的种类
}

//在包子 包中我们创建了一个结构体,但是该结构体外部无法访问使用,因为小写字母开头,只能内部使用

func (this *baozi) Product() {
	switch this.Kind {
	case "rou":
		fmt.Println("生产了一个肉包")
	case "cai":
		fmt.Println("生产了一个菜包")
	default:
		fmt.Println("生产了一个未知包")
	}
}

// 在包子 包中提供了一个函数NewBaozi,只能通过调用NewBaozi创建baozi结构体
func NewBaozi(kind string) *baozi {
	return &baozi{kind}
}