java实现BIO服务器端网络编程(在springboot中使用)

  1. 创建一个类BIOServer
@Component
public  class BioServer {
    @Autowired
    AsyncService asyncService;

    public ServerSocket  serverSocket = null;
    public  void BioServerListen() throws IOException {
        try{
            serverSocket = new ServerSocket(8888);
            System.out.println("服务启动了");
            while(true){
                //监听,等待客户端连接
                final Socket socket = serverSocket.accept();
                System.out.println("连接到一个客户端");
                //创建一个线程,与之通信
                asyncService.dataHandler(socket);
            }
        }catch (IOException exception){

        }
    }
}
  1. AsyncService(为了偷懒没有接口+实现类的形式),调用其中使用了 异步方法处理,从而实现多线程可连接多个客户端
package com.mql.nettybio.BioServer;


import com.mql.nettybio.mapper.AdcpMapper;
import com.mql.nettybio.bean.ADCP_DataInfo;
import com.mql.nettybio.config.GetDataInfo;
import org.apache.log4j.Logger;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.scheduling.annotation.Async;
import org.springframework.stereotype.Component;

import java.io.IOException;
import java.io.InputStream;
import java.net.Socket;
import java.text.ParseException;


@Component
public class AsyncService {
    @Autowired
    AdcpMapper adcpMapper;//mapper接口,可忽略

	//日志操作,可忽略
    private Logger logger = Logger.getLogger(AsyncService.class);

	//开启异步方法的注解 里面的名字将于下文中的Bean实例名字必须相同
    @Async("asyncServiceExecutor")
    public void dataHandler(Socket socket){
        System.out.println("Spring自带的线程池" + Thread.currentThread().getName());
         InputStream inputStream = null;
        try {
            byte[] bytes = new byte[1024];
            inputStream = socket.getInputStream();
            while(true){
                int read = inputStream.read(bytes);
                if(read!=-1){
                    String dataTemp = new String (bytes,0,read,"GBK");

                    logger.info(dataTemp);
					//这里是我获取到数据添加到数据库,可忽略
                    /*ADCP_DataInfo dataInfo = GetDataInfo.getDataInfo(dataTemp.trim());

                    if(dataInfo!=null){
                        if(adcpMapper.insertDataInfo(dataInfo)==1){
                            logger.info("新增成功");
                        }else {
                            logger.info("新增失败");
                        }
                    }else {
                        logger.info("无效数据");
                    }*/

                }else {
                    break;
                }
            }


        } catch (IOException | ParseException exception) {
            exception.printStackTrace();
            logger.error(exception);
        }finally {
            System.out.println("关闭和client的连接");
            logger.info("关闭和client的连接");
            try{
                if(inputStream!=null){
                    inputStream.close();
                }
            }catch (Exception exception){
                exception.printStackTrace();
            }

            try{
                if(socket!=null){
                    socket.close();
                }
            }catch (Exception exception){
                exception.printStackTrace();
            }
        }
    }
}

  1. 异步线程的配置类
package com.mql.nettybio.config;

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.annotation.EnableAsync;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;

import java.util.concurrent.Executor;
import java.util.concurrent.ThreadPoolExecutor;

@Configuration
@EnableAsync
public class ExecutorConfig {


    @Bean //asyncServiceExecutor与上文中的注解中名字相同
    public Executor asyncServiceExecutor() {

        //申请一个线程池 threadPoolTas
        ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
        //配置核心线程数
        executor.setCorePoolSize(5);
        //配置最大线程数
        executor.setMaxPoolSize(10);
        //配置队列大小
        executor.setQueueCapacity(99999);
        //配置线程池中的线程的名称前缀
        executor.setThreadNamePrefix("async-service-");

        // rejection-policy:当pool已经达到max size的时候,如何处理新任务
        // CALLER_RUNS:不在新线程中执行任务,而是有调用者所在的线程来执行
        executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
        //执行初始化
        executor.initialize();
        return executor;
    }
}
  1. 在SpiringBoot启动类中调用
package com.mql.nettybio;

import com.mql.nettybio.BioServer.BioServer;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.scheduling.annotation.EnableAsync;
import org.springframework.scheduling.annotation.EnableScheduling;

import java.io.IOException;

@SpringBootApplication

@EnableScheduling  //表示开启异步线程使用(不可省略),会导致异步方法无法起作用
public class NettyBioApplication {


