前言

Redis是企业级系统高并发、高可用架构中非常重要的一个环节。Redis主要解决了关系型数据库并发量低的问题,有助于缓解关系型数据库在高并发场景下的压力,提高系统的吞吐量(具体Redis是如何提高系统的性能、吞吐量,后面会专门讲)。

而我们在Redis的实际使用过程中,难免会遇到缓存与数据库双写时数据不一致的问题,这也是我们必须要考虑的问题。如果还有同学不了解这个问题,可以搬小板凳来听听啦。

一、对象的实例化

1.创建对象的方式

  • new:最常见的方式(本质是构造器)

    变形1 : Xxx的静态方法

    变形2 : XxBuilder/XxoxFactory的静态方法

  • Class的newInstance():反射的方式,只能调用空参的构造器,权限必须是public

  • Constructor的newInstance(Xxx):反射的方式,位于java.lang.reflect.Constructor 可以调用空参、带参的构造器,权限没有要求

  • 使用clone() :不调用任何构造器,当前类需要实现Cloneable接口,实现clone()

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

  • 第三方库Objenesis

2.创建对象的步骤

1.判断对象对应的类是否加载、链接、初始化

虚拟机遇到一条new指令,首先去检查这个指令的参数能否在Metaspace的常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已经被加载、解析和初始化。( 即判断类元信息是否存在)。

如果没有,那么在双亲委派模式下,使用当前类加载器以ClassLoader+包名+类名为Key进行查找对应的.class文件。如果没有找到文件,则抛出ClassNotFoundException异常,如果找到,则进行类加载,并生成对应的Class类对象

2.为对象分配内存

首先计算对象占用空间大小,接着在堆中划分一块内存给新对象。 如果实例成员变量是引用变量,仅分配引用变量空间即可,即4个字节大小。

  • 如果内存规整,使用指针碰撞
    如果内存是规整的,那么虚拟机将采用的是指针碰撞法(BumpThePointer)来为对象分配内存。意思是所有用过的内存在一边,空闲的内存在另外一边,中间放着一个指针作为分界点的指示器,分配内存就仅仅是把指针向空闲那边挪动一段与对象大小相等的距离罢了。如果垃圾收集器选择的是Serial、ParNew这种基于压缩算法的,虚拟机采用这种分配方式。所以一般使用带有compact(整理)过程的收集器时,使用指针碰撞。
  • 如果内存不规整,虚拟机需要维护一个列表,使用空闲列表分配
    如果内存不是规整的,已使用的内存和未使用的内存相互交错,那么虛拟机将采用的是空闲列表法来为对象分配内存。意思是虚拟机维护了一个列表,记录上哪些内存块是可用的,再分配的时候从列表中找到一块足够大的空间划分给对象实例,并更新列表上的内容。这种分配方式成为“空闲列表(Free List) ”。

说明:选择哪种分配方式由Java堆是否规整决定,而Java堆是否规整又由所采用的垃圾收集器是否带有压缩整理功能决定。

3.处理并发安全问题

在分配内存空间时,另外一个问题是及时保证new对象时候的线程安全性:创建对象是非常频繁的操作,虚拟机需要解决并发问题。虚拟机采用 了两种方式解决并发问题:

  • CAS ( Compare And Swap )失败重试、区域加锁:保证指针更新操作的原子性;
  • TLAB把内存分配的动作按照线程划分在不同的空间之中进行,即每个线程在Java堆中预先分配一小块内存,称为本地线程分配缓冲区,(TLAB ,Thread Local Allocation Buffer) 虚拟机是否使用TLAB,可以通过一XX:+/一UseTLAB参数来 设定。
4.初始化分配到的空间

Java给对象的属性赋值的操作有如下四种:

  • 属性的默认初始化
  • 显式初始化
  • 代码块中初始化
  • 构造器中初始化

内存分配结束,虚拟机将分配到的内存空间都初始化为零值(不包括对象头)。这一步保证了对象的实例字段在Java代码中可以不用赋初始值就可以直接使用,程序能访问到这些字段的数据类型所对应的零值。

5.设置对象的对象头

将对象的所属类(即类的元数据信息)、对象的HashCode和对象的GC信息、锁信息等数据存储在对象的对象头中。这个过程的具体设置方式取决于JVM实现。

6.执行init方法进行初始化

在Java程序的视角看来,初始化才正式开始。初始化成员变量,执行实例化代码块,调用类的构造方法,并把堆内对象的首地址赋值给引用变量。 因此一般来说(由字节码中是否跟随有invokespecial指令所决定),new指令之 后会接着就是执行方法,把对象按照程序员的意愿进行初始化,这样一个真正可用的对象才算完全创建出来。

二、对象的内存布局

1.对象头(Header)

包含两部分

  • 运行时元数据

    • 哈希值( HashCode )
    • GC分代年龄
    • 锁状态标志
    • 线程持有的锁
    • 偏向线程ID
    • 偏向时间戳
  • 类型指针:指向类元数据的InstanceKlass,确定该对象所属的类型

说明:如果是数组,还需记录数组的长度

2.实例数据(Instance Data)

