0

0

如何用C++创建临时文件 tmpnam安全替代方案与使用示例

P粉602998670

P粉602998670

发布时间:2025-07-20 10:03:02

|

1032人浏览过

|

来源于php中文网

原创

不推荐使用 tmpnam 的原因是存在安全漏洞,容易引发竞态条件。1. tmpfile 可自动管理临时文件生命周期,安全性高但无法获取文件路径;2. mkstemp 通过修改模板字符串生成唯一文件名并返回已打开的文件描述符,避免竞态条件;3. windows 平台可使用 gettempfilename 创建临时文件并在使用后手动删除;4. 跨平台项目建议封装不同实现并采用 raii 管理资源,及时清理临时文件以避免安全隐患。

如何用C++创建临时文件 tmpnam安全替代方案与使用示例

在C++中创建临时文件,很多人第一时间想到的是 tmpnam 函数。但其实这个函数已经被标记为不推荐使用(deprecated),主要原因是它存在安全漏洞,容易引发竞态条件,导致程序被恶意利用。

如何用C++创建临时文件 tmpnam安全替代方案与使用示例

如果你希望安全、可靠地创建临时文件,可以考虑使用标准库提供的替代方案或系统调用接口。下面介绍几种更安全的替代方法及使用示例。


为什么不要用 tmpnam?

tmpnam 的问题在于它只是生成一个“看起来安全”的临时文件名,并不实际创建文件。这就带来了一个隐患:如果两个线程或者两个进程几乎同时调用了这个函数,它们可能会生成相同的文件名。此时如果其中一个程序删除了另一个刚创建的临时文件,就可能造成数据丢失甚至安全问题。

立即学习C++免费学习笔记(深入)”;

如何用C++创建临时文件 tmpnam安全替代方案与使用示例

此外,攻击者还可以通过预测生成的文件名来提前创建同名文件,从而实施符号链接攻击等手段。


安全替代方案一:使用 tmpfile

如果你不需要保留临时文件的内容,只需要一个临时的缓冲区用于读写,那么最简单的方式就是使用 tmpfile 函数:

如何用C++创建临时文件 tmpnam安全替代方案与使用示例
#include 

int main() {
    FILE* tmp = tmpfile();
    if (tmp == nullptr) {
        // 处理错误
        return 1;
    }

    fprintf(tmp, "这是临时内容");
    rewind(tmp);

    char buffer[100];
    fgets(buffer, sizeof(buffer), tmp);
    puts(buffer);

    fclose(tmp);  // 文件会自动删除
}

优点:

红墨
红墨

一站式小红书图文生成器

下载
  • 系统自动管理生命周期
  • 不需要手动清理
  • 安全性高

缺点:

  • 无法获取文件路径,不适合需要共享文件的场景

安全替代方案二:使用 mkstemp

如果你想获得一个临时文件的路径,并且保证其安全性,可以使用 mkstemp 函数。它是 POSIX 标准的一部分,在 Linux 和 macOS 上都可以使用。

使用方式如下:

#include 
#include 
#include 
#include 

int main() {
    char templatePath[] = "/tmp/mytempXXXXXX";
    int fd = mkstemp(templatePath);

    if (fd == -1) {
        std::cerr << "创建失败\n";
        return 1;
    }

    // 可以用 fd 操作文件,也可以转换为 FILE*
    FILE* fp = fdopen(fd, "w+");
    if (fp) {
        fprintf(fp, "写入一些内容");
        fclose(fp);
    }

    // 用完记得删除文件(可选)
    unlink(templatePath);
}

关键点:

  • 模板字符串最后必须是六个 X
  • mkstemp 会修改模板字符串的内容,填入随机字符
  • 返回的文件描述符是已打开的,避免了竞态条件

Windows 平台可用方案:GetTempFileName

如果你是在 Windows 平台上开发,可以使用 Win32 API 提供的 GetTempFileName 函数:

#include 
#include 

int main() {
    char path[MAX_PATH];
    DWORD result = GetTempFileNameA(".", "myprefix", 0, path);

    if (result == 0) {
        std::cerr << "创建失败\n";
        return 1;
    }

    // 使用完成后删除文件
    DeleteFileA(path);
}

说明:

  • 第一个参数是目录前缀(. 表示当前目录)
  • 第二个参数是前缀字符串
  • 第三个为非零时会在返回后立即删除文件
  • 最后一个参数用来接收生成的路径

跨平台建议与注意事项

  • 如果你的项目需要跨平台兼容性,可以封装不同平台的实现逻辑。
  • 使用 RAII 或智能指针管理临时文件资源是个好习惯。
  • 创建完临时文件后,应尽快使用并及时删除,避免占用磁盘空间或留下安全隐患。
  • 在多线程或多进程环境中,尤其要小心临时文件的命名和访问权限控制。

基本上就这些。tmpnam 虽然方便,但确实有安全缺陷。换用 mkstemptmpfile 是更好的选择,特别是在服务端或涉及用户输入的程序中。

相关专题

更多
js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

248

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

205

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1435

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

609

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

547

2024.03.22

php中定义字符串的方式
php中定义字符串的方式

php中定义字符串的方式:单引号;双引号;heredoc语法等等。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

539

2024.04.29

go语言字符串相关教程
go语言字符串相关教程

本专题整合了go语言字符串相关教程,阅读专题下面的文章了解更多详细内容。

158

2025.07.29

c++字符串相关教程
c++字符串相关教程

本专题整合了c++字符串相关教程,阅读专题下面的文章了解更多详细内容。

77

2025.08.07

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

7

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 6.3万人学习

Git 教程
Git 教程

共21课时 | 2.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号