类文件

文件对象是底层平台中文件的表示形式。

类文件扩展模块 FileTest,支持诸如 File.exist? 等单例方法。

关于示例

此处许多示例使用这些变量

# English text with newlines.
text = <<~EOT
  First line
  Second line

  Fourth line
  Fifth line
EOT

# Russian text.
russian = "\u{442 435 441 442}" # => "тест"

# Binary data.
data = "\u9990\u9991\u9992\u9993\u9994"

# Text file.
File.write('t.txt', text)

# File with Russian text.
File.write('t.rus', russian)

# File with binary data.
f = File.new('t.dat', 'wb:UTF-16')
f.write(data)
f.close

访问模式

方法 File.newFile.open 为给定的文件路径创建文件对象。

字符串访问模式

方法 File.newFile.open 都可以采用字符串参数 mode,它

读/写模式

读/写 mode 确定

这些表格总结

Read/Write Modes for Existing File

|------|-----------|----------|----------|----------|-----------|
| R/W  | Initial   |          | Initial  |          | Initial   |
| Mode | Truncate? |  Read    | Read Pos |  Write   | Write Pos |
|------|-----------|----------|----------|----------|-----------|
| 'r'  |    No     | Anywhere |    0     |   Error  |     -     |
| 'w'  |    Yes    |   Error  |    -     | Anywhere |     0     |
| 'a'  |    No     |   Error  |    -     | End only |    End    |
| 'r+' |    No     | Anywhere |    0     | Anywhere |     0     |
| 'w+' |    Yes    | Anywhere |    0     | Anywhere |     0     |
| 'a+' |    No     | Anywhere |   End    | End only |    End    |
|------|-----------|----------|----------|----------|-----------|

Read/Write Modes for \File To Be Created

|------|----------|----------|----------|-----------|
| R/W  |          | Initial  |          | Initial   |
| Mode |  Read    | Read Pos |  Write   | Write Pos |
|------|----------|----------|----------|-----------|
| 'w'  |   Error  |    -     | Anywhere |     0     |
| 'a'  |   Error  |    -     | End only |     0     |
| 'w+' | Anywhere |    0     | Anywhere |     0     |
| 'a+' | Anywhere |    0     | End only |    End    |
|------|----------|----------|----------|-----------|

请注意,对于不存在的文件,不允许模式 'r''r+'(引发异常)。

在表格中

现有文件的读/写模式
要创建的文件的读/写模式

请注意,对于不存在的文件,不允许模式 'r''r+'(引发异常)。

数据模式

要指定数据是作为文本数据还是作为二进制数据处理,可以在上述任何字符串读/写模式后缀以下内容

如果两者都没有给出,则流默认为文本数据。

示例

File.new('t.txt', 'rt')
File.new('t.dat', 'rb')

当指定数据模式时,不能省略读/写模式,并且数据模式必须在文件创建模式(如果给出)之前

File.new('t.dat', 'b')   # Raises an exception.
File.new('t.dat', 'rxb') # Raises an exception.

文件创建模式

可以将以下内容后缀到上述任何可写字符串模式

示例

File.new('t.tmp', 'wx')

当指定文件创建模式时,不能省略读/写模式,并且文件创建模式必须在数据模式之后

File.new('t.dat', 'x')   # Raises an exception.
File.new('t.dat', 'rxb') # Raises an exception.

整数访问模式

当模式为整数时,它必须是以下常量中的一个或多个,这些常量可以通过按位 OR 运算符 | 组合

示例

File.new('t.txt', File::RDONLY)
File.new('t.tmp', File::RDWR | File::CREAT | File::EXCL)

注意:Method IO#set_encoding 不允许将模式指定为整数。

以整数指定的文件创建模式

这些常量也可以按位或到整数模式中

以整数指定的数据模式

数据模式不能指定为整数。当流访问模式给定为整数时,数据模式始终为文本,从不为二进制。

请注意,虽然有一个常量 File::BINARY,但在整数流模式中设置其值没有任何效果;这是因为,如 File::Constants 中所述,File::BINARY 值禁用行代码转换,但不会更改外部编码。

编码

上述任何字符串模式都可以指定编码 - 仅外部编码或外部和内部编码 - 通过附加一个或两个编码名称,用冒号分隔

f = File.new('t.dat', 'rb')
f.external_encoding # => #<Encoding:ASCII-8BIT>
f.internal_encoding # => nil
f = File.new('t.dat', 'rb:UTF-16')
f.external_encoding # => #<Encoding:UTF-16 (dummy)>
f.internal_encoding # => nil
f = File.new('t.dat', 'rb:UTF-16:UTF-16')
f.external_encoding # => #<Encoding:UTF-16 (dummy)>
f.internal_encoding # => #<Encoding:UTF-16>
f.close

数组 Encoding.name_list 中提供了许多编码名称

Encoding.name_list.take(3) # => ["ASCII-8BIT", "UTF-8", "US-ASCII"]

设置外部编码后,读取的字符串在读取时会标记为该编码,而写入的字符串在写入时会转换为该编码。

当外部和内部编码都设置时,读取的字符串会从外部编码转换为内部编码,而写入的字符串会从内部编码转换为外部编码。有关转码输入和输出的更多详细信息,请参阅 编码

如果外部编码是 'BOM|UTF-8''BOM|UTF-16LE''BOM|UTF16-BE',Ruby 会检查输入文档中的 Unicode BOM 以帮助确定编码。对于 UTF-16 编码,文件打开模式必须为二进制。如果找到 BOM,则会将其剥离,并使用 BOM 中的外部编码。

请注意,BOM 样式编码选项不区分大小写,因此 'bom|utf-8' 也是有效的。

文件权限

File 对象具有权限,这是一个八进制整数,表示底层平台中实际文件的权限。

请注意,文件权限与文件流(文件对象)的模式截然不同。

在文件对象中,权限可用,其中方法 mode 返回权限,尽管其名称如此

f = File.new('t.txt')
f.lstat.mode.to_s(8) # => "100644"

在基于 Unix 的操作系统上,三个低阶八进制数字表示所有者 (6)、组 (4) 和世界 (4) 的权限。每个八进制数字中的三位分别表示读取、写入和执行权限。

因此,权限 0644 表示所有者具有读写访问权限,而组和世界具有只读访问权限。请参阅手册页 open(2)chmod(2)

对于目录,执行位的含义发生了变化:设置时,可以搜索目录。

权限中的高阶位可能指示文件类型(普通、目录、管道、套接字等)和其他各种特殊功能。

在非 Posix 操作系统上,权限可能仅包括只读或读写,在这种情况下,其余权限将类似于典型值。例如,在 Windows 上,默认权限为 0644;唯一可以进行的更改是将文件设为只读,报告为 0444

对于在底层平台中实际创建文件的某个方法(而不是仅仅创建文件对象),可以指定权限

File.new('t.tmp', File::CREAT, 0644)
File.new('t.tmp', File::CREAT, 0444)

也可以更改权限

f = File.new('t.tmp', File::CREAT, 0444)
f.chmod(0644)
f.chmod(0444)

文件常量

可以在模块 File::Constants 中找到用于文件和 IO 方法的各种常量;File::Constants.constants 返回其名称的数组。

此处的内容

首先,其他地方的内容。类文件

