您当前的位置:首页 > 电脑百科 > 程序开发 > 语言 > JAVA

JDK17 与 JDK11 特性差异浅谈

时间:2024-01-26 13:25:53  来源:51CTO  作者:政采云技术
从 JDK11 到 JDK17 ,JAVA 的发展经历了一系列重要的里程碑。其中最重要的是 JDK17 的发布,这是一个长期支持(LTS)版本,它将获得长期的更新和支持,有助于保持程序的稳定性和可靠性。此外,Java 的性能也有了显著的提升。这些进步都反映了 Java 在持续改进和适应现代编程需求方面的承诺。

1.1 switch 表达式语法变化

  1. 在 JDK12 之前如果 switch 忘记写 break 将导致贯穿,在 JDK12 中对 switch 的这一贯穿性做了改进。你只要将 case 后面的冒号改成箭头,那么你即使不写break也不会贯穿了。
  2. switch 可作为表达式,不再是单独的语句。
  3. 当你把 switch 中的 case 后的冒号改为箭头之后,此时 switch 就不会贯穿了,但在某些情况下,程序本来就希望贯穿比如我就希望两个 case 共用一个执行体。JDK12 的 switch 中的 case 也支持多值匹配,这样程序就变得更加简洁了。
  4. JDK13 引入了一个新的 yield 语句来产生一个值,该值成为封闭的 switch 表达式的值。yield 和 return 的主要区别在于它们如何控制程序的流程。return 会结束当前的方法或函数,并将控制权返回给调用者。而 yield 则会暂时离开当前的 switch 表达式,将一个值返回给调用者,然后再回到 switch 表达式的地方继续执行。
public class Demo{
  public static void mAIn(String[] args){
    var score = 'C';
    // 执行switch分支语句
    String s = switch (score){
      case 'A', 'B' -> "上等";
      case 'C' -> "中等";
      case 'D', 'E' -> "下等"; 
      default -> {
        if (score > 100) {
          yield "数据不能超过100";
        } else {
          yield score + "此分数低于0分";
        }
      }
    } 
  }
}

1.2 微基准测试套件

JMH ,即 Java Microbenchmark Harness ,是专门用于代码微基准测试的工具套件。

JMH 典型的应用场景

  1. 想定量地知道某个方法需要执行多长时间,以及执行时间和输入参数的相关性。
  2. 一个接口有两种不同实现,希望比较哪种实现性能更好。

JMH 使用案例

增加 JMH 的依赖

<properties> 
 <jmh.version>1.14.1</jmh.version>
</properties>
<dependencies>
  <dependency>
    <groupId>org.openjdk.jmh</groupId>
    <artifactId>jmh-core</artifactId>
    <version>${jmh.version}</version>
  </dependency>
  <dependency>
    <groupId>org.openjdk.jmh</groupId>
    <artifactId>jmh-generator-annprocess</artifactId>
    <version>${jmh.version}</version>
    <scope>provided</scope>
  </dependency>
</dependencies>

代码编写

import org.openjdk.jmh.annotations.*;

@State(Scope.Thread)
public class MyBenchmark {
  @Benchmark
  @BenchmarkMode(Mode.All)
  public void testMethod() {
    try {
      Thread.sleep(300);
    } catch (InterruptedException e) {
      e.printStackTrace();
    }
  }

  @Benchmark
  @BenchmarkMode(Mode.All)
  public void testMethod2() {
    try {
      Thread.sleep(600);
    } catch (InterruptedException e) {
      e.printStackTrace();
    }
  }
}
 
import org.openjdk.jmh.runner.Runner;
import org.openjdk.jmh.runner.options.Options;
import org.openjdk.jmh.runner.options.OptionsBuilder;

