热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

从两个易错的笔试题深入理解自增运算符

目录自增运算符:例题一:例题二:【阿里巴巴-笔试】自增运算符:不是原子操作;前缀自增自减法(a,--

目录

自增运算符:

例题一:

例题二:【阿里巴巴-笔试】



自增运算符:

不是原子操作;

前缀自增自减法(++a,--a): 先进行自增或者自减运算,再进行表达式运算;

后缀自增自减法(a++,a--): 先进行表达式运算,再进行自增或者自减运算;

int i = 0; i = i++;

0: iconst_0

把int型的0值压入操作数栈

1: istore_1

将一个数值从操作数栈上存储到局部变量表中

2: iload_1

将一个局部变量加载到操作数栈上

3: iinc 1by 1

iinc是对int类型的值进行自增操作,后面第一个数值1表示,局部变量表的index值,说明要对此值执行iinc操作,第二个数值1表示要增加的数值。(这时局部变量表index为1的值因为执行了自增操作变为1了,但是操作数栈中栈顶的值仍然是0)

6: istore_1

将一个数值从操作数栈上存储到局部变量表中

7: iload_1

将一个局部变量加载到操作数栈上


从执行顺序可以看到,这里第1和第6执行了2次将0赋值给变量i的操作(=号赋值),i++操作是在这两次操作之间执行的,自增操作是对局部变量表中的值进行自增,而栈顶的值没有发生变化,这里需要注意的是保存这个初始值的地方是操作数栈而不是局部变量表,最后再将栈顶的值覆盖到局部变量表i所在的索引位置中去。

x = -1;

y = x++ + ++x;

x 执行 x++ ,进行加运算的时候x = -1 (先运算,后自增),此时 x = 0,++x 结果是1(先自增,后运算),变大时变成-1 + 1,所以 y = 0;

例题一:

检查程序,是否存在问题,如果存在指出问题所在,如果不存在,说明输出结果。

package algorithms.com.guan.javajicu;
public class Inc { public static void main(String[] args) { Inc inc = new Inc(); int i = 0; inc.fermin(i); i= i ++; System.out.println(i);} void fermin(int i){ i++; }
}

正确答案: 0

单独看容易让人迷惑的int i = 0; i = i++;这句代码,使用jclasslib查看字节码;

0: iconst_0

把int型的0值压入操作数栈

1: istore_1

将一个数值从操作数栈上存储到局部变量表中

2: iload_1

将一个局部变量加载到操作数栈上

3: iinc 1by 1

iinc是对int类型的值进行自增操作,后面第一个数值1表示,局部变量表的index值,说明要对此值执行iinc操作,第二个数值1表示要增加的数值。(这时局部变量表index为1的值因为执行了自增操作变为1了,但是操作数栈中栈顶的值仍然是0)

6: istore_1

将一个数值从操作数栈上存储到局部变量表中

7: iload_1

将一个局部变量加载到操作数栈上


这里第1和第6执行了2次将0赋值给变量i的操作(=号赋值),i++操作是在这两次操作之间执行的,自增操作是对局部变量表中的值进行自增,而栈顶的值没有发生变化,这里需要注意的是保存这个初始值的地方是操作数栈而不是局部变量表,最后再将栈顶的值覆盖到局部变量表i所在的索引位置中去。

而fermin方法中的i因为不是引用变量,所以不会改变原有的i;最终答案是0

例题二:【阿里巴巴-笔试】

以下代码的输出结果是?【阿里巴巴-笔试】

public class Test{
static{int x=5;
}
static int x,y;
public static void main(String args[]){x--;myMethod( );System.out.println(x+y+ ++x);
}
public static void myMethod( ){y=x++ + ++x;}
}

prints:3

静态语句块中x为局部变量,不影响静态变量x的值

x和y为静态变量,默认初始值为0,属于当前类,其值得改变会影响整个类运行

y = x++ + ++x;

x 执行 x++ ,进行加运算的时候x = -1 (先运算,后自增),此时 x = 0,++x 结果是1(先自增,后运算),变大时变成-1 + 1,所以 y = 0,x = 1;

然后x+y+ ++x 等于3;


推荐阅读
  • 1:有如下一段程序:packagea.b.c;publicclassTest{privatestaticinti0;publicintgetNext(){return ... [详细]
  • 本文介绍了Java并发库中的阻塞队列(BlockingQueue)及其典型应用场景。通过具体实例,展示了如何利用LinkedBlockingQueue实现线程间高效、安全的数据传递,并结合线程池和原子类优化性能。 ... [详细]
  • 本文介绍了如何通过 Maven 依赖引入 SQLiteJDBC 和 HikariCP 包,从而在 Java 应用中高效地连接和操作 SQLite 数据库。文章提供了详细的代码示例,并解释了每个步骤的实现细节。 ... [详细]
  • 本文介绍如何使用阿里云的fastjson库解析包含时间戳、IP地址和参数等信息的JSON格式文本,并进行数据处理和保存。 ... [详细]
  • andr ... [详细]
  • 本文提供了使用Java实现Bellman-Ford算法解决POJ 3259问题的代码示例,详细解释了如何通过该算法检测负权环来判断时间旅行的可能性。 ... [详细]
  • 本文详细探讨了JDBC(Java数据库连接)的内部机制,重点分析其作为服务提供者接口(SPI)框架的应用。通过类图和代码示例,展示了JDBC如何注册驱动程序、建立数据库连接以及执行SQL查询的过程。 ... [详细]
  • Java 中 Writer flush()方法,示例 ... [详细]
  • 本文介绍了如何使用 Spring Boot DevTools 实现应用程序在开发过程中自动重启。这一特性显著提高了开发效率,特别是在集成开发环境(IDE)中工作时,能够提供快速的反馈循环。默认情况下,DevTools 会监控类路径上的文件变化,并根据需要触发应用重启。 ... [详细]
  • Java 中的 BigDecimal pow()方法,示例 ... [详细]
  • Java 类成员初始化顺序与数组创建
    本文探讨了Java中类成员的初始化顺序、静态引入、可变参数以及finalize方法的应用。通过具体的代码示例,详细解释了这些概念及其在实际编程中的使用。 ... [详细]
  • 本文介绍如何利用动态规划算法解决经典的0-1背包问题。通过具体实例和代码实现,详细解释了在给定容量的背包中选择若干物品以最大化总价值的过程。 ... [详细]
  • 主要用了2个类来实现的,话不多说,直接看运行结果,然后在奉上源代码1.Index.javaimportjava.awt.Color;im ... [详细]
  • Scala 实现 UTF-8 编码属性文件读取与克隆
    本文介绍如何使用 Scala 以 UTF-8 编码方式读取属性文件,并实现属性文件的克隆功能。通过这种方式,可以确保配置文件在多线程环境下的一致性和高效性。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
author-avatar
幸福的FRN
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有