注意事项

对进程有了深入理解后,我们编写实际应用可能遇到这些坑,这里总结一下。

创建目录权限

如果你想创建一个目录并授予777权限,你需要怎么做?查看Go的API文档我们可以这样写。源文件为mkdir.go。

package main
import (
  "fmt"
  "os"
)
func main() {
    err := os.MkdirAll("/tmp/gotest/", 0777)
    if err != nil {
      panic(err)
    }
    fmt.Println("Mkdir /tmp/gotest/")
}

运行结果

➜  understand_linux_process_examples git:(master) ✗ ll /tmp/
drwxr-xr-x   2 tobe  wheel    68B Dec 30 10:06 gotest
➜  understand_linux_process_examples git:(master) ✗ umask
022

正确做法

代码在mkdir_umask.go中。

package main
import (
  "fmt"
  "os"
  "syscall"
)
func main() {
    mask := syscall.Umask(0)
    defer syscall.Umask(mask)
    err := os.MkdirAll("/tmp/gotest/", 0777)
    if err != nil {
      panic(err)
    }
    fmt.Println("Mkdir /tmp/gotest/")
}

注意事项: 这并不是Go的Bug,包括Linux系统调用都是这样的,创建目录除了给定的权限还要加上系统的Umask,Go也是如实遵循这种约定。如果你想达到你的预期权限,知道Umask及其用法是必须的。

捕获SIGKILL

SIGKILL是常见的Linux信号,我们使用kill命令杀掉进程也就是像进程发送SIGKILL信号。

和其他信号不同,SIGKILL和SIGSTOP是不可被Catch的,因此下面的代码是能编译通过但也是无效的,更多细节可以参考golang/go#9463.

c := make(chan os.Signal, 1)
signal.Notify(c, syscall.SIGKILL, syscall.SIGSTOP)

注意事项: 这是Linux内核的限制,这种限制也是为了让操作系统有可能控制进程的生命周期,理解后我们也不应该去尝试捕获SIGKILL。不过还是有人这样去做,最后结果也不符合预期,这需要我们对底层有足够的理解。

系统调用sendfile

Sendfile是Linux实现的系统调用,可以通过避免文件在内核态和用户态的拷贝来优化文件传输的效率。

其中大名鼎鼎的分布式消息队列服务Kafka就使用sendfile来优化效率,具体用法可参见其官方文档

优化策略

在普通进程中,要从磁盘拷贝数据到网络,其实是需要通过系统调用,进程也会反复在用户态和内核态切换,频繁的数据传输在此有效率问题。因此我们必须意识到Linux给我们提供了sendfile这样的系统调用,可以提高进程的数据传输效率。

参考书籍

下一节:进程的概念大家都很熟悉,但你是否能准确说出僵尸进程的含义呢?还有COW(Copy On Write)、Flock(File Lock)、Epoll和Namespace的概念又是否了解过呢?