Java 基础学习第一弹

这篇具有很好参考价值的文章主要介绍了Java 基础学习第一弹。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

1. equels和==的区别

  equals方法用于比较对象的内容是否相等,可以根据自定义的逻辑来定义相等的条件,而==操作符用于比较对象的引用是否相等,即它们是否指向同一块内存地址。equals方法是一个

实例方法,可以被所有的Java对象调用,而==操作符可以用于比较对象的引用或基本数据类型的值。equals方法的行为可以被重写,以适应特定的比较需求,而==操作符的行为不可修改。

2. 垃圾回收机制

  垃圾回收是一种在堆内存中找出哪些对象在被使用,还有哪些对象没被使用,并且将后者回收掉的机制。所谓使用中的对象,指的是程序中还有引用的对象;而未使用中的对象,指的是程

序中已经没有引用的对象,该对象占用的内存也可以被回收掉。垃圾回收的第一步是标记。垃圾回收器此时会找出内存哪些在使用中,哪些不是。垃圾回收的第二步是清除,这一步会删掉标记

出的未引用对象。内存分配器会保留指向可用内存中的引用,以分配给新的对象。垃圾回收的第三步是压缩,为了提升性能,删除了未引用对象后,还可以将剩下的已引用对象放在一起(压  

缩),这样就能更简单快捷地分配新对象了。逐一标记和压缩  Java 虚拟机中的所有对象非常低效:分配的对象越多,垃圾回收需要的时间就越久。不过,根据统计,大部分的对象,其实用没

多久就不用了。

Java 堆(Java Heap)是 JVM 所管理的内存中最大的一块,堆又是垃圾收集器管理的主要区域,这里我们主要分析一下 Java 堆的结构。

Java 堆主要分为 2 个区域-年轻代与老年代,其中年轻代又分 Eden 区和 Survivor 区,其中 Survivor 区又分 From 和 To 2 个区。可能这时候大家会有疑问,为什么需要 Survivor 区,为什么 Survivor 还要分 2 个区。

大多数情况下,对象会在新生代 Eden 区中进行分配。当 Eden 区没有足够空间进行分配时,虚拟机会发起一次 Minor GC,Minor GC 相比 Major GC 更频繁,回收速度也更快。

通过 Minor GC 之后,Eden 会被清空,Eden 区中绝大部分对象会被回收,而那些无需回收的存活对象,将会进到 Survivor 的 From 区(若 From 区不够,则直接进入 Old 区)。

Survivor 区相当于是 Eden 区和 Old 区的一个缓冲,类似于我们交通灯中的黄灯。Survivor 又分为 2 个区,一个是 From 区,一个是 To 区。每次执行 Minor GC,会将 Eden 区和 From 存活的对象放到 Survivor 的 To 区(如果 To 区不够,则直接进入 Old 区)。

之所以有 Survivor 区是因为如果没有 Survivor 区,Eden 区每进行一次 Minor GC,存活的对象就会被送到老年代,老年代很快就会被填满。而有很多对象虽然一次 Minor GC 没有消灭,但其实也并不会蹦跶多久,或许第二次,第三次就需要被清除。这时候移入老年区,很明显不是一个明智的决定。

所以,Survivor 的存在意义就是减少被送到老年代的对象,进而减少 Major GC 的发生。Survivor 的预筛选保证,只有经历 16 次 Minor GC 还能在新生代中存活的对象,才会被送到老年代。

设置两个 Survivor 区最大的好处就是解决内存碎片化。

我们先假设一下,Survivor 如果只有一个区域会怎样。Minor GC 执行后,Eden 区被清空了,存活的对象放到了 Survivor 区,而之前 Survivor 区中的对象,可能也有一些是需要被清除的。问题来了,这时候我们怎么清除它们?在这种场景下,我们只能标记清除,而我们知道标记清除最大的问题就是内存碎片,在新生代这种经常会消亡的区域,采用标记清除必然会让内存产生严重的碎片化。因为 Survivor 有 2 个区域,所以每次 Minor GC,会将之前 Eden 区和 From 区中的存活对象复制到 To 区域。第二次 Minor GC 时,From 与 To 职责互换,这时候会将 Eden 区和 To 区中的存活对象再复制到 From 区域,以此反复。

