簡體   English   中英

如何在win32中使用GetSaveFileName保存文件?

[英]how to save file with GetSaveFileName in win32?

我寫這個代碼來獲取fileName來保存我的文件:

#include "stdafx.h"
#include <windows.h>


int _tmain(int argc, _TCHAR* argv[])
{            
    OPENFILENAME ofn;

    char szFileName[MAX_PATH] = "";

    ZeroMemory(&ofn, sizeof(ofn));

    ofn.lStructSize = sizeof(ofn); 
    ofn.hwndOwner = NULL;
    ofn.lpstrFilter = (LPCWSTR)L"Text Files (*.txt)\0*.txt\0All Files (*.*)\0*.*\0";
    ofn.lpstrFile = (LPWSTR)szFileName;
    ofn.nMaxFile = MAX_PATH;
    ofn.Flags = OFN_EXPLORER | OFN_FILEMUSTEXIST | OFN_HIDEREADONLY;
    ofn.lpstrDefExt = (LPCWSTR)L"txt";

    GetSaveFileName(&ofn);
    printf("the path is : %s\n", ofn.lpstrFile);
    getchar();
    return 0;
}

但輸出是:

 the path is : H 

為什么? 難道我做錯了什么 ?
我在Windows 7上使用Visual Studio 2008。

根本問題在於:

char szFileName[MAX_PATH] = "";
...
ofn.lpstrFile = (LPWSTR)szFileName;
ofn.nMaxFile = MAX_PATH;

這將創建一個MAX_PATH字符的緩沖區,但它告訴GetSaveFileName函數它是MAX_PATH 字符的緩沖區。 當有人選擇長路徑名時,這很可能會崩潰(或者無聲地踐踏內存)。

贈品是演員。 不要欺騙編譯器或庫。 他們不喜歡這樣,他們最終會報復。 用這個替換這些行:

WCHAR szFileName[MAX_PATH] = L"";
...
ofn.lpstrFile = szFileName;  // no cast needed
ofn.nMaxFile = MAX_PATH;

現在,所選文件名將作為寬字符串返回。 Tony The Lion的答案是正確的,因為您需要使用wprintf而不是printf來打印寬字符串:

wprintf(L"the path is : %s\n", ofn.lpstrFile);  // though I'd use szFileName at this point

如果您需要8位字符而不是寬字符的字符串,則可以使用WideCharToMultiByte。 但我總是堅持使用寬字符API。

除非你確切知道它的作用以及為什么在你的特定情況下它是必要的,否則永遠不要施放。

這一行:

printf("the path is : %s\n", ofn.lpstrFile);

應該使用printf的寬字符版本。

wprintf(L"the path is : %s\n", ofn.lpstrFile);

你錯了,這是一個簡單的C指針/堆棧問題。

// WRONG:
char szFileName[MAX_PATH] = "";

這會混淆數組和指針,您在堆棧上聲明了一個數組,但隨后將其內存地址更改為指向數據部分中的空字符串。 換句話說,緩沖區溢出。

// RIGHT:
char szFileName[MAX_PATH];
ZeroMemory(szFileName, MAX_PATH);

這在堆棧上聲明了一個字符數組,並將所有元素初始化為null終止符。

希望有所幫助!

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM