Gox语言中,1.06a版本以上,可以利用内置的github.com/topxeq/afero包来将内存中虚拟出一个文件系统。这样,如果内存足够大,完全可以利用它来提升一些文件IO处理的效率。
该包实现的文件系统操作和Go语言(Golang)的os标准包中的基本保持一致,并提供了一些附加方法以便提高代码效率。
下面是内存虚拟文件系统的一些常规操作,包括新建文件系统、新建或删除文件、复制文件(包括在虚拟文件系统和真实文件系统之间复制)、创建目录和子目录等。
// 创建一个内存文件系统对象
fs = github_topxeq_afero.NewMemFS()
// 在根目录下新建一个文件a.txt
f, err = fs.Create(`/a.txt`)
// 输出一下创建结果,f是文件对象,如果发生错误,err中包含错误对象
// plvsr用于连续输出多个对象信息
plvsr(f, err)
// 在该文件中写入字符串abc
f.WriteString("abc")
// 关闭文件(也可以用defer,但会在脚本最后才执行,所以不太适合)
f.Close()
// 以二进制方式从该文件中载入字节数组进行检验
rs, err := fs.LoadBytesFromFile(`/a.txt`)
// 输出获得的字节数组
plvsr(rs, err)
// 直接创建一个二级子目录(一级目录会自动创建)
fs.MkdirAll("/tmp/data", 0x666)
// 复制真实文件系统的某个文件到内存文件系统,-force参数表示如果文件存在则覆盖
err = fs.CopyFileFrom(`d:\tmpx\sp\sel1.txt`, fs.Join(`/tmp/data`, `sel1.txt`), "-force")
plvsr(err)
// 打开根目录
root, err = fs.Open(`/`)
plvsr(root, err)
// 读取目录内所有文件名的方法之一
plvsr(root.Readdirnames(-1))
// 直接从文件中载入字符串(注意一定是UTF-8编码的)
rs, err = fs.LoadStringFromFile(`/tmp/data/sel1.txt`)
plvsr(rs, err)
// 列出该目录下所有文件(不包含子目录名和子目录下的文件)
pl("ls files: %v", fs.Ls("/"))
// 列出该目录下所有文件(不包含所有子目录本身,但包含各级子目录下的文件)
pl("ls files recursively: %v", fs.Lsr("/"))
// 创建另一个子目录(如果没有上级目录会报错)
fs.Mkdir("/tmp/next", 0x666)
// 在内存文件系统中复制文件
err = fs.CopyFile(fs.Join(`/tmp/data`, `sel1.txt`), fs.Join(`/tmp/next`, `sel1bak.txt`))
// 如果有错误则退出程序执行
checkErrf("failed to copy file: %v", err)
// 再次列出内存文件系统所有文件以验证
// 按通配符列出符合条件的文件
dirs, err = fs.Glob("/tmp/*/*.txt")
plvsr(dirs, err)
// 删除一个文件
err = fs.RemoveFile(`/tmp/data/sel1.txt`)
// 验证删除是否成功
pl("ls files recursively: %v", fs.Lsr("/"))
// 从内存文件系统中复制文件到真实文件系统(没有-force参数则如果存在文件会报错)
err = fs.CopyFileTo(fs.Join(`/tmp/next`, `sel1bak.txt`), path_filepath.Join(`d:\tmpx\sp`, `testsel1.txt`))
// 如果有错误则退出程序执行
checkErrf("failed to copy file: %v", err)
代码中已经给出了详细的文档内注释,它的运行结果是:
C:\Users\Administrator>d:
D:\>cd tmpx
D:\tmpx>cd sp
D:\tmpx\sp>gox -gopath memfs.gox
&mem.File{at:0, readDirCount:0, closed:false, readOnly:false, fileData:(*mem.FileData)(0xc000068cc0)}
<nil>
[]byte{0x61, 0x62, 0x63}
<nil>
<nil>
&mem.File{at:0, readDirCount:0, closed:false, readOnly:true, fileData:(*mem.FileData)(0xc000068d20)}
<nil>
[]interface {}{[]string{"a.txt", "tmp"}, interface {}(nil)}
"60\r\n30\r\n10\r\n40\r\n15\r\n50\r\n20\r\n12\r\n16\r\n52\r\n13\r\n51\r\n62\r\n14\r\n11"
<nil>
ls files: [\a.txt]
ls files recursively: [\a.txt \tmp\data\sel1.txt]
[]string{"\\tmp\\data\\sel1.txt", "\\tmp\\next\\sel1bak.txt"}
<nil>
ls files recursively: [\a.txt \tmp\next\sel1bak.txt]
D:\tmpx\sp>
从代码中可以看出,整体操作还是比较方便的。
注意:
Gox语言是脱胎于Go语言(Golang)的开源脚本语言,解释执行,但相比Go语言更贴近高级语言,语法硬性限制也少一些;是一门偏向快速应用的语言,也可以说是一个集成工具;
-
Gox语言主要优势有三点:
- 第一,Gox语言本身只有一个可执行文件,绿色免配置,下载即可使用,无需安装Go语言环境,无需编译,非常适合快速制作原型以及云服务器上的远程开发;
- 第二,Gox中可以直接使用绝大多数Go语言标准库中的对象和方法函数,也内置了很多常用、优秀的第三方库,充分发挥Go语言多年积累的资源优势;
- 第三,与很多其他主流语言不同,Gox语言着力解决了GUI图形界面编程的问题,内置了基于Giu(imgui)、LCL、Sciter的三套图形界面编程库,直接可以进行快捷高效的图形界面开发(LCL、Sciter只需分别下载一个动态链接库文件,执行和分发时附带上即可),特别适合编写演示原型系统。
作为脚本语言,Gox语言性能肯定不如Go语言这样的编译型语言快,但由于Gox语言与Go语言的紧密联系,Gox语言编写的脚本可以很容易的改写成Go语言代码,编译执行后就可以发挥Go语言的速度优势了。因此,Gox语言也比较适合做初期的Go语言调试,还有一个更直接的方式是使用Gotx(在Gox官网上也有下载),这是使用完全和Go语言一样语法的解释器,可以理解成集成了Go语言标准库和不少第三方库的解释执行的Go语言,一样也不需要搭建Go语言环境。Gotx与Gox的区别在于,Gotx仍然遵循Go语言的文法,代码相对复杂一些,限制也多一些,但改写回Go语言准备编译执行时,基本上没有成本。
Gox的官网在这里,也可以在浏览器搜索引擎中直接搜索“gox语言”,Github页面在这里,在这里可以看到很多Gox语言的学习指南和实际应用实例。
有疑问加站长微信联系(非本文作者)