此处,类文件提供对以下内容有用的方法

创建

查询

路径

时间

类型

内容

设置

其他

常量

ALT_SEPARATOR

特定于平台的备用分隔符

PATH_SEPARATOR

路径列表分隔符

SEPARATOR

分隔路径中的目录部分

分隔符

分隔路径中的目录部分

公共类方法

absolute_path(file_name [, dir_string] ) → abs_file_name 单击以切换源

将路径名转换为绝对路径名。相对路径从进程的当前工作目录引用,除非给定 dir_string,在这种情况下,它将用作起始点。如果给定的路径名以“~”开头,则不会展开它,而是将其视为普通目录名。

File.absolute_path("~oracle/bin")       #=> "<relative_path>/~oracle/bin"
static VALUE
s_absolute_path(int c, const VALUE * v, VALUE _)
{
    return rb_file_s_absolute_path(c, v);
}
absolute_path?(file_name) → true 或 false 单击以切换源

如果 file_name 是绝对路径,则返回 true,否则返回 false

File.absolute_path?("c:/foo")     #=> false (on Linux), true (on Windows)
static VALUE
s_absolute_path_p(VALUE klass, VALUE fname)
{
    VALUE path = rb_get_path(fname);

    if (!rb_is_absolute_path(RSTRING_PTR(path))) return Qfalse;
    return Qtrue;
}
atime(file_name) → time 点击切换源

返回指定文件作为 Time 对象的最后访问时间。

file_name 可以是 IO 对象。

File.atime("testfile")   #=> Wed Apr 09 08:51:48 CDT 2003
static VALUE
rb_file_s_atime(VALUE klass, VALUE fname)
{
    struct stat st;

    if (rb_stat(fname, &st) < 0) {
        int e = errno;
        FilePathValue(fname);
        rb_syserr_fail_path(e, fname);
    }
    return stat_atime(&st);
}
basename(file_name [, suffix] ) → base_name 点击切换源

返回 file_name 中给定的文件名(首先去除尾部分隔符之后)的最后一个组件,可以使用 File::SEPARATORFile::ALT_SEPARATOR 作为分隔符,当 File::ALT_SEPARATOR 不为 nil 时。如果给定 suffix 且出现在 file_name 的末尾,则将其移除。如果 suffix 为 “.*”,则将移除任何扩展名。

File.basename("/home/gumby/work/ruby.rb")          #=> "ruby.rb"
File.basename("/home/gumby/work/ruby.rb", ".rb")   #=> "ruby"
File.basename("/home/gumby/work/ruby.rb", ".*")    #=> "ruby"
static VALUE
rb_file_s_basename(int argc, VALUE *argv, VALUE _)
{
    VALUE fname, fext, basename;
    const char *name, *p;
    long f, n;
    rb_encoding *enc;

    fext = Qnil;
    if (rb_check_arity(argc, 1, 2) == 2) {
        fext = argv[1];
        StringValue(fext);
        enc = check_path_encoding(fext);
    }
    fname = argv[0];
    FilePathStringValue(fname);
    if (NIL_P(fext) || !(enc = rb_enc_compatible(fname, fext))) {
        enc = rb_enc_get(fname);
        fext = Qnil;
    }
    if ((n = RSTRING_LEN(fname)) == 0 || !*(name = RSTRING_PTR(fname)))
        return rb_str_new_shared(fname);

    p = ruby_enc_find_basename(name, &f, &n, enc);
    if (n >= 0) {
        if (NIL_P(fext)) {
            f = n;
        }
        else {
            const char *fp;
            fp = StringValueCStr(fext);
            if (!(f = rmext(p, f, n, fp, RSTRING_LEN(fext), enc))) {
                f = n;
            }
            RB_GC_GUARD(fext);
        }
        if (f == RSTRING_LEN(fname)) return rb_str_new_shared(fname);
    }

    basename = rb_str_new(p, f);
    rb_enc_copy(basename, fname);
    return basename;
}
birthtime(file_name) → time 点击切换源

返回指定文件的创建时间。

file_name 可以是 IO 对象。

File.birthtime("testfile")   #=> Wed Apr 09 08:53:13 CDT 2003

如果平台没有创建时间,则引发 NotImplementedError

RUBY_FUNC_EXPORTED VALUE
rb_file_s_birthtime(VALUE klass, VALUE fname)
{
    statx_data st;

    if (rb_statx(fname, &st, STATX_BTIME) < 0) {
        int e = errno;
        FilePathValue(fname);
        rb_syserr_fail_path(e, fname);
    }
    return statx_birthtime(&st, fname);
}
blockdev?(filepath) → true or false 点击切换源

如果 filepath 指向块设备,则返回 true,否则返回 false

File.blockdev?('/dev/sda1')       # => true
File.blockdev?(File.new('t.tmp')) # => false
static VALUE
rb_file_blockdev_p(VALUE obj, VALUE fname)
{
#ifndef S_ISBLK
#   ifdef S_IFBLK
#       define S_ISBLK(m) (((m) & S_IFMT) == S_IFBLK)
#   else
#       define S_ISBLK(m) (0)  /* anytime false */
#   endif
#endif

#ifdef S_ISBLK
    struct stat st;

    if (rb_stat(fname, &st) < 0) return Qfalse;
    if (S_ISBLK(st.st_mode)) return Qtrue;

#endif
    return Qfalse;
}
chardev?(filepath) → true or false 点击切换源

如果 filepath 指向字符设备,则返回 true,否则返回 false

File.chardev?($stdin)     # => true
File.chardev?('t.txt')     # => false
static VALUE
rb_file_chardev_p(VALUE obj, VALUE fname)
{
#ifndef S_ISCHR
#   define S_ISCHR(m) (((m) & S_IFMT) == S_IFCHR)
#endif

    struct stat st;

    if (rb_stat(fname, &st) < 0) return Qfalse;
    if (S_ISCHR(st.st_mode)) return Qtrue;

    return Qfalse;
}
chmod(mode_int, file_name, ... ) → integer 点击切换源

将指定文件上的权限位更改为由 mode_int 表示的位模式。实际效果取决于操作系统(请参阅本节开头)。在 Unix 系统上,请参阅 chmod(2) 了解更多详情。返回处理的文件数。

File.chmod(0644, "testfile", "out")   #=> 2
static VALUE
rb_file_s_chmod(int argc, VALUE *argv, VALUE _)
{
    mode_t mode;

    apply2args(1);
    mode = NUM2MODET(*argv++);

    return apply2files(chmod_internal, argc, argv, &mode);
}
chown(owner_int, group_int, file_name, ...) → integer 点击切换源

将指定文件的所有者和组更改为给定的数字所有者和组 ID。只有具有超级用户权限的进程才能更改文件的拥有者。文件的当前所有者可以将文件组更改为所有者所属的任何组。忽略nil或 -1 所有者或组 ID。返回已处理的文件数。

File.chown(nil, 100, "testfile")
static VALUE
rb_file_s_chown(int argc, VALUE *argv, VALUE _)
{
    struct chown_args arg;

    apply2args(2);
    arg.owner = to_uid(*argv++);
    arg.group = to_gid(*argv++);

    return apply2files(chown_internal, argc, argv, &arg);
}
ctime(file_name) → time 点击切换源代码

