• Http 调用netty 服务,服务调用客户端,伪同步响应.ProtoBuf 解决粘包,半包问题.


    实际情况是: 公司需要开发一个接口给新产品使用,需求如下

    1.有一款硬件设备,客户用usb接上电脑就可以,但是此设备功能比较单一,所以开发一个服务器程序,辅助此设备业务功能

    2.解决方案,使用Socket调用此设备

    3.增强此设备功能,增加Socket客户端连接到Socket服务端

    4.Http请求,同步响应

    测试注意:

    1.nettyServer 在ubuntu下编码,使用Epoll

    2.Http请求的测试最好运行再Linux 下进行,因为Windows 可能会因为并发高的时候占满端口限制,HttpClient或者RestTemplate 请求不了.

    3.ProtoBuf 插件无论再Windows,还是linux同样适用,在linux 下,会自动下载 protoc-3.5.1-linux-x86_64.exe

    简单的流程如下

    解决方案:

    1.使用Netty框架

    2.使用ProtoBuf,配合Netty 对ProtoBuf解决半包问题

    3.Future 实现伪同步响应

    4.SpringBoot + jetty

    pom.xml 添加ProtoBuf依赖以及插件

     <properties>
            <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
            <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
            <java.version>1.8</java.version>
            <grpc.version>1.11.0</grpc.version>
            <protobuf.version>3.5.1</protobuf.version>
        </properties>
      <dependency>
                <groupId>com.google.protobuf</groupId>
                <artifactId>protobuf-java</artifactId>
                <version>${protobuf.version}</version>
            </dependency>
            <dependency>
                <groupId>io.grpc</groupId>
                <artifactId>grpc-netty</artifactId>
                <version>${grpc.version}</version>
                <scope>provided</scope>
            </dependency>
            <dependency>
                <groupId>io.grpc</groupId>
                <artifactId>grpc-protobuf</artifactId>
                <version>${grpc.version}</version>
                <scope>provided</scope>
            </dependency>
            <dependency>
                <groupId>io.grpc</groupId>
                <artifactId>grpc-stub</artifactId>
                <version>${grpc.version}</version>
                <scope>provided</scope>
            </dependency>
    
            <dependency>
                <groupId>com.googlecode.protobuf-java-format</groupId>
                <artifactId>protobuf-java-format</artifactId>
                <version>1.4</version>
            </dependency>

    插件

        <build>
            <extensions>
                <extension>
                    <groupId>kr.motd.maven</groupId>
                    <artifactId>os-maven-plugin</artifactId>
                    <version>1.5.0.Final</version>
                </extension>
            </extensions>
            <plugins>
                <plugin>
                    <groupId>org.apache.maven.plugins</groupId>
                    <artifactId>maven-compiler-plugin</artifactId>
                    <configuration>
                        <source>1.8</source>
                        <target>1.8</target>
                    </configuration>
                </plugin>
                <plugin>
                    <groupId>org.xolstice.maven.plugins</groupId>
                    <artifactId>protobuf-maven-plugin</artifactId>
                    <version>0.5.1</version>
                    <configuration>
                        <protocArtifact>com.google.protobuf:protoc:${protobuf.version}:exe:${os.detected.classifier}</protocArtifact>
                        <pluginId>grpc-java</pluginId>
                        <pluginArtifact>io.grpc:protoc-gen-grpc-java:${grpc.version}}:exe:${os.detected.classifier}</pluginArtifact>
                    </configuration>
                    <executions>
                        <execution>
                            <goals>
                                <goal>compile</goal>
                                <goal>compile-custom</goal>
                            </goals>
                        </execution>
                    </executions>
                </plugin>
            </plugins>
        </build>

     屏蔽Tomcat 使用 Jetty

            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-starter-web</artifactId>
                <exclusions>
                    <exclusion>
                        <groupId>org.springframework.boot</groupId>
                        <artifactId>spring-boot-starter-tomcat</artifactId>
                    </exclusion>
                </exclusions>
            </dependency>
    
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-starter-jetty</artifactId>
            </dependency>

    编写proto:再/src/main中创建文件夹 proto,创建一个Message.proto

    文件内容

    syntax = "proto3";
    option java_package = "com.lzw.netty";
    option java_outer_classname = "MessageProto";
    message Message {
    
        int32 type = 1;
    
        sfixed64 id = 2;
    
        string msgBody = 3;
    
        enum Type {
            ACTIVE = 0;
            MESSAGE = 1;
        }
    
    }

    生成java 文件

    文件目录,挪到自己需要的包下面

    服务端代码

    /**
     * User: laizhenwei
     * Date: 2018-03-26 Time: 21:46
     * Description:
     */
    public class EchoServer {
    
        //缓存ResponseFuture
        public static Map<Long, ResponseFuture<MessageProto.Message>> responseFutureMap = new HashMap<>();
    
        private final int port;
    
        public EchoServer(int port) {
            this.port = port;
        }
    
        public void start() throws InterruptedException {
    
            EventLoopGroup bossGroup = new EpollEventLoopGroup(1);
            EventLoopGroup workerGroup = new EpollEventLoopGroup();
            ServerBootstrap bootstrap = new ServerBootstrap();
            bootstrap.group(bossGroup,workerGroup).channel(EpollServerSocketChannel.class)
                    .localAddress(new InetSocketAddress(port))
                    .childHandler(new MyServerChannelInitializer());
            try {
                ChannelFuture f = bootstrap.bind().sync();
                //清理不可预知而失败的脏数据
                f.channel().eventLoop().scheduleAtFixedRate(() -> {
                    long nowTime = System.currentTimeMillis();
                    responseFutureMap.entrySet().stream().filter(e -> (nowTime - e.getValue().getBeginTime()) > 60000).map(e -> e.getKey()).forEach(k->responseFutureMap.remove(k));
                }, 300, 300, TimeUnit.SECONDS);
                f.channel().closeFuture().sync();
            } finally {
                bossGroup.shutdownGracefully().sync();
                workerGroup.shutdownGracefully().sync();
            }
        }
    }

    ContextHelper缓存ChannelHandlerContext

    /**
     * User: laizhenwei
     * Date: 2018-03-26 Time: 21:46
     * Description: 缓存客户端的ChannelHandlerContext
     */
    public class ContextHelper {
    
        private final static Map<String, ChannelHandlerContext> clientMap = new ConcurrentHashMap<>();
    
        public static Map<String, ChannelHandlerContext> getClientMap() {
            return Collections.unmodifiableMap(clientMap);
        }
    
        public static ChannelHandlerContext get(String id){
            return clientMap.get(id);
        }
    
        public static void add(String id, ChannelHandlerContext ctx) {
            clientMap.put(id, ctx);
        }
    
        public static void remove(String id) {
            clientMap.remove(id);
        }
    }

    MyServerHandler

    /**
     * User: laizhenwei
     * Date: 2018-03-26 Time: 21:46
     * Description:
     */
    @Slf4j
    @ChannelHandler.Sharable
    public class MyServerHandler extends SimpleChannelInboundHandler<MessageProto.Message> {
    
        @Override
        protected void channelRead0(ChannelHandlerContext channelHandlerContext, MessageProto.Message msg) {
            String message = msg.getMsgBody();
            if ((MessageProto.Message.Type.ACTIVE_VALUE) == msg.getType()) {
                Attribute<String> attribute = channelHandlerContext.channel().attr(AttributeKey.valueOf("userName"));
                //连接上以后获取消息参数,设置到channelAttr
                String userName = message.split(":")[1];
                attribute.setIfAbsent(userName);
                //缓存channelHandlerContext
                ContextHelper.add(userName, channelHandlerContext);
            } else if (MessageProto.Message.Type.MESSAGE_VALUE == msg.getType()) {
                ResponseFuture<MessageProto.Message> resutl = EchoServer.responseFutureMap.get(msg.getId());
                if (resutl == null)
                    log.warn("result is null ! msgId:" + msg.getId());
                MessageProto.Message message1 = MessageProto.Message.newBuilder().setId(msg.getId()).setType(MessageProto.Message.Type.MESSAGE_VALUE).setMsgBody("接收成功!msg:" + message).build();
                resutl.setResponse(message1);
            }
    //        System.out.println("Client->Server:" + channelHandlerContext.channel().remoteAddress() + " send " + msg.getMsgBody());
        }
    
    
        @Override
        public void channelInactive(ChannelHandlerContext ctx){
            Attribute<String> attribute = ctx.channel().attr(AttributeKey.valueOf("userName"));
            ContextHelper.remove(attribute.get());
        }
    
    
        @Override
        public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) {
            cause.printStackTrace();
        }
    }

    ChannelInitializer,添加 Netty 支持 ProtoBuf 的拆包处理,以及编码解码

    /**
     * User: laizhenwei
     * Date: 2018-03-26 Time: 21:46
     * Description:
     */
    public class MyServerChannelInitializer extends ChannelInitializer<SocketChannel> {
    
        @Override
        protected void initChannel(SocketChannel socketChannel) throws Exception {
            socketChannel.pipeline()
                    .addLast(new ProtobufVarint32FrameDecoder())
                    .addLast(new ProtobufDecoder(MessageProto.Message.getDefaultInstance()))
                    .addLast(new ProtobufVarint32LengthFieldPrepender())
                    .addLast(new ProtobufEncoder())
                    .addLast(new MyServerHandler());
        }
    
    }

    ResponseFuture

    @NoArgsConstructor
    public class ResponseFuture<T> implements Future<T> {
        // 因为请求和响应是一一对应的,因此初始化CountDownLatch值为1。
        private CountDownLatch latch = new CountDownLatch(1);
        // 响应结果
        private T response;
        // Futrue的请求时间,用于计算Future是否超时
        private long beginTime = System.currentTimeMillis();
    
        @Override
        public boolean cancel(boolean mayInterruptIfRunning) {
            return false;
        }
    
        @Override
        public boolean isCancelled() {
            return false;
        }
    
        @Override
        public boolean isDone() {
            if (response != null)
                return true;
            return false;
        }
    
        // 获取响应结果,直到有结果才返回。
        @Override
        public T get() throws InterruptedException {
            latch.await();
            return this.response;
        }
    
        // 获取响应结果,直到有结果或者超过指定时间就返回。
        @Override
        public T get(long timeout, TimeUnit unit) throws InterruptedException {
            if (latch.await(timeout, unit))
                return this.response;
            return null;
        }
    
        // 用于设置响应结果,并且做countDown操作,通知请求线程
        public void setResponse(T response) {
            this.response = response;
            latch.countDown();
        }
    
        public long getBeginTime() {
            return beginTime;
        }
    }

    ApplicationStartup SpringBoot 完全启动以后,运行Netty服务

    /**
     * User: laizhenwei
     * Date: 2018-03-26 Time: 21:46
     * Description:
     */
    @Component
    public class ApplicationStartup implements CommandLineRunner {
    
        @Override
        public void run(String... args) throws Exception {
            new EchoServer(5000).start();
        }
    }

    客户端 EchoClient

    /**
     * User: laizhenwei
     * Date: 2018-03-27 Time: 21:50
     * Description:
     */
    public class EchoClient {
    
        private final String host;
    
        private final int port;
    
    
        public EchoClient(String host,int port){
            this.host = host;
            this.port = port;
        }
    
        public void start(String userName) throws InterruptedException {
            EventLoopGroup group = new NioEventLoopGroup();
    
            Bootstrap b = new Bootstrap();
            b.group(group).channel(NioSocketChannel.class)
                    .remoteAddress(new InetSocketAddress(host,port))
                    .handler(new ChannelInitializer<SocketChannel>(){
                        @Override
                        protected void initChannel(SocketChannel socketChannel){
                            socketChannel.attr(AttributeKey.valueOf("userName")).setIfAbsent(userName);
                            socketChannel.pipeline()
                                    .addLast(new ProtobufVarint32FrameDecoder())
                                    .addLast(new ProtobufDecoder(MessageProto.Message.getDefaultInstance()))
                                    .addLast(new ProtobufVarint32LengthFieldPrepender())
                                    .addLast(new ProtobufEncoder())
                                    .addLast(new MyClientHandler());
                        }
                    });
    
           try {
               ChannelFuture f = b.connect().sync();
               f.channel().closeFuture().sync();
           }finally {
               group.shutdownGracefully().sync();
           }
        }
    
        public static void main(String[] args){
            threadRun("Athos");
            threadRun("Nero");
            threadRun("Dante");
            threadRun("Vergil");
            threadRun("lzw");
            threadRun("Churchill");
            threadRun("Peter");
            threadRun("Bob");
        }
    
        private static void threadRun(String userName){
            new Thread(()-> {
                try {
                    new EchoClient("192.168.1.8",5000).start(userName);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }).start();
        }
    
    }

     MyClientHandler

    /**
     * User: laizhenwei
     * Date: 2018-04-09 Time: 11:20
     * Description:
     */
    @ChannelHandler.Sharable
    public class MyClientHandler extends SimpleChannelInboundHandler<MessageProto.Message> {
    
        @Override
        public void channelActive(ChannelHandlerContext ctx) {
            Attribute<Object> attribute = ctx.channel().attr(AttributeKey.valueOf("userName"));
            String m = "userName:" + attribute.get();
            MessageProto.Message.Builder builder = MessageProto.Message.newBuilder();
            builder.setType(MessageProto.Message.Type.ACTIVE_VALUE).setMsgBody(m);
            ctx.writeAndFlush(builder.build());
        }
    
        @Override
        protected void channelRead0(ChannelHandlerContext channelHandlerContext, MessageProto.Message msg) {
            MessageProto.Message.Builder builder = MessageProto.Message.newBuilder();
            //把接收到的消息写回到服务端
            builder.setId(msg.getId()).setType(MessageProto.Message.Type.MESSAGE_VALUE).setMsgBody(msg.getMsgBody());
            channelHandlerContext.channel().writeAndFlush(builder.build());
        }
    
        @Override
        public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) {
            cause.printStackTrace();
            ctx.close();
        }
    
    }

    JunitTest

        @Test
        public void testRest() throws InterruptedException {
            final Gson gson = new Gson();
            AtomicLong atomicLong = new AtomicLong(0);
    
            ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
            executor.setCorePoolSize(50);
            executor.setMaxPoolSize(50);
            executor.setQueueCapacity(512);
            executor.setThreadNamePrefix("Executor-");
            executor.setAllowCoreThreadTimeOut(false);
            executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
            executor.setWaitForTasksToCompleteOnShutdown(true);
            executor.initialize();
            String[] userNames = {"Athos", "Nero", "Dante"
                    , "Vergil", "lzw", "Churchill"
                    , "Peter", "Bob"};
    
    //        String[] userNames = {"Athos"};
    
            RestTemplate restTemplate = new RestTemplate();
            HttpHeaders httpHeaders = new HttpHeaders();
            httpHeaders.setAccept(Arrays.asList(MediaType.APPLICATION_JSON_UTF8));
            httpHeaders.add("connection", "keep-alive");
    //        httpHeaders.setConnection("close");
            List<CompletableFuture<Boolean>> futures = new ArrayList<>();
            long begin = System.nanoTime();
            Arrays.stream(userNames).forEach(userName -> new Thread(() -> {
                for (int i = 0; i < 100000; i++) {
                    futures.add(CompletableFuture.supplyAsync(() -> {
                        long currentId = atomicLong.getAndIncrement();
                        MultiValueMap<String, String> params = new LinkedMultiValueMap<>();
                        params.add("userName", userName);
                        params.add("msg", "你好啊!" + currentId);
                        HttpEntity<MultiValueMap<String, String>> httpEntity = new HttpEntity<>(params, httpHeaders);
                        String response = restTemplate.postForObject("http://192.168.91.130:8010/process", httpEntity, String.class);
                        if (response != null) {
                            Map<String, Object> responseMap;
                            responseMap = gson.fromJson(response, HashMap.class);
                            return responseMap.get("msgBody").equals("接收成功!msg:你好啊!" + currentId);
                        }
                        return false;
                    }, executor));
                }
            }).start());
    
            while(futures.size()!=(100000*userNames.length)){
                TimeUnit.MILLISECONDS.sleep(500);
            }
    
            List<Boolean> result = futures.stream().map(CompletableFuture::join).collect(Collectors.toList());
    
            System.out.println((System.nanoTime() - begin) / 1000000);
    
            result.stream().filter(r -> !r).forEach(r -> System.out.println(r));
    
        }

    1.启动NettyServer

    2.启动NettyClient

    3.启动N个JunitTest windows 启动5个,Linux 启动5个

    看看server输出,从请求到响应非常迅速

    Client 多个线程也没有看到输出有false,证明伪同步响应成功

  • 相关阅读:
    java基础--日期--练习集锦
    java基础-数字与字符串-练习集锦
    java-多态-之疑惑
    java-接口与多态-
    java基础 -继承-练习集锦
    java基础-对象-练习集锦
    java基础--数组--练习集锦
    hibernate第二天
    jquery的一些笔记
    struts入门2
  • 原文地址:https://www.cnblogs.com/sweetchildomine/p/8798493.html
Copyright © 2020-2023  润新知