无符号字符的字符串

时间:2013-11-29 23:34:25

标签: c++ string encryption

这是一个有趣的。我正在编写AES加密算法,并设法让它进行准确的加密。当我尝试将结果写入文件时出现问题。我收到的文件输出不正确。十六进制值会被破坏,它通常是荒谬的(即使是加密标准)。

我通过在将加密输出发送到文件之前对其进行采样来进行一些调试。我发现我在某处遇到了某种类型的溢出。当正确的十六进制值应该是9e时,我会得到ffffff9e。它只会对高于7F的十六进制值执行此操作,即“扩展”字符集中的字符未正确处理。这在我的项目中也发生在我的早期,然后问题是使用char [] []容器而不是unsigned char [] []容器。

我的代码使用字符串在用户界面和AES加密类之间传递加密数据。我猜测std :: strings不支持扩展字符集。所以我的问题是:有没有办法实例化一个无符号字符串,还是我必须找到一种方法来替换我对字符串的所有使用?

3 个答案:

答案 0 :(得分:13)

std::string实际上只是一个typedef,类似于:

namespace std { 
   typedef basic_string<char> string;
}

unsigned char

创建变体相当容易
typedef basic_string<unsigned char> ustring;

但是,您必须更改代码以使用ustring(或您喜欢的任何名称)而不是std::string

根据您编写代码的方式,可能不需要编辑所有代码。特别是,如果你有类似的东西:

namespace crypto { 
   using std::string;

   class AES { 
      string data;
      // ..
    };
}

您可以通过仅更改using声明来更改字符串类型:

namespace unsigned_types { 
    typedef std::basic_string<unsigned char> string;
}

// ...

namespace crypto {
    using unsigned_types::string;

    class AES {
        string data;
    };
}

另请注意,模板的不同实例化是完全独立的类型,即使它们被实例化的类型是相关的,因此您可以在charunsigned char之间隐式转换这一事实这意味着您将在basic_string<char>basic_string<unsigned char>之间获得匹配的隐式转换。

答案 1 :(得分:2)

std::string仅仅是std::basic_string<>模板的特化,所以你可以简单地做一个

typedef std::basic_string<unsigned char> ustring;

得到你想要的东西。


请注意,C / C ++标准没有定义char是有符号还是无符号变体,因此任何直接将char强制转换为更大类型的程序都会调用实现定义的行为。

答案 2 :(得分:1)

首先将您的值转换为unsigned char

char input = 250;                                    // just an example

unsigned int n = static_cast<unsigned char>(input);  // NOT: "unsigned int n = input;"
//               ^^^^^^^^^^^^^^^^^^^^^^^^^^

问题是您的char恰好是签名,因此其值不是您想要的“字节值” - 您必须转换为{{1}得到那个。