返回指定文件的更改时间(目录信息更改的时间,而不是文件本身)。

file_name 可以是 IO 对象。

请注意,在 Windows(NTFS)上,返回创建时间(出生时间)。

File.ctime("testfile")   #=> Wed Apr 09 08:53:13 CDT 2003
static VALUE
rb_file_s_ctime(VALUE klass, VALUE fname)
{
    struct stat st;

    if (rb_stat(fname, &st) < 0) {
        int e = errno;
        FilePathValue(fname);
        rb_syserr_fail_path(e, fname);
    }
    return stat_ctime(&st);
}
delete(file_name, ...) → integer 点击切换源代码
unlink(file_name, ...) → integer

删除指定的文件,返回作为参数传递的名称数。引发任何错误的异常。由于底层实现依赖于unlink(2)系统调用,因此引发的异常类型取决于其错误类型(参见linux.die.net/man/2/unlink),并且具有 e.g. Errno::ENOENT 的形式。

另请参见Dir::rmdir

static VALUE
rb_file_s_unlink(int argc, VALUE *argv, VALUE klass)
{
    return apply2files(unlink_internal, argc, argv, 0);
}
directory?(path) → true 或 false 点击切换源代码

给定字符串object,如果path是通往目录或指向目录的符号链接的字符串路径,则返回true;否则返回false

File.directory?('.')              # => true
File.directory?('foo')            # => false
File.symlink('.', 'dirlink')      # => 0
File.directory?('dirlink')        # => true
File.symlink('t,txt', 'filelink') # => 0
File.directory?('filelink')       # => false

参数path可以是IO对象。

VALUE
rb_file_directory_p(VALUE obj, VALUE fname)
{
#ifndef S_ISDIR
#   define S_ISDIR(m) (((m) & S_IFMT) == S_IFDIR)
#endif

    struct stat st;

    if (rb_stat(fname, &st) < 0) return Qfalse;
    if (S_ISDIR(st.st_mode)) return Qtrue;
    return Qfalse;
}
dirname(file_name, level = 1) → dir_name 点击切换源代码

返回file_name中给定文件名除了最后一个之外的所有组件(首先去掉尾部分隔符)。当File::ALT_SEPARATOR不为nil时,可以使用File::SEPARATORFile::ALT_SEPARATOR作为分隔符来形成文件名。

File.dirname("/home/gumby/work/ruby.rb")   #=> "/home/gumby/work"

如果给定了level,则不仅删除一个组件,还要删除最后level个组件。

File.dirname("/home/gumby/work/ruby.rb", 2) #=> "/home/gumby"
File.dirname("/home/gumby/work/ruby.rb", 4) #=> "/"
static VALUE
rb_file_s_dirname(int argc, VALUE *argv, VALUE klass)
{
    int n = 1;
    if ((argc = rb_check_arity(argc, 1, 2)) > 1) {
        n = NUM2INT(argv[1]);
    }
    return rb_file_dirname_n(argv[0], n);
}
zero?(file_name) → true 或 false 点击切换源代码

如果指定的文件存在且大小为零,则返回true

file_name 可以是 IO 对象。

static VALUE
rb_file_zero_p(VALUE obj, VALUE fname)
{
    struct stat st;

    if (rb_stat(fname, &st) < 0) return Qfalse;
    return RBOOL(st.st_size == 0);
}
executable?(file_name) → true 或 false 点击切换源代码

如果指定的文件可由该进程的有效用户和组 ID 执行,则返回true。参见 eaccess(3)。

Windows 不支持单独于读取权限的执行权限。在 Windows 上,只有当文件以 .bat、.cmd、.com 或 .exe 结尾时,才认为该文件可执行。

请注意,即使文件不可由有效用户/组执行,某些操作系统级别的安全功能也可能导致此函数返回 true。

static VALUE
rb_file_executable_p(VALUE obj, VALUE fname)
{
    return RBOOL(rb_eaccess(fname, X_OK) >= 0);
}
executable_real?(file_name) → true 或 false 点击切换源代码

如果指定的文件可由该进程的实际用户和组 ID 执行,则返回true。参见 access(3)。

Windows 不支持单独于读取权限的执行权限。在 Windows 上,只有当文件以 .bat、.cmd、.com 或 .exe 结尾时,才认为该文件可执行。

请注意,即使文件不可由实际用户/组执行,某些操作系统级别的安全功能也可能导致此函数返回 true。

static VALUE
rb_file_executable_real_p(VALUE obj, VALUE fname)
{
    return RBOOL(rb_access(fname, X_OK) >= 0);
}
exist?(file_name) → true 或 false 点击切换源

如果指定的文件存在,则返回 true

file_name 可以是 IO 对象。

“文件存在”表示 stat() 或 fstat() 系统调用成功。

static VALUE
rb_file_exist_p(VALUE obj, VALUE fname)
{
    struct stat st;

    if (rb_stat(fname, &st) < 0) return Qfalse;
    return Qtrue;
}
expand_path(file_name [, dir_string] ) → abs_file_name 点击切换源

将路径名转换为绝对路径名。除非给定 dir_string,否则相对路径将从进程的当前工作目录引用,在这种情况下,它将用作起始点。给定的路径名可以以“~”开头,它扩展为进程所有者的主目录(环境变量 HOME 必须设置正确)。“~user”扩展为指定用户的家目录。

File.expand_path("~oracle/bin")           #=> "/home/oracle/bin"

以下是一个使用 dir_string 的简单示例。

File.expand_path("ruby", "/usr/bin")      #=> "/usr/bin/ruby"

以下是一个更复杂的示例,它还解析父目录。假设我们在 bin/mygem 中,并且想要 lib/mygem.rb 的绝对路径。

File.expand_path("../../lib/mygem.rb", __FILE__)
#=> ".../path/to/project/lib/mygem.rb"

因此,它首先解析 __FILE__ 的父目录,即 bin/,然后转到父目录,即项目的根目录,并追加 lib/mygem.rb

static VALUE
s_expand_path(int c, const VALUE * v, VALUE _)
{
    return rb_file_s_expand_path(c, v);
}
extname(path) → string 点击切换源

返回扩展名(从最后一个句点开始的 path 中的文件名部分)。

如果 path 是一个点文件,或以句点开头,则起始点不会作为扩展名的开头处理。

当句点是 path 中的最后一个字符时,也将返回一个空字符串。

在 Windows 上,尾随的点将被截断。

File.extname("test.rb")         #=> ".rb"
File.extname("a/b/d/test.rb")   #=> ".rb"
File.extname(".a/b/d/test.rb")  #=> ".rb"
File.extname("foo.")            #=> "" on Windows
File.extname("foo.")            #=> "." on non-Windows
File.extname("test")            #=> ""
File.extname(".profile")        #=> ""
File.extname(".profile.sh")     #=> ".sh"
static VALUE
rb_file_s_extname(VALUE klass, VALUE fname)
{
    const char *name, *e;
    long len;
    VALUE extname;

    FilePathStringValue(fname);
    name = StringValueCStr(fname);
    len = RSTRING_LEN(fname);
    e = ruby_enc_find_extname(name, &len, rb_enc_get(fname));
    if (len < 1)
        return rb_str_new(0, 0);
    extname = rb_str_subseq(fname, e - name, len); /* keep the dot, too! */
    return extname;
}
file?(file) → true 或 false 点击切换源