这种机制最大的好处就是,整个过程中,永远有一个 Survivor space 是空的,另一个非空的 Survivor space 是无碎片的。那么,Survivor 为什么不分更多块呢?比方说分成三个、四个、五个?显然,如果 Survivor 区再细分下去,每一块的空间就会比较小,容易导致 Survivor 区满,两块 Survivor 区可能是经过权衡之后的最佳方案。

老年代占据着 2/3 的堆内存空间,只有在 Major GC 的时候才会进行清理,每次 GC 都会触发“Stop-The-World”。内存越大,STW 的时间也越长,所以内存也不仅仅是越大就越好。在内存担保机制下,无法安置的对象会直接进到老年代,以下几种情况也会进入老年代。

1)大对象,指需要大量连续内存空间的对象,这部分对象不管是不是“朝生夕死”,都会直接进到老年代。这样做主要是为了避免在 Eden 区及 2 个 Survivor 区之间发生大量的内存复制。

2)长期存活对象,虚拟机给每个对象定义了一个对象年龄(Age)计数器。正常情况下对象会不断的在 Survivor 的 From 区与 To 区之间移动,对象在 Survivor 区中每经历一次 Minor GC,年龄就增加 1 岁。当年龄增加到 15 岁时,这时候就会被转移到老年代。当然,这里的 15,JVM 也支持进行特殊设置。

3)动态对象年龄,虚拟机并不重视要求对象年龄必须到 15 岁,才会放入老年区,如果 Survivor 空间中相同年龄所有对象大小的总合大于 Survivor 空间的一半,年龄大于等于该年龄的对象就可以直接进去老年区,无需等你“成年”。

这其实有点类似于负载均衡,轮询是负载均衡的一种,保证每台机器都分得同样的请求。看似很均衡,但每台机的硬件不通,健康状况不同,我们还可以基于每台机接受的请求数,或每台机的响应时间等,来调整我们的负载均衡算法。

3. String、StringBuffer、StringBuilder的区别

在Java中,StringStringBufferStringBuilder都是用于处理字符串的类,但它们在性能、线程安全性和可变性方面存在一些区别。

String(字符串)String是Java中最常用的字符串类,它是不可变的(immutable)。这意味着一旦创建了一个String对象,它的值就不能被修改。每次对String的操作(例如连接、替换等)都会创建一个新的String对象。这种不可变性使得String具有线程安全性,适合在多线程环境下使用。然而,频繁的字符串操作可能会导致内存开销较大,因为每次操作都会创建新的对象。

String str = "Hello";
str += " World"; // 创建了一个新的String对象
```

StringBuffer(字符串缓冲区)StringBuffer是可变的(mutable)字符串类,它可以进行多次修改而无需创建新的对象。StringBuffer是线程安全的,适用于多线程环境下的字符串操作。它提供了多个方法用于对字符串进行修改、连接、插入和删除等操作。

StringBuffer sb = new StringBuffer("Hello");
sb.append(" World"); // 在原对象上进行修改,无需创建新对象
```

由于`StringBuffer`是线程安全的,它的执行速度相对较慢。因此,如果在单线程环境下进行字符串操作,推荐使用`StringBuilder`,因为它的执行速度更快。

StringBuilder(字符串构建器)StringBuilder也是可变的字符串类,类似于StringBuffer,它可以进行多次修改而无需创建新的对象。StringBuilder不是线程安全的,因此在多线程环境下使用时需要进行外部同步。由于不需要额外的线程安全检查,StringBuilder的执行速度相对较快。

