为什么?
举例:
#include <iostream>
#include <vector>
int main(){
std::vector<int> vec{1, 2, 3};
for(auto i = 0; i<vec.size(); i++){
std::cout<<vec[i]<<std::endl;
}
return 0;
}
编译结果:
<source>: In function 'int main()':
<source>:8:22: warning: comparison of integer expressions of different signedness: 'int' and 'std::vector<int>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
8 | for(auto i = 0; i<vec.size(); i++){
| ~^~~~~~~~~~~
或者,我们将for(auto i = 0; i<vec.size(); i++){
改成for(auto i = 0, s = vec.size(); i<s; i++){
,得到编译结果
<source>: In function 'int main()':
<source>:8:9: error: inconsistent deduction for 'auto': 'int' and then 'long unsigned int'
8 | for(auto i = 0, s = vec.size(); i<s; i++){
| ^~~~
<source>:8:38: warning: comparison of integer expressions of different signedness: 'int' and 'long unsigned int' [-Wsign-compare]
8 | for(auto i = 0, s = vec.size(); i<s; i++){
| ~^~
第一种编译结果只是类型不匹配,第二种编译结果是auto
自动推导出错。这给我们日常的编程工作带来一些不便。
怎么办?
这里借用cppreference的例子:
static_assert(std::is_same_v<decltype(0UZ), std::size_t>);
static_assert(std::is_same_v<decltype(0Z), std::make_signed_t<std::size_t>>);
对于有符号的std::size_t
,使用z
或者Z
作为后缀。
对于无符号的std::size_t
,使用z
、Z
和u
、U
的集合,即uz
、uZ
、Uz
或者UZ
。