public class BenchmarkRunner {
  public static void main(String[] args) throws Exception {
    Options opt = new OptionsBuilder()
          .include(MyBenchmark.class.getSimpleName())
          .forks(1)
          .warmupIterations(5)
          .measurementIterations(5)
          .build();
    new Runner(opt).run();
  }
}
//以下这些方法都是JMH的一部分,可以在任何版本的JMH中使用。
//include(SimpleBenchmark.class.getSimpleName()) :这个方法表示你想要运行哪个类的基准测试。
//exclude("xxx") :这个方法表示你想要在基准测试中排除哪个方法。
//forks(1) :这个方法表示你想要进行多少轮的基准测试。每一轮测试都会在一个新的 JVM 进程中进行,以确保每轮测试的环境是独立的。
//warmupIterations(5) :这个方法表示你想要进行多少次预热迭代。预热迭代是为了让 JVM 达到稳定状态,预热迭代的结果不会被计入最终的基准测试结果。
//measurementIterations(5) :这个方法表示你想要进行多少次正式的基准测试迭代,这些迭代的结果会被用来计算基准测试的最终结果。

结果输出(只截取了一部分)

图片图片

图片图片

相关注解

@BenchmarkMode

对应 Mode 选项,可用于类或者方法上,需要注意的是,这个注解的 value 是一个数组,可以把几种 Mode 集合在一起执行,还可以设置为 Mode.All ,即全部执行一遍。

图片图片

  • 吞吐量 (thrpt) :单位时间内完成的操作次数,也可以理解为每秒能处理的事务数。
  • 平均时间 (avgt) :每次操作所需的平均时间。
  • 样本时间 (sample) :随机取样,最后输出取样结果的分布,例如“99%的调用在xxx毫秒以内,99.99%的调用在xxx毫秒以内”。
  • 单次启动时间 (ss) :SingleShotTime ,度量一次操作的运行时间,即每次迭代前都会重新初始化状态。
@State

类注解,JMH 测试类必须使用 @State 注解,State 定义了一个类实例的生命周期,可以类比 SpringBean 的 Scope 。由于 JMH 允许多线程同时执行测试,不同的选项含义如下:

Scope.Thread :默认的 State ,每个测试线程分配一个实例。

Scope.Benchmark :所有测试线程共享一个实例,用于测试有状态实例在多线程共享下的性能。

Scope.Group :每个线程组共享一个实例。

如果你想测试一个对象在多线程环境下的行为,你可以选择 Scope.Benchmark 。如果你想要每个线程都有自己的状态,你可以选择 Scope.Thread 。如果你想要在同一线程组内的所有线程共享状态,你可以选择 Scope.Group 。

@OutputTimeUnit

benchmark 结果所使用的时间单位,可用于类或者方法注解,使用 java.util.concurrent.TimeUnit 中的标准时间单位。

@Benchmark

方法注解,表示该方法是需要进行 benchmark 的对象。

1.3 生成类数据共享特性优化

背景:在同一个物理机上启动多个 JVM 时,如果每个虚拟机都单独装载自己需要的所有类,启动成本和内存占用是比较高的。所以引入了类数据共享机制 ( Class Data Sharing ,简称 CDS ) 的概念,通过把一些核心类在每个 JVM 间共享,每个 JVM 只需要装载自己的应用类即可。

JDK12 之前想要利用 CDS 的用户,必须 -Xshare:dump 作为额外的步骤来运行。

JDK12 针对 64 位平台使其默认生成类数据共享 ( CDS ) 归档。

好处:JVM启动时间减少了。因为核心类是共享的,所以 JVM 的内存占用也减少了。

JDK13 则支持在应用运行之后进行动态归档。需要使用 -XX:ArchiveClassesAtExit=filename.jsa 选项来指定一个文件,JVM 会在退出时将应用程序类和标准库类的元数据写入这个文件。然后,在下一次启动 JVM 时,你可以使用 -XX:SharedArchiveFile=filename.jsa 选项来指定刚才创建的文件。

好处:这个特性允许 JVM 在运行时捕获类的元数据,然后在下一次 JVM 启动时重用这些元数据,从而提高启动速度和减少内存占用。

