基于SpringBoot实现多线程多主机TCP通信

 更新时间:2025年09月19日 14:50:15   作者:IT界Tony哥  
这篇文章主要为大家详细介绍了如何基于SpringBoot实现多线程多主机TCP通信,包括发送数据和接收应答并解析,感兴趣的小伙伴可以跟随小编一起学习一下

下面我将介绍如何使用 Spring Boot 实现多线程、多主机的 TCP 通信,包括发送数据和接收应答并解析。

1. 项目结构

src/main/java/com/example/tcpdemo/
├── config
│   └── TcpClientConfig.java
├── controller
│   └── TcpController.java
├── handler
│   ├── TcpClientHandler.java
│   └── TcpResponseHandler.java
├── model
│   └── TcpHost.java
├── service
│   ├── TcpClientService.java
│   └── impl
│       └── TcpClientServiceImpl.java
└── TcpDemoApplication.java

2. 核心代码实现

2.1 配置类

// TcpClientConfig.java
@Configuration
public class TcpClientConfig {
    
    @Value("${tcp.client.thread-pool-size:10}")
    private int threadPoolSize;
    
    @Bean
    public ExecutorService tcpClientExecutor() {
        return Executors.newFixedThreadPool(threadPoolSize);
    }
}

2.2 TCP主机模型

// TcpHost.java
@Data
@AllArgsConstructor
@NoArgsConstructor
public class TcpHost {
    private String host;
    private int port;
    private String name; // 主机标识名称
}

2.3 TCP客户端处理器

// TcpClientHandler.java
@Component
public class TcpClientHandler {
    
    @Autowired
    private ExecutorService tcpClientExecutor;
    
    @Autowired
    private TcpResponseHandler responseHandler;
    
    public void sendToMultipleHosts(List<TcpHost> hosts, String message) {
        hosts.forEach(host -> {
            tcpClientExecutor.execute(() -> {
                try (Socket socket = new Socket(host.getHost(), host.getPort());
                     PrintWriter out = new PrintWriter(socket.getOutputStream(), true);
                     BufferedReader in = new BufferedReader(new InputStreamReader(socket.getInputStream()))) {
                    
                    // 发送数据
                    out.println(message);
                    System.out.println("Sent to " + host.getName() + ": " + message);
                    
                    // 接收响应
                    String response = in.readLine();
                    responseHandler.handleResponse(host, response);
                    
                } catch (IOException e) {
                    System.err.println("Error communicating with " + host.getName() + ": " + e.getMessage());
                }
            });
        });
    }
}

2.4 响应处理器

// TcpResponseHandler.java
@Component
public class TcpResponseHandler {
    
    public void handleResponse(TcpHost host, String response) {
        // 这里实现你的响应解析逻辑
        System.out.println("Received from " + host.getName() + ": " + response);
        
        // 示例解析逻辑
        if (response != null) {
            // 假设响应格式为 "status|data"
            String[] parts = response.split("\|");
            if (parts.length == 2) {
                String status = parts[0];
                String data = parts[1];
                System.out.println("Parsed response - Status: " + status + ", Data: " + data);
            }
        }
    }
}

2.5 TCP服务接口

// TcpClientService.java
public interface TcpClientService {
    void sendToHosts(List<TcpHost> hosts, String message);
}

2.6 TCP服务实现

// TcpClientServiceImpl.java
@Service
public class TcpClientServiceImpl implements TcpClientService {
    
    @Autowired
    private TcpClientHandler tcpClientHandler;
    
    @Override
    public void sendToHosts(List<TcpHost> hosts, String message) {
        tcpClientHandler.sendToMultipleHosts(hosts, message);
    }
}

2.7 控制器

// TcpController.java
@RestController
@RequestMapping("/api/tcp")
public class TcpController {
    
    @Autowired
    private TcpClientService tcpClientService;
    
    @PostMapping("/send")
    public ResponseEntity<String> sendMessage(@RequestBody Map<String, Object> request) {
        List<Map<String, Object>> hostsInfo = (List<Map<String, Object>>) request.get("hosts");
        String message = (String) request.get("message");
        
        List<TcpHost> hosts = hostsInfo.stream()
            .map(info -> new TcpHost(
                (String) info.get("host"),
                (Integer) info.get("port"),
                (String) info.get("name")))
            .collect(Collectors.toList());
        
        tcpClientService.sendToHosts(hosts, message);
        
        return ResponseEntity.ok("Messages sent successfully");
    }
}

3. 应用配置

application.propertiesapplication.yml中添加配置:

# TCP客户端线程池大小
tcp.client.thread-pool-size=20

# 服务器端口
server.port=8080

4. 使用示例

4.1 启动Spring Boot应用

// TcpDemoApplication.java
@SpringBootApplication
public class TcpDemoApplication {
    public static void main(String[] args) {
        SpringApplication.run(TcpDemoApplication.class, args);
    }
}

4.2 发送请求

使用Postman或curl发送POST请求:

curl -X POST http://localhost:8080/api/tcp/send \
-H "Content-Type: application/json" \
-d '{
    "hosts": [
        {"host": "127.0.0.1", "port": 12345, "name": "Server1"},
        {"host": "127.0.0.1", "port": 12346, "name": "Server2"}
    ],
    "message": "Hello TCP Server"
}'

5. 高级功能扩展

5.1 连接池管理

// 可以扩展为使用连接池管理TCP连接
@Component
public class TcpConnectionPool {
    private Map<String, Socket> connectionPool = new ConcurrentHashMap<>();
    
