1、针对对象的操作码

实例化一个新对象需要通过new操作码来实现。

对象的创建

操作码

操作数

说明

new

index

在堆中创建一个新的对象,将其引用压入栈

new操作码后面紧跟一个无符号16位数,表示常量池中的一个索引。在特定偏移量位置处的常量池入口给出了新对象所属类的信息。如果还没有这些信息,那么虚拟机会解析这个常量池入口。它会为这个堆中的对象建立一个新的实例,用默认初始化对象实例变量,然后把新对象的引用压入栈。

存取实例变量

操作码

操作数

说明

putfield

index

设置对象字段(由index指定)的值,值value和对象引用objectref均从栈中获得

getfield

index

将对象字段(由index指定)压入栈,对象引用objectref栈中取得

存取类变量

操作码

操作数

说明

putstatic

index

设置静态字段(由index指定)的值,值value从栈中获得

getstatic

index

将静态字段(由index指定)压入栈

putfield和getfield这两个操作码只在字段是实例变量的情况下才执行,putstatic和getstatic对静态变量进行存取操作。操作数表示常量池的索引。这个索引所指向的常量池入口包含了该字段的所属类、名字和类型等信息。如果还没有这些信息,虚拟机会解析这个常量池入口。

例如下面代码:

public class TestA {

int x;

int y;

}

public class TestMain {

/**

@param args

*/

public static void main(String[] args) {

// TODO Auto-generated method stub

TestA testA = new TestA();

testA.x = 3;

testA.y = 4;

}

}

用javap工具查看其字节码指令为:

Compiled from "TestMain.java"

public class TestMain extends java.lang.Object{

public TestMain();

Code:

0: aload_0

1: invokespecial #8; //Method java/lang/Object."<init>":()V

4: return

public static void main(java.lang.String[]);

Code:

0: new #16; //class TestA 新建 TestA对象

3: dup //

4: invokespecial #18; //Method TestA."<init>":()V 调用构造方法

7: astore_1 //存入位置为1的局部变量

8: aload_1 //取出位置为1的局部变量压入栈

9: iconst_3 //常量3入栈

10: putfield #19; //Field TestA.x:I 赋值

13: aload_1

14: iconst_4

15: putfield #23; //Field TestA.y:I

18: return

}

2、针对数组的操作码

创建数组

操作码

操作数

说明

newarray

atype

从栈中弹出数组长度,使用atype所指定的基本数据类型分配新数组,将数组的对象引用压入栈

anewarray

index

从栈中弹出数组长度,是哟index所指定的类分配新对象数组,将新数组的对象引用压入栈

multianewarray

index,dimensions

从栈中弹出数组的维数,使用由index所指定的类分配新多维数组,将新数组的对象引用压入栈

atype的值

数组类型

atype

T_BOOLEAN

4

T_CHAR

5

T_FLOAT

6

T_DOUBLE

7

T_BYTE

8

T_SHORT

9

T_INT

10

T_LONG

11

需要注意的是,当数组类型显示声明为boolean类型时,Java虚拟机中创建数组的指令会以位为单位进行操作。无论虚拟机对于boolean数组使用哪一种内部实现,都会使用存取byte数组元素的操作码访问boolean数组的元素。

获取数组长度

操作码

操作数

说明

arraylength

(无)

从栈中弹出一个数组的对象引用,将数组长度压入栈

arraylength指令从栈顶端弹出一个数组引用,然后把这个数组的长度压入栈。

获取数组元素,虚拟机从栈中弹出数组的索引和数组引用,再将位于给定数组的指定索引位置压入栈。

获取数组元素

操作码

操作数

说明

baload

(无)

将byte类型或者boolean类型的数组的索引index和数组引用arrayref弹出栈,将arrayref[index]压入栈

caload

(无)

将char类型的数组的索引index和数组引用arrayref弹出栈,将arrayref[index]压入栈

saload

(无)

将short类型的数组的索引index和数组引用arrayref弹出栈,将arrayref[index]压入栈

iaload

(无)

将int类型的数组的索引index和数组引用arrayref弹出栈,将arrayref[index]压入栈

laload

(无)

将long类型的数组的索引index和数组引用arrayref弹出栈,将arrayref[index]压入栈

faload

(无)

将float类型的数组的索引index和数组引用arrayref弹出栈,将arrayref[index]压入栈

daload

(无)

将double类型的数组的索引index和数组引用arrayref弹出栈,将arrayref[index]压入栈

aaload

(无)

将对象引用类型的数组的索引index和数组引用arrayref弹出栈,将arrayref[index]压入栈

操作码

操作数

说明

bastore

(无)

将byte类型或者boolean类型的值、数组的索引index和数组引用arrayref弹出栈,赋值为arrayref[index] = value

castore

(无)

将char类型的值value、数组的索引index和数组引用arrayref弹出栈,赋值为arrayref[index] = value

sastore

(无)

将short类型的值value、数组的索引index和数组引用arrayref弹出栈,赋值为arrayref[index] = value

iastore

(无)

将int类型的数组的值value、索引index和数组引用arrayref弹出栈,赋值为arrayref[index] = value

lastore

(无)

将long类型的值value、数组的索引index和数组引用arrayref弹出栈,赋值为arrayref[index] = value

fastore

(无)

将float类型的值value、数组的索引index和数组引用arrayref弹出栈,赋值为arrayref[index] = value

