java–流已经启动或关闭

本文介绍 Java8 中 Stream 的一次性特性,并演示如何通过 Supplier 方法解决 Stream 重复使用的问题,避免 IllegalStateException 异常。

在 Java 8, Stream 不能被重新使用, 一旦它被使用或者消费了, stream 将被关闭.

1. Example – Stream 已经被关闭!

 查看下面的例子 , 它将被抛出一个 IllegalStateException, 说 “stream is closed”.

TestJava8.java
package com.mkyong.java8;

import java.util.Arrays;
import java.util.stream.Stream;

public class TestJava8 {

    public static void main(String[] args) {

        String[] array = {"a", "b", "c", "d", "e"};
        Stream<String> stream = Arrays.stream(array);

        // loop a stream
        stream.forEach(x -> System.out.println(x));

        // reuse it to filter again! throws IllegalStateException
        long count = stream.filter(x -> "b".equals(x)).count();
        System.out.println(count);

    }

}

Output

java.lang.IllegalStateException: stream has already been operated upon or closed
	at java.util.stream.AbstractPipeline.(AbstractPipeline.java:203)
	at java.util.stream.ReferencePipeline.(ReferencePipeline.java:94)
	at java.util.stream.ReferencePipeline$StatelessOp.(ReferencePipeline.java:618)
	at java.util.stream.ReferencePipeline$2.(ReferencePipeline.java:163)
	at java.util.stream.ReferencePipeline.filter(ReferencePipeline.java:162)
	at com.hostingcompass.whois.range.run.TestJava8.main(TestJava8.java:25)
	at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
	at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
	at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
	at java.lang.reflect.Method.invoke(Method.java:498)
	at com.intellij.rt.execution.application.AppMain.main(AppMain.java:144)

2. Example – 重新运用一个 stream

无论因为什么原因, 你真的想重新运用一个Stream, 可以使用下面的方式解决:

TestJava8.java
package com.mkyong.java8;

import java.util.function.Supplier;
import java.util.stream.Stream;

public class TestJava8 {

    public static void main(String[] args) {

        String[] array = {"a", "b", "c", "d", "e"};

        Supplier<Stream<String>> streamSupplier = () -> Stream.of(array);

        //get new stream
        streamSupplier.get().forEach(x -> System.out.println(x));

        //get another new stream
        long count = streamSupplier.get().filter(x -> "b".equals(x)).count();
        System.out.println(count);

    }

}

Output

a
b
c
d
e
1
每一个get() 将返回一个新的 stream.
启动Java进程有多种方式,以下为不同场景下的启动方法: - **在Android系统上**:Android上可通过`app_process`启动Java进程,该进程是从`/system/bin/sh(shell)` fork出来的,其进程启动路径为`/init–>/sbin/adbd–>/system/bin/sh–>app_process(Java进程)` [^1]。 - **以服务进程方式启动**:在服务目录下(jar包目录下,以服务名`transfer`为例),执行`nohup java -jar transfer.jar &`命令启动服务。使用这种方式运行的程序日志会输出到当前目录下的`nohup.out`文件,使用`ctrl+c`中断关闭窗口都不会中断程序的执行 [^2]。 - **通过`Runtime`类**:每个Java应用程序都有一个类实例`Runtime`,允许应用程序与运行环境进行交互,当前运行时间可以从`getRuntime`方法中获取,不过应用程序无法创建自己的此类实例 [^3]。 以下是通过`Runtime`类执行外部命令启动Java进程的示例代码: ```java import java.io.IOException; public class JavaProcessLauncher { public static void main(String[] args) { try { // 启动Java进程,这里以启动一个简单的Java程序为例 Process process = Runtime.getRuntime().exec("java -version"); // 获取进程的输入,可用于读取进程的输出信息 java.io.InputStream inputStream = process.getInputStream(); java.io.BufferedReader reader = new java.io.BufferedReader(new java.io.InputStreamReader(inputStream)); String line; while ((line = reader.readLine()) != null) { System.out.println(line); } // 等待进程执行完毕 int exitCode = process.waitFor(); System.out.println("Process exited with code " + exitCode); } catch (IOException | InterruptedException e) { e.printStackTrace(); } } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值