在C++中,当无符号数和有符号数进行减法运算时,通常有符号数会被转换为无符号数

一段程序引发的问题:

cpp 复制代码
 for (int i = 0; i <= s.length() - L; ++ i) {
       string str = s.substr(i, L);
        if (++ hash[str] == 2) {
            res.push_back(str);
        }
    }
}
bash 复制代码
terminate called after throwing an instance of 'std::out_of_range'
  what():  basic_string::substr: __pos (which is 2) > this->size() (which is 1)

在C++中,当无符号数和有符号数进行减法运算时,通常有符号数会被转换为无符号数,然后进行无符号算术运算,结果也是一个无符号数。这是由C++的整数提升和算术转换规则所决定的,被称为 "usual arithmetic conversions"。

例如:

cpp 复制代码
unsigned int a = 5;
int b = 10;
auto c = a - b;

在这个例子中,b 被转换为一个相应的无符号数,然后再与 a 相减。如果 b 的值足够大以至于在转换为无符号数时不会导致负数,那么得到的结果是正常的。但是如果 b 较大(例如在本例中的10),转换结果会是一个非常大的无符号数,因为无符号整数不能表示负数。这个转换是按照模运算的方式来处理的,即加上或减去足够大的2的幂次,使结果适合无符号类型的表示范围。

回到上面的代码段中,s.length() 是一个 size_t 类型,即无符号类型。当与 L 这个有符号整数做减法时,L 会被转换为 size_t 类型,然后执行减法操作,结果也是 size_t 类型。如果 s.length() 小于 L,则会发生上述的整数环绕(underflow),导致计算出一个非常大的无符号数,从而导致循环条件错误地评估为 true

相关推荐
秋92 分钟前
学霸圈公认的 10 种高效学习习惯:从低效到顶尖的底层逻辑
人工智能·学习·算法
并不喜欢吃鱼14 分钟前
从零开始C++----四.vector的使用与底层实现
开发语言·c++
极简车辆控制16 分钟前
泵控式电液主动悬架系统分层控制研究_论文复现
算法·汽车
沐雪轻挽萤24 分钟前
17. C++17新特性-并行算法 (Parallel Algorithms)
java·开发语言·c++
A7bert77731 分钟前
【YOLOv8部署至RDK X5】模型训练→转换bin→Sunrise 5部署
c++·人工智能·python·深度学习·yolo·机器学习
扶苏xw38 分钟前
【分组背包】
算法·动态规划
李兆龙的博客42 分钟前
从一到无穷大 #68 Agent Memory 全景:大模型智能体记忆机制的形态、动态与前沿
大数据·人工智能·算法
cwplh1 小时前
平衡树学习笔记
数据结构·笔记·学习·算法
wen__xvn1 小时前
天梯赛L2刷题(也就写写水题骗骗自己了)
算法
EllinY1 小时前
扩展欧几里得算法 exgcd 详解
c++·笔记·数学·算法·exgcd