判断奇数应优先用 (n & 1) != 0,因其无符号歧义、补码下恒成立;若用 % 需注意负数行为及 INT_MIN 边界问题,且现代编译器已将 % 2 优化为位运算,性能差异可忽略。

用 % 判断奇偶数最直接,但负数会出问题
在 C++ 里,% 是取模运算符,不是数学意义上的“取余”,对负数的处理依赖于编译器实现(C++98/03 不强制规定符号),C++11 起才统一为“向零截断”,但实际行为仍可能因平台或优化级别微调。所以 n % 2 == 1 看似简单,遇到 n = -3 时可能返回 -1,导致判断失败。
实操建议:
- 判断奇数优先用
(n & 1) != 0—— 位运算无符号歧义,整数补码下恒成立 - 若必须用
%,写成abs(n) % 2 == 1(注意abs对INT_MIN未定义行为,慎用于边界值) - 对无符号类型(如
unsigned int)可放心用n % 2,因为无负值
% 和 & 性能差多少?编译器早优化掉了
很多人以为 & 1 比 % 2 快很多,其实现代编译器(GCC/Clang/MSVC)在 -O2 及以上都会把 n % 2 自动转成位运算,生成的汇编几乎一样。性能不是选它的理由,语义清晰性和负数鲁棒性才是。
验证方式:写个简单函数,用 g++ -S -O2 看汇编输出,你会发现两者都变成 testl , %eax 或类似指令。
立即学习“C++免费学习笔记(深入)”;
- 别为这点“优化”牺牲可读性,除非你在裸金属或极端嵌入式环境手写汇编
- 如果代码要跑在老编译器(比如 VC6)上,
& 1更稳妥 -
%在非 2 的幂时无法被优化成位运算,比如% 3就真得算
模板函数封装奇偶判断时,要注意类型退化
写个通用函数如 template<typename t> bool is_odd(T n)</typename> 很常见,但容易忽略:当传入 char 或 short 时,它们会先整型提升为 int,而 unsigned char 提升为 unsigned int —— 这会导致 & 1 依然安全,但若内部用了 static_cast<int>(n) % 2</int>,就可能把原本的无符号语义搞丢。
- 统一用
static_cast<long long>(n) & 1</long>避免提升歧义(long long足够宽,且位运算是定义良好的) - 避免在模板里对
n做有符号/无符号混用的算术操作 - 如果只处理非负数,加个
static_assert(std::is_unsigned_v<t> || n >= 0, "...")</t>提前报错更友好
输入流读入后立刻判断奇偶,别忘了检查 failbit
新手常写 cin >> n; if (n % 2) ...,但如果用户输了个字母,n 保持原值(甚至未初始化),后续判断完全不可靠。
- 务必检查流状态:
if (cin >> n && !cin.fail()) - 或者用
cin >> n ? is_odd(n) : false这类短路表达式 - 读入失败后记得
cin.clear()和cin.ignore(...)清缓冲区,否则下次读还卡住










