• JRebel热加载插件激活、启动报错解决方法以及java类加载、热加载原理


    idea安装好热加载插件 JRebel,启动后报错java.lang.OutOfMemoryError: PermGen space解决方法
    报错原因是因为内存溢出了,也就是内存不足,方法就是增加内存,添加如下配置:

    参数解释:
    -Xms,表示程序启动时,JVM 堆的初始化最小尺寸参数;
    -Xmx,表示程序启动时,JVM 堆的初始化最大尺寸参数;
    -XX:PermSize,表示程序启动时,JVM 方法区的初始化最小尺寸参数;
    -XX:MaxPermSize,表示程序启动时,JVM 方法区的初始化最大尺寸参数。


    激活JRebel
    生成 GUID 的网址:https://www.guidgen.com/

    用这个网址 + 生成的 GUID 激活

    https://jrebel.qekang.com/
    

    例如:

    https://jrebel.qekang.com/cb2546bb-9d43-4115-bf4b-10539349efed
    

    设置离线模式 来防止失效

    File -> Settings -> JRebel -> [Work offline]按钮
    

    既然用了这个热加载,就了解一下这个热加载原理

    一、java类加载

    java的类加载过程
    一个java类文件到虚拟机里的对象,要经过如下过程:

    首先我们编写好了的java源代码通过java编译器,将java源代码文件编译成class字节码,类加载器读取class字节码,再将类转化为实例,对实例newInstance就可以生成对象。
    类加载器ClassLoader功能,也就是将class字节码转换为类的实例。在java应用中,所有的实例都是由类加载器,加载而来。一般在系统中,类的加载都是由系统自带的类加载器完成,而且对于同一个全限定名的java类(如com.csiar.soc.HelloWorld),只能被加载一次,而且无法被卸载。


    加载class字节码的工作是由类加载器实例去实现的,类加载器支持通过文件目录,jar,zip,网络等多种途径,加载class字节码文件。
    JVM启动后就默认有三个类加载器实例,负责去加载不同位置的class。
    1.核心类库加载器 BootStrap ClassLoader,负责加载jdk安装目录下lib文件夹里面的jar包,我们的String.class,System.class这些类都放在这个目录下面,启动jvm就会去加载,必不可少。
    2.拓展类库加载器 Extension ClassLoader,负责加载jdk安装目录下lib/ext文件夹里面的jar包,这里面是一些jdk的拓展jar包,比如zipfs.jar这样的包或工具类。拓展的意思就是在某些情况下,这些jar包不加载也不影响jvm工作。
    3.应用程序代码加载器 Application ClassLoader,负责加载我们自己写的程序代码,通过java命令 -cp 或者 -classpath告诉jvm我们的代码class存放位置。如果我们的程序是jar包运行,你可以在jar包 META-INF目录MANIFEST.MF文件里面看到一个Class-Path: .配置,这就是指定代码位置的。


    java类加载的阶段

    加载阶段
    找到类的静态存储结构,并加载到虚拟机里面,然后转换成方法区的运行时数据结构,生成class对象,加载阶段,用户可以自定义类加载器参与进来。
    验证阶段
    主要确保字节码安全的,确保不会对虚拟机安全造成危害,可以通过JVM启动参数来禁用一些验证,但不推荐修改设置,参数禁用可能会对虚拟机安全造成一些危害。
    准备阶段
    确定内存布局,初始化内存变量,注意点:赋初始值,不会执行程序自己定义的赋值操作,比如定义了一个私有变量:private static int count = 12,在准备阶段并不是把count初始为了12,这里是会赋初始值,int初始值为0,所以会把这私有静态变量赋值为0,而不是12。
    解析阶段
    这个阶段主要是将符号引用变为直接引用。
    初始化阶段
    调用程序自定义的代码。比如private static int count = 12, count在本阶段将会被初始化为12,而不是之前准备阶段的0,初始化阶段会生成clean int 方法,这个方法由编译器自动收集类中的所有类变量的赋值、动作和静态语句块中的语句合并,同一个类加载器中,只会将一个类型初始化一次。

    Java虚拟机没有强制约束什么时候开始初始化阶段,但规定了5种情况必须立即初始化,当然这之前的几种操作都是已经运行了的。5种情况如下:

    1.遇到new、 get static 、post static、 invoke static这四条字节码指令的时候,如果类没有初始化,需要触发初始化,注意的是final修饰的类,会在编译期的时候,将结果放在常量池,即使调用也不会触发初始化,因为final修饰的是常量,会把常量放在常量池,调用常量不会触发初始化这个阶段。
    2.使用java.long.reflect包里方法,即对类进行反射调用的时候,如果类没初始化的话,需要初始化。
    3.当初始化一个子类的时候,如果父类还没有初始化,需要先初始化父类,再初始化子类。
    4.虚拟机启动的时候,用户制定一个要执行的主类,虚拟机会先初始化这个主类,例子:我们写的java程序,在某一个类里面写了一个main方法,通过运行这个main方法启动这个程序,虚拟机会先初始化这个main方法所在的类。
    5.使用jdk1.7动态语言支持的时候,如果java.lang.invoke.methondhandler类实例解析的最后结果是ref_getstatic、ref_putstatic、ref_invokestatic方法句柄的时候,如果句柄对应的类没有初始化,那就需要先初始化句柄对应的类。

    二、Java类加载器的特点

    1.由AppClass Loader(系统类加载器)开始加载指定的类;
    2.类加载器将加载任务交给其父类,如果其父类找不到,再交给自己去加载(即双亲委派);
    3.Bootstrap Loader (启动类加载器)是最顶级的类加载器。

    三、热加载原理

     热加载的实现原理主要依赖java的类加载机制,在实现方式可以概括为在容器启动的时候起一个后台线程,定时的检测类文件的时间戳变化,如果类的时间戳变化了,则将类重新载入。
    对比反射机制,反射是在运行时获取类信息,通过动态的调用来改变程序行为; 热加载则是在运行时通过重新加载改变类信息,直接改变程序行为。

    四、参考博文

    (1) https://www.cnblogs.com/cdcr/p/9737166.html (热加载插件启动时报内存溢出解决方法)
    (2) https://blog.csdn.net/wing_93/article/details/78561736 (热部署原理解析以及热部署实现)
    (3) https://blog.csdn.net/u013159507/article/details/82960847 (类加载过程)

  • 相关阅读:
    一道微分方程应用题中的“微”妙解答 高等数学
    Do Visual Studio Database Project *.refactorlog Files Belong in Source Control?
    WPF binding not updating the view
    Extend IQueryable<T> Where() as OR instead of AND relationship
    How can I add an item to a IEnumerable<T> collection?
    What are the Navigation Properties in Entity Framework
    Is there a quick way to find all columns in SQL Server 2008 R2 that are encrypted/have encrypted data?
    Cannot convert from an IEnumerable<T> to an ICollection<T>
    Why Navigation Properties are virtual by default in EF
    k8smtu设置不当引发的线上故障
  • 原文地址:https://www.cnblogs.com/jasonboren/p/12205790.html
Copyright © 2020-2023  润新知