    public static void main(String[] args) throws IOException {
        SpringApplication.run(NettyBioApplication.class, args).getBean(BioServer.class).BioServerListen();
        //这里不能用new BioServer() 去调用BioServerListen(); 因为new出来的实例没有被ioc容器接管,会报错
    }

}
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在Java,网络IO的实现方式有BIO、NIO、AIO三种。 1. BIO(Blocking I/O) BIOJava最早的一种网络IO实现方式。它通过阻塞式的方式来进行网络IO操作,即一个线程在进行IO操作时会一直阻塞,直到IO操作完成。这种方式的优点是编程简单易懂,但缺点是并发性能较差,不能满足高并发的需求。 2. NIO(Non-blocking I/O) NIO是Java在1.4版本引入的一种新的网络IO实现方式。NIO的核心是Selector(选择器),它能够监控多个通道的状态,当某个通道有数据可读或可写时,Selector会通知相应的线程进行处理。NIO采用非阻塞式的方式进行IO操作,可以支持更高的并发性能。但是,NIO编程难度较大,需要掌握较多的概念和技巧。 3. AIO(Asynchronous I/O) AIO是Java在1.7版本引入的一种新的网络IO实现方式。AIO采用异步的方式进行IO操作,即一个线程在进行IO操作时不需要一直阻塞,而是可以继续执行其他任务,当IO操作完成后再由系统通知线程进行处理。AIO可以支持更高的并发性能,并且编程模型比NIO更加简单。但是,AIO的兼容性不如NIO,需要在操作系统和JVM等多个方面进行支持。 ### 回答2: 在Java,网络IO可以通过不同的方式实现,这些方式包括BIO(阻塞IO)、NIO(非阻塞IO)和AIO(异步IO)。 BIO是最早的实现方式,也是最简单直观的方式。它的特点是使用阻塞模式,即当一个线程在进行IO操作时,其他线程必须等待IO操作完成才能继续执行。在BIO,每个连接都需要一个独立的线程来处理,这可能导致服务器资源浪费,无法满足高并发的需求。 NIO是在Java 1.4引入的新IO模型,相较于BIO,它具有更高的并发处理能力。NIO使用了多路复用器(Selector)来管理多个连接,一个线程可以通过一个选择器同时处理多个连接的IO操作,从而避免了每个连接都需要一个独立线程的问题。 AIO是在Java 1.7引入的新IO模型,也被称为NIO.2。AIO是基于事件和回调机制的,它的特点是IO操作后不需要对应的线程进行阻塞等待,而是通过回调方式来通知IO操作已经完成。AIO适合处理连接数较多且连接时间较长的场景,例如聊天服务器。 综上所述,BIO、NIO和AIO是Java实现网络IO的三种方式。BIO适用于连接数较小的场景,NIO适用于连接数较多但连接时间较短的场景,而AIO适用于连接数较多且连接时间较长的场景。选择适合的IO模型能够提高服务器的并发处理能力和效率。 ### 回答3: JAVA有三种主要的网络IO实现方式,分别是BIO(Blocking IO)、NIO(Non-blocking IO)和AIO(Asynchronous IO)。 BIOJAVA IO最传统的模型,采用阻塞方式进行IO操作。在BIO模型,每个连接都需要一个独立的线程来处理,当有大量的连接时,就需要大量线程,会导致性能下降。 NIO是JAVA IO的改进版本,引入了通道(Channel)和缓冲区(Buffer)的概念。NIO采用非阻塞方式进行IO操作,可以提高IO的效率。在NIO模型,一个线程可以处理多个连接,通过选择器(Selector)来监听多个通道的事件,当有事件发生时,线程可以进行处理。 AIO是JAVA NIO的进一步改进,引入了异步通道(AsynchronousChannel)和回调机制。AIO采用异步方式进行IO操作,可以在IO操作完成之后再通知线程进行处理,而不需要线程一直等待IO操作完成。AIO适用于高并发的场景,可以大大提高系统的吞吐量和性能。 总结来说,BIO适用于连接数较少且业务处理较简单的场景;NIO适用于连接数较多但每个连接的并发处理量不大的场景;AIO适用于连接数较多且每个连接的并发处理量大的场景。选择合适的IO模型可以根据业务需求和实际场景进行选择,来提高系统的性能和并发能力。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值