1.4 G1 垃圾收集器和 ZGC 功能增强

  1. 在 JDK12 之前,一旦开始执行垃圾收集,即使可能会超过 -XX:MaxGCPauseMillis 参数设定的值,也不会停止。JDK12 中,如果 G1 垃圾收集器有可能超过预期的暂停时间,则可以终止。G1 垃圾收集器发现反复标记过多的区域后,G1 就会切换到更增量的一种 Mix GC 。它将待回收的集合分为两个部分:强制回收和可选回收。强制回收的部分是 G1 无法增量回收的部分(比如年轻代)也可以包含能增量回收的部分(比如老年代)来提升性能,它占待回收集合的 80% 。当 G1 回收完强制回收部分后,如果还有多余的时间,将会更细粒度的处理可选回收部分,每次只会处理一个区域,避免超过用户指定的时间。在处理完一个区域后,G1 会根据剩余的时间来决定是否继续处理剩余的可选部分。
  2. JDK12 中,如果应用程序活动非常低,G1 可以使其能够在空闲时自动将 Java 堆内存返还给操作系统。
  3. JDK13 中,ZGC 也能够主动释放未使用内存给操作系统,但可以通过 -XX : -ZUncommit 参数来显示关闭此功能。ZGC 支持最大堆大小为 16TB ,可以满足大多数大型服务器的需求。
  4. ZGC 是在 JDK11 中引入的垃圾回收器,但一直都是实验版本,在 JDK15 中正式上线,如果你的应用程序需要处理非常大的堆或者更低的暂停时间,那么 ZGC 可能是一个更好的选择。如果你对兼容性和稳定性有更高的要求,因为 G1 经过长时间的验证和优化,可能 G1 更适合。

1.5 ShenandoahGC

添加一个名为 Shenandoah 的新垃圾收集算法,通过与正在运行的 Java 线程同时进行疏散工作来减少 GC 暂停时间,最终目标旨在针对 JVM 上的内存收回实现低停顿的需求。

Shenandoah 是以实验特性在 JDK12 中引入的。在 JDK15 中正式上线。

使用 Shenandoah 的暂停时间与堆大小无关,这意味着无论堆是 200MB 还是 200GB ,都将具有相同的一致暂停时间。与 ZGC 类似,Shenandoah GC 主要目标是 99.9% 的暂停小于 10ms ,暂停与堆大小无关等。

ZGC 和 ShenandoahGC 的一些主要区别:

  1. 设计目标:ZGC 和 ShenandoahGC 都是为了实现低延迟的垃圾收集而设计的。
  2. 并发回收:ShenandoahGC 实现了并发回收,这意味着它可以在应用线程运行的同时进行垃圾收集,从而减少了垃圾收集对应用性能的影响。
  3. 内存管理:ShenandoahGC 使用名为“连接矩阵”的全局数据结构来记录跨 Region 的引用关系,降低了处理跨代指针时的记忆集维护消耗。而 ZGC 的 Region 可以动态创建和销毁,容量也可以动态调整。
  4. 开发者:Shenandoah 由 RedHat 开发,而 ZGC 由 Oracle 开发。

使用方法:要启用/使用 Shenandoah GC,需要以下 JVM 选项: -XX:+UnlockExperimentalVMOptions -XX:+UseShenandoahGC。作为实验性功能,Shenandoah 构建系统会自动禁用不受支持的配置。

1.6 String 新增方法

  1. transform(Function):对字符串进行处理后返回。
var rs = "test".transform(s -> s + "Java").transform(s -> s.toUpperCase()); 
// TESTJAVA
  1. indent:该方法允许我们调整String实例的缩进。
String result = "Javanjavantest".indent(3);
/*结果会缩进三格
   Java
   java
   test
*/

1.7 Files 新增 mismatch 方法

返回内容第一次不匹配的字符位置索引。

