前言 Android应用中JNI代码,是作为本地方法运行的.而大部分情况下,这些JNI方法均需要传递Dalvik虚拟机实例作为第一个参数.例如,你需要用虚拟机实例来创建jstring和其他的Java对象.查找类或成员变量等.大部分情况下,在你用JNI接口从Java层调用Native层中的代码时,你并不需要在native代码中自己初始化一个Dalvik虚拟机实例.但是,如果你在搞逆向或者写exp,你总是需要钻研各种非常规的情况. 最近,我在逆向时需要在native代码中手动创建虚拟机实例用于在JN…
1.Java虚拟机栈 java方法执行时的内存模型 1.1 栈帧 每个方法都会在虚拟机栈中创建一个对应的栈帧,用于存储局部变量表,操作数栈,动态链接,方法出口等信息. 一个方法的调用到结束就对应这一个栈帧从虚拟机栈入栈到出栈. 1.2 局部变量表 存放编译期可知在方法中各种基本数据类型和对象的引用,基本数据类型直接存值(字面量),引用类型存指向对象的指针或者代表对象的句柄或者与对象位置有关的信息. 1.2.1 double long 型数值会占据2个局部变量空间(slot)其余数据类型只占一个.…
1.Java虚拟机栈(Java Virtual Machine Stacks) 线程私有,它的生命周期与线程相同.描述的是Java方法执行的内存模型:每个方法在执行的同时都会创建一个栈帧(Stack Frame)用于存储局部变量表. 操作数栈.动态链接.方法出口等信息.每一个方法从调用直至执行完成的过程,对应着一个栈帧在虚拟机中入栈到出栈的过程. 有人会把Java内存区分为堆内存(Heap)和栈内存(Stack),这种分发太过粗糙,Java内存区域的划分实际上远比这个要复杂很多, 这种划分只是说…
找到eclipse目录下的eclipse.ini,可以看到如下内容: -startup plugins/org.eclipse.equinox.launcher_1.1.0.v20100507.jar –launcher.library plugins/org.eclipse.equinox.launcher.win32.win32.x86_1.1.0.v20100503 -product org.eclipse.epp.package.jee.product –launcher.default…
最 近一直在用eclipse开发android程序,今天不知怎么的启动eclipse时就会出现Failed to create java virtual machine,无法打开eclipse程序,折腾了半天,最好发现是eclipse.ini文件配置问题,重新将配置文件进行恢复后,eclipse即 可正常打开. 方法/步骤 在桌面上的eclipse快捷打开图标上单击右键,然后选择“属性”: 在打开的eclipse属性菜单里,找到快捷方式选项卡,然后打开备注下方的“打开文件位置”: 进入eclip…
Sun/Oracle JDK的HotSpot VM中,直到JDK7都有“持久代”(Permanent Generation,简称PermGen).也称为方法区.Oracle JDK8的HotSpot VM去掉“持久代”,以“元数据区”(Metaspace)替代之. Oracle是在JDK7的时候宣布移除PermGen内存区域,但是知道JDK8才最终移除. 在JDK7之前,所谓的Permanet Generation内存区域其实包含了两个部分: 方法区 Internded String 方法区主要…
今天开发模块时,遇到这个问题,本来是版本的问题,jdk1.6的版本有点低,与cxf框架不兼容,需要用到jdk1.7,结果安装了jdk1.7之后,客户方要求必须用jdk1.6,要统一,所以卸载jdk1.7,结果就出现了上述的问题,于是百度一下,得到的结果是eclipse.ini里面参数给改一样,结果还是不行. 环境变量以及classpath等都折腾了,不行,于是果断卸载jdk,从新安装,可以了. 谢天谢地,又可以工作了.小问题累死人,尤其是版本不兼容这类蛋疼的问题.…
Java虚拟机的内存管理主要分两点:内存分配以及内存回收.· 一.内存分配图: 注: 所占区域的大小与实际的内存大小比例并无直接关系. 解读: 1.如图,分成两种颜色的内存区域,其中蓝色的是线程隔离的数据区,也就是说每一个线程都有自己的这么一个区域存放自己的数据,而青色区域则是线程共享的,里面的数据为所有线程共有,原则上都有权限访问. 2.程序计数器: 用途:用来给程序导航指路的.这个是一块较小的内存空间,可以看做是当前线程执行的字节码的行号指示器.理解就是虚拟机把java源代码编译成了字节码,…
1. 虚方法调用 在上一篇中我曾经提到,Java 里所有非私有实例方法调用都会被编译成 invokevirtual 指令,而接口方法调用都会被编译成 invokeinterface 指令.这两种指令,均属于 Java 虚拟机中的虚方法调用. 在绝大多数情况下,Java 虚拟机需要根据调用者的动态类型,来确定虚方法调用的目标方法.这个过程我们称之为动态绑定.那么,相对于静态绑定的非虚方法调用来说,虚方法调用更加耗时. 在 Java 虚拟机中,静态绑定包括用于调用静态方法的 invokestatic…
重载与重写 在 Java 程序里,如果同一个类中出现多个名字相同,并且参数类型相同的方法,那么它无法通过编译.也就是说,在正常情况下,如果我们想要在同一个类中定义名字相同的方法,那么它们的参数类型必须不同.这些方法之间的关系,我们称之为重载. 小知识:这个限制可以通过字节码工具绕开.也就是说,在编译完成之后,我们可以再向 class 文件中添加方法名和参数类型相同,而返回类型不同的方法.当这种包括多个方法名相同.参数类型相同,而返回类型不同的方法的类,出现在 Java 编译器的用户类路径上时,它…