这是一个有趣的。我正在编写AES加密算法,并设法让它进行准确的加密。当我尝试将结果写入文件时出现问题。我收到的文件输出不正确。十六进制值会被破坏,它通常是荒谬的(即使是加密标准)。
我通过在将加密输出发送到文件之前对其进行采样来进行一些调试。我发现我在某处遇到了某种类型的溢出。当正确的十六进制值应该是9e时,我会得到ffffff9e。它只会对高于7F的十六进制值执行此操作,即“扩展”字符集中的字符未正确处理。这在我的项目中也发生在我的早期,然后问题是使用char [] []容器而不是unsigned char [] []容器。
我的代码使用字符串在用户界面和AES加密类之间传递加密数据。我猜测std :: strings不支持扩展字符集。所以我的问题是:有没有办法实例化一个无符号字符串,还是我必须找到一种方法来替换我对字符串的所有使用?
答案 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;
};
}
另请注意,模板的不同实例化是完全独立的类型,即使它们被实例化的类型是相关的,因此您可以在char
和unsigned 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}得到那个。