System.out.println(Files.mismatch(Path.of("a.txt"),Path.of("b.txt")));

1.8 核心库 java.text 支持压缩数字格式

NumberFormat 添加了对 ”紧凑形式格式化数字“ 的支持。

”紧凑数字格式“是指以简短或人类可读形式表示的数字。

例如,在 en_US 语言环境中,1000 可以格式化为 “ 1K ”,1000000 可以格式化为 “ 1M ”,具体取决于指定的样式 NumberFormat.Style 。紧凑数字格式由 LDML 的 Compact Number 格式规范定义。要获取实例,请使用 NumberFormat 紧凑数字格式所给出的工厂方法之一。

NumberFormat fmt = NumberFormat.getCompactNumberInstance(Locale.US, NumberFormat.Style.SHORT);
String result = fmt.format(1000);
//1K

var cnf = NumberFormat.getCompactNumberInstance(Locale.CHINA,NumberFormat.Style.SHORT);
System.out.println(cnf.format(5_0000));
//"5万"
System.out.println(cnf.format(7_9200));
//"7.9万"
System.out.println(cnf.format(8_000_000));
//"800万"
System.out.println(cnf.format(9L << 30));
//"96亿"
System.out.println(cnf.format(6L << 50));
//"5637142兆"
System.out.println(cnf.format(6L << 60));
//"6917529京"

1.9 JDK17 支持到 Unicode13

JDK12 支持 Unicode11.0

JDK13 支持 Unicode12.1

从 JDK14 到 JDK17 均是支持 Unicode13.0

1.10 NullPointerExceptions 升级

JDK14 之前,从报错中我们只能得到错误出现的行数,但在 JDK14 之后,会清晰的告诉你哪个对象空指针了。

Exception in thread "main" java.lang.NullPointerException: 
Cannot invoke "String.charAt(int)" because "str" is null
at com.qf.jdk14.Test.main(Test.java:11)

1.11 文本块特性

背景:在 Java 中,在字符串文字中嵌入 html ,XML ,SQL 或 JSON 片段通常需要先进行转义和串联的大量编辑,然后才能编译包含该片段的代码。该代码段通常难以阅读且难以维护。

Java 的文本块特性是在 JDK15 中正式实现的。这个特性首先在 JDK13 中以预览版的形式发布,然后在 JDK14 中改进并再次以预览版的形式发布。这一特性提高了 Java 程序书写大段字符串文本的可读性和方便性。

文本块的开头定界符是由三个双引号 """ 开始,从新的一行开始字符串的内容,以 """ 结束。如果结束的 """ 另起一行时,字符串内容最后会留有一新行。

使用案例

String query = "SELECT `EMP_ID`, `LAST_NAME` FROM `EMPLOYEE_TB`n" +
               "WHERE `CITY` = 'INDIANAPOLIS'n" +
               "ORDER BY `ID`, `LAST_NAME`;";
//使用文本块语法
String query = """
               SELECT `EMP_ID`, `LAST_NAME` FROM `EMPLOYEE_TB`
               WHERE `CITY` = 'INDIANAPOLIS'
               ORDER BY `EMP_ID`, `LAST_NAME`;""";
 
String html = "<html>n" +
              "    <body>n" +
              "        <p>Hello, world</p>n" +
              "    </body>n" +
              "</html>n";
//使用文本块语法
String html = """
              <html>
                  <body>
                      <p>Hello, world</p>
                  </body>
              </html>
              """;

缩进示例

Java 编译器会自动删除不需要的缩进:

  • 每行结尾的空格都会删除。
  • 每行开始的共有的空格会自动删除。
  • 只保留相对缩进。
  • 新行 """ 结束时,将 """ 向左调整,则可以给所有行前加相应数量的空格。将 """ 向右调整,没有作用。
System.out.println("""
        Hello,
          multiline
        text blocks!
    """);
// 结果
// >     Hello,
// >       multiline
// >     text blocks!

