Java 11的新功能和增强功能

Java 11(于2018年9月发布)包含许多重要且有用的更新。让我们看看它为开发人员和架构师带来的新功能和改进。

1. HTTP客户端API

Java有HttpURLConnection很长时间用于HTTP通信的类。但是随着时间的 Stream 逝,要求变得越来越复杂,对应用程序的要求也越来越高。在Java 11之前,开发人员不得不求助于功能丰富的库,例如Apache HttpComponentsOkHttp等。

我们看到Java 9发行版包含一个HttpClient作为实验性功能的实现。随着时间的 Stream 逝,它已经成为Java 11的最终功能。现在,Java应用程序可以进行HTTP通信,而无需任何外部依赖。

1.1。如何使用HttpClient

java.net.http模块的典型HTTP交互看起来像-

  • 创建HttpClient实例并根据需要进行配置。
  • 创建HttpRequest的实例并填充信息。
  • 将请求传递给客户端,执行请求,并检索HttpResponse的实例。
  • 处理中包含的信息HttpResponse

HTTP API可以处理同步和异步通信。让我们来看一个简单的例子。

1.2。同步请求示例

请注意,http客户端API如何使用构建器模式创建复杂对象。

import java.io.IOException;
import java.net.URI;
import java.net.http.HttpClient;
import java.net.http.HttpRequest;
import java.net.http.HttpResponse;
import java.time.Duration;

HttpClient httpClient = HttpClient.newBuilder()
				        .connectTimeout(Duration.ofSeconds(10))
				        .build();                                  

try 
{
    String urlEndpoint = "https://postman-echo.com/get";
    URI uri = URI.create(urlEndpoint + "?foo1=bar1&foo2=bar2");
    HttpRequest request = HttpRequest.newBuilder()
						            .uri(uri)
						            .build();                              
    HttpResponse<String> response = httpClient.send(request,
            							HttpResponse.BodyHandlers.ofString()); 
} catch (IOException | InterruptedException e) {
    throw new RuntimeException(e);
}

System.out.println("Status code: " + response.statusCode());                            
System.out.println("Headers: " + response.headers().allValues("content-type"));               
System.out.println("Body: " + response.body()); 

1.2。异步请求示例

如果我们不想等待响应,那么异步通信很有用。我们提供了回调处理程序,可在响应可用时执行。

注意,使用sendAsync()方法发送异步请求。

import java.io.IOException;
import java.net.URI;
import java.net.http.HttpClient;
import java.net.http.HttpRequest;
import java.net.http.HttpResponse;
import java.time.Duration;
import java.util.List;
import java.util.concurrent.CompletableFuture;
import java.util.stream.Stream;
import static java.util.stream.Collectors.toList;

final List<URI> uris = Stream.of(
				        "https://www.google.com/",
				        "https://www.github.com/",
				        "https://www.yahoo.com/"
				        ).map(URI::create).collect(toList());      

HttpClient httpClient = HttpClient.newBuilder()
				        .connectTimeout(Duration.ofSeconds(10))
				        .followRedirects(HttpClient.Redirect.ALWAYS)
				        .build();

CompletableFuture[] futures = uris.stream()
					        .map(uri -> verifyUri(httpClient, uri))
					        .toArray(CompletableFuture[]::new);     

CompletableFuture.allOf(futures).join();           

private CompletableFuture<Void> verifyUri(HttpClient httpClient, 
                                          URI uri) 
{
    HttpRequest request = HttpRequest.newBuilder()
						            .timeout(Duration.ofSeconds(5))
						            .uri(uri)
						            .build();

    return httpClient.sendAsync(request,HttpResponse.BodyHandlers.ofString())
			            .thenApply(HttpResponse::statusCode)
			            .thenApply(statusCode -> statusCode == 200)
			            .exceptionally(ex -> false)
			            .thenAccept(valid -> 
			            {
			                if (valid) {
			                    System.out.println("[SUCCESS] Verified " + uri);
			                } else {
			                    System.out.println("[FAILURE] Could not " + "verify " + uri);
			                }
			            });                                    
}

2.启动不编译的单文件程序

传统上,对于我们要执行的每个程序,我们都需要先对其进行编译。出于测试目的,小型程序似乎不必要地耗时。