如果指定 file 存在并且是一个常规文件,则返回 true

file 可以是一个 IO 对象。

如果 file 参数是一个符号链接,它将解析符号链接并使用链接引用的文件。

static VALUE
rb_file_file_p(VALUE obj, VALUE fname)
{
    struct stat st;

    if (rb_stat(fname, &st) < 0) return Qfalse;
    return RBOOL(S_ISREG(st.st_mode));
}
fnmatch( pattern, path, [flags] ) → (true 或 false) 点击切换源

如果pathpattern匹配,则返回true。该模式不是正则表达式;相反,它遵循类似于shell文件名扩展的规则。它可能包含以下元字符

*

匹配任何文件。可以通过扩展中的其他值进行限制。等效于regexp中的/.*/x

*

匹配所有常规文件

c*

匹配所有以c开头的文件

*c

匹配所有以c结尾的文件

*c*

匹配所有包含c的文件(包括开头或结尾)。

要匹配隐藏文件(以.开头),请设置File::FNM_DOTMATCH标志。

**

递归匹配目录或扩展匹配文件。

?

匹配任何一个字符。等效于regexp中的/.{1}/

[set]

匹配set中的任何一个字符。行为完全像Regexp中的字符集,包括集否定([^a-z])。

\

转义下一个元字符。

{a,b}

如果启用了File::FNM_EXTGLOB标志,则匹配模式a和模式b。行为像Regexp并集((?:a|b))。

flagsFNM_XXX常量的按位OR。相同的扩展模式和标志由Dir::glob使用。

示例

File.fnmatch('cat',       'cat')        #=> true  # match entire string
File.fnmatch('cat',       'category')   #=> false # only match partial string

File.fnmatch('c{at,ub}s', 'cats')                    #=> false # { } isn't supported by default
File.fnmatch('c{at,ub}s', 'cats', File::FNM_EXTGLOB) #=> true  # { } is supported on FNM_EXTGLOB

File.fnmatch('c?t',     'cat')          #=> true  # '?' match only 1 character
File.fnmatch('c??t',    'cat')          #=> false # ditto
File.fnmatch('c*',      'cats')         #=> true  # '*' match 0 or more characters
File.fnmatch('c*t',     'c/a/b/t')      #=> true  # ditto
File.fnmatch('ca[a-z]', 'cat')          #=> true  # inclusive bracket expression
File.fnmatch('ca[^t]',  'cat')          #=> false # exclusive bracket expression ('^' or '!')

File.fnmatch('cat', 'CAT')                     #=> false # case sensitive
File.fnmatch('cat', 'CAT', File::FNM_CASEFOLD) #=> true  # case insensitive
File.fnmatch('cat', 'CAT', File::FNM_SYSCASE)  #=> true or false # depends on the system default

File.fnmatch('?',   '/', File::FNM_PATHNAME)  #=> false # wildcard doesn't match '/' on FNM_PATHNAME
File.fnmatch('*',   '/', File::FNM_PATHNAME)  #=> false # ditto
File.fnmatch('[/]', '/', File::FNM_PATHNAME)  #=> false # ditto

File.fnmatch('\?',   '?')                       #=> true  # escaped wildcard becomes ordinary
File.fnmatch('\a',   'a')                       #=> true  # escaped ordinary remains ordinary
File.fnmatch('\a',   '\a', File::FNM_NOESCAPE)  #=> true  # FNM_NOESCAPE makes '\' ordinary
File.fnmatch('[\?]', '?')                       #=> true  # can escape inside bracket expression

File.fnmatch('*',   '.profile')                      #=> false # wildcard doesn't match leading
File.fnmatch('*',   '.profile', File::FNM_DOTMATCH)  #=> true  # period by default.
File.fnmatch('.*',  '.profile')                      #=> true

File.fnmatch('**/*.rb', 'main.rb')                  #=> false
File.fnmatch('**/*.rb', './main.rb')                #=> false
File.fnmatch('**/*.rb', 'lib/song.rb')              #=> true
File.fnmatch('**.rb', 'main.rb')                    #=> true
File.fnmatch('**.rb', './main.rb')                  #=> false
File.fnmatch('**.rb', 'lib/song.rb')                #=> true
File.fnmatch('*',     'dave/.profile')              #=> true

File.fnmatch('**/foo', 'a/b/c/foo', File::FNM_PATHNAME)     #=> true
File.fnmatch('**/foo', '/a/b/c/foo', File::FNM_PATHNAME)    #=> true
File.fnmatch('**/foo', 'c:/a/b/c/foo', File::FNM_PATHNAME)  #=> true
File.fnmatch('**/foo', 'a/.b/c/foo', File::FNM_PATHNAME)    #=> false
File.fnmatch('**/foo', 'a/.b/c/foo', File::FNM_PATHNAME | File::FNM_DOTMATCH) #=> true
# File dir.rb, line 502
def fnmatch(pattern, path, flags = 0)
end
别名:fnmatch?
fnmatch?( pattern, path, [flags] ) → (true or false)
别名:fnmatch
ftype(file_name) → string 单击以切换源

标识命名文件类型;返回字符串是“file”、“directory”、“characterSpecial”、“blockSpecial”、“fifo”、“link”、“socket”或“unknown”之一。

File.ftype("testfile")            #=> "file"
File.ftype("/dev/tty")            #=> "characterSpecial"
File.ftype("/tmp/.X11-unix/X0")   #=> "socket"
static VALUE
rb_file_s_ftype(VALUE klass, VALUE fname)
{
    struct stat st;

    FilePathValue(fname);
    fname = rb_str_encode_ospath(fname);
    if (lstat_without_gvl(StringValueCStr(fname), &st) == -1) {
        rb_sys_fail_path(fname);
    }

    return rb_file_ftype(&st);
}
grpowned?(file_name) → true or false 单击以切换源

如果命名文件存在并且调用进程的有效组ID是该文件的所有者,则返回true。在Windows上返回false

file_name 可以是 IO 对象。

static VALUE
rb_file_grpowned_p(VALUE obj, VALUE fname)
{
#ifndef _WIN32
    struct stat st;

    if (rb_stat(fname, &st) < 0) return Qfalse;
    if (rb_group_member(st.st_gid)) return Qtrue;
#endif
    return Qfalse;
}
identical?(file_1, file_2) → true or false 单击以切换源

如果命名文件相同,则返回true

file_1file_2可以是IO对象。

open("a", "w") {}
p File.identical?("a", "a")      #=> true
p File.identical?("a", "./a")    #=> true
File.link("a", "b")
p File.identical?("a", "b")      #=> true
File.symlink("a", "c")
p File.identical?("a", "c")      #=> true
open("d", "w") {}
p File.identical?("a", "d")      #=> false
static VALUE
rb_file_identical_p(VALUE obj, VALUE fname1, VALUE fname2)
{
#ifndef _WIN32
    struct stat st1, st2;

    if (rb_stat(fname1, &st1) < 0) return Qfalse;
    if (rb_stat(fname2, &st2) < 0) return Qfalse;
    if (st1.st_dev != st2.st_dev) return Qfalse;
    if (st1.st_ino != st2.st_ino) return Qfalse;
    return Qtrue;
#else
    extern VALUE rb_w32_file_identical_p(VALUE, VALUE);
    return rb_w32_file_identical_p(fname1, fname2);
#endif
}
join(string, ...) → string 单击以切换源

