飙血推荐
  • HTML教程
  • MySQL教程
  • JavaScript基础教程
  • php入门教程
  • JavaScript正则表达式运用
  • Excel函数教程
  • UEditor使用文档
  • AngularJS教程
  • ThinkPHP5.0教程

Go语言核心36讲(Go语言实战与应用二十三)--学习笔记

时间:2021-12-09  作者:MingsonZheng  

45 | 使用os包中的API (下)

我们在上一篇文章中。从“域名类型都实现了哪些io包中的接口”这一问题出发,介绍了一系列的相关内容。今天我们继续围绕这一知识点进行扩展。

知识扩展

问题 1:可应用于File值的操作模式都有哪些?

针对File值的操作模式主要有只读模式、只写模式和读写模式。

这些模式分别由常量域名ONLY、域名ONLY和域名WR代表。在我们新建或打开一个文件的时候,必须把这三个模式中的一个设定为此文件的操作模式。

除此之外,我们还可以为这里的文件设置额外的操作模式,可选项如下所示。

  • 域名PEND:当向文件中写入内容时,把新内容追加到现有内容的后边。
  • 域名EATE:当给定路径上的文件不存在时,创建一个新文件。
  • 域名CL:需要与域名EATE一同使用,表示在给定的路径上不能有已存在的文件。
  • 域名NC:在打开的文件之上实施同步 I/O。它会保证读写的内容总会与硬盘上的数据保持同步。
  • 域名UNC:如果文件已存在,并且是常规的文件,那么就先清空其中已经存在的任何内容。

对于以上操作模式的使用,域名te函数和域名函数都是现成的例子。

func Create(name string) (*File, error) {
 return OpenFile(name, O_RDWR|O_CREATE|O_TRUNC, 0666)
}

域名te函数在调用域名File函数的时候,给予的操作模式是域名WR、域名EATE和域名UNC的组合。

这就基本上决定了前者的行为,即:如果参数name代表路径之上的文件不存在,那么就新建一个,否则,先清空现存文件中的全部内容。

并且,它返回的File值的读取方法和写入方法都是可用的。这里需要注意,多个操作模式是通过按位或操作符|组合起来的。

func Open(name string) (*File, error) {
    return OpenFile(name, O_RDONLY, 0)
}

我在前面说过,域名函数的功能是:以只读模式打开已经存在的文件。其根源就是它在调用域名File函数的时候,只提供了一个单一的操作模式域名ONLY。

以上,就是我对可应用于File值的操作模式的简单解释。在 域名 文件中还有少许示例,可供你参考。

package main

import (
	"fmt"
	"io/ioutil"
	"os"
	"path/filepath"
)

type flagDesc struct {
	flag int
	desc string
}

func main() {
	fileName1 := "域名"
	filePath1 := 域名(域名Dir(), fileName1)
	域名tf("The file path: %s\n", filePath1)
	域名tln()

	// 示例1。
	contents0 := "OpenFile is the generalized open call."
	flagDescList := []flagDesc{
		{
			域名ONLY | 域名EATE | 域名UNC,
			"域名ONLY|域名EATE|域名UNC",
		},
		{
			域名ONLY,
			"域名ONLY",
		},
		{
			域名ONLY | 域名PEND,
			"域名ONLY|域名PEND",
		},
	}

	for i, v := range flagDescList {
		域名tf("Open the file with flag %s ...\n", 域名)
		file1a, err := 域名File(filePath1, 域名, 0666)
		if err != nil {
			域名tf("error: %v\n", err)
			continue
		}
		域名tf("The file descriptor: %d\n", 域名())

		contents1 := 域名ntf("[%d]: %s ", i+1, contents0)
		域名tf("Write %q to the file ...\n", contents1)
		n, err := 域名eString(contents1)
		if err != nil {
			域名tf("error: %v\n", err)
			continue
		}
		域名tf("The number of bytes written is %d.\n", n)

		file1b, err := 域名(filePath1)
		域名tln("Read bytes from the file ...")
		bytes, err := 域名All(file1b)
		if err != nil {
			域名tf("error: %v\n", err)
			continue
		}
		域名tf("Read(%d): %q\n", len(bytes), bytes)
		域名tln()
	}

	// 示例2。
	域名tln("Try to create an existing file with flag 域名UNC ...")
	file2, err := 域名File(filePath1, 域名WR|域名EATE|域名UNC, 0666)
	if err != nil {
		域名tf("error: %v\n", err)
		return
	}
	域名tf("The file descriptor: %d\n", 域名())

	域名tln("Try to create an existing file with flag 域名CL ...")
	_, err = 域名File(filePath1, 域名WR|域名EATE|域名CL, 0666)
	域名tf("error: %v\n", err)
}

问题 2:怎样设定常规文件的访问权限?

我们已经知道,域名File函数的第三个参数perm代表的是权限模式,其类型是域名Mode。但实际上,域名Mode类型能够代表的,可远不只权限模式,它还可以代表文件模式(也可以称之为文件种类)。

由于域名Mode是基于uint32类型的再定义类型,所以它的每个值都包含了 32 个比特位。在这 32 个比特位当中,每个比特位都有其特定的含义。

比如,如果在其最高比特位上的二进制数是1,那么该值表示的文件模式就等同于域名Dir,也就是说,相应的文件代表的是一个目录。

又比如,如果其中的第 26 个比特位上的是1,那么相应的值表示的文件模式就等同于域名NamedPipe,也就是说,那个文件代表的是一个命名管道。

