我使用了一些类似于下面几行的代码。如果目录被阻止,因为某些应用程序在其中打开了一个文件,则重命名可能会失败。
err := os.Rename("C:/temp/inUse", "c:/temp/Renamed")
if err != nil {
fmt.Println(err)
...
}
Run Code Online (Sandbox Code Playgroud)
当我检查err此内容时,我能够检测到这一事实:
重命名 C:/temp/inUse c:/temp/Renamed: Der Prozess kann nicht auf die Datei zugreifen, da sie von einem anderen Prozess verwendet wird。
(这意味着“该进程无法访问该文件,因为它被不同的进程使用”)但是此消息因操作系统语言而异。
是否可以通过唯一的“错误代码”来检测问题?我不介意该解决方案是否特定于 Windows 并且不适用于 Go 支持的其他操作系统。
理想情况下,我可以做类似的事情 if err.Err == 32 { fmt.Println( "Your directory is in use please ...
返回的错误os.Rename类型*os.LinkError使您可以从操作系统访问底层错误。您应该能够使用它来区分您遇到的特定错误。
您需要首先将错误转换为 *os.LinkError。
例如,尝试将文件夹 ( /Users/t/pprof) 重命名为写保护的另一个名称 ( /Users/t/pprof2):
func TestRename(t *testing.T) {
err := os.Rename("/Users/t/pprof", "/Users/t/pprof2")
if err != nil {
e := err.(*os.LinkError)
t.Logf("Op: ", e.Op)
t.Logf("Old: ", e.Old)
t.Logf("New: ", e.New)
t.Logf("Err: ", e.Err)
}
}
Run Code Online (Sandbox Code Playgroud)
提供以下输出:
Op: %!(EXTRA string=rename)
Old: %!(EXTRA string=/Users/t/pprof)
New: %!(EXTRA string=/Users/t/pprof2)
Err: %!(EXTRA syscall.Errno=operation not permitted)
Run Code Online (Sandbox Code Playgroud)
操作系统错误代码可以作为Err成员访问,但会根据您运行的操作系统而有所不同。
该Err成员的类型为syscall.Errno。要进一步检查实际错误,您需要首先将其转换为该类型:
oserr := e.Err.(syscall.Errno)
Run Code Online (Sandbox Code Playgroud)
现在可以与包中声明oserr的值进行比较。如果您在该页面上搜索,就会找到它们。ErrnosyscallENOENT
例如,您可以通过执行以下操作来检查特定错误:
switch oserr {
case syscall.ENOENT:
// Handle this error
default:
// Handle other errors
}
}
Run Code Online (Sandbox Code Playgroud)
一般来说,在调试此类问题时使用 fmt.Printf 非常方便。在上面的例子中:
fmt.Println(err)
Run Code Online (Sandbox Code Playgroud)
印刷
rename /Users/t/pprof /Users/t/pprof2: operation not permitted
Run Code Online (Sandbox Code Playgroud)
然而
fmt.Printf("%#v\n", err)
Run Code Online (Sandbox Code Playgroud)
印刷
&os.LinkError{Op:"rename", Old:"/Users/t/pprof", New:"/Users/t/pprof2", Err:0x1}
Run Code Online (Sandbox Code Playgroud)
提供有关实际错误的详细信息,而不仅仅是其字符串表示形式。