使用 "/" 连接字符串,返回一个新字符串。

File.join("usr", "mail", "gumby")   #=> "usr/mail/gumby"
static VALUE
rb_file_s_join(VALUE klass, VALUE args)
{
    return rb_file_join(args);
}
lchmod(mode_int, file_name, ...) → integer 单击以切换源代码

等同于 File::chmod,但不会跟随符号链接(因此它将更改与链接关联的权限,而不是链接引用的文件)。通常不可用。

static VALUE
rb_file_s_lchmod(int argc, VALUE *argv, VALUE _)
{
    mode_t mode;

    apply2args(1);
    mode = NUM2MODET(*argv++);

    return apply2files(lchmod_internal, argc, argv, &mode);
}
lchown(owner_int, group_int, file_name,..) → integer 单击以切换源代码

等同于 File::chown,但不会跟随符号链接(因此它将更改与链接关联的所有者,而不是链接引用的文件)。通常不可用。返回参数列表中的文件数。

static VALUE
rb_file_s_lchown(int argc, VALUE *argv, VALUE _)
{
    struct chown_args arg;

    apply2args(2);
    arg.owner = to_uid(*argv++);
    arg.group = to_gid(*argv++);

    return apply2files(lchown_internal, argc, argv, &arg);
}
lstat(filepath) → stat 单击以切换源代码

类似于 File::stat,但不会跟随最后一个符号链接;而是返回链接本身的 File::Stat 对象。

File.symlink('t.txt', 'symlink')
File.stat('symlink').size  # => 47
File.lstat('symlink').size # => 5
static VALUE
rb_file_s_lstat(VALUE klass, VALUE fname)
{
#ifdef HAVE_LSTAT
    struct stat st;

    FilePathValue(fname);
    fname = rb_str_encode_ospath(fname);
    if (lstat_without_gvl(StringValueCStr(fname), &st) == -1) {
        rb_sys_fail_path(fname);
    }
    return rb_stat_new(&st);
#else
    return rb_file_s_stat(klass, fname);
#endif
}
lutime(atime, mtime, file_name, ...) → integer 单击以切换源代码

将每个命名文件的访问和修改时间设置为前两个参数。如果文件是符号链接,则此方法作用于链接本身,而不是其引用的内容;有关相反的行为,请参见 File.utime。返回参数列表中的文件名数。

static VALUE
rb_file_s_lutime(int argc, VALUE *argv, VALUE _)
{
    return utime_internal_i(argc, argv, TRUE);
}
mkfifo(file_name, mode=0666) → 0 单击以切换源代码

使用名称 file_name 创建 FIFO 特殊文件。mode 指定 FIFO 的权限。它以通常的方式由进程的 umask 修改:创建的文件的权限为 (mode & ~umask)。

static VALUE
rb_file_s_mkfifo(int argc, VALUE *argv, VALUE _)
{
    VALUE path;
    struct mkfifo_arg ma;

    ma.mode = 0666;
    rb_check_arity(argc, 1, 2);
    if (argc > 1) {
        ma.mode = NUM2MODET(argv[1]);
    }
    path = argv[0];
    FilePathValue(path);
    path = rb_str_encode_ospath(path);
    ma.path = RSTRING_PTR(path);
    if (rb_thread_call_without_gvl(nogvl_mkfifo, &ma, RUBY_UBF_IO, 0)) {
        rb_sys_fail_path(path);
    }
    return INT2FIX(0);
}
mtime(file_name) → time 单击以切换源代码

将命名文件的修改时间作为 Time 对象返回。

file_name 可以是 IO 对象。

File.mtime("testfile")   #=> Tue Apr 08 12:58:04 CDT 2003
static VALUE
rb_file_s_mtime(VALUE klass, VALUE fname)
{
    struct stat st;

    if (rb_stat(fname, &st) < 0) {
        int e = errno;
        FilePathValue(fname);
        rb_syserr_fail_path(e, fname);
    }
    return stat_mtime(&st);
}
new(path, mode = 'r', perm = 0666, **opts) → file 单击以切换源代码

根据给定的 mode 打开给定 path 处的文件;为该文件创建并返回一个新的 File 对象。

除非 filename 是 tty,否则新的 File 对象为缓冲模式(或非同步模式)。请参见 IO#flushIO#fsyncIO#fdatasyncIO#sync=

参数 path 必须是有效的文件路径

f = File.new('/etc/fstab')
f.close
f = File.new('t.txt')
f.close

可选参数 mode(默认为“r”)必须指定有效模式;请参见 访问模式

f = File.new('t.tmp', 'w')
f.close
f = File.new('t.tmp', File::RDONLY)
f.close

可选参数 perm(默认为 0666)必须指定有效权限,请参见 文件权限

f = File.new('t.tmp', File::CREAT, 0644)
f.close
f = File.new('t.tmp', File::CREAT, 0444)
f.close

可选关键字参数 opts 指定

static VALUE
rb_file_initialize(int argc, VALUE *argv, VALUE io)
{
    if (RFILE(io)->fptr) {
        rb_raise(rb_eRuntimeError, "reinitializing File");
    }
    if (0 < argc && argc < 3) {
        VALUE fd = rb_check_to_int(argv[0]);

        if (!NIL_P(fd)) {
            argv[0] = fd;
            return rb_io_initialize(argc, argv, io);
        }
    }
    rb_open_file(argc, argv, io);

    return io;
}
open(path, mode = 'r', perm = 0666, **opts) → file 单击以切换源
open(path, mode = 'r', perm = 0666, **opts) {|f| ... } → object

通过 File.new 创建新的 File 对象,并使用给定的参数。

如果没有给定块,则返回 File 对象。

如果给定了块,则使用 File 对象调用该块,并返回该块的值。

static VALUE
rb_io_s_open(int argc, VALUE *argv, VALUE klass)
{
    VALUE io = rb_class_new_instance_kw(argc, argv, klass, RB_PASS_CALLED_KEYWORDS);

    if (rb_block_given_p()) {
        return rb_ensure(rb_yield, io, io_close, io);
    }

    return io;
}
owned?(file_name) → true or false 单击以切换源

如果指定的文件存在,并且调用进程的有效用户 ID 是该文件的所有者,则返回 true

file_name 可以是 IO 对象。

static VALUE
rb_file_owned_p(VALUE obj, VALUE fname)
{
    struct stat st;

    if (rb_stat(fname, &st) < 0) return Qfalse;
    return RBOOL(st.st_uid == geteuid());
}
path(path) → string 单击以切换源

返回路径的字符串表示形式

File.path(File::NULL)           #=> "/dev/null"
File.path(Pathname.new("/tmp")) #=> "/tmp"
static VALUE
rb_file_s_path(VALUE klass, VALUE fname)
{
    return rb_get_path(fname);
}
pipe?(filepath) → true or false 单击以切换源

如果 filepath 指向管道,则返回 true,否则返回 false

