日本免费高清视频-国产福利视频导航-黄色在线播放国产-天天操天天操天天操天天操|www.shdianci.com

學(xué)無先后,達(dá)者為師

網(wǎng)站首頁 編程語言 正文

關(guān)于Golang獲取當(dāng)前項(xiàng)目絕對(duì)路徑的問題_Golang

作者:快樂的GTD吧 ? 更新時(shí)間: 2022-06-20 編程語言

導(dǎo)讀

由于Golang是編譯型語言(非腳本型語言),如果你想在Golang程序中獲取當(dāng)前執(zhí)行目錄將是一件非常蛋疼的事情。以前大家最折中的解決方案就是通過啟動(dòng)傳參或是環(huán)境變量將路徑手動(dòng)傳遞到程序,而今天我在看日志庫的時(shí)候發(fā)現(xiàn)了一種新的解決方案。
Go程序兩種不同的執(zhí)行方式

用Go編寫的程序有兩種執(zhí)行方式,go run和go build

通常的做法是go run用于本地開發(fā),用一個(gè)命令中快速測(cè)試代碼確實(shí)非常方便;在部署生產(chǎn)環(huán)境時(shí),我們會(huì)通過go build構(gòu)建出二進(jìn)制文件然后上傳到服務(wù)器再去執(zhí)行。

兩種啟動(dòng)方式會(huì)產(chǎn)生什么問題?

那么兩種啟動(dòng)方式下,獲取到當(dāng)前執(zhí)行路徑會(huì)產(chǎn)生什么問題?

話不多說,我們直接上代碼

我們編寫獲取當(dāng)前可執(zhí)行文件路徑的方法

package main

