jvm 05JVM - 对象的创建 ,oop模型,字符串常量池

01.JVM - 对象的创建

1、对象的创建的方式

Java语言中,对象创建的方式有六种:

  • new关键字:最常见的形式、Xxx的静态方法、XxxBuilder、XxxFactory的静态方法。

  • Class类的newInstance()方法:通过反射的方式创建对象,调用类的无参构造器进行对象的创建,且其访问权限为public。

  • Constructor的newInstance()方法:通过反射的方式创建对象,调用类的无参、有参构造器进行对象的创建,对构造器访问权限没有要求。

  • 使用clone()方法:不调用任何构造器,但是要求当前类实现Cloneable接口,重写clone()方法。

  • 使用序列化:从文件、网络中获取一个对象的二进制流。

  • 使用第三方库:Objenesis。

02.oop模型

2.1oop模型

1、非数组对象 InstaceOopDesc

2、数组对象 arrayOopDesc

复制代码
    2.1 基本数据类型数组 typeArrayOopDesc

    2.2 引用类型数组 objArrayOopDesc

3、MarkOopDesc

复制代码
    存放锁信息、分代年龄等

2.2 oop和Klass关系

1、InstanceKlass是JVM中表示类的对象的数据结构。JVM在加载class时,会创建instanceKlass,表示其元数据,包括常量池,字段,方法等,存放在方法区,instanceKlass是JVM中的数据结构;

2、 InstanceOopDesc是JVM中表示实例对象的数据结构。在new一个对象时,JVM创建instanceOopDesc来表示这个对象,存放在堆中,其引用存放在栈中;instanceOopDesc对应Java中的对象实例;

3、new操作返回的instanceOopDesc类型指针指向instanceKlass,而instanceKlass指向对应类型的Class实例的instanceOopDesc;

4、在JDK6中,Class对象存放在方法区,JDK7和JDK8中,Class对象存放在Java堆中。

03.字符串常量池

即String Pool,但是JVM中对应的类是StringTable,底层实现是一个hashtable,看代码

cpp 复制代码
class StringTable : public Hashtable<oop, mtSymbol> {
......

Key的生成方式

1、通过String的内容+长度生成hash值

2、将hash值转为key

cpp 复制代码
hashValue = hash_string(name, len);
index = hash_to_index(hashValue);
cpp 复制代码
// Pick hashing algorithm
unsigned int StringTable::hash_string(const jchar* s, int len) {
  return use_alternate_hashcode() ? AltHashing::murmur3_32(seed(), s, len) :
                                    java_lang_String::hash_code(s, len);
}
cpp 复制代码
  // Bucket handling
  int hash_to_index(unsigned int full_hash) {
    int h = full_hash % _table_size;
    assert(h >= 0 && h < _table_size, "Illegal hash value");
    return h;
  }

Value的生成方式

将Java的String类的实例instanceOopDesc封装成HashtableEntry

String的两种创建方式:

cpp 复制代码
第一种方式是在常量池中获取字符串对象。
第二种方式是直接在堆空间创建一个新的字符串对象。
cpp 复制代码
//先检查字符串常量池中有没有"apesource",如果字符产常量池中没有,则创建一个,然后str1指向字符串床常量池中的对象,如果有,则直接将str1指向"apesource"
String str1 = "apesource";
String str2 = new String("apesource");//堆中创建一个新的对象
String str3 = new String("apesource");//堆中创建一个新的对象

System.out.println(str1 == str2);//false
System.out.println(str2 == str3);//false

字符串拼接:

cpp 复制代码
public class TestString_5 {

    public static void main(String[] args) {

    }

    public static void test2() {
        String s1 = "1";
        String s2 = "1";

        String s = s1 + s2;
    }
}

用到了StringBuilder

return new String(this.value, 0, this.count);

不会去查常量池的

双引号 + new String

cpp 复制代码
public class TestString_6 {

    public static void main(String[] args) {

    }

    public static void test2() {
        String s1 = "1";
        String s2 = new String("1");

        String s = s1 + s2;
    }
}
相关推荐
初学小白...6 小时前
线程同步机制及三大不安全案例
java·开发语言·jvm
凤山老林8 小时前
还在用JDK8?JDK8升级JDK11:一次价值千万的升级指南
java·开发语言·jvm·spring boot·后端·jdk
2501_938790078 小时前
详解 JVM 中的对象创建过程:类加载检查、内存分配、初始化的完整流程
jvm
宸津-代码粉碎机10 小时前
Java内部类内存泄露深度解析:原理、场景与根治方案(附GC引用链分析)
java·开发语言·jvm·人工智能·python
杨筱毅10 小时前
【底层机制】Android GC -- 为什么要有GC?GC的核心原理?理解GC的意义
android·jvm·gc
東雪木10 小时前
Java基础语言进阶学习——1,JVM内存模型(堆、栈、方法区)
java·jvm·学习
小满、18 小时前
JVM 执行引擎:字节码是如何被执行的
jvm·字节码·jvm执行引擎
无敌最俊朗@1 天前
SQLite 约束 (Constraints) 面试核心知识点
java·开发语言·jvm
milanyangbo1 天前
谁生?谁死?从引用计数到可达性分析,洞悉GC的决策逻辑
java·服务器·开发语言·jvm·后端·算法·架构
m0_748231311 天前
深入JVM:让Java性能起飞的核心原理与优化策略
java·开发语言·jvm