前缀++i先自增后取值,后缀i++先取值后自增;同一表达式多次修改同一变量会导致逻辑混乱且可读性差,应避免。
Java里 ++ 和 -- 有前缀和后缀两种形式,行为完全不同:前缀(如 ++i)先修改再取值,后缀(如 i++)先取值再修改。这个差异在赋值、函数调用或复杂表达式中会立刻暴露。
常见错误现象:int i = 5; int j = i++ + ++i; 结果不是 11 或 12,而是 12 —— 因为执行顺序是:取 i 原值 5 → i 变成 6 → ++i 把 i 变成 7 并取 7 → 最终 j = 5 + 7。
i++ 或 ++i 效果一样,但习惯上用 i++;性能上无差别,现代 JVM 都会优化掉冗余操作if (i++ ,i 总是多加一次,后续逻辑可能错位
++ 和 -- 的优先级高于算术运算符(如 +、-),但低于括号和点号。这意味着它们会先于相邻的加减乘除被计算,但不会“跳过”整个子表达式。
例如:int a = 2, b = 3; int x = a++ + b * 2; 等价于 x = (a++) + (b * 2),不是 (a++ + b) * 2。这里 a++ 先取 2,再让 a 变成 3;b * 2 是 6;最终 x = 2 + 6 = 8。
++ 和 -- 与一元 +/- 优先级相同,所以 -i++ 解析为 -(i++),不是 (-i)++(后者非法)arr[i++] = arr[++i] 是未定义行为倾向 —— i 被修改两次且无序列点,Java 虽规范明确,但极难推理for 循环头里写 i++ 或 ++i 完全等效,因为递增动作本身不参与循环条件或主体的值计算;但在 while 或 do-while 中,若把自增放在条件里(如 while (arr[i++] != null)),就要格外注意边界和越界风险。
ArrayIndexOutOfBoundsException
i++,否则可能跳过下一轮迭代,或造成无限循环编译后,i++ 和 ++i 都会生成多条字节码指令:加载变量、复制栈顶值、执行加法、存回变量。区别只在是否保留原始值用于后续计算。可以用 javap -c 查看验证。
例如 int j = i++; 会生成 iload_1(载入 i)、dup(复制)、iinc 1 1(i 加 1)、istore_2(j 存 旧值);而 int j = ++i; 是 iload_1、iinc 1 1、iload_1、is(j 存 新值)。
Integer i = 0; i++;),每次都会新建对象i++ 不是原子操作(读-改-写三步),必须用 AtomicInteger 或同步块保护最常被忽略的一点:自增/自减只对变量有效,不能作用于字面量、表达式或 final 字段。写 5++、(a + b)++ 或 final int x = 1; x++ 都会在编译时报错 unexpected type 或 cannot assign a value to final variable。