说明:它是对象真正存储的有效信息,包括程序代码中定义的各种类型的字段(包括从父类继承下来的和本身拥有的字段) 规则:

  • 相同宽度的字段总被分配在一起
  • 父类中定义的变量会出现在子类之前
  • 如果CompactFields参数为true(默认为true),子类的窄变量可能插入到父类变量的空隙

3.对齐填充(Padding)

不是必须的,也没特别含义,仅仅起到占位符作用

最后

针对最近很多人都在面试,我这边也整理了相当多的面试专题资料,也有其他大厂的面经。希望可以帮助到大家。

下面的面试题答案都整理成文档笔记。也还整理了一些面试资料&最新2021收集的一些大厂的面试真题(都整理成文档,小部分截图),有需要的可以戳这里免费领取

最新整理电子书

有需要的可以戳这里免费领取

[外链图片转存中…(img-eUXjDy4V-1624080302461)]

最新整理电子书

java对象数组赋值相关推荐

  1. 关于Java对象数组赋值的问题

    在写Java大作业时需要对对象数组赋值,而我的程序在赋值完后,我发现对象数组中的每个对象都相同. 定义对象数组 读取文件中的数据,设置一个临时变量stu储存每一个对象的信息并赋值 上诉代码每次赋完值后 ...

  2. java对象数组赋值时空指针问题

    今天学习java对象数组的时候遇到一个问题,在对数组元素赋值的时候出现空指针报错,经过咨询大佬后问题得到解决,现在记录一下 对于引用类型的数组,默认值为null,相当于里面是[null,null],需 ...

  3. 关于Java对象数组赋值

    对象 package test.service.bean;import java.util.List;public class MyRole {private String key;public My ...

  4. json 解析 转java对象数组对象数组对象_json字符串转java对象数组

    需要引入json-lib-2.2-jdk15.jar和ezmorph-1.0.6.jar包 String itemStar = request.getParameter("itemStar& ...

  5. java 对象数组定义_Java对象数组定义与用法详解

    本文实例讲述了Java对象数组定义与用法.分享给大家供大家参考,具体如下: 所谓的对象数组,就是指包含了一组相关的对象,但是在对象数组的使用中一定要清楚一点:数组一定要先开辟空间,但是因为其是引用数据 ...

  6. 登录、注册界面(Java对象数组实现简易版)

    注册.登录界面(Java对象数组实现) 此代码为基础简易版,涉及简单的面向对象思想,用到对象数组.成员变量.成员方法.简单封装等. 一.需求   有注册.登录.退出功能,注册的账号存于对象数组中,对象 ...

  7. java从键盘为数组赋值,java给数组赋值

    java 动态数组赋值,java对象数组详解,java二维数组赋值,java给数组赋值 java数组动态赋值,从零学java笔录-第24篇 图解一维数组在内存中,java二维数组赋值,java给数组赋 ...

  8. java中数组赋值方法

    Java中数组赋值方法,以及为什么不能使用a = b //这里定义两个长度相同的数组,其中数组二为空 int[] arr1 = new int[]{1,2,3,4,5,6,0,0,0,0}; int[ ...

  9. java对象数组和Array List

    java对象数组 在preson类的基础上 public static void main (String args){Person[] array = new Person[3];Person on ...

最新文章

  1. 【牛客每日一题】4.16 逆序对 ( 数学 , 排列组合 ,快速幂 , 快速乘 )
  2. linux内存之buff/cache
  3. Spring scope解惑
  4. mysql 数据库安装命令_教你MySQL数据库的编译安装以及命令详解(5.7版本)
  5. android 消息循环滚动条,Android ViewPager实现循环滚动
  6. Apache整合Tomcat
  7. JS的常用正则表达式 验证密码
  8. kafka2.3.1+zookeeper3.5.6+kafka-manager2.0.0.2集群部署(centos7.7)
  9. java 一切object_javaObject类
  10. jdk 安装_Linux安装JDK操作手册
  11. [每日一题] OCP1z0-047 :2013-08-08   相关子查询中EXISTS的使用.....................................28...
  12. Java手写线程池(不带返回值、带返回值)
  13. Linux运维之道(大量经典案例、问题分析,运维案头书,红帽推荐)
  14. 挑战程序设计竞赛——Ants(POJ No.1852)
  15. android--手机桌面添加网址链接图标(解决方式)
  16. WPF实现选项卡效果(3)——自定义动态添加的AvalonDock选项卡内容
  17. 【PFC】PFC测试指令
  18. 「2017ACM/ICPC亚洲区域赛北京站现场赛G」Liaoning Ship's Voyage【计算几何】
  19. c语言中的fab是什么意思,FAB法则中的F是指什么
  20. Kafka 如何给集群配置Scram账户认证

热门文章

  1. mysql 用户赋予全部权限_mysql 给用户赋值权限
  2. linux文字转图形界面,Linux字符界面转图形界面
  3. 通过java将Excel表格导入数据到数据库
  4. 8个优秀的前端UI框架
  5. 10行Python代码让你秒变撩妹达人
  6. mysql 设置mvcc_mysql的MVCC机制
  7. 2019暑期组队排位总结
  8. 随机生成4个数字php,PHP随机生成4位数字的方法
  9. 任务计划程序使用方法
  10. 5g时代,选华为p30 or 苹果?