[C/C++]_[初级]_[关于编译时出现有符号-无符号不匹配的警告-sizeof使用注意事项]
Posted infoworld
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了[C/C++]_[初级]_[关于编译时出现有符号-无符号不匹配的警告-sizeof使用注意事项]相关的知识,希望对你有一定的参考价值。
场景
- 自从有了
C++11
的auto
关键字之后,声明变量经常使用它。之后有一次发现编译控制台大量输出有符号/无符号不匹配
的警告信息。怎么回事?
说明
-
这里的有符号和无符号指的就是整数的类型,无符号的整数无论怎么运算都不会出现负数。
-
计算机里存储无符号的类型是通过原码,也就是二进制格式存储。而有符号的正数是通过原码存储,负数是通过补码存储的。
-
问题就是出现了如下的一种错误, 在使用比较运算符
<
是操作数的类型不匹配,一个是有符号,一个是无符号。可能导致错误的比较结果。因为有无符号的等级更高些,有符号的操作数会被转换为无符号再进行比较。这里的警告就是提醒如果有负值的时候比较的结果可能就是错误的。
testcpp.cpp(55): warning C4018: “<”: 有符号/无符号不匹配
C
的sizeof()
函数返回的就是unsigned int
类型,而这个函数经常用在对数组求大小时。当我们使用auto
声明变量size
的时候,默认就是sizeof
的类型了。
int ns[] = 4,5,6,7,78,8 ;
auto size = sizeof(ns) / sizeof(int); //unsigned int
- 这时候如果使用
size
在循环里做减法希望得到负数的时候就会出问题,因为unsigned int
类型不会有负数。比如对数组倒序求解,i-0
时候等于4294967295
,这个循环就会进入死循环。
for (auto i = size - 1; i >= 0; --i)
cout << i << endl;
assert(i < size);
方案
- 最好的方法就是在求解
size
的时候不要使用auto
,使用int
类型让它隐式转换。
int size = sizeof(ns) / sizeof(int);
- 其他使用
unsigned int
类型的时候,如果和常量整数(默认是int
有符号)或变量进行运算时,都要转换为统一的类型再运算。
例子
#include <string.h>
#include <iostream>
#include <bitset>
#include <assert.h>
using namespace std;
void TestUnsignedInt()
unsigned char a = 1;
unsigned char b = 3;
unsigned char res = a - b; //00000001 + 11111101(-3的补码)
unsigned char ma = 255;
cout << (unsigned)ma << endl;
cout << (unsigned)res << endl; //254
cout << bitset<8>(res) << endl; //11111110
void TestUnsignedSizeofFunc()
int ns[] = 4,5,6,7,78,8 ;
// sizeof()返回值是unsigned int类型
// 方法1: 要把auto改为int,推荐,这样没有编译警告。
auto size = sizeof(ns) / sizeof(int); //unsigned int
cout << "++++++" << endl;
for (auto i = 0; i < size; i++)
cout << i << endl;
cout << "------" << endl;
// 当 0 -1 的时候等于 4294967295
// 方法2: 要把 auto 改为 int
for (auto i = size - 1; i >= 0; --i)
cout << i << endl;
assert(i < size);
int main()
TestUnsignedInt();
TestUnsignedSizeofFunc();
以上是关于[C/C++]_[初级]_[关于编译时出现有符号-无符号不匹配的警告-sizeof使用注意事项]的主要内容,如果未能解决你的问题,请参考以下文章
[C/C++]_[初级]_[关于编译时出现有符号-无符号不匹配的警告-sizeof使用注意事项]
[C/C++]_[初级]_[关于编译时出现有符号-无符号不匹配的警告-sizeof使用注意事项]
[C/C++]_[初级]_[关于把字符串string作为字节数组存储的注意点]
[C/C++]_[初级]_[static_cast,reinterpret_cast,dynimic_cast的使用场景和区别]