高效使用内存,这些你得知道

简介: 堆内存用的越少,堆被填满的几率就越低,新生代回 收的次数更少,对象的晋升年龄也就不会很频繁地增加,这意味着对象被提升到老年代的 可能性也降低了,因此, Full GC会减少,降低了GC发生的频率「下面将介绍几种减少内存使用的方式」对象会占用一定数量的堆内存,所以要减少内存使用,最简单的方式就是让对象小一些减少对象大小有两种方式:减少实例变量的个数,或者减少实例变量的大小注意这里的减少变量大小主要是对每个对象选择更小的类型,避免空间的浪费「同时对于对象大小的计算,有几个注意点:」1.对象大小未必和你计算的一样,因为对象会被填充到 8 字节的整数倍2.对象内部即使为 null 的实例变量也会占用

之后更新文章的频率会变高,不过可能不会只更新技术文章了

最近也在阅读“金字塔原理“和”高效能人士的七个习惯“这两本书,之后也会做个读书心得分享

前言

首先有一个结论:

「减少堆内存的使用可以更高效使用内存」

这句话怎么理解呢?

堆内存用的越少,堆被填满的几率就越低,新生代回 收的次数更少,对象的晋升年龄也就不会很频繁地增加,这意味着对象被提升到老年代的 可能性也降低了,因此, Full GC会减少,降低了GC发生的频率

「下面将介绍几种减少内存使用的方式」


减少对象大小

对象会占用一定数量的堆内存,所以要减少内存使用,最简单的方式就是让对象小一些

减少对象大小有两种方式:减少实例变量的个数,或者减少实例变量的大小

注意这里的减少变量大小主要是对每个对象选择更小的类型,避免空间的浪费

「同时对于对象大小的计算,有几个注意点:」

1.对象大小未必和你计算的一样,因为对象会被填充到 8 字节的整数倍

2.对象内部即使为 null 的实例变量也会占用空间


延迟初始化

有时候需要采用延迟初始化来降低初始化类和创建对象的开销

注意我们一般不会在线程安全的代码上引入延迟初始化,这样会增加同步操作的开销

同时对于使用了线程安全对象的代码,如果要采用延迟初始化,也应该使用双重检查锁,比如大家熟悉的单例的写法:

public class LazyDoubleCheckSingleton {
    private volatile static LazyDoubleCheckSingleton lazyDoubleCheckSingleton = null;
    private LazyDoubleCheckSingleton(){
    }
    public static LazyDoubleCheckSingleton getInstance() {
        if (lazyDoubleCheckSingleton == null) {
            synchronized (LazyDoubleCheckSingleton.class) {
                if (lazyDoubleCheckSingleton == null) {
                    lazyDoubleCheckSingleton =  new LazyDoubleCheckSingleton();
                }
            }
        }
        return lazyDoubleCheckSingleton;
    }
}

「尽早清理空值数据」

延迟初始化的对象如果一直不用的话,通过将变量的值设置为 null ,实现尽早清理, 从而使对象可以更快地被垃圾收集器回收

可以看看JDK 中 ArrayList 类的 remove() 方法的实现:

public E remove(int index) {
        //检查下标是否越界
        rangeCheck(index);
        modCount++;
        E oldValue = elementData(index);
        //将index+1以及之后的元素向前移动以为,覆盖被删除的值
        int numMoved = size - index - 1;
        if (numMoved > 0)
            System.arraycopy(elementData, index+1, elementData, index,
                             numMoved);
        //将最后一个位置的元素清空
        elementData[--size] = null; // clear to let GC do its work
        return oldValue;
    }

关键就是对于不需要再引用的元素,应该主动将其设置为 null


不可变对象

「使用不可变的对象常量可以很好的减少空间的浪费」

在 Java 中,很多对象类型都是不可变的

包括那些有相应的基本类型的类,如 Integer 、 Double 和 Boolean 等, 以及其他一些基于数值的类型, 如 BigDecimal

最常见的 Java 对象当属不可变的 String

「举个Boolean类的例子说说」

在 Java 中,其实只需要两个 Boolean 示例,一个表 示 true , 一个表示 false

Boolean 类有一个 public 的构造 器,应用喜欢创建多少这类对象就能创建多少,即使它们和两个Boolean 对象其 中之一是完全相同的

更好的设计方案应该是,让 Boolean 类只有一个 private 的构造器, 通过 static 方法根据其参数返回 Boolean.TRUEBoolean.FALSE ,就可以防止它们占用应用中额外的堆空间

public static final Boolean TRUE = new Boolean(true);
public static final Boolean FALSE = new Boolean(false);


字符串常量池

字符串是最常见的 Java 对象;应用的堆中几乎到处都是字符串

如果有大量的字符串是相同的,那很大一部分空间都是浪费的

「所以有了字符串常量池的概念」

我们知道String 常见的创建方式有两种,new String() 的方式和直接赋值的方式

直接赋值的方式会先去字符串常量池中查找是否已经有此值,如果有则把引用地址直接指向此值,否则会先在常量池中创建,然后再把引用指向此值;而 new String()的方式一定会先在堆上创建一个字符串对象,然后再去常量池中查询此字符串的值是否已经存在,如果不存在会先在常量池中创建此字符串,然后把引用的值指向此字符串

String s1 = new String("Java");
String s2 = s1.intern();
String s3 = "Java";
System.out.println(s1 == s2); // false
System.out.println(s2 == s3); // true


