zoukankan      html  css  js  c++  java
  • jvm总结

    总结工作中遇到的JVM知识

    不定期更新

     一、有关创建字符串时到底创建几个对象,创建的字符串引用比较。

        public static void stringTest() {
            //创建两个变量,一个放入常量池中的"abc",一个是堆中的a
            String a = new String("abc");
            //创建一个对象,堆中的b
            String b = new String("abc");
            //直接拷贝的常量池中的"abc"对象,所以没有创建
            String c = "abc";
            //同上
            String d = "abc";
            
            //说明不是从常量池中取得的对象a=="abc" false
            System.out.println("a=="abc" " + (a=="abc"));
            //说明new一个String()总会(最少)创建一个新的对象a==c false
            System.out.println("a==c " + (a==c));
            //说明,即便有new过相同的字符串,也不会把常量池的引用拷贝过来,new就是创建新的对象b=="abc" false
            System.out.println("b=="abc" " + (b=="abc"));
            //同上b==c false
            System.out.println("b==c " + (b==c));
            //说明也不会将已经存在的常量池的引用拷贝过来b==a false
            System.out.println("b==a " + (b==a));
            //直接赋值的情况下,是将常量池的引用拷贝过来c==d true
            System.out.println("c==d " + (d==c));
            //同一个引用c=="abc" true
            System.out.println("c=="abc" " + (c=="abc"));
            /**
             * 在java中,堆保存对象,栈保存相应的数据。所以可以看出,String a = new String("abc");是创建了两个对象的,一个是"abc",保存在常量池中。
             * 一个是a引用。而后面的String b = new String("abc");其实只创建了一个对象,"abc"已经存在常量池了,就没有创建,只创建了一个b
             */
        }

    额外补充:String.equals(Object o);方法会先判断是否是一个对象,然后判断是否属于String类型,最后判断长度和每个char的内容是否相同。源码:

    public boolean equals(Object anObject) {
            if (this == anObject) {
                return true;
            }
            if (anObject instanceof String) {
                String anotherString = (String)anObject;
                int n = value.length;
                if (n == anotherString.value.length) {
                    char v1[] = value;
                    char v2[] = anotherString.value;
                    int i = 0;
                    while (n-- != 0) {
                        if (v1[i] != v2[i])
                            return false;
                        i++;
                    }
                    return true;
                }
            }
            return false;
        }

    二、有关jvm参数设置

    年轻代=新生代+两个幸存者区

    -XX:PermSize 设置永久代的初始大小

    -XX:MaxPermSize 设置永久代的最大值

    –XX:NewRatio 设置新生代和老年代的比例

    -Xms设置堆的最小空间大小。

    -Xmx设置堆的最大空间大小。

    -Xmn:设置年轻代大小

    -XX:NewSize设置新生代最小空间大小。

    -XX:MaxNewSize设置新生代最大空间大小。

    -Xss设置每个线程的堆栈大小

    -XX:+UseParallelGC:选择垃圾收集器为并行收集器。此配置仅对年轻代有效。即上述配置下,年轻代使用并发收集,而年老代仍旧使用串行收集。

    -XX:ParallelGCThreads=20:配置并行收集器的线程数,即:同时多少个线程一起进行垃圾回收。此值最好配置与处理器数目相等。

    样例:

    -XX:PermSize=64MB      最小尺寸,初始分配

    -XX:MaxPermSize=256MB        最大允许分配尺寸

    XX:+CMSClassUnloadingEnabled -XX:+CMSPermGenSweepingEnabled 设置垃圾不回收

    -Xmx3550m           设置JVM最大可用内存为3550M。

    jvm内存空间可大概分为

    • 堆内存
    • 方法区
    • JAVA虚拟机栈
    • 本地方法栈
    • 程序计数器

    堆内存可以划分为新生代和老年代。

    新生代中还可以再次划分为Eden(新生代)区、From Survivor(幸存者1)区和To Survivor(幸存者2)区。

    其中Java 堆和方法区是线程共享的;虚拟机栈和本地方法栈以及程序计数器是线程私有的。

    1.堆内存

    此内存区域的唯一目的就是存放对象实例,几乎所有的对象实例都在这里分配内存。

    但是注意永久代并不属于堆内存中的一部分,同时jdk1.8之后永久代已经被移除。永久代和老年代是不一样的。

    新生代与老年代的比例的值为 1:2(默认)。而伊甸园eden区和幸存者区是8:1:1的关系。

    2.虚拟机栈

    描述的是java方法执行的内存模型:每个方法被执行的时候都会创建一个"栈帧",用于存储局部变量表(包括参数)、操作栈、方法出口等信息。

    每个方法被调用到执行完的过程,就对应着一个栈帧在虚拟机栈中从入栈到出栈的过程。

    3.本地方法栈

    由名可知,试运行native方法时,用于保存方法运行信息的地方。和虚拟机栈的功能类似。

    4.程序计数器

    程序计数器是用于标识当前线程执行的字节码文件的行号指示器。多线程情况下,每个线程都具有各自独立的程序计数器,所以该区域是非线程共享的内存区域。

    当执行java方法时候,计数器中保存的是字节码文件的行号;当执行Native方法时,计数器的值为空。

    5.方法区:

    方法区也称"永久代",它用于存储虚拟机加载的类信息、常量、静态变量、是各个线程共享的内存区域。

    -server选项下默认MaxPermSize为64m

    -client选项下默认MaxPermSize为32m

    6.基本概念:转自<https://blog.csdn.net/Luomingkui1109/article/details/72820232>

    JVM是可运行Java代码的假想计算机,包括一套字节码指令集、一组寄存器、一个栈、一个垃圾回收,堆 和 一个存储方法域。JVM是运行在操作系统之上的,它与硬件没有直接的交互。

       运行过程:

    我们都知道Java源文件,通过编译器,能够生产相应的.Class文件,也就是字节码文件,而字节码文件又通过Java虚拟机中的解释器,编译成特定机器上的机器码 。

    也就是如下:

    1. Java源文件—->编译器—->字节码文件
    2. 字节码文件—->JVM—->机器码

    每一种平台的解释器是不同的,但是实现的虚拟机是相同的,这也就是Java为什么能够跨平台的原因了 ,当一个程序从开始运行,这时虚拟机就开始实例化了,多个程序启动就会存在多个虚拟机实例。
    程序退出或者关闭,则虚拟机实例消亡,多个虚拟机实例之间数据不能共享。

    7.类加载过程

    Java文件->编译为class文件->加载类->验证->准备->解析->初始化->使用->卸载。

  • 相关阅读:
    第12组 Alpha冲刺(4/6)
    第12组 Alpha冲刺(3/6)
    第12组 Alpha冲刺(2/6)
    2019软件工程实践——第四次作业
    2019软件工程实践——第三次作业
    2019软件工程实践——第二次作业
    2019软件工程实践——第一次作业
    博客园 自定义背景图片(包括动图)
    软件工程 实验一 GIT代码版本管理
    WordCounter项目(基于javase)
  • 原文地址:https://www.cnblogs.com/chxwkx/p/11270148.html
Copyright © 2011-2022 走看看