dastore

(无)

将double类型的值value、数组的索引index和数组引用arrayref弹出栈,赋值为arrayref[index] = value

aastore

(无)

将对象引用类型的值value、数组的索引index和数组引用arrayref弹出栈,,赋值为arrayref[index] = value

例如如下代码:

public class TestMain {

/**

@param args

*/

public static void main(String[] args) {

// TODO Auto-generated method stub

int a[] = new int[3];

for (int i = 0 ; i < 3;i++){

a[i] = i;

}

}

}

用javap工具查看其字节码为:

Compiled from "TestMain.java"

public class TestMain extends java.lang.Object{

public TestMain();

Code:

0: aload_0

1: invokespecial #8; //Method java/lang/Object."<init>":()V

4: return

public static void main(java.lang.String[]);

Code:

0: iconst_3

1: newarray int

3: astore_1

4: iconst_0

5: istore_2

6: goto 16

9: aload_1

10: iload_2

11: iload_2

12: iastore

13: iinc 2, 1

16: iload_2

17: iconst_3

18: if_icmplt 9

21: return

}

《深入Java虚拟机学习笔记》- 第15章 对象和数组的更多相关文章

  1. java JDK8 学习笔记——第15章 通用API

    第十五章 通用API 15.1 日志 15.1.1 日志API简介 1.java.util.logging包提供了日志功能相关类与接口,不必额外配置日志组件,就可在标准Java平台使用是其好处.使用日 ...

  2. 《深入Java虚拟机学习笔记》- 第19章 方法的调用与返回

    <深入Java虚拟机学习笔记>- 第19章 方法的调用与返回

  3. 《深入Java虚拟机学习笔记》- 第16章 控制流

    <深入Java虚拟机学习笔记>- 第16章 控制流

  4. 《深入Java虚拟机学习笔记》- 第17章 异常

    <深入Java虚拟机学习笔记>- 第17章 异常

  5. 《深入Java虚拟机学习笔记》- 第13章 逻辑运算

    <深入Java虚拟机学习笔记>- 第13章 浮点运算

  6. 《深入Java虚拟机学习笔记》- 第14章 浮点运算

    <深入Java虚拟机学习笔记>- 第13章 浮点运算

  7. 《深入Java虚拟机学习笔记》- 第8章 连接模型

    Java虚拟机学习笔记(八)连接模型

  8. 《深入Java虚拟机学习笔记》- 第4章 网络移动性

    Java虚拟机学习笔记(四)网络移动性

  9. 《深入Java虚拟机学习笔记》- 第2章 平台无关

    Java虚拟机学习笔记(二)平台无关

  10. java JDK8 学习笔记——第16章 整合数据库

    第十六章 整合数据库 16.1 JDBC入门 16.1.1 JDBC简介 1.JDBC是java联机数据库的标准规范.它定义了一组标准类与接口,标准API中的接口会有数据库厂商操作,称为JDBC驱动程 ...

随机推荐

  1. iOS通过http post上传图片 (转)

    转载自:http://www.cocoachina.com/bbs/read.php?tid=89985 由于iOS无法通过html表单来上传图片,因此想要上传图片,必须实现http请求,而不能像其他 ...

  2. 【BZOJ 1066】[SCOI2007]蜥蜴

    Description 在一个r行c列的网格地图中有一些高度不同的石柱,一些石柱上站着一些蜥蜴,你的任务是让尽量多的蜥蜴逃到边界外. 每行每列中相邻石柱的距离为1,蜥蜴的跳跃距离是d,即蜥蜴可以跳到平 ...

  3. angularApi网站用vue重构

    最近在博客园上看到不少关于vue的文章但感觉都是在简单原生写法上,真正vue在实际开发中的优点组件化,spa应用,路由好像都没涉及到,我在学angular1的时候发现没有中文版的api,于是本人不才弄 ...

  4. [转载]js javascript 判断字符串是否包含某字符串,String对象中查找子字符,indexOf

    var Cts = "bblText"; if(Cts.indexOf("Text") > 0 ) { alert('Cts中包含Text字符串'); }

  5. EhCache 分布式缓存/缓存集群(转)

    开发环境: System:Windows JavaEE Server:tomcat5.0.2.8.tomcat6 JavaSDK: jdk6+ IDE:eclipse.MyEclipse 6.6 开发 ...

  6. leetcode3 Two Sum III – Data structure design

    Question: Design and implement a TwoSum class. It should support the following operations: add and f ...

  7. 学习笔记:shared_ptr陷阱

    条款1:不要把一个原生指针给多个shared_ptr管理 int* ptr = new int; shared_ptr<int> p1(ptr); shared_ptr<int> ...

  8. DJANGO技巧两则:模拟MKDIR -P及配合NGINX上传大文件不使超时

    这都是在开发当哪遇到的问题,网上转转,作个记录: http://blog.chinaunix.net/uid-25525723-id-1596574.html http://bookshadow.co ...

  9. Tomcat Java内存溢出 PermGen space 解决方案

    -Xms300m -Xmx400m -XX:PermSize=128M -XX:MaxNewSize=256m -XX:MaxPermSize=256M

  10. Android:控件布局(单帧布局)FrameLayout

    FrameLayout:所有控件位于左上角,并且直接覆盖前面的子元素. 在最上方显示的层加上: android:clickable="true" 可以避免点击上层触发底层. 实例: ...