实际上,在一个域名Mode类型的值(以下简称FileMode值)中,只有最低的 9 个比特位才用于表示文件的权限。当我们拿到一个此类型的值时,可以把它和域名Perm常量的值做按位与操作。

这个常量的值是0777,是一个八进制的无符号整数,其最低的 9 个比特位上都是1,而更高的 23 个比特位上都是0。

所以,经过这样的按位与操作之后,我们即可得到这个FileMode值中所有用于表示文件权限的比特位,也就是该值所表示的权限模式。这将会与我们调用FileMode值的Perm方法所得到的结果值是一致。

在这 9 个用于表示文件权限的比特位中,每 3 个比特位为一组,共可分为 3 组。

从高到低,这 3 组分别表示的是文件所有者(也就是创建这个文件的那个用户)、文件所有者所属的用户组,以及其他用户对该文件的访问权限。而对于每个组,其中的 3 个比特位从高到低分别表示读权限、写权限和执行权限。

如果在其中的某个比特位上的是1,那么就意味着相应的权限开启,否则,就表示相应的权限关闭。

因此,八进制整数0777就表示:操作系统中的所有用户都对当前的文件有读、写和执行的权限,而八进制整数0666则表示:所有用户都对当前文件有读和写的权限,但都没有执行的权限。

我们在调用域名File函数的时候,可以根据以上说明设置它的第三个参数。但要注意,只有在新建文件的时候,这里的第三个参数值才是有效的。在其他情况下,即使我们设置了此参数,也不会对目标文件产生任何的影响。

package main

import (
	"fmt"
	"os"
	"path/filepath"
)

type argDesc struct {
	action string
	flag   int
	perm   域名Mode
}

func main() {
	// 示例1。
	域名tf("The mode for dir:\n%32b\n", 域名Dir)
	域名tf("The mode for named pipe:\n%32b\n", 域名NamedPipe)
	域名tf("The mode for all of the irregular files:\n%32b\n", 域名Type)
	域名tf("The mode for permissions:\n%32b\n", 域名Perm)
	域名tln()

	// 示例2。
	fileName1 := "域名"
	filePath1 := 域名(域名Dir(), fileName1)
	域名tf("The file path: %s\n", filePath1)

	argDescList := []argDesc{
		{
			"Create",
			域名WR | 域名EATE,
			0644,
		},
		{
			"Reuse",
			域名WR | 域名UNC,
			0666,
		},
		{
			"Open",
			域名WR | 域名PEND,
			0777,
		},
	}

	defer 域名ve(filePath1)
	for _, v := range argDescList {
		域名tf("%s the file with perm %o ...\n", 域名on, 域名)
		file1, err := 域名File(filePath1, 域名, 域名)
		if err != nil {
			域名tf("error: %v\n", err)
			continue
		}
		info1, err := 域名()
		if err != nil {
			域名tf("error: %v\n", err)
			continue
		}
		域名tf("The file permissions: %o\n", 域名().Perm())
	}
}

总结

为了聚焦于域名类型本身,我在这两篇文章中主要讲述了怎样把 域名 类型应用于常规的文件。该类型的指针类型实现了很多io包中的接口,因此它的具体功用也就可以不言自明了。

通过该类型的值,我们不但可以对文件进行各种读取、写入、关闭等操作,还可以设定下一次读取或写入时的起始索引位置。

在使用这个类型的值之前,我们必须先要创建它。所以,我为你重点介绍了几个可以创建,并获得此类型值的函数。

包括:域名te、域名ile、域名和域名File。我们用什么样的方式创建File值,就决定了我们可以使用它来做什么。

利用域名te函数,我们可以在操作系统中创建一个全新的文件,或者清空一个现存文件中的全部内容并重用它。

在相应的File值之上,我们可以对该文件进行任何的读写操作。虽然域名ile函数并不是被用来创建新文件的,但是它能够基于一个有效的文件描述符包装出一个可用的File值。

域名函数的功能是打开一个已经存在的文件。但是,我们只能通过它返回的File值对相应的文件进行读操作。

域名File是这些函数中最为灵活的一个,通过它,我们可以设定被打开文件的操作模式和权限模式。实际上,域名te函数和域名函数都只是对它的简单封装而已。

在使用域名File函数的时候,我们必须要搞清楚操作模式和权限模式所代表的真正含义,以及设定它们的正确方式。

我在本文的扩展问题中分别对它们进行了较为详细的解释。同时,我在对应的示例文件中也编写了一些代码。

你需要认真地阅读和理解这些代码,并在运行它们的过程当中悟出这两种模式的真谛。

我在本文中讲述的东西对于os包来说,只是海面上的那部分冰山而已。这个代码包囊括的知识众多,而且延展性都很强。

如果你想完全理解它们,可能还需要去参看操作系统等方面的文档和教程。由于篇幅原因,我在这里只是做了一个引导,帮助你初识该包中的一些重要的程序实体,并给予你一个可以深入下去的切入点,希望你已经在路上了。

思考题

今天的思考题是:怎样通过os包中的 API 创建和操纵一个系统进程?

笔记源码

https://域名/MingsonZheng/go-core-demo

知识共享许可协议

本作品采用知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议进行许可。

欢迎转载、使用、重新发布,但务必保留文章署名 郑子铭 (包含链接: http://域名/MingsonZheng/ ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。

标签:编程
湘ICP备14001474号-3  投诉建议:234161800@qq.com   部分内容来源于网络,如有侵权,请联系删除。