import (
"fmt"
"log"
"os"
"path/filepath"
)
func main() {
fmt.Println("getCurrentAbPathByExecutable = ", getCurrentAbPathByExecutable())
}
// 獲取當(dāng)前執(zhí)行程序所在的絕對(duì)路徑
func getCurrentAbPathByExecutable() string {
exePath, err := os.Executable()
if err != nil {
log.Fatal(err)
res, _ := filepath.EvalSymlinks(filepath.Dir(exePath))
return res

首先通過go run啟動(dòng)

D:\Projects\demo>go run main.go
getCurrentAbPathByExecutable = C:\Users\XXX\AppData\Local\Temp\go-build216571510\b001\exe

再嘗試go build執(zhí)行

D:\Projects\demo>go build & demo.exe
getCurrentAbPathByExecutable = D:\Projects\demo

通過對(duì)比執(zhí)行結(jié)果,我們發(fā)現(xiàn)兩種執(zhí)行方式,我們獲取到了不同的路徑。而且很明顯,go run獲取到的路徑是錯(cuò)誤的。

原因: 這是由于go run會(huì)將源代碼編譯到系統(tǒng)TEMP或TMP環(huán)境變量目錄中并啟動(dòng)執(zhí)行;而go build只會(huì)在當(dāng)前目錄編譯出可執(zhí)行文件,并不會(huì)自動(dòng)執(zhí)行。

我們可以簡(jiǎn)單理解為,go run main.go等價(jià)于go build & ./main

雖然兩種執(zhí)行方式最終都是一樣的過程:源碼->編譯->可執(zhí)行文件->執(zhí)行輸出,但他們的執(zhí)行目錄卻完全不一樣了。
新的方案誕生

這是在我今天查看服務(wù)日志(zap庫)的時(shí)候,突然反應(yīng)過來一件事情。比如下面是一條簡(jiǎn)單的日志,而服務(wù)是通過go run啟動(dòng)的,但日志庫卻把我正確的程序路徑D:/Projects/te-server/modules/es/es.go:139給打印出來了

2021-03-26 17:47:06 D:/Projects/te-server/modules/es/es.go:139 update es index {"index": "tags", "data": "[200 OK] {"acknowledged":true}"}

于是我馬上去翻看zap源碼,發(fā)現(xiàn)是通過runtime.Caller()實(shí)現(xiàn)的,其實(shí)所有Golang日志庫都會(huì)有runtime.Caller()這個(gè)調(diào)用。

我開心的以為找到了最終答案,然后寫代碼試了下:

package main

import (
"fmt"
"path"
"runtime"
)
func main() {
fmt.Println("getCurrentAbPathByCaller = ", getCurrentAbPathByCaller())
}
// 獲取當(dāng)前執(zhí)行文件絕對(duì)路徑(go run)
func getCurrentAbPathByCaller() string {
var abPath string
_, filename, _, ok := runtime.Caller(0)
if ok {
abPath = path.Dir(filename)
return abPath

首先在windows下面go run 和go build試一下

D:\Projects\demo>go run main.go
getCurrentAbPathByCaller = D:/Projects/demo

D:\Projects\demo>go build & demo.exe
getCurrentAbPathByCaller = D:/Projects/demo

嗯~~ 結(jié)果完全正確!

然后我再把構(gòu)建好的程序扔到linux再運(yùn)行后,它把我windows的路徑給打印出來了 --!

[root@server app]# chmod +x demo
[root@server app]# ./demo
getCurrentAbPathByCaller = D:/Projects/demo

沒想到白白高興一場(chǎng),這個(gè)時(shí)候我就在想,既然go run時(shí)可以通過runtime.Caller()獲取到正確的結(jié)果,go build時(shí)也可以通過 os.Executable()來獲取到正確的路徑;

那如果我能判定當(dāng)前程序是通過go run還是go build執(zhí)行的,選擇不同的路徑獲取方法,所有問題不就迎刃而解了嗎。
區(qū)分程序是go run還是go build執(zhí)行

Go沒有提供接口讓我們區(qū)分程序是go run還是go build執(zhí)行,但我們可以換個(gè)思路來實(shí)現(xiàn):

根據(jù)go run的執(zhí)行原理,我們得知它會(huì)源代碼編譯到系統(tǒng)TEMP或TMP環(huán)境變量目錄中并啟動(dòng)執(zhí)行;

那我們可以直接在程序中對(duì)比os.Executable()獲取到的路徑是否與環(huán)境變量TEMP設(shè)置的路徑相同, 如果相同,說明是通過go run啟動(dòng)的,因?yàn)楫?dāng)前執(zhí)行路徑是在TEMP目錄;不同的話自然是go build的啟動(dòng)方式。

下面是完整代碼:

package main

import (
"fmt"
"log"
"os"
"path"
"path/filepath"
"runtime"
"strings"
)
func main() {
fmt.Println("getTmpDir(當(dāng)前系統(tǒng)臨時(shí)目錄) = ", getTmpDir())
fmt.Println("getCurrentAbPathByExecutable(僅支持go build) = ", getCurrentAbPathByExecutable())
fmt.Println("getCurrentAbPathByCaller(僅支持go run) = ", getCurrentAbPathByCaller())
fmt.Println("getCurrentAbPath(最終方案-全兼容) = ", getCurrentAbPath())
}
// 最終方案-全兼容
func getCurrentAbPath() string {
dir := getCurrentAbPathByExecutable()
if strings.Contains(dir,getTmpDir()) {
return getCurrentAbPathByCaller()
return dir
// 獲取系統(tǒng)臨時(shí)目錄,兼容go run
func getTmpDir() string {
dir := os.Getenv("TEMP")
if dir == "" {
dir = os.Getenv("TMP")
res, _ := filepath.EvalSymlinks(dir)
return res
// 獲取當(dāng)前執(zhí)行文件絕對(duì)路徑
func getCurrentAbPathByExecutable() string {
exePath, err := os.Executable()
if err != nil {
log.Fatal(err)
res, _ := filepath.EvalSymlinks(filepath.Dir(exePath))
// 獲取當(dāng)前執(zhí)行文件絕對(duì)路徑(go run)
func getCurrentAbPathByCaller() string {
var abPath string
_, filename, _, ok := runtime.Caller(0)
if ok {
abPath = path.Dir(filename)
return abPath

在windows執(zhí)行

D:\Projects\demo>go run main.go
getTmpDir(當(dāng)前系統(tǒng)臨時(shí)目錄) = C:\Users\XXX\AppData\Local\Temp
getCurrentAbPathByExecutable(僅支持go build) = C:\Users\XXX\AppData\Local\Temp\go-build456189690\b001\exe
getCurrentAbPathByCaller(僅支持go run) = D:/Projects/demo
getCurrentAbPath(最終方案-全兼容) = D:/Projects/demo
D:\Projects\demo>go build & demo.exe
getTmpDir(當(dāng)前系統(tǒng)臨時(shí)目錄) = C:\Users\XXX\AppData\Local\Temp
getCurrentAbPathByExecutable(僅支持go build) = D:\Projects\demo
getCurrentAbPathByCaller(僅支持go run) = D:/Projects/demo
getCurrentAbPath(最終方案-全兼容) = D:\Projects\demo

在windows編譯后上傳到Linux執(zhí)行

[root@server app]# pwd
/data/app
[root@server app]# ./demo
getTmpDir(當(dāng)前系統(tǒng)臨時(shí)目錄) = .
getCurrentAbPathByExecutable(僅支持go build) = /data/app
getCurrentAbPathByCaller(僅支持go run) = D:/Projects/demo
getCurrentAbPath(最終方案-全兼容) = /data/app

對(duì)比結(jié)果,我們可以看到,在不同的系統(tǒng)中,不同的執(zhí)行方式,我們封裝的getCurrentAbPath方法最終都輸出的正確的結(jié)果,perfect!

原文鏈接:https://www.cnblogs.com/jiftle/p/16177738.html

欄目分類
最近更新