替换字符串的二进制部分?

时间:2013-10-21 21:33:52

标签: c++ c++builder

我有一个包含一些二进制数据的字符串。 xml格式的字符串,所以在我要处理它之前,我需要转换base64格式的二进制数据。 我正在使用一个名为findXMLTag的函数,它将在给定包含它的xml标记的情况下找到数据的开头和结尾的位置。 现在我能够将该数据转换为base64,但是我在使用新的base64数据替换旧的二进制数据时遇到了问题。

问题是我不能使用任何类型的字符串,因为当它找到一个空字符时,它会将它视为字符串的终止点,但实际上因为我有二进制数据存储在字符串中然后那个空字符可以是我的二进制数据的一部分。

所以我想我正在寻找某种二元替代品,我无法弄清楚如何让它发挥作用。 提前感谢您的任何帮助。

这是我用来在xml字符串中找到数据的开头和结尾的代码。

std::vector<TForm1::Pair> TForm1::findXMLTag(char *XMLString, char* XMLTag, int XMLSize)
{
    void *found = XMLString;
    int XMLTagLen = strlen(XMLTag);
    std::vector<TForm1::Pair> result;
    TForm1::Pair pair;
    AnsiString XMLTagEnd = "</";
    XMLTagEnd += &XMLTag[1];

    while(found = memmem(XMLString, XMLSize - ((char*)found - XMLString), XMLTag, XMLTagLen))
    {
        if(found == NULL)
            return result;

        found = (char*)found + XMLTagLen;

        pair.start = int((char*)found - XMLString);

        found = memmem(found, XMLSize - ((char*)found - XMLString), XMLTagEnd.c_str(), XMLTagEnd.Length());

        pair.end = int((char*)found - XMLString);

        found = (char*)found + XMLTagEnd.Length();

        result.push_back(pair);
    }

    return result;
}

3 个答案:

答案 0 :(得分:2)

将C风格的答案翻译成C ++,我们留下一个单行,这是安全的(对于有效的索引),高效且可读:

std::string binary_replace(
    std::string const& bin, unsigned bin_start, unsigned bin_end,
    std::string const& replace_with
) {
    assert(bin_start < bin.size() and bin_end < bin.size());
    return bin.substr(0, bin_start) + replace_with + bin.substr(bin_end);
}

为此目的使用replace function可以使这更简单:

std::string binary_replace(
    std::string bin, unsigned bin_start, unsigned bin_end,
    std::string const& replace_with
) {
    assert(bin_start < bin.size() and bin_end < bin.size());
    return bin.replace(bin_start, bin_end - bin_start, replace_with);
}

(注意binreplace修改后,{{1}}按值传递。)

基本上可以直接替代C ++中的大多数C字符串函数 - 在这种情况下,请查看documentation of std::basic_string::substr

答案 1 :(得分:1)

这是一个可能对您有所帮助的自包含示例。请注意,没有错误或边界检查,它只是为了演示一个概念。

#include <iostream>
#include <vector>
#include <string>
using namespace std;

// stub for real base64_encode
std::string base64_encode(const string &data)
{
    return "FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF";
}

// search without terminating on NULL
size_t binary_find(const string &str, size_t offset, const string &s)
{
    for (size_t i=offset; i<str.length(); i++)
        if (str.compare(i, string::npos, s) == 0)
            return i;
    return string::npos;
}

int main()
{
    string tag = "<data>";
    string endtag = "</data>";
    string xml("<data>\0\0\0\0\0\0\0\0\0\0</data>", 23);
    size_t start = xml.find(tag) + tag.length();
    size_t end = binary_find(xml, start, endtag);
    string binary = xml.substr(start, end-start);
    string base64 = base64_encode(binary);
    xml.replace(start, end-start, base64);
    cout << xml << endl;
}

答案 2 :(得分:-1)

char *binary_replace(char *binString,  int _strlen, int binDataStart, int binDataEnd, char* replaceWith)
{
    char *buffer = (char*)malloc( (strlen(replaceWith)+(_strlen - (binDataEnd-binDataStart)))*sizeof(char) );

    memcpy(buffer, binString, binDataStart);

    strcat(buffer, replaceWith);

    memmove(buffer+binDataStart+strlen(replaceWith), binString+binDataEnd, _strlen - binDataEnd);

    return buffer;
}

我知道这不是c ++,但它解决了我的问题。