1.12 重新实现旧版 Socket API

背景:现在已有的 java.NET.Socket 和 java.net.ServerSocket 以及它们的实现类,都可以回溯到 JDK1.0 时代了。原始 socket 的维护和调试都很痛苦。实现类还使用了线程栈作为 I/O 的缓冲,导致在某些情况下还需要增加线程栈的大小。该实现还存在几个并发问题,需要彻底解决。在未来的网络世界,要快速响应,不能阻塞本地方法线程,当前的实现不适合使用了。

JDK13 全新实现的 NIOSocketImpl 来替换 JDK1 的 SocketImpl 和 PlainSocketImpl。

  • 它便于维护和调试,与 NewI/O (NIO) 使用相同的 JDK 内部结构,因此不需要使用系统本地代码。
  • 它与现有的缓冲区缓存机制集成在一起,这样就不需要为 I/O 使用线程栈。
  • 它使用 java.util.concurrent 锁,而不是 synchronized 同步方法,增强了并发能力。
  • 新的实现是 JDK13 中的默认实现,但是旧的实现还没有删除,可以通过设置参数 -Djdk.net.usePlainSocketImpl=true 来切换到旧版本。

1.13 Hidden Classes

通常我们在使用大型的框架或者 lambda 表达式的时候,会动态生成很多类。但是不幸的是标准的定义类的API:ClassLoader::defineClass 和 Lookup::defineClass 不能够区分出这些类是动态生成(运行时生成)的还是静态生成(编译生成)的。

一般来说动态生成的类生命周期更短,并且其可⻅性要更低。但是现有的 JDK 并没有这个功能。

所有有了 HiddenClasses 的提案,通过 HiddenClasses ,不管是 JDK 还是 JDK 外部的框架,在生成动态类的时候都可以定义为 HiddenClasses,这样可以更加有效的控制这些动态生成类的生命周期和可⻅性。

1.14 instanceof 关键词

instanceof关键词主要用来判断某个对象是不是某个类的实例。

比如,有时候我们要处理一个类似这样的数据集:

Map<String, Object> data = new HashMap<>();
data.put("test", "111");
data.put("test2", 222);

JDK16 之前需要先判断获取的 value 是否是 String ,再做强制类型转换:

Object value = data.get("test");
if (value instanceof String)
{
  String s = (String) value;
  System.out.println(s.substring(1));
}

在 JDK16 的增强之后,对于 instanceof 的判断以及类型转换可以合二为一了:

Object value = data.get("test");
if (value instanceof String s)
{
 System.out.println(s.substring(1));
}

1.15 档案类

Records 的目标是扩展 Java 语言语法,Records 为声明类提供了一种紧凑的语法,通过对类做这样的声明,编译器可以通过自动创建所有方法并让所有字段参与 hashCode() 等方法。其目的是为了充当不可变数据的透明载体的类。

旧方法定义实体类,代码如下:

public final class User {
  final String name;
  final int age;
  public User(String name, int age) {
    this.name = name;
    this.age = age;
  }
  @Override
  public String toString() {
    return "User{" +
          "name='" + name + ''' +
          ", age=" + age +
          '}';
  }
  @Override
  public boolean equals(Object o) {
    if (this == o) return true;
    if (o == null || getClass() != o.getClass()) return false;
    User user = (User) o;
    return age == user.age && Objects.equals(name, user.name);
  }
  @Override
  public int hashCode() {
   return Objects.hash(name, age);
  }
}

通过 Record 类方式,一句话就可以实现以上功能,代码如下:

public record User(String username, String password) {}

在 JDK16 之前的版本中,我们不能在类名后面直接写参数来定义类的状态。这是 JDK16 引入 record 类的一个新特性。

调用 Record 类方式,如下:

public class App {
  public static void main(String[] args) {
    User user = new User("user", "123456");
    System.out.println(user.username());
  }
}

注意事项:

  • record 类不允许使用 abstract 关键字定义为抽象
  • 所有成员变量均为 final 修饰,不允许再次赋值
  • 允许出现静态变量/实例方法/静态方法
  • 允许出现其他构造方法,但必须调用 record 构造方法
  • Record 不允许 extends 继承其他类

1.16 密封类

在 JDK15 中,Java 提出了密封类( Sealed Classes )的概念,在 JDK17 中被正式确认。密封类允许类和接口定义其允许的子类型。因此,如果一个类没有显式地使用 sealed 、non-sealed 或 final 关键字,那么它的默认权限就是 non-sealed 。

以下是一个密封类的代码示例:

sealed class Human permits Kyrie, LeBron, George {
  public void printName() {
   System.out.println("Default");
  }
}
non-sealed class Kyrie extends Human {
  public void printName() {
   System.out.println("Bob");
  }
}
sealed class LeBron extends Human {
  public void printName() {
   System.out.println("Mike");
  }
}
final class George extends Human {
  public void printName() {
   System.out.println("Yannick");
  }
}

在这个例子中,Human 是一个密封类,它只允许 Kyrie,LeBron 和 George 这三个类继承。这样,我们就可以更精确地控制哪些类可以继承 Human 类,从而提高代码的安全性和可读性。

1.17 统一日志异步刷新

在 JDK17 中,引入了一项新特性:统一日志异步刷新。

先将日志写入缓存,然后再异步地刷新到日志文件,这样写日志的操作就不会阻塞执行业务逻辑的线程,从而提高了程序的运行效率。这个特性对于需要大量日志输出,并且对性能有较高要求的应用来说,是一个非常实用的改进。可以通过传递命令行选项 -Xlog:async 来开启此功能。

总结

从 JDK11 到 JDK17 ,Java 的发展经历了一系列重要的里程碑。其中最重要的是 JDK17 的发布,这是一个长期支持(LTS)版本,它将获得长期的更新和支持,有助于保持程序的稳定性和可靠性。此外,Java 的性能也有了显著的提升。这些进步都反映了 Java 在持续改进和适应现代编程需求方面的承诺。

参考文档

  1. https://openjdk.org/projects/jdk/
  2. JDK12: JDK12新功能深度解析_jdk12新特性-CSDN博客
  3. switch statement - What does the new keyword "yield" mean in Java 13? - Stack Overflow
  4. New Features in Java 13 | Baeldung
  5. Java 11 and 12 - New Features (packtpub.com)
  6. Java 16 and IntelliJ IDEA | The IntelliJ IDEA Blog (jetbrains.com)
  7. Sealed Class in Java - GeeksforGeeks


Tags:JDK17   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,不构成投资建议。投资者据此操作,风险自担。如有任何标注错误或版权侵犯请与我们联系(Email:2595517585@qq.com),我们将及时更正、删除。
▌相关推荐
JDK17 与 JDK11 特性差异浅谈
从 JDK11 到 JDK17 ,Java 的发展经历了一系列重要的里程碑。其中最重要的是 JDK17 的发布,这是一个长期支持(LTS)版本,它将获得长期的更新和支持,有助于保持程序的稳定性和可靠性...【详细内容】
2024-01-26  Tags: JDK17  点击:(2)  评论:(0)  加入收藏
JDK11 升级 JDK17 最全实践干货来了
1、前言 上篇文章给大家带来了 JDK8 升级 JDK11 的最全实践,相信大家阅读后已经对 JDK11 有了比较深入的了解。2021 年 9 月 14 日,Oracle 发布了可以长期支持的 JDK17 版本,那...【详细内容】
2023-11-15  Tags: JDK17  点击:(158)  评论:(0)  加入收藏
▌简易百科推荐
JDK17 与 JDK11 特性差异浅谈
从 JDK11 到 JDK17 ,Java 的发展经历了一系列重要的里程碑。其中最重要的是 JDK17 的发布,这是一个长期支持(LTS)版本,它将获得长期的更新和支持,有助于保持程序的稳定性和可靠性...【详细内容】
2024-01-26  政采云技术  51CTO  Tags:JDK17   点击:(2)  评论:(0)  加入收藏
Java并发编程高阶技术
随着计算机硬件的发展,多核处理器的普及和内存容量的增加,利用多线程实现异步并发成为提升程序性能的重要途径。在Java中,多线程的使用能够更好地发挥硬件资源,提高程序的响应...【详细内容】
2024-01-19  大雷家吃饭    Tags:Java   点击:(14)  评论:(0)  加入收藏
这篇文章彻底让你了解Java与RPA
前段时间更新系统的时候,发现多了一个名为Power Automate的应用,打开了解后发现是一个自动化应用,根据其描述,可以自动执行所有日常任务,说的还是比较夸张,简单用了下,对于office、...【详细内容】
2024-01-17  Java技术指北  微信公众号  Tags:Java   点击:(16)  评论:(0)  加入收藏
Java 在 2023 年仍然流行的 25 个原因
译者 | 刘汪洋审校 | 重楼学习 Java 的过程中,我意识到在 90 年代末 OOP 正值鼎盛时期,Java 作为能够真正实现这些概念的语言显得尤为突出(尽管我此前学过 C++,但相比 Java 影响...【详细内容】
2024-01-10  刘汪洋  51CTO  Tags:Java   点击:(17)  评论:(0)  加入收藏
一文搞定JMM核心原理
您可以使用Java synchronized块。同步块保证在任何给定时间只有一个线程可以进入代码的给定关键部分。同步块还保证在同步块内访问的所有变量都将从主存储器中读入,当线程退...【详细内容】
2024-01-09  鲁大猿  微信公众号  Tags:JMM   点击:(11)  评论:(0)  加入收藏
如何用Java实现自动化测试和质量控制?
使用Java实现自动化测试和质量控制是现代软件开发中的重要环节。下面将详细介绍如何使用Java实现自动化测试和质量控制。一、自动化测试概述自动化测试是指使用软件工具和脚...【详细内容】
2024-01-03  编程技术汇  今日头条  Tags:Java   点击:(21)  评论:(0)  加入收藏
如何提高 Java 代码的可重用性
译者 | 刘汪洋审校 | 重楼对于软件开发者而言,编写可重用的代码是一项基本而重要的技能。每位工程师都应掌握如何尽可能地提高代码的复用性。当前,一些开发人员可能会认为微服...【详细内容】
2024-01-03    51CTO  Tags:Java   点击:(17)  评论:(0)  加入收藏
Java类的加载顺序是怎样的
1、虚拟机首次加载类时,会对静态初始块,静态对象,静态方法进行一次初始化。2、只有调用new方法时才会创建实例。3、类实例创建过程:首先执行父类初始化块部分,然后是父类的构造方...【详细内容】
2024-01-03  程序员界的彭于晏  今日头条  Tags:Java类   点击:(8)  评论:(0)  加入收藏
掌握Java并发编程,避免无处不在的竞态条件
掌握Java并发编程是编写高效、可靠的多线程应用程序的关键。竞态条件(Race Condition)是多线程环境下常见的问题,可能导致数据不一致、死锁等严重后果。下面将介绍Java并发编程...【详细内容】
2023-12-29  编程技术汇  今日头条  Tags:Java   点击:(41)  评论:(0)  加入收藏
深入掌握Java线程池调度策略,优化任务执行
在Java开发中,线程池是一种重要的并发处理机制。合理地使用线程池可以提高系统性能、响应速度和资源利用率。下面将深入掌握Java线程池的调度策略,介绍线程池的原理和常用的调...【详细内容】
2023-12-29  编程技术汇  今日头条  Tags:Java线程池   点击:(28)  评论:(0)  加入收藏
站内最新
站内热门
站内头条