File.mkfifo('tmp/fifo')
File.pipe?('tmp/fifo') # => true
File.pipe?('t.txt')    # => false
static VALUE
rb_file_pipe_p(VALUE obj, VALUE fname)
{
#ifdef S_IFIFO
#  ifndef S_ISFIFO
#    define S_ISFIFO(m) (((m) & S_IFMT) == S_IFIFO)
#  endif

    struct stat st;

    if (rb_stat(fname, &st) < 0) return Qfalse;
    if (S_ISFIFO(st.st_mode)) return Qtrue;

#endif
    return Qfalse;
}
readable?(file_name) → true or false 单击以切换源

如果此进程的有效用户和组 ID 可以读取指定的文件,则返回 true。请参见 eaccess(3)。

请注意,即使有效用户/组无法读取该文件,某些操作系统级别的安全功能也可能导致此方法返回 true。

static VALUE
rb_file_readable_p(VALUE obj, VALUE fname)
{
    return RBOOL(rb_eaccess(fname, R_OK) >= 0);
}
readable_real?(file_name) → true or false 单击以切换源

如果此进程的实际用户和组 ID 可以读取指定的文件,则返回 true。请参见 access(3)。

请注意,即使实际用户/组无法读取该文件,某些操作系统级别的安全功能也可能导致此方法返回 true。

static VALUE
rb_file_readable_real_p(VALUE obj, VALUE fname)
{
    return RBOOL(rb_access(fname, R_OK) >= 0);
}
realdirpath(pathname [, dir_string]) → real_pathname 单击以切换源

返回实际文件系统中pathname的真实(绝对)路径名。真实路径名不包含符号链接或无用的点。

如果给定dir_string,则将其用作解释相对路径名的基本目录,而不是当前目录。

真实路径名的最后一个组件可能不存在。

static VALUE
rb_file_s_realdirpath(int argc, VALUE *argv, VALUE klass)
{
    VALUE basedir = (rb_check_arity(argc, 1, 2) > 1) ? argv[1] : Qnil;
    VALUE path = argv[0];
    FilePathValue(path);
    return rb_realpath_internal(basedir, path, 0);
}
realpath(pathname [, dir_string]) → real_pathname 单击以切换源

返回实际文件系统中pathname的真实(绝对)路径名,不包含符号链接或无用的点。

如果给定dir_string,则将其用作解释相对路径名的基本目录,而不是当前目录。

调用此方法时,路径名的所有组件都必须存在。

static VALUE
rb_file_s_realpath(int argc, VALUE *argv, VALUE klass)
{
    VALUE basedir = (rb_check_arity(argc, 1, 2) > 1) ? argv[1] : Qnil;
    VALUE path = argv[0];
    FilePathValue(path);
    return rb_realpath_internal(basedir, path, 1);
}
rename(old_name, new_name) → 0 单击以切换源

将给定文件重命名为新名称。如果无法重命名文件,则引发SystemCallError

File.rename("afile", "afile.bak")   #=> 0
static VALUE
rb_file_s_rename(VALUE klass, VALUE from, VALUE to)
{
    struct rename_args ra;
    VALUE f, t;

    FilePathValue(from);
    FilePathValue(to);
    f = rb_str_encode_ospath(from);
    t = rb_str_encode_ospath(to);
    ra.src = StringValueCStr(f);
    ra.dst = StringValueCStr(t);
#if defined __CYGWIN__
    errno = 0;
#endif
    if ((int)(VALUE)rb_thread_call_without_gvl(no_gvl_rename, &ra,
                                         RUBY_UBF_IO, 0) < 0) {
        int e = errno;
#if defined DOSISH
        switch (e) {
          case EEXIST:
            if (chmod(ra.dst, 0666) == 0 &&
                unlink(ra.dst) == 0 &&
                rename(ra.src, ra.dst) == 0)
                return INT2FIX(0);
        }
#endif
        syserr_fail2(e, from, to);
    }

    return INT2FIX(0);
}
setgid?(file_name) → true 或 false 单击以切换源

如果指定的文件设置了 setgid 位,则返回true

file_name 可以是 IO 对象。

static VALUE
rb_file_sgid_p(VALUE obj, VALUE fname)
{
#ifdef S_ISGID
    return check3rdbyte(fname, S_ISGID);
#else
    return Qfalse;
#endif
}
setuid?(file_name) → true 或 false 单击以切换源

如果指定的文件设置了 setuid 位,则返回true

file_name 可以是 IO 对象。

static VALUE
rb_file_suid_p(VALUE obj, VALUE fname)
{
#ifdef S_ISUID
    return check3rdbyte(fname, S_ISUID);
#else
    return Qfalse;
#endif
}
size(file_name) → integer 单击以切换源

返回file_name的大小。

file_name 可以是 IO 对象。

static VALUE
rb_file_s_size(VALUE klass, VALUE fname)
{
    struct stat st;

    if (rb_stat(fname, &st) < 0) {
        int e = errno;
        FilePathValue(fname);
        rb_syserr_fail_path(e, fname);
    }
    return OFFT2NUM(st.st_size);
}
size?(file_name) → Integer 或 nil 单击以切换源

如果file_name不存在或大小为零,则返回nil,否则返回文件的大小。

file_name 可以是 IO 对象。

static VALUE
rb_file_size_p(VALUE obj, VALUE fname)
{
    struct stat st;

    if (rb_stat(fname, &st) < 0) return Qnil;
    if (st.st_size == 0) return Qnil;
    return OFFT2NUM(st.st_size);
}
socket?(filepath) → true 或 false 单击以切换源

如果filepath指向套接字,则返回true,否则返回false

require 'socket'
File.socket?(Socket.new(:INET, :STREAM)) # => true
File.socket?(File.new('t.txt'))          # => false
static VALUE
rb_file_socket_p(VALUE obj, VALUE fname)
{
#ifndef S_ISSOCK
#  ifdef _S_ISSOCK
#    define S_ISSOCK(m) _S_ISSOCK(m)
#  else
#    ifdef _S_IFSOCK
#      define S_ISSOCK(m) (((m) & S_IFMT) == _S_IFSOCK)
#    else
#      ifdef S_IFSOCK
#        define S_ISSOCK(m) (((m) & S_IFMT) == S_IFSOCK)
#      endif
#    endif
#  endif
#endif

#ifdef S_ISSOCK
    struct stat st;

    if (rb_stat(fname, &st) < 0) return Qfalse;
    if (S_ISSOCK(st.st_mode)) return Qtrue;
#endif

    return Qfalse;
}
split(file_name) → array 单击以切换源

将给定的字符串拆分为目录和文件组件,并以一个包含两个元素的数组返回它们。另请参见 File::dirnameFile::basename

File.split("/home/gumby/.profile")   #=> ["/home/gumby", ".profile"]
static VALUE
rb_file_s_split(VALUE klass, VALUE path)
{
    FilePathStringValue(path);          /* get rid of converting twice */
    return rb_assoc_new(rb_file_dirname(path), rb_file_s_basename(1,&path,Qundef));
}
stat(filepath) → stat 单击以切换源

