技术就是一层窗户纸,能捅破便是拨开云天见月明,捅不破就是一叶障目,不见泰山呀!
今天碰到了个问题,卡了我半天。
malformed module path "XXXX": missing dot in first path element
问题原因
因为在 go1.13 中, go module 名称规范要求路径的第一部分必须满足域名规范,否则可能汇报类似
malformed module path "xxxx": missing dot in first path element 这样的错误。
解决方法
使用go mod 的replace语句对相关的包进行替换。
项目的目录结构:
kafka模块的go mod:
taillog模块的go mod:
main包的go mod:
替换之前他会报错,说找不到http://github.com/wind-zhou/logagent/kafka这个包,经过relace替换后便编译成功。
我这里具体的机理还没有弄明白,但我猜测他可能会以为引入的包是第三方的包,因此回去GitHub上找,用replace之后,便告诉编译器这个包在本地。
下面对go mod的本地的包的操作进行详细解释。
go mod 如何导入本地的包
接下来的内容转载自: https://www.liwenzhou.com/posts/Go/import_local_package_in_go_module/
go module 是Go1.11版本之后官方推出的版本管理工具,并且从Go1.13 版本开始,go module 将是Go语言默认的依赖管理工具。到今天Go1.14 版本推出之后Go modules 功能已经被正式推荐在生产环境下使用了。
这几天已经有很多教程讲解如何使用go module ,以及如何使用go module 导入gitlab私有仓库,我这里就不再啰嗦了。但是最近我发现很多小伙伴在群里问如何使用go module 导入本地包,作为初学者大家刚开始接触package的时候肯定都是先在本地创建一个包,然后本地调用一下,然后就被卡住了。。。
这里就详细介绍下如何使用go module 导入本地包。
前提
假设我们现在有moduledemo 和mypackage 两个包,其中moduledemo 包中会导入mypackage 包并使用它的New 方法。
mypackage/mypackage.go 内容如下:
package mypackage
import "fmt"
func New(){
fmt.Println("mypackage.New")
}
我们现在分两种情况讨论:
在同一个项目下
注意:在一个项目(project)下我们是可以定义多个包(package)的。
目录结构
现在的情况是,我们在moduledemo/main.go 中调用了mypackage 这个包。
moduledemo
├── go.mod
├── main.go
└── mypackage
└── mypackage.go
导入包
这个时候,我们需要在moduledemo/go.mod 中按如下定义:
module moduledemo
go 1.14
然后在moduledemo/main.go 中按如下方式导入mypackage
package main
import (
"fmt"
"moduledemo/mypackage" // 导入同一项目下的mypackage包
)
func main() {
mypackage.New()
fmt.Println("main")
}
举个例子
举一反三,假设我们现在有文件目录结构如下:
└── bubble
├── dao
│ └── mysql.go
├── go.mod
└── main.go
其中bubble/go.mod 内容如下:
module github.com/q1mi/bubble
go 1.14
bubble/dao/mysql.go 内容如下:
package dao
import "fmt"
func New(){
fmt.Println("mypackage.New")
}
bubble/main.go 内容如下:
package main
import (
"fmt"
"github.com/q1mi/bubble/dao"
)
func main() {
dao.New()
fmt.Println("main")
}
不在同一个项目下
目录结构
├── moduledemo
│ ├── go.mod
│ └── main.go
└── mypackage
├── go.mod
└── mypackage.go
导入包
这个时候,mypackage 也需要进行module初始化,即拥有一个属于自己的go.mod 文件,内容如下:
module mypackage
go 1.14
然后我们在moduledemo/main.go 中按如下方式导入:
import (
"fmt"
"mypackage"
)
func main() {
mypackage.New()
fmt.Println("main")
}
因为这两个包不在同一个项目路径下,你想要导入本地包,并且这些包也没有发布到远程的github或其他代码仓库地址。这个时候我们就需要在go.mod 文件中使用replace 指令。
在调用方也就是packagedemo/go.mod 中按如下方式指定使用相对路径来寻找mypackage 这个包。
module moduledemo
go 1.14
require "mypackage" v0.0.0
replace "mypackage" => "../mypackage"
举个例子
最后我们再举个例子巩固下上面的内容。
我们现在有文件目录结构如下:
├── p1
│ ├── go.mod
│ └── main.go
└── p2
├── go.mod
└── p2.go
p1/main.go 中想要导入p2.go 中定义的函数。
p2/go.mod 内容如下:
module liwenzhou.com/q1mi/p2
go 1.14
p1/main.go 中按如下方式导入
import (
"fmt"
"liwenzhou.com/q1mi/p2"
)
func main() {
p2.New()
fmt.Println("main")
}
因为我并没有把liwenzhou.com/q1mi/p2 这个包上传到liwenzhou.com 这个网站,我们只是想导入本地的包,这个时候就需要用到replace 这个指令了。
p1/go.mod 内容如下:
module github.com/q1mi/p1
go 1.14
require "liwenzhou.com/q1mi/p2" v0.0.0
replace "liwenzhou.com/q1mi/p2" => "../p2"
此时,我们就可以正常编译p1 这个项目了。
说再多也没用,自己动手试试吧。
|
请发表评论