System.out和System.err的并发

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

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对象,所以线程不同步。同时使用进行打印,会出现打印内容交叉的结果。
public class APP { public static void main(String[] args) throws Exception { System.out.println("1. 登录"); System.out.println("2. 注册"); System.out.print("请选择操作: "); int Select = Common.sc.nextInt(); if (Select == 2) { registerAccount(); return; } System.out.println("请输入账号"); String account = Common.sc.next(); System.out.println("请输入密码"); String password =Common.sc.next(); Class.forName("com.mysql.cj.jdbc.Driver"); Connection conn = DriverManager.getConnection( "jdbc:mysql://117.72.189.79 : 3306/jx2510" , "root" , "Fbb608425@"); String sql = "select * from user where account = ? and password = ?"; PreparedStatement ps = conn.prepareStatement(sql); ps.setString(1, account); ps.setString(2, password); ResultSet rs = ps.executeQuery(); if (rs.next()) { System.out.println("登录成功"); System.out.println(); System.out.println("请输入地图X轴边界"); int mapX = Common.sc.nextInt(); System.out.println("请输入地图Y轴边界"); int mapY = Common.sc.nextInt(); System.out.println("请输入老鼠数量"); int mouseNumber = Common.sc.nextInt(); GameMap gameMap = new GameMap(mapX, mapY, mouseNumber); Player player = new Player(); while (true) { gameMap.putMouse(); gameMap.showMap(); player.getPlayerInput(); MouseFather hitMouse = gameMap.getMap()[player.getHitX()][player.getHitY()]; if (hitMouse != null) { System.out.println("恭喜你,打中了..."); hitMouse.cry(); int type = hitMouse.getType(); player.setScore(player.getScore() + type); } else { System.out.println("很可惜,没打中..."); player.setScore(player.getScore() - 1); } System.out.println("当前玩家的分数为: " + player.getScore()); if (player.getScore() == 0 || player.getScore() >= 10) { System.out.println("游戏结束 " + (player.getScore() == 0 ? "游戏失败" : "游戏胜利")); System.out.println("是否重新开始游戏? 1-重新开始 其他-结束游戏"); int choice = Common.sc.nextInt(); if (choice == 1) { player.setScore(3); } else { System.out.println("再见,欢迎再次游玩"); break; } } gameMap.freshMap(); } }else { System.out.println("登陆失败"); } rs.close(); ps.close(); conn.close(); } public static void registerAccount() throws Exception { System.out.print("请输入账号: "); String account = Common.sc.next(); System.out.print("请输入密码: "); String password = Common.sc.next(); Class.forName("com.mysql.cj.jdbc.Driver"); try (Connection conn = DriverManager.getConnection( "jdbc:mysql://117.72.189.79:3306/jx2510", "root", "Fbb608425@")) { String checkSql = "SELECT COUNT(*) FROM user WHERE account = ?"; try (PreparedStatement checkPs = conn.prepareStatement(checkSql)) { checkPs.setString(1, account); try (ResultSet rs = checkPs.executeQuery()) { if (rs.next() && rs.getInt(1) > 0) { System.out.println("账号已存在,请选择其他账号!"); return; } } } String insertSql = "INSERT INTO user(account, password) VALUES(?, ?)"; try (PreparedStatement insertPs = conn.prepareStatement(insertSql)) { insertPs.setString(1, account); insertPs.setString(2, password); int affectedRows = insertPs.executeUpdate(); if (affectedRows > 0) { System.out.println("注册成功!"); } else { System.out.println("注册失败,请重试"); } } }catch (SQLException e) { System.err.println("数据库错误: " + e.getMessage()); e.printStackTrace(); } } }代码差错
最新发布
11-12
public class Todolist { import java.io.DataInputStream; import java.io.InputStream; import java.net.Socket; import java.util.ArrayList; import java.util.HashMap; import java.util.Scanner; public class SeverReaderThread extends Thread{ //下面是有参构造器,在这里再创建一个csoc接收处理 private Socket csoc; // 使用 HashMap 存储任务 private static HashMap<Integer, String> tasks = new HashMap<>(); private static int nextId = 1; private static int maxTasks; private static ArrayList<String> invalidInputs = new ArrayList<>(); //使用数组存储任务,定义为static,静态代码块:初始化最大任务数 static { Scanner sc = new Scanner(System.in); maxTasks = sc.nextInt(); // 吃掉换行符 sc.nextLine(); System.out.println("确定好了一共 " + maxTasks + " 条任务哦!"); } public SeverReaderThread(Socket csoc){ //把送进来的csoc管道给到上面的socket,把当前客户管道交给了一个独立的线程对象。 this.csoc=csoc; } @Override public void run() { //进阶任务:处理异常,得到输入流,包装成数据输入流 try{ InputStream is=csoc.getInputStream(); DataInputStream dis=new DataInputStream(is); while(true){ String msg=dis.readUTF(); //基础任务进阶任务,一共八个功能(有一个是defult里面记录用户错误输入的需求) // 客户端发送的消息一次性以空格拆分为最多三部分 // parts[0]=命令, [1]=ID, [2]=内容(可选) String[] parts = msg.split(" ", 3); // 第一部分:命令名 String command = parts[0]; // 第二部分:任务编号 int taskId = Integer.parseInt(parts[1]); // 第三部分:任务内容(可能没有) String content = (parts.length > 2) ? parts[2] : ""; switch (command) { case "add": if (tasks.size() >= maxTasks) { System.out.println("任务列表已满!"); } else { tasks.put(taskId, content); System.out.println("已添加 -> ID:" + taskId + ", 内容:'" + content + "'"); if (taskId >= nextId) { nextId = taskId + 1; } } break; case "list": System.out.println("--- 当前任务列表 ---"); if (tasks.isEmpty()) { System.out.println("暂无任务"); } else { for (int id : tasks.keySet()) { System.out.println(id + ": " + tasks.get(id)); } } break; case "modify": tasks.put(taskId, content); System.out.println("任务" + taskId + "已修改为: '" + content + "'"); break; case "delete": tasks.remove(taskId); System.out.println("任务" + taskId + "已删除"); break; case "clear": tasks.clear(); nextId = 1; System.out.println("所有任务已清空"); break; case "count": int count = tasks.size(); System.out.println("当前共有 " + count + " 个任务(最多可存 " + maxTasks + " 条)"); break; case "help": System.out.println("==== 帮助菜单 ===="); System.out.println("可用命令如下:"); System.out.println(" add 1 买菜 —— 添加ID=1的任务"); System.out.println(" modify 2 新内容 —— 修改ID=2的任务"); System.out.println(" delete 3 —— 删除ID=3的任务"); System.out.println(" list —— 查看全部任务"); System.out.println(" clear —— 清空所有任务"); System.out.println(" count —— 显示任务总数"); System.out.println(" help —— 显示此帮助信息"); break; default: // 把用户输入的原始消息保存到列表中 invalidInputs.add(msg); System.out.println("未知命令: '" + command + "'"+"抱歉" + "我目前只有这么多功能."); // 打印当前已记录多少条无效输入 System.out.println("但是我当前已记录 " + invalidInputs.size() + " 条功能之外的需求" + "我会努力增加的"); } } } catch (Exception e) { throw new RuntimeException(e); } } } }上面是我原有的代码,在我原有代码上面修改,变成一个完整的程序,尽量少的修改,简单初学这可以学会,提供最终完整答案,表明修改了哪里,只用java,每一行都解释
10-18
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
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值