7

基本上,我有一个程序,它有一个 4 meg 的压缩文件,它必须将此文件解码为未压缩的 ~ 100 meg,然后将其压缩回 ~4 meg 文件。我需要将这个 100 兆的中间文件存储在驱动器的某个位置(不想将它保存在内存中)。

程序是用 C 语言编写的,将在 MS Windows 7 上执行。在解压缩时,不会为程序提供保证文件夹(具有写访问权限)(包含源文件的文件夹可能是只读的,而包含目标文件的文件夹可能不是尚未指定)。

事实证明,这不是一件容易的事:

1) 我读过一个 C 函数,该函数创建一个临时文件,该文件在关闭或程序终止时会消失。但是,据我了解,它会尝试在磁盘 C 上的根目录中创建文件,因此如果用户对此没有权限(普通用户没有权限),这显然会失败

2)我有一个想法使用环境/系统变量 TEMP 并在那里创建一个文件,但是在没有调整的随机 Win7 PC 上查看,我看到这个变量指向 c:/windows/temp,并且该文件夹具有特定权限对于“用户”——也就是说,他们有权读取、执行、创建和写入文件,但不能删除它们、检查它们的属性等。这意味着,我假设,如果程序以用户权限运行,它将能够创建文件但无法删除它,因此“删除”它的唯一方法是打开文件进行写入然后关闭它,使其成为一个长度为 0 的文件。这也是不希望的,我不知道如何从 C 中查询系统变量

3)所以,基本上,我现在唯一的想法是创建一个打开文件的函数:

  • 如果可能,尝试在输出目录中创建一个临时文件
  • 如果失败,尝试在输入目录中创建一个临时文件
  • 如果失败,尝试从系统变量在 TEMP 目录中创建一个临时文件
  • 如果失败,则尝试从系统变量在 TMP 目录中创建临时文件

和一个删除功能:

  • 尝试删除()文件(通过其存储在某处的名称)
  • 如果失败,它会尝试打开文件进行写入,然后关闭它,所以它变成一个 0 字节的文件

有更好的想法吗?

任何帮助表示赞赏,谢谢!

PS:程序不得使用任何外部库,如 MFC 之类的,只能使用内置的标准 C 函数

4

2 回答 2

6

GetTempPath

检索为临时文件指定的目录的路径。

GetTempFileName

为临时文件创建一个名称。如果生成了唯一的文件名,则创建一个空文件并释放它的句柄;否则,只会生成一个文件名。

这两个为您提供了获取临时文件的位置和名称的简单方法。

UPD:MSDN 上的代码示例:创建和使用临时文件

于 2012-03-15T09:52:55.417 回答
0
#include <windows.h>
#include <iostream>
#include <chrono>
#include <string>
#include <cstdio>
#include <chrono>

using namespace std;

int FileExists(string& filepath)
{
  DWORD dwAttrib = GetFileAttributes(filepath.c_str());
  return (dwAttrib != INVALID_FILE_ATTRIBUTES &&
         !(dwAttrib & FILE_ATTRIBUTE_DIRECTORY));
}

int GetTemporaryFilePath(
    string  filePrefix,
    string  fileExt,
    string& TmpFilePath /*return*/)
{
    if (fileExt[0] == '.')
        fileExt.erase(0,1);

    char TempPath[MAX_PATH] = { 0 };

    if (!GetTempPath(MAX_PATH, TempPath))
        return -1;

    uint16_t tickint = 0;

    while(1) {
        const int nowlen = 17; char nowstr[nowlen];
        const int ticklen = 5; char tickstr[ticklen];

        // Milliseconds since 1970
        auto ms = chrono::duration_cast<chrono::milliseconds>(
            chrono::system_clock::now().time_since_epoch()
        );
        __int64 nowint = ms.count();

        snprintf(nowstr,  nowlen,  "%016" "I64" "x", nowint);
        snprintf(tickstr, ticklen, "%04x", tickint);

        TmpFilePath = string(TempPath)
                    + filePrefix
                    + "."   + string(nowstr)
                    + "."   + string(tickstr)
                    + "."   + fileExt;

        if (!FileExists(TmpFilePath)) {
            //Touch File
            FILE* w = fopen(TmpFilePath.c_str(), "w");
            fclose(w);
            break;
        }
        tickint++;
     }

     return 0;
}

int main()
{
    string TmpFilePath;
    GetTemporaryFilePath("MyFile", ".txt", TmpFilePath);
    cout << "TmpFilePath: " << TmpFilePath << endl;
    return 0;
}
于 2017-09-20T16:58:18.977 回答