System.out和System.err的并发

本文探讨了Java中System.out和System.err并发打印时出现的内容交叉现象,并提供了通过设置两者为同一输出流来解决该问题的方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

System.out和System.err的并发

最近在开发时发现,同时使用System.out和System.err打印内容,会出现多线程的现象。

代码如下:

public class SystemDemo {

    public static void main(String[] args) {

        for (int i = 0; i < 5; i++) {
            System.out.println("+++++");
            // System.out.flush();
            System.err.println("-----");
        }
    }
}

运行结果类似于: 

查阅资料的解释是:

  1. 标准输出和标准出错的一个区别是,标准输出往往是带缓存的,而标准出错没有缓存。 但是我们通过添加System.out.flush发现,现象依然存在。 --> 所以不是缓存的原因

  2. 另一个解释是,They are different OutputStreams. 建议的方法:

       System.setErr(System.out);

    或者

       System.setOut(System.err);

    发现的确是可以的:

    private static void test02() {
        System.setErr(System.out);
        // System.setOut(System.err);
        for (int i = 0; i < 5; i++) {
            System.out.println("+++++");
            System.err.println("-----");
        }
    }

    运行结果: 

总结:

  • System.out和System.err使用不同的printstream对象,所以线程不同步。同时使用进行打印,会出现打印内容交叉的结果。
package org.yqq.server.smpp; import org.jsmpp.session.SMPPServerSession; import org.springframework.stereotype.Component; import javax.annotation.PreDestroy; import java.io.IOException; import java.net.ServerSocket; import java.net.Socket; import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; @Component public class SmppServer implements Runnable { // @Value("${smpp.server.port:2775}") private int port = 7788; // @Value("${smpp.server.systemid:systemid}") private String systemId = "systemid"; // @Value("${smpp.server.password:password}") private String password = "123"; private final ExecutorService executor = Executors.newCachedThreadPool(); private ServerSocket serverSocket; private volatile boolean running = true; // 跟踪活动会话 private final ConcurrentHashMap<SMPPServerSession, Boolean> activeSessions = new ConcurrentHashMap<>(); @Override public void run() { try { serverSocket = new ServerSocket(port); System.out.println("SMPP Server started on port " + port); //优雅关闭 serverSocket.setSoTimeout(1000); while (running) { Socket socket = serverSocket.accept(); System.out.println("New client connected: " + socket.getInetAddress()); executor.execute(new SessionHandler(socket,activeSessions)); } } catch (IOException e) { if (running) { System.err.println("Server error: " + e.getMessage()); } } } @PreDestroy public void stop() { running = false; // 关闭所有活动会话 activeSessions.keySet().forEach(session -> { try { if (session.getSessionState().isBound()) { session.unbind(); } session.close(); } catch (Exception e) { System.err.println("Error closing session: " + e.getMessage()); } }); executor.shutdown(); if (serverSocket != null) { try { serverSocket.close(); } catch (IOException e) { System.err.println("Error closing server socket: " + e.getMessage()); } } System.out.println("SMPP Server stopped"); } } 执行run方法会报错 具体的这一行 Socket socket = serverSocket.accept(); 然后进入了catch 执行了 System.err.println("Server error: " + e.getMessage()); 控制台输出Server error: Accept timed out
最新发布
08-08
package 网络.核心类; import java.net.*; import java.io.*; import java.util.Scanner; public class Client { public static void main(String[] args) { try (Socket socket = new Socket("localhost", 12345); // 连接到服务器的IP端口 PrintWriter out = new PrintWriter(socket.getOutputStream(), true); // 用于发送数据 BufferedReader in = new BufferedReader(new InputStreamReader(socket.getInputStream())); // 用于接收数据 Scanner scanner = new Scanner(System.in)) { // 用于读取键盘输入 System.out.println("Connected to server. Type your message and press enter to send. Type 'exit' to quit."); while (true) { System.out.print("Client: "); String userInput = scanner.nextLine(); // 读取用户输入 if (userInput.equalsIgnoreCase("exit")) { System.out.println("Exiting client..."); break; // 退出循环 } // TODO: 扩展用户输入处理,支持不同命令(如 "echo <消息>"、"time"、"quit")。例如,提示用户可用命令并根据输入类型发送适当的消息。 // 提示:您可以在这里添加代码来解析用户输入,并修改发送逻辑。 out.println(userInput); // 通过Socket发送消息到服务器 String response = in.readLine(); // 通过Socket读取服务器响应 System.out.println("Server response: " + response); // 显示响应 // TODO: 扩展响应处理,根据服务器返回的不同响应类型进行处理(如显示时间或错误消息)。确保处理 null 或异常情况。 } } catch (IOException e) { System.err.println("Connection error: " + e.getMessage()); // 处理连接异常 } } } package 网络.核心类; import java.net.*; import java.io.*; public class Server { public static void main(String[] args) { try (ServerSocket serverSocket = new ServerSocket(12345)) { // 绑定到端口12345 System.out.println("Server started. Listening on port 12345..."); while (true) { // 无限循环,等待客户端连接 Socket clientSocket = serverSocket.accept(); // 接受客户端连接,返回一个Socket对象 System.out.println("Client connected: " + clientSocket.getInetAddress()); // 使用Socket处理客户端通信(这里使用独立的线程处理每个客户端,避免阻塞) new Thread(() -> { try (BufferedReader in = new BufferedReader(new InputStreamReader(clientSocket.getInputStream())); PrintWriter out = new PrintWriter(clientSocket.getOutputStream(), true)) { String message; while ((message = in.readLine()) != null) { // 读取客户端发送的消息 System.out.println("Received from client: " + message); // TODO: 扩展消息处理逻辑,支持简单协议。例如,根据消息类型执行操作: // - 如果消息以 "echo" 开头,返回回显消息。 // - 如果消息是 "time",返回当前系统时间。 // - 如果消息是 "quit",断开连接。 // 提示:使用 if-else 语句或 switch 表达式来处理不同命令。确保处理无效命令,返回错误消息。 out.println("Echo: " + message); // 默认回显逻辑,学生需要修改此部分 } } catch (IOException e) { System.err.println("Error handling client: " + e.getMessage()); } finally { try { clientSocket.close(); // 关闭Socket } catch (IOException e) { e.printStackTrace(); } } }).start(); // 启动新线程处理客户端 } } catch (IOException e) { System.err.println("Server error: " + e.getMessage()); // 处理异常,如端口被占用 } } }补充完善上述代码
06-12
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值