    public Socket getConnection(TcpHost host) throws IOException {
        String key = host.getHost() + ":" + host.getPort();
        if (!connectionPool.containsKey(key) || connectionPool.get(key).isClosed()) {
            connectionPool.put(key, new Socket(host.getHost(), host.getPort()));
        }
        return connectionPool.get(key);
    }
    
    public void closeAll() {
        connectionPool.values().forEach(socket -> {
            try {
                socket.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        });
        connectionPool.clear();
    }
}

5.2 超时设置

// 在TcpClientHandler中添加超时设置
public void sendToMultipleHosts(List<TcpHost> hosts, String message) {
    hosts.forEach(host -> {
        tcpClientExecutor.execute(() -> {
            try {
                Socket socket = new Socket();
                socket.connect(new InetSocketAddress(host.getHost(), host.getPort()), 5000); // 5秒连接超时
                socket.setSoTimeout(10000); // 10秒读取超时
                
                try (PrintWriter out = new PrintWriter(socket.getOutputStream(), true);
                     BufferedReader in = new BufferedReader(new InputStreamReader(socket.getInputStream()))) {
                    
                    out.println(message);
                    System.out.println("Sent to " + host.getName() + ": " + message);
                    
                    String response = in.readLine();
                    responseHandler.handleResponse(host, response);
                }
            } catch (SocketTimeoutException e) {
                System.err.println("Timeout when communicating with " + host.getName());
            } catch (IOException e) {
                System.err.println("Error communicating with " + host.getName() + ": " + e.getMessage());
            }
        });
    });
}

5.3 自定义协议解析

// 扩展TcpResponseHandler实现更复杂的协议解析
public void handleResponse(TcpHost host, String response) {
    try {
        // 示例:解析JSON格式响应
        JSONObject jsonResponse = new JSONObject(response);
        String status = jsonResponse.getString("status");
        String data = jsonResponse.getString("data");
        long timestamp = jsonResponse.getLong("timestamp");
        
        System.out.printf("Response from %s - Status: %s, Data: %s, Time: %tF %<tT%n",
            host.getName(), status, data, new Date(timestamp));
        
    } catch (JSONException e) {
        System.err.println("Invalid response format from " + host.getName());
    }
}

6. 注意事项

  • ​线程安全​​:确保在多线程环境下共享资源的线程安全
  • ​资源释放​​:正确关闭Socket、流等资源
  • ​异常处理​​:合理处理各种网络异常
  • ​性能优化​​:根据实际需求调整线程池大小
  • ​日志记录​​:添加详细的日志记录以便排查问题

这个实现提供了基本的TCP多线程通信框架,你可以根据实际需求进行扩展和优化。

到此这篇关于基于SpringBoot实现多线程多主机TCP通信的文章就介绍到这了,更多相关SpringBoot TCP通信内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Spring boot2X Consul如何通过RestTemplate实现服务调用

    Spring boot2X Consul如何通过RestTemplate实现服务调用

    这篇文章主要介绍了spring boot2X Consul如何通过RestTemplate实现服务调用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2019-12-12
  • Java中Elasticsearch的核心概念详解

    Java中Elasticsearch的核心概念详解

    这篇文章主要介绍了Java中Elasticsearch的核心概念详解,Elasticsearch 是一个分布式、免费和开放的搜索和分析引擎,适用于所有类型的数据,包括文本、数字、地理空间、结构化和非结构化数据,需要的朋友可以参考下
    2023-07-07
  • Java详细讲解IO流的Writer与Reader操作

    Java详细讲解IO流的Writer与Reader操作

    Writer与Reader类不能直接调用,需要使用多带的方法调用它们的子类,在他们的前边加上一个File即可如(FileWriter或FileReader)的多态方法进行其调用,并且他们也是抽象类调用需要连接接口Exception,它们的优点在于可以直接写入或读出内容,不需要使用byte转八进制
    2022-05-05
  • 详解SpringBoot时间参数处理完整解决方案

    详解SpringBoot时间参数处理完整解决方案

    这篇文章主要介绍了详解SpringBoot时间参数处理完整解决方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-12-12
  • JDK反序列化时修改类的全限定性名解析

    JDK反序列化时修改类的全限定性名解析

    这篇文章主要介绍了JDK反序列化时修改类的全限定性名解析,具有一定借鉴价值,需要的朋友可以参考下。
    2017-12-12
  • Spring项目中使用Cache Redis实现数据缓存

    Spring项目中使用Cache Redis实现数据缓存

    这篇文章主要为大家介绍了项目中使用Spring Cache Redis实现数据缓存,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-06-06
  • Java JVM内存区域详解

    Java JVM内存区域详解

    下面小编就为大家带来一篇基于jvm java内存区域的介绍。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2021-11-11
  • 简单分析Java的求值策略原理

    简单分析Java的求值策略原理

    在本篇文章里小编给大家整理的是一篇关于简单分析Java的求值策略原理内容,有需要的朋友们可以学习下。
    2021-06-06
  • IntelliJ IDEA2020.1 Mac maven sdk 全局配置

    IntelliJ IDEA2020.1 Mac maven sdk 全局配置

    这篇文章主要介绍了IntelliJ IDEA2020.1 Mac maven sdk 全局配置,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-06-06
  • java判断是否空最简单的方法

    java判断是否空最简单的方法

    在本篇文章里小编给大家整理的一篇关于java判断是否空最简单的方法,有兴趣的读者们可以参考下。
    2019-12-12

最新评论