StringBuilder sb = new StringBuilder("Hello");
sb.append(" World"); // 在原对象上进行修改,无需创建新对象
```

总结:

  • 如果需要频繁操作字符串,并且在多线程环境下使用,应该使用StringBuffer
  • 如果需要频繁操作字符串,但在单线程环境下使用,应该使用StringBuilder,因为它的执行速度更快。
  • 如果不需要频繁操作字符串,或者字符串是不可变的,可以使用String

4. 操作字符串常见的类及方法

String类:String是Java中最常用的字符串类,它提供了许多方法来处理字符串。以下是一些示例:

String str1 = "Hello";
String str2 = "World";

// 连接字符串
String result1 = str1 + str2; // 结果为 "HelloWorld"

// 获取字符串长度
int length = str1.length(); // 结果为 5

// 检查字符串是否为空
boolean isEmpty = str1.isEmpty(); // 结果为 false

// 检查字符串是否包含指定字符
boolean contains = str1.contains("H"); // 结果为 true

// 提取子字符串
String subStr = str1.substring(1, 4); // 结果为 "ell"

// 替换字符
String replacedStr = str1.replace("H", "J"); // 结果为 "Jello"

// 拆分字符串
String[] parts = str1.split("l"); // 结果为 ["He", "", "o"]

// 转换为大写或小写
String upperCase = str1.toUpperCase(); // 结果为 "HELLO"
String lowerCase = str1.toLowerCase(); // 结果为 "hello"

StringBuilder类:StringBuilder用于构建可变字符串,它提供了一系列方法来进行字符串的拼接和修改。以下是一些示例:

StringBuilder sb = new StringBuilder();

// 追加字符串
sb.append("Hello");
sb.append("World");

// 插入字符串
sb.insert(5, " ");

// 替换字符串
sb.replace(6, 11, "Java");

// 删除字符
sb.deleteCharAt(5);

// 反转字符串
sb.reverse();

String result2 = sb.toString(); // 结果为 "avaJdlroW"

StringBuffer类:StringBufferStringBuilder类似,也用于构建可变字符串。不同的是,StringBuffer是线程安全的,适用于多线程环境下的字符串操作。以下是一个示例:

StringBuffer buffer = new StringBuffer();

buffer.append("Hello");
buffer.append("World");

String result3 = buffer.toString(); // 结果为 "HelloWorld"

5. Static的用法和作用

static 是Java中的一个关键字,可以应用于变量、方法和代码块。它具有以下几种用法和作用:

静态变量(Static Variables):使用 static 关键字声明的变量称为静态变量,也称为类变量。静态变量属于类而不是实例,它在类加载时被初始化,并且在整个程序执行期间保持不变。静态变量可以通过类名直接访问,无需创建类的实例。静态变量常用于表示在类的所有实例之间共享的数据。

public class MyClass {
    static int count = 0; // 静态变量

    public MyClass() {
        count++; // 每次创建实例时,静态变量 count 自增
    }
}

静态方法(Static Methods):使用 static 关键字声明的方法称为静态方法。静态方法属于类而不是实例,它可以在类加载时直接调用,无需创建类的实例。静态方法只能访问静态变量和调用其他静态方法,不能直接访问实例变量或调用实例方法。

public class MathUtils {
    public static int add(int a, int b) { // 静态方法
        return a + b;
    }
}

静态代码块(Static Initialization Blocks):静态代码块用于在类加载时执行一些初始化操作。它使用 static 关键字定义,并用花括号括起来的代码块。静态代码块只执行一次,且在类的第一次使用时执行。

public class MyClass {
    static {
        // 静态代码块
        // 执行一些初始化操作
    }
}

静态代码块通常用于初始化静态变量或执行其他与类相关的初始化操作。

静态导入(Static Import):静态导入用于在代码中直接使用静态成员(变量或方法),而无需使用类名限定符。通过使用 import static 语法,可以导入静态成员,使其在代码中可直接访问。

import static java.lang.Math.PI;

public class MyClass {
    public double calculateArea(double radius) {
        return PI * radius * radius; // 直接使用静态变量 PI,无需使用 Math.PI
    }
}

  文章来源地址https://www.toymoban.com/news/detail-711395.html

到了这里,关于Java 基础学习第一弹的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 计算机网络 基础面试第一弹

    DNS解析(Domain Name System Resolution)是将域名转换为IP地址的过程。 以下是DNS解析的详细过程: 发起解析请求:当您在浏览器中输入一个域名时,您的操作系统会发起一个DNS解析请求。该请求被发送到您计算机上配置的本地DNS服务器,通常由您的互联网服务提供商(ISP)提供。

    2024年02月08日
    浏览(32)
  • 软考A计划-系统架构师-学习笔记-第一弹

    点击跳转专栏=Unity3D特效百例 点击跳转专栏=案例项目实战源码 点击跳转专栏=游戏脚本-辅助自动化 点击跳转专栏=Android控件全解手册 点击跳转专栏=Scratch编程案例 点击跳转=软考全系列 专注于 Android/Unity 和各种游戏开发技巧,以及 各种资源分享 (网站、工具、素材、源码、

    2024年02月08日
    浏览(31)
  • 6.6---每天一点点--- Java基础之==和equals区别并举例说明(总复习+深入版)---详细+举例(小白也能看懂)

    1.== 为运算符,equal 为String数据类型的比较方法 相同内容的对象地址不一定相同,但相相同地址的对象内容一定相同;== 比较的是值是否相等,equal比较的是是否是同一个对象 2.把对象想象成一个房子,而引用则是门牌号。如果两个引用指向同一个房子,那么它们就是相等的

    2024年02月08日
    浏览(32)
  • 互联网Java工程师面试题·Java 面试篇·第一弹

    目录 1、Java 中能创建 volatile 数组吗? 2、volatile 能使得一个非原子操作变成原子操作吗? 3、volatile 修饰符的有过什么实践? 4、volatile 类型变量提供什么保证? 5、10 个线程和 2 个线程的同步代码,哪个更容易写? 6、你是如何调用 wait()方法的?使用 if 块还是循环?为什

    2024年02月07日
    浏览(27)
  • 互联网Java工程师面试题·Elasticsearch 篇·第一弹

    目录 1、elasticsearch 了解多少,说说你们公司 es 的集群架构,索引数据大小,分片有多少,以及一些调优手段 。 1.1  设计阶段调优 1.2  写入调优 1.3  查询调优 1.4  其他调优 2、elasticsearch 的倒排索引是什么 3、elasticsearch 索引数据多了怎么办,如何调优,部署 3.1 动态索引

    2024年02月06日
    浏览(33)
  • “==“和equals方法究竟有什么区别?

    ==操作符专门用来比较两个变量的值是否相等,也就是用于比较变量所对应的内存中所存储的数值是否相同,要比较两个基本类型的数据或两个引用变量是否相等,只能用 ==操作符。 如果一个变量指向的数据是对象类型的,那么,这时候涉及了两块内存,对象本身占用一块内

    2023年04月17日
    浏览(24)
  • BigDecimal的equals方法和compareTo方法的区别

    面试经常会问道为什么不能用BigDecimal的equals方法做等值比较? 其实BigDecimal的equals方法和compareTo并不一样,equals方法会比较两部分内容,分别是值(value)和标度(scale),而对于1.0和1这两个数字,他们的值虽然一样,但是精度是不一样的,在使用equals比较的时候会返回fals

    2024年02月04日
    浏览(24)
  • Java 中 == 与 equals() 的区别

    2024年02月15日
    浏览(32)
  • JAVA中.equals()与 ==的区别

            “==”是 运算符 ,如果是基本数据类型,则比较 存储的值 ;如果是引用数据类型,则比较 所指向对象的地址值 。         equals是 Object的方法 ,比较的是 所指向的对象的地址值 ,一般情况下,重写之后比较的是 对象的值 。 八大数据类型: 整数 byte、sho

    2024年02月08日
    浏览(32)
  • c# ==操作符和equals方法的区别

    在C#中,\\\"==\\\"操作符和Equals()方法有着不同的用途和行为。 \\\"==\\\"操作符: \\\" “操作符用于比较两个对象的值是否相等。当使用” \\\"操作符比较两个引用类型的对象时,它会比较它们的引用是否指向相同的内存地址。 对于值类型(如int、double等),\\\"==\\\"操作符比较它们的实际值。如

    2024年01月23日
    浏览(32)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包