Hotspot JVM 底层C/C++ 源码 入门4--oop-klass模型

Hotspot虚拟机在内部使用两组类来表示Java的类和对象

  • oop(ordinary object pointer),描述对象的实例信息
  • klass,描述java类,是虚拟机内部Java类型结构
typedef class oopDesc*                            oop;
typedef class   instanceOopDesc*            instanceOop;
typedef class   arrayOopDesc*                    arrayOop;
typedef class     objArrayOopDesc*            objArrayOop;
typedef class     typeArrayOopDesc*            typeArrayOop;

class Klass;
class   InstanceKlass;
class     InstanceMirrorKlass;
class     InstanceClassLoaderKlass;
class     InstanceRefKlass;
class   ArrayKlass;
class     ObjArrayKlass;
class     TypeArrayKlass;

一般来说,这几种模型分别由于描述Java类类型类型指针(引用)

比如: 

ClassA a = new ClassA();

JVM加载ClassA到方法区,创建一个instanceKlass来保存ClassA这个类对象的信息(变量,方法,父类..)即instanceKlass等价于java里的class对象.

而instanceOop这个“普通对象指针”对象中包含了一个字段,指向instanceKlass这个实例(oop-klass模型)

在JVM实例化ClassA时,会另外创建一个instanceOop,储存ClassA实例对象的成员变量,instanceOop中也有一个字段指向instanceKlass

通过这个指针,JVM可以在运行期获取类的信息

oopDesc

为所有oop对象的顶级父类

class oopDesc {
  friend class VMStructs;
 private:
  volatile markOop  _mark;
  union _metadata {//注意是union
    Klass*      _klass;
    narrowKlass _compressed_klass;
  } _metadata;

  // Fast access to barrier set.  Must be initialized.
  static BarrierSet* _bs;
//...
}

先不管VMStructs和_bs,oopDesc类只剩下两个成员变量_mark,_metadata(使用Klass*为例)

_mark是一种标记,线程状态,并发锁,GC信息

_metadata标记元数据(变量,方法,父类,接口...)java类在JVM解析字节码是会还原出java源代码所定义的数据结构信息并保留在内存中,比如反射,而_metadata便指向java类的数据结构的内存位置

Handle模型

class Handle VALUE_OBJ_CLASS_SPEC{
    private:
        oop* _handle;
    //...
}

handle主要用于封装oop和klass,因此声明handle是直接讲oop或者klass传递进去,同时JVM执行java类的方法是最终也是通过handle拿到对应的oop和klass

inline Handle::Handle(oop obj){
    if (obj == NULL){
        _handle = NULL;
    }else{
        _handle = Thread::current()->handle_area()->allocate_handle(obj);
    }
}

这个构造函数结构oop类型参数,将其保存到当前线程在堆申请的handleArea表中.

oop和klass被handle封装之后,JVM内部大部分堆oop和klass的函数调用都要经过Handle一次中间寻址,因此重载了->

oop non_null_obj() const{
    assert(_handle != NULL,"resolving NULL handle");
    return *_handle;
}
oop operator ->() const {
    return non_null_obj();
}

klass和oop的转换

为了方便GC回收,每一个klass实例都要封装位oop,具体为:先分配oop,接着将klass实例分配到oop对象头后面

Klass* klass_part() const {
	return (Klass*)((address)this + klass_part_offset_in_bytes());
}
klassOop as_klassOop() const{
	return (klassOop)(((char*)this) - sizeof(klassOopDesc));
}

 

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
众所周知,Java编译后的Jar包和Class文件,可以轻而易举的使用反编译工具(如JD-GUI)进行反编译,拿到源码。为了保护自己发布的Jar包和Class文件,采用的方式大多是混淆方式,这种方式对于Class文件的加密是不彻底的,还是能够通过分析得出核心算法。本工具是采用jvmti方式对Class文件进行加密,使用C++生成加密和解密库,先用加密库对Jar包进行加密,将加密后的Jar包及解密库文件发布出去,执行时候需要JVM引入解密库文件,解密后执行。c++的.dll文件和.so文件的破解难度是很大的,这就能有效的保护软件和代码的知识产权. 使用方法: 1.打开windows命令行(运行=>cmd=>回车),在命令行中 进入 EncryptJar目录 2.执行 java -jar encrypt.jar 3.输入h,然后回车,可以看到帮助菜单 4.输入3,然后按回车键,进入加入jar文件功能 5.输入要加密的jar文件的路径 6.提示输入秘钥(key)的时候,直接回车,不要输入任何字符(否则后面classhook将不可解密加密后的jar包) 7.输入目标路径(加密后的jar文件路径,此处要注意:jar文件名要保持相同,将加密后的文件保存到不同的目录) 8.将加密后的jar包,替换原来的没有加密的jar包,与要发布的程序一起进行发布.(一般替换lib目录下对应的jar包即可) 9.加密后的jar包运行方法: windows下: 拷贝libClassHook.dll文件到程序的根目录(通常为要执行的jar程序的根目录) 使用以下命令启动程序: java -agentlib:libClassHook -jar xxxxxxxxxxx.jar 则在运行过程中会自动进行解密操作(解密过程是运行过程中用c++的dll进行解密的,可以有效防止破解class文件) 如果执行过程报错,可将程序根目录添加到环境变量path中去 Linux下: 拷贝libClassHook.so到程序的根目录(通常为要执行的jar程序的根目录) 使用以下命令启动程序: java -agentlib:ClassHook -jar xxxxxxxxxxx.jar (这里要删除掉lib,linux系统下会自动补全) 则在运行过程中会自动进行解密操作(解密过程是运行过程中用c++的dll进行解密的,可以有效防止破解class文件) 如果执行过程报错,可以在程序根目录下执行以下语句:export LD_LIBRARY_PATH=`pwd`:$LD_LIBRARY_PATH 或将libClassHook.so 拷贝到/usr/lib目录中去。 支持操作系统:加密请在windows64位系统并安装了64位jdk环境下进行。 需要解密运行的程序支持LINUX(64位)和windows(64位)安装了JDK1.8以上的系统。 测试程序: (t_lib目录下的jar包为经过加密的jar包) java -agentlib:libClassHook -jar test.jar

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值