返回 filepath 处文件的 File::Stat 对象(请参见 File::Stat

File.stat('t.txt').class # => File::Stat
static VALUE
rb_file_s_stat(VALUE klass, VALUE fname)
{
    struct stat st;

    FilePathValue(fname);
    fname = rb_str_encode_ospath(fname);
    if (stat_without_gvl(RSTRING_PTR(fname), &st) < 0) {
        rb_sys_fail_path(fname);
    }
    return rb_stat_new(&st);
}
sticky?(file_name) → true 或 false 单击以切换源

如果指定文件设置了粘滞位,则返回 true

file_name 可以是 IO 对象。

static VALUE
rb_file_sticky_p(VALUE obj, VALUE fname)
{
#ifdef S_ISVTX
    return check3rdbyte(fname, S_ISVTX);
#else
    return Qfalse;
#endif
}
truncate(file_name, integer) → 0 单击以切换源

将文件 file_name 截断为最多 integer 个字节长。并非在所有平台上都可用。

f = File.new("out", "w")
f.write("1234567890")     #=> 10
f.close                   #=> nil
File.truncate("out", 5)   #=> 0
File.size("out")          #=> 5
static VALUE
rb_file_s_truncate(VALUE klass, VALUE path, VALUE len)
{
    struct truncate_arg ta;
    int r;

    ta.pos = NUM2OFFT(len);
    FilePathValue(path);
    path = rb_str_encode_ospath(path);
    ta.path = StringValueCStr(path);

    r = (int)(VALUE)rb_thread_call_without_gvl(nogvl_truncate, &ta,
                                                RUBY_UBF_IO, NULL);
    if (r < 0)
        rb_sys_fail_path(path);
    return INT2FIX(0);
}
umask() → integer 单击以切换源
umask(integer) → integer

返回此进程的当前 umask 值。如果给出了可选参数,则将 umask 设置为该值并返回前一个值。Umask 值从默认权限中减去,因此 umask 为 0222 将使每个人都无法读取文件。

File.umask(0006)   #=> 18
File.umask         #=> 6
static VALUE
rb_file_s_umask(int argc, VALUE *argv, VALUE _)
{
    mode_t omask = 0;

    switch (argc) {
      case 0:
        omask = umask(0);
        umask(omask);
        break;
      case 1:
        omask = umask(NUM2MODET(argv[0]));
        break;
      default:
        rb_error_arity(argc, 0, 1);
    }
    return MODET2NUM(omask);
}
utime(atime, mtime, file_name, ...) → integer 单击以切换源

将每个指定文件的访问时间和修改时间设置为前两个参数。如果文件是符号链接,则此方法对其引用的内容而不是链接本身执行操作;有关相反的行为,请参见 File.lutime。返回参数列表中的文件名数量。

static VALUE
rb_file_s_utime(int argc, VALUE *argv, VALUE _)
{
    return utime_internal_i(argc, argv, FALSE);
}
world_readable?(file_name) → 整数或 nil 点击切换源

如果其他人可读file_name,则返回一个表示file_name文件权限位的整数。否则返回nil。位含义取决于平台;在 Unix 系统上,请参见stat(2)

file_name 可以是 IO 对象。

File.world_readable?("/etc/passwd")           #=> 420
m = File.world_readable?("/etc/passwd")
sprintf("%o", m)                              #=> "644"
static VALUE
rb_file_world_readable_p(VALUE obj, VALUE fname)
{
#ifdef S_IROTH
    struct stat st;

    if (rb_stat(fname, &st) < 0) return Qnil;
    if ((st.st_mode & (S_IROTH)) == S_IROTH) {
        return UINT2NUM(st.st_mode & (S_IRUGO|S_IWUGO|S_IXUGO));
    }
#endif
    return Qnil;
}
world_writable?(file_name) → 整数或 nil 点击切换源

如果其他人可写file_name,则返回一个表示file_name文件权限位的整数。否则返回nil。位含义取决于平台;在 Unix 系统上,请参见stat(2)

file_name 可以是 IO 对象。

File.world_writable?("/tmp")                  #=> 511
m = File.world_writable?("/tmp")
sprintf("%o", m)                              #=> "777"
static VALUE
rb_file_world_writable_p(VALUE obj, VALUE fname)
{
#ifdef S_IWOTH
    struct stat st;

    if (rb_stat(fname, &st) < 0) return Qnil;
    if ((st.st_mode & (S_IWOTH)) == S_IWOTH) {
        return UINT2NUM(st.st_mode & (S_IRUGO|S_IWUGO|S_IXUGO));
    }
#endif
    return Qnil;
}
writable?(file_name) → true 或 false 点击切换源

如果此进程的有效用户和组 ID 可写指定文件,则返回true。请参见 eaccess(3)。

请注意,即使有效用户/组不可写文件,某些操作系统级安全功能也可能导致此方法返回 true。

static VALUE
rb_file_writable_p(VALUE obj, VALUE fname)
{
    return RBOOL(rb_eaccess(fname, W_OK) >= 0);
}
writable_real?(file_name) → true 或 false 点击切换源

如果此进程的实际用户和组 ID 可写指定文件,则返回true。请参见 access(3)。

请注意,即使实际用户/组不可写文件,某些操作系统级安全功能也可能导致此方法返回 true。

static VALUE
rb_file_writable_real_p(VALUE obj, VALUE fname)
{
    return RBOOL(rb_access(fname, W_OK) >= 0);
}
zero?(file_name) → true 或 false 点击切换源代码

如果指定的文件存在且大小为零,则返回true

file_name 可以是 IO 对象。

static VALUE
rb_file_zero_p(VALUE obj, VALUE fname)
{
    struct stat st;

    if (rb_stat(fname, &st) < 0) return Qfalse;
    return RBOOL(st.st_size == 0);
}

公有实例方法

atime → 时间 点击切换源

返回file的最后访问时间(Time 对象),如果file未被访问,则返回纪元。

File.new("testfile").atime   #=> Wed Dec 31 18:00:00 CST 1969
static VALUE
rb_file_atime(VALUE obj)
{
    rb_io_t *fptr;
    struct stat st;

    GetOpenFile(obj, fptr);
    if (fstat(fptr->fd, &st) == -1) {
        rb_sys_fail_path(fptr->pathv);
    }
    return stat_atime(&st);
}
birthtime → 时间 点击切换源

返回file的出生时间。

File.new("testfile").birthtime   #=> Wed Apr 09 08:53:14 CDT 2003

如果平台没有创建时间,则引发 NotImplementedError

static VALUE
rb_file_birthtime(VALUE obj)
{
    rb_io_t *fptr;
    statx_data st;

    GetOpenFile(obj, fptr);
    if (fstatx_without_gvl(fptr->fd, &st, STATX_BTIME) == -1) {
        rb_sys_fail_path(fptr->pathv);
    }
    return statx_birthtime(&st, fptr->pathv);
}
chmod(mode_int) → 0 点击切换源

file的权限位更改为mode_int表示的位模式。实际效果取决于平台;在 Unix 系统上,请参见chmod(2)了解详情。跟随符号链接。另请参见 File#lchmod。

f = File.new("out", "w");
f.chmod(0644)   #=> 0
static VALUE
rb_file_chmod(VALUE obj, VALUE vmode)
{
    rb_io_t *fptr;
    mode_t mode;
#if !defined HAVE_FCHMOD || !HAVE_FCHMOD
    VALUE path;
#endif

    mode = NUM2MODET(vmode);

    GetOpenFile(obj, fptr);
#ifdef HAVE_FCHMOD
    if (fchmod(fptr->fd, mode) == -1) {
        if (HAVE_FCHMOD || errno != ENOSYS)
            rb_sys_fail_path(fptr->pathv);
    }
    else {
        if (!HAVE_FCHMOD) return INT2FIX(0);
    }
#endif
#if !defined HAVE_FCHMOD || !HAVE_FCHMOD
    if (NIL_P(fptr->pathv)) return Qnil;
    path = rb_str_encode_ospath(fptr->pathv);
    if (chmod(RSTRING_PTR(path), mode) == -1)
        rb_sys_fail_path(fptr->pathv);
#endif

    return INT2FIX(0);
}
chown(owner_int, group_int ) → 0 点击切换源

file的所有者和组更改为给定的数字所有者和组 ID。只有具有超级用户权限的进程才能更改文件的拥有者。文件的当前所有者可以将文件的组更改为所有者所属的任何组。忽略nil或 -1 所有者或组 ID。跟随符号链接。另请参见 File#lchown。

File.new("testfile").chown(502, 1000)
static VALUE
rb_file_chown(VALUE obj, VALUE owner, VALUE group)
{
    rb_io_t *fptr;
    rb_uid_t o;
    rb_gid_t g;
#ifndef HAVE_FCHOWN
    VALUE path;
#endif

    o = to_uid(owner);
    g = to_gid(group);
    GetOpenFile(obj, fptr);
#ifndef HAVE_FCHOWN
    if (NIL_P(fptr->pathv)) return Qnil;
    path = rb_str_encode_ospath(fptr->pathv);
    if (chown(RSTRING_PTR(path), o, g) == -1)
        rb_sys_fail_path(fptr->pathv);
#else
    if (fchown(fptr->fd, o, g) == -1)
        rb_sys_fail_path(fptr->pathv);
#endif

    return INT2FIX(0);
}
ctime → 时间 点击切换源

返回file的更改时间(即,文件目录信息更改的时间,而不是文件本身)。

请注意,在 Windows(NTFS)上,返回创建时间(出生时间)。

File.new("testfile").ctime   #=> Wed Apr 09 08:53:14 CDT 2003
static VALUE
rb_file_ctime(VALUE obj)
{
    rb_io_t *fptr;
    struct stat st;

    GetOpenFile(obj, fptr);
    if (fstat(fptr->fd, &st) == -1) {
        rb_sys_fail_path(fptr->pathv);
    }
    return stat_ctime(&st);
}
flock(locking_constant) → 0 或 false 点击切换源代码

根据给定的 locking_constant(下表中值的按位 OR)锁定或解锁文件。

并非在所有平台上都可用。

如果指定了 File::LOCK_NB 且操作将被阻塞,则返回 false;否则返回 0

锁定常量
常量 锁定 效果
File::LOCK_EX 独占 一次只能有一个进程对 self 持有独占锁。
File::LOCK_NB 非阻塞 不阻塞;可以使用按位 OR 运算符 | 与其他 File::LOCK_SHFile::LOCK_EX 结合使用。
File::LOCK_SH 共享 多个进程可以同时对 self 持有共享锁。
File::LOCK_UN 解锁 移除此进程持有的现有锁。


示例

# Update a counter using an exclusive lock.
# Don't use File::WRONLY because it truncates the file.
File.open('counter', File::RDWR | File::CREAT, 0644) do |f|
  f.flock(File::LOCK_EX)
  value = f.read.to_i + 1
  f.rewind
  f.write("#{value}\n")
  f.flush
  f.truncate(f.pos)
end

# Read the counter using a shared lock.
File.open('counter', 'r') do |f|
  f.flock(File::LOCK_SH)
  f.read
end
static VALUE
rb_file_flock(VALUE obj, VALUE operation)
{
    rb_io_t *fptr;
    int op[2], op1;
    struct timeval time;

    op[1] = op1 = NUM2INT(operation);
    GetOpenFile(obj, fptr);
    op[0] = fptr->fd;

    if (fptr->mode & FMODE_WRITABLE) {
        rb_io_flush_raw(obj, 0);
    }
    while ((int)rb_thread_io_blocking_region(rb_thread_flock, op, fptr->fd) < 0) {
        int e = errno;
        switch (e) {
          case EAGAIN:
          case EACCES:
#if defined(EWOULDBLOCK) && EWOULDBLOCK != EAGAIN
          case EWOULDBLOCK:
#endif
            if (op1 & LOCK_NB) return Qfalse;

            time.tv_sec = 0;
            time.tv_usec = 100 * 1000;  /* 0.1 sec */
            rb_thread_wait_for(time);
            rb_io_check_closed(fptr);
            continue;

          case EINTR:
#if defined(ERESTART)
          case ERESTART:
#endif
            break;

          default:
            rb_syserr_fail_path(e, fptr->pathv);
        }
    }
    return INT2FIX(0);
}
lstat → stat 点击切换源代码

File#stat 类似,但不会跟随最后一个符号链接;而是返回链接本身的 File::Stat 对象

File.symlink('t.txt', 'symlink')
f = File.new('symlink')
f.stat.size  # => 47
f.lstat.size # => 11
static VALUE
rb_file_lstat(VALUE obj)
{
#ifdef HAVE_LSTAT
    rb_io_t *fptr;
    struct stat st;
    VALUE path;

    GetOpenFile(obj, fptr);
    if (NIL_P(fptr->pathv)) return Qnil;
    path = rb_str_encode_ospath(fptr->pathv);
    if (lstat_without_gvl(RSTRING_PTR(path), &st) == -1) {
        rb_sys_fail_path(fptr->pathv);
    }
    return rb_stat_new(&st);
#else
    return rb_io_stat(obj);
#endif
}
mtime → time 点击切换源代码

返回 file 的修改时间。

File.new("testfile").mtime   #=> Wed Apr 09 08:53:14 CDT 2003
static VALUE
rb_file_mtime(VALUE obj)
{
    rb_io_t *fptr;
    struct stat st;

    GetOpenFile(obj, fptr);
    if (fstat(fptr->fd, &st) == -1) {
        rb_sys_fail_path(fptr->pathv);
    }
    return stat_mtime(&st);
}
size → integer 点击切换源代码

以字节为单位返回 file 的大小。

File.new("testfile").size   #=> 66
static VALUE
file_size(VALUE self)
{
    return OFFT2NUM(rb_file_size(self));
}
truncate(integer) → 0 点击切换源代码

file 截断为最多 integer 个字节。文件必须打开以供写入。并非在所有平台上都可用。

f = File.new("out", "w")
f.syswrite("1234567890")   #=> 10
f.truncate(5)              #=> 0
f.close()                  #=> nil
File.size("out")           #=> 5
static VALUE
rb_file_truncate(VALUE obj, VALUE len)
{
    rb_io_t *fptr;
    struct ftruncate_arg fa;

    fa.pos = NUM2OFFT(len);
    GetOpenFile(obj, fptr);
    if (!(fptr->mode & FMODE_WRITABLE)) {
        rb_raise(rb_eIOError, "not opened for writing");
    }
    rb_io_flush_raw(obj, 0);
    fa.fd = fptr->fd;
    if ((int)rb_thread_io_blocking_region(nogvl_ftruncate, &fa, fa.fd) < 0) {
        rb_sys_fail_path(fptr->pathv);
    }
    return INT2FIX(0);
}