除此之外编译器还会对 String 字符串做一些优化,例如以下代码:

String s1 = "Ja" + "va";
String s2 = "Java";
System.out.println(s1 == s2);

虽然 s1 拼接了多个字符串,但对比的结果却是 true,我们使用反编译工具,看到的结果如下:

Compiled from "StringExample.java"
public class com.lagou.interview.StringExample {
  public com.lagou.interview.StringExample();
    Code:
       0: aload_0
       1: invokespecial #1                  // Method java/lang/Object."<init>":()V
       4: return
    LineNumberTable:
      line 3: 0
  public static void main(java.lang.String[]);
    Code:
       0: ldc           #2                  // String Java
       2: astore_1
       3: ldc           #2                  // String Java
       5: astore_2
       6: getstatic     #3                  // Field java/lang/System.out:Ljava/io/PrintStream;
       9: aload_1
      10: aload_2
      11: if_acmpne     18
      14: iconst_1
      15: goto          19
      18: iconst_0
      19: invokevirtual #4                  // Method java/io/PrintStream.println:(Z)V
      22: return
    LineNumberTable:
      line 5: 0
      line 6: 3
      line 7: 6
      line 8: 22
}

从编译代码 #2 可以看出,代码 “Ja”+”va” 被直接编译成了 “Java” ,因此 s1==s2 的结果才是 true,这就是编译器对字符串优化的结果


最后

「喜欢的话,希望帮忙点赞,转发下哈,谢谢」

相关文章
|
12月前
|
缓存 运维 监控
CPU被打满/CPU 100%:高效应对策略与技术干货分享
【10月更文挑战第3天】在信息技术高速发展的今天,无论是开发人员、运维人员还是数据分析师,都可能遇到CPU被打满(即CPU使用率达到100%)的情况。这不仅会影响系统的响应速度,严重时甚至会导致服务中断。本文将从诊断、分析与解决三个方面,详细介绍处理CPU 100%问题的技术干货。
545 3
|
7月前
|
弹性计算 Linux 云计算
阿里云操作系统控制台——ECS操作及云计算应用实践
本文详细介绍了云服务器ECS的使用流程,包括开通服务、系统配置、权限管理、组件安装及内存全景诊断等关键步骤。通过开通阿里云操作系统服务、授予RAM用户权限和安装必要组件,可实现对服务器的有效管理与维护。在内存诊断部分,展示了如何发起诊断并解析结果,帮助精准定位内存问题。此外,文章还讲解了利用ECS训练模型的操作方法,从上传文件到终端命令执行,直至完成模型训练。最后总结指出,掌握这些技能不仅提升了对云服务器架构的理解,还为实际业务提供了高效解决方案,展现了ECS在数据处理与分析中的重要价值。
330 8
阿里云操作系统控制台——ECS操作及云计算应用实践
|
10月前
|
存储 缓存 监控
Docker容器性能调优的关键技巧,涵盖CPU、内存、网络及磁盘I/O的优化策略,结合实战案例,旨在帮助读者有效提升Docker容器的性能与稳定性。
本文介绍了Docker容器性能调优的关键技巧,涵盖CPU、内存、网络及磁盘I/O的优化策略,结合实战案例,旨在帮助读者有效提升Docker容器的性能与稳定性。
929 7
|
10月前
|
存储 Windows
如何删除DMP文件
如何删除DMP文件
1464 12
|
机器学习/深度学习 人工智能 算法
探索机器学习中的支持向量机(SVM)算法
【5月更文挑战第27天】在数据科学和人工智能的领域中,支持向量机(SVM)是一种强大的监督学习模型,它基于统计学习理论中的VC维理论和结构风险最小化原理。本文将详细介绍SVM的工作原理、核心概念以及如何在实际问题中应用该算法进行分类和回归分析。我们还将讨论SVM面临的挑战以及如何通过调整参数和核技巧来优化模型性能。
|
运维 Kubernetes 调度
阿里云容器服务 ACK One 分布式云容器企业落地实践
3年前的云栖大会,我们发布分布式云容器平台ACK One,随着3年的发展,很高兴看到ACK One在混合云,分布式云领域帮助到越来越多的客户,今天给大家汇报下ACK One 3年来的发展演进,以及如何帮助客户解决分布式领域多云多集群管理的挑战。
阿里云容器服务 ACK One 分布式云容器企业落地实践
|
机器学习/深度学习 自然语言处理 搜索推荐
云上智能客服机器人:重塑客户服务体验的新篇章
未来,云上智能客服机器人将继续深化深度学习技术的应用,通过跨领域的知识融合和模型训练提升其在复杂场景下的理解和决策能力。同时,机器人将更加注重多模态交互技术的发展以提供更加自然流畅的交互体验。 4.2 情感智能与人性化服务 随着情感智能技术的不断发展,云上智能客服机器人将更加注重情感交互和人性化服务。机器人将能够识别用户的情感状态和需求偏好提供更加贴心和温暖的服务体验。
613 7
|
前端开发 JavaScript UED
深度解析Qt背景设计:从基础到高级,从Widget到Quick(三)
深度解析Qt背景设计:从基础到高级,从Widget到Quick
483 0
|
数据安全/隐私保护
阿里云域名购买至备案流程
阿里云域名购买至备案流程
|
XML Java 应用服务中间件
SpringBoot配置加载,各配置文件优先级对比
SpringBoot配置加载,各配置文件优先级对比
651 0