5

ファイルからデータを読み取るのに少し問題があります。wstring だけでなく、任意のサイズ (サイズはバイト単位) の生データのチャンクも読み取れるようにしたいと考えています。

std::wfstream stream(file.c_str());

std::wstring comType;
stream >> comType;

int comSize;
stream >> comSize;

char *comData = new char[comSize];
memset(comData, 0, comSize);
stream.read(comData, comSize); 
//error C2664 : 'std::basic_istream<_Elem,_Traits>::read' 
//            : cannot convert parameter 1 from 'char *' to 'wchar_t *'

おそらく、間違ったストリームを使用しているか、それらの線に沿った何かを使用しています。基本的に、wstring、データのサイズ (バイト数は任意) を読み取り、その後にそのバイト数のコンポーネント データを読みたいと考えています。明らかに、テンプレートは wchar_t を想定しているため、char を読み取ることはできません。

wchar_t を読み取ることはできますが、データが sizeof(wchar_t) によって整列されていることを確認する必要があります。そうしないと、ストリームが破損する可能性があります。シナリオは、データが 15 バイトの場合です。次のデータ チャンクを読み取れるようにするには、16 バイトを読み取り、不要なバイトをマスクし、ストリームを 15 バイトのオフセットまでシークする必要があります (可能な場合は、テンプレート化された wchar_t を使用しますか?)。

明らかに、私がやろうとしていることを達成するためのより良い方法があるはずです.

4

3 に答える 3

2

stream.read の問題は、wfstream で wchar_t を「文字単位」として使用することです。fstream を使用すると、「文字単位」として char が使用されます。

これは、ワイド文字を読みたい場合に機能します。

wchar_t *comData = new wchar_t[comSize];
stream.read(comData, comSize);

また、15 バイトのデータをワイド ストリームで読み取ることはできません。これは、最小単位が少なくとも 2 バイトであるため (以下を参照)、sizwof(wchar_t) * n のチャンクしか読み取ることができないためです。

しかし、アプリケーションの移植性を懸念している場合は、wchar_t の幅に関する標準がないため、wfstream/wchar_t は最善の解決策ではない可能性があります (たとえば、Windows では wchar_t は 16 ビットであり、多くの UNIX/Linux システムでは 32 ビットです)。

テキストをワイド文字として保存する際の 2 つ目の問題はエンディアンです。テキストの保存には UTF-8 を使用することをお勧めします。

于 2008-10-02T12:55:26.320 に答える
1

あなたの要件を考えると、wfstreamが道だとは思いません。次のコード スニペットのようなものを使用することを検討してください。

#include "stdafx.h"
#include <fstream>
#include <iostream>

int _tmain(int argc, _TCHAR* argv[])
{
    std::wstring str(L"hello");
    size_t size1 = str.length();
    char data[] = { 0x10, 0x20, 0x30 };
    size_t size2 = 3;

    FILE* output = NULL;
    if (_wfopen_s(&output, L"c:\\test.bin", L"wb") == 0) {
        fwrite(&size1, sizeof(size_t), 1, output);
        fwrite(str.c_str(), size1 * sizeof(wchar_t), 1, output);
        fwrite(&size2, sizeof(size_t), 1, output);
        fwrite(data, size2, 1, output);

        fclose(output);
    }

    FILE* input = NULL;
    if (_wfopen_s(&input, L"c:\\test.bin", L"rb") == 0) {
        fread(&size1, sizeof(size_t), 1, input);
        wchar_t* wstr = new wchar_t[size1 + 1];
        fread(wstr, size1 * sizeof(wchar_t), 1, input);
        std::wstring str(wstr, size1);
        delete[] wstr;
        fread(&size2, sizeof(size_t), 1, input);
        char* data1 = new char[size2];
        fread(data1, size2, 1, input);

        std::wcout << str.c_str() << std::endl;
        for (size_t i = 0; i < size2; ++i) {
            std::wcout << std::hex << "0x" << int(data1[i]) << std::endl;
        }

        delete[] data1;

        fclose(input);
    }

    return 0;
}

これは以下を出力します:

hello
0x10
0x20
0x30
于 2008-10-02T11:52:07.037 に答える
0
# ifdef UNICODE
#     define tfstream wfstream
# else
#     define tfstream fstream
# endif

tfstream fs( _T("filename.bin"), tfstream::binary );
byte buffer[1023];
fs.read( buffer, sizeof(buffer) )

_T("filename.bin") と tfstream は UI 表現だと思います。バッファと read() は DATA LOGIC 式です。wfstream は、バッファを wchar_t 型に制限してはなりません。UI は DATA LOGIC と混在させてはなりません! wfstream はここで間違ったことをします

于 2014-01-15T02:53:15.897 に答える