Java 11对其进行了更改,现在我们可以执行单个文件中包含的Java源代码,而无需先对其进行编译。

public class HelloWorld {
    public static void main(String[] args) {
        System.out.println("Hello World!");
    }
}

要执行上述类,请直接使用java命令运行它。

$ java HelloWorld.java

Hello World!

需要注意的是该程序不能使用任何外部的依赖比其他java.base module。并且程序只能是单文件程序

3.字符串API的更改

3.1。String.repeat(整数)

此方法仅重复一个字符串 n时间。它返回一个字符串,其值是重复N次给定字符串的串联。

如果此字符串为空或count为零,则返回空字符串。

public class HelloWorld 
{
    public static void main(String[] args) 
    {
    	String str = "1".repeat(5);

        System.out.println(str);	//11111
    }
}

3.2。String.isBlank()

此方法指示字符串是空还是仅包含空格。以前,我们一直在Apache的中使用它StringUtils.java

public class HelloWorld 
{
    public static void main(String[] args) 
    {
    	"1".isBlank();	//false

        "".isBlank();	//true

        "    ".isBlank();	//true
    }
}

3.3。String.strip()

此方法需要除去前导和尾随空白。我们可以通过使用去除刚刚主人公更加具体String.stripLeading()或只是尾部字符使用String.stripTrailing() 

public class HelloWorld 
{
    public static void main(String[] args) 
    {
    	"   hi  ".strip();	//"hi"

       "   hi  ".stripLeading();	//"hi   "

       "   hi  ".stripTrailing();	//"   hi"
    }
}

3.4。String.lines()

此方法有助于将多行文本作为Stream处理

public class HelloWorld 
{
    public static void main(String[] args) 
    {
    	String testString = "hello\nworld\nis\nexecuted";

	    List<String> lines = new ArrayList<>();

	    testString.lines().forEach(line -> lines.add(line));

	    assertEquals(List.of("hello", "world", "is", "executed"), lines);
    }
}

4. Collection.toArray(IntFunction<t[]></t[]>

在Java 11之前,将集合转换为数组并不容易。Java 11使转换更加方便。

public class HelloWorld 
{
    public static void main(String[] args) 
    {
    	List<String> names = new ArrayList<>();
	    names.add("alex");
	    names.add("brian");
	    names.add("charles");

	    String[] namesArr1 = names.toArray(new String[names.size()]);		//Before Java 11

	    String[] namesArr2 = names.toArray(String[]::new);					//Since Java 11
    }
}

5. Files.readString()和Files.writeString()

使用这些重载方法,Java 11的目标是减少大量样板代码,从而使文件读写更加容易。

public class HelloWorld 
{
    public static void main(String[] args) 
    {
    	//Read file as string
    	URI txtFileUri = getClass().getClassLoader().getResource("helloworld.txt").toURI();

    	String content = Files.readString(Path.of(txtFileUri),Charset.defaultCharset());

    	//Write string to file
    	Path tmpFilePath = Path.of(File.createTempFile("tempFile", ".tmp").toURI());

    	Path returnedFilePath = Files.writeString(tmpFilePath,"Hello World!", 
    								Charset.defaultCharset(), StandardOpenOption.WRITE);
    }
}

6. Optional.isEmpty()

可选的是包含或不包含非null值的容器对象。如果不存在任何值,则该对象被认为是空的。

如果存在值,则先前存在的方法isPresent()返回true,否则返回false。有时,它迫使我们编写不可读的负面条件。

isEmpty()方法与isPresent()method 相反,false如果存在值,则返回false true

因此,在任何情况下我们都不会写负面条件。适当时使用这两种方法中的任何一种。

public class HelloWorld 
{
    public static void main(String[] args) 
    {
    	String currentTime = null;

	    assertTrue(!Optional.ofNullable(currentTime).isPresent());	//It's negative condition
	    assertTrue(Optional.ofNullable(currentTime).isEmpty());		//Write it like this

	    currentTime = "12:00 PM";

	    assertFalse(!Optional.ofNullable(currentTime).isPresent());	//It's negative condition
	    assertFalse(Optional.ofNullable(currentTime).isEmpty());	//Write it like this
    }
}

将有关Java 11中这些新API更改的问题交给我。

学习愉快!

参考:Java 11发布文档

saigon has written 1440 articles

Leave a Reply