Java 从零开始手写 RPC—如何实现客户端调用服务端?

开发 后端
写完了客户端和服务端,那么如何实现客户端和服务端的调用呢?下面就让我们一起来看一下。希望本文对你有所帮助!

[[429717]]

写完了客户端和服务端,那么如何实现客户端和服务端的调用呢?

下面就让我们一起来看一下。

接口定义

计算方法

package com.github.houbb.rpc.common.service; 
 
 
import com.github.houbb.rpc.common.model.CalculateRequest; 
import com.github.houbb.rpc.common.model.CalculateResponse; 
 
 
/** 
 * <p> 计算服务接口 </p> 
 * 
 * <pre> Created: 2018/8/24 下午4:47  </pre> 
 * <pre> Project: fake  </pre> 
 * 
 * @author houbinbin 
 * @since 0.0.1 
 */ 
public interface Calculator { 
 
 
    /** 
     * 计算加法 
     * @param request 请求入参 
     * @return 返回结果 
     */ 
    CalculateResponse sum(final CalculateRequest request); 
 
 

  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.

pojo

对应的参数对象:

  • CalculateRequest
package com.github.houbb.rpc.common.model; 
 
 
import java.io.Serializable
 
 
/** 
 * <p> 请求入参 </p> 
 * 
 * <pre> Created: 2018/8/24 下午5:05  </pre> 
 * <pre> Project: fake  </pre> 
 * 
 * @author houbinbin 
 * @since 0.0.3 
 */ 
public class CalculateRequest implements Serializable { 
 
 
    private static final long serialVersionUID = 6420751004355300996L; 
 
 
    /** 
     * 参数一 
     */ 
    private int one; 
 
 
    /** 
     * 参数二 
     */ 
    private int two; 
 
 
    public CalculateRequest() { 
    } 
 
 
    public CalculateRequest(int one, int two) { 
        this.one = one; 
        this.two = two; 
    } 
 
 
    //getter setter toString 
 
 

  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.
  • 29.
  • 30.
  • 31.
  • 32.
  • 33.
  • 34.
  • 35.
  • 36.
  • 37.
  • 38.
  • 39.
  • 40.
  • 41.
  • 42.
  • 43.
  • 44.
  • 45.
  • 46.
  • 47.
  • CalculateResponse
package com.github.houbb.rpc.common.model; 
 
 
import java.io.Serializable
 
 
/** 
 * <p> 请求入参 </p> 
 * 
 * <pre> Created: 2018/8/24 下午5:05  </pre> 
 * <pre> Project: fake  </pre> 
 * 
 * @author houbinbin 
 * @since 0.0.3 
 */ 
public class CalculateResponse implements Serializable { 
 
 
    private static final long serialVersionUID = -1972014736222511341L; 
 
 
    /** 
     * 是否成功 
     */ 
   private boolean success; 
 
 
    /** 
     * 二者的和 
     */ 
   private int sum
 
 
    public CalculateResponse() { 
    } 
 
 
    public CalculateResponse(boolean success, int sum) { 
        this.success = success; 
        this.sum = sum
    } 
 
 
    //getter setter toString 

  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.
  • 29.
  • 30.
  • 31.
  • 32.
  • 33.
  • 34.
  • 35.
  • 36.
  • 37.
  • 38.
  • 39.
  • 40.
  • 41.
  • 42.
  • 43.
  • 44.
  • 45.

客户端

核心部分

RpcClient 需要添加对应的 Handler,调整如下:

Bootstrap bootstrap = new Bootstrap(); 
ChannelFuture channelFuture = bootstrap.group(workerGroup) 
        .channel(NioSocketChannel.class) 
        .option(ChannelOption.SO_KEEPALIVE, true
        .handler(new ChannelInitializer<Channel>(){ 
            @Override 
            protected void initChannel(Channel ch) throws Exception { 
                ch.pipeline() 
                        .addLast(new LoggingHandler(LogLevel.INFO)) 
                        .addLast(new CalculateRequestEncoder()) 
                        .addLast(new CalculateResponseDecoder()) 
                        .addLast(new RpcClientHandler()); 
            } 
        }) 
        .connect(RpcConstant.ADDRESS, port) 
        .syncUninterruptibly(); 
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.

netty 中的 handler 泳道设计的非常优雅,让我们的代码可以非常灵活地进行拓展。

接下来我们看一下对应的实现。

RpcClientHandler

package com.github.houbb.rpc.client.handler; 
 
 
import com.github.houbb.log.integration.core.Log; 
import com.github.houbb.log.integration.core.LogFactory; 
import com.github.houbb.rpc.client.core.RpcClient; 
import com.github.houbb.rpc.common.model.CalculateRequest; 
import com.github.houbb.rpc.common.model.CalculateResponse; 
import io.netty.channel.ChannelHandlerContext; 
import io.netty.channel.SimpleChannelInboundHandler; 
 
 
/** 
 * <p> 客户端处理类 </p> 
 * 
 * <pre> Created: 2019/10/16 11:30 下午  </pre> 
 * <pre> Project: rpc  </pre> 
 * 
 * @author houbinbin 
 * @since 0.0.2 
 */ 
public class RpcClientHandler extends SimpleChannelInboundHandler { 
 
 
    private static final Log log = LogFactory.getLog(RpcClient.class); 
 
 
    @Override 
    public void channelActive(ChannelHandlerContext ctx) throws Exception { 
        CalculateRequest request = new CalculateRequest(1, 2); 
 
 
        ctx.writeAndFlush(request); 
        log.info("[Client] request is :{}", request); 
    } 
 
 
    @Override 
    protected void channelRead0(ChannelHandlerContext ctx, Object msg) throws Exception { 
        CalculateResponse response = (CalculateResponse)msg; 
        log.info("[Client] response is :{}", response); 
    } 
 
 

  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.
  • 29.
  • 30.
  • 31.
  • 32.
  • 33.
  • 34.
  • 35.
  • 36.
  • 37.
  • 38.
  • 39.
  • 40.
  • 41.
  • 42.
  • 43.
  • 44.
  • 45.

这里比较简单,channelActive 中我们直接发起调用,入参的对象为了简单,此处固定写死。

channelRead0 中监听服务端的相应结果,并做日志输出。

CalculateRequestEncoder

请求参数是一个对象,netty 是无法直接传输的,我们将其转换为基本对象:

package com.github.houbb.rpc.client.encoder; 
 
 
import com.github.houbb.rpc.common.model.CalculateRequest; 
import io.netty.buffer.ByteBuf; 
import io.netty.channel.ChannelHandlerContext; 
import io.netty.handler.codec.MessageToByteEncoder; 
 
 
/** 
 * @author binbin.hou 
 * @since 0.0.3 
 */ 
public class CalculateRequestEncoder extends MessageToByteEncoder<CalculateRequest> { 
 
 
    @Override 
    protected void encode(ChannelHandlerContext ctx, CalculateRequest msg, ByteBuf out) throws Exception { 
        int one = msg.getOne(); 
        int two = msg.getTwo(); 
 
 
        out.writeInt(one); 
        out.writeInt(two); 
    } 
 
 

  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.

CalculateResponseDecoder

针对服务端的响应,也是同理。

我们需要把基本的类型,封装转换为我们需要的对象。

package com.github.houbb.rpc.client.decoder; 
 
 
import com.github.houbb.rpc.common.model.CalculateResponse; 
import io.netty.buffer.ByteBuf; 
import io.netty.channel.ChannelHandlerContext; 
import io.netty.handler.codec.ByteToMessageDecoder; 
 
 
import java.util.List; 
 
 
/** 
 * 响应参数解码 
 * @author binbin.hou 
 * @since 0.0.3 
 */ 
public class CalculateResponseDecoder extends ByteToMessageDecoder { 
 
 
    @Override 
    protected void decode(ChannelHandlerContext ctx, ByteBuf in, List<Object> out) throws Exception { 
        boolean success = in.readBoolean(); 
        int sum = in.readInt(); 
 
 
        CalculateResponse response = new CalculateResponse(success, sum); 
        out.add(response); 
    } 
 
 

  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.
  • 29.
  • 30.
  • 31.
  • 32.

服务端

设置处理类

RpcServer 中的处理类要稍微调整一下,其他的保持不变。

ServerBootstrap serverBootstrap = new ServerBootstrap(); 
serverBootstrap.group(workerGroup, bossGroup) 
        .channel(NioServerSocketChannel.class) 
        // 打印日志 
        .handler(new LoggingHandler(LogLevel.INFO)) 
        .childHandler(new ChannelInitializer<Channel>() { 
            @Override 
            protected void initChannel(Channel ch) throws Exception { 
                ch.pipeline() 
                        .addLast(new CalculateRequestDecoder()) 
                        .addLast(new CalculateResponseEncoder()) 
                        .addLast(new RpcServerHandler()); 
            } 
        }) 
        // 这个参数影响的是还没有被accept 取出的连接 
        .option(ChannelOption.SO_BACKLOG, 128) 
        // 这个参数只是过一段时间内客户端没有响应,服务端会发送一个 ack 包,以判断客户端是否还活着。 
        .childOption(ChannelOption.SO_KEEPALIVE, true); 
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.

RpcServerHandler

一开始这里是空实现,我们来添加一下对应的实现。

package com.github.houbb.rpc.server.handler; 
 
 
import com.github.houbb.log.integration.core.Log; 
import com.github.houbb.log.integration.core.LogFactory; 
import com.github.houbb.rpc.common.model.CalculateRequest; 
import com.github.houbb.rpc.common.model.CalculateResponse; 
import com.github.houbb.rpc.common.service.Calculator; 
import com.github.houbb.rpc.server.service.CalculatorService; 
import io.netty.channel.ChannelHandlerContext; 
import io.netty.channel.SimpleChannelInboundHandler; 
 
 
/** 
 * @author binbin.hou 
 * @since 0.0.1 
 */ 
public class RpcServerHandler extends SimpleChannelInboundHandler { 
 
 
    private static final Log log = LogFactory.getLog(RpcServerHandler.class); 
 
 
    @Override 
    public void channelActive(ChannelHandlerContext ctx) throws Exception { 
        final String id = ctx.channel().id().asLongText(); 
        log.info("[Server] channel {} connected " + id); 
    } 
 
 
    @Override 
    protected void channelRead0(ChannelHandlerContext ctx, Object msg) throws Exception { 
        final String id = ctx.channel().id().asLongText(); 
 
 
        CalculateRequest request = (CalculateRequest)msg; 
        log.info("[Server] receive channel {} request: {} from ", id, request); 
 
 
        Calculator calculator = new CalculatorService(); 
        CalculateResponse response = calculator.sum(request); 
 
 
        // 回写到 client 端 
        ctx.writeAndFlush(response); 
        log.info("[Server] channel {} response {}", id, response); 
    } 
 
 

  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.
  • 29.
  • 30.
  • 31.
  • 32.
  • 33.
  • 34.
  • 35.
  • 36.
  • 37.
  • 38.
  • 39.
  • 40.
  • 41.
  • 42.
  • 43.
  • 44.
  • 45.
  • 46.
  • 47.
  • 48.
  • 49.
  • 50.

读取到客户端的访问之后,我们获取到计算的入参 CalculateRequest,然后调用 sum 方法,获取到对应的 CalculateResponse,将结果通知客户端。

CalculateRequestDecoder

这里和客户端是一一对应的,我们首先把 netty 传递的基本类型转换为 CalculateRequest 对象。

package com.github.houbb.rpc.server.decoder; 
 
 
import com.github.houbb.rpc.common.model.CalculateRequest; 
import io.netty.buffer.ByteBuf; 
import io.netty.channel.ChannelHandlerContext; 
import io.netty.handler.codec.ByteToMessageDecoder; 
 
 
import java.util.List; 
 
 
/** 
 * 请求参数解码 
 * @author binbin.hou 
 * @since 0.0.3 
 */ 
public class CalculateRequestDecoder extends ByteToMessageDecoder { 
 
 
    @Override 
    protected void decode(ChannelHandlerContext ctx, ByteBuf in, List<Object> out) throws Exception { 
        int one = in.readInt(); 
        int two = in.readInt(); 
 
 
        CalculateRequest request = new CalculateRequest(one, two); 
        out.add(request); 
    } 
 
 

  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.
  • 29.
  • 30.
  • 31.
  • 32.

CalculateResponseEncoder

这里和客户端类似,我们需要把 response 转换为基本类型进行网络传输。

package com.github.houbb.rpc.server.encoder; 
 
 
import com.github.houbb.rpc.common.model.CalculateResponse; 
import io.netty.buffer.ByteBuf; 
import io.netty.channel.ChannelHandlerContext; 
import io.netty.handler.codec.MessageToByteEncoder; 
 
 
/** 
 * @author binbin.hou 
 * @since 0.0.3 
 */ 
public class CalculateResponseEncoder extends MessageToByteEncoder<CalculateResponse> { 
 
 
    @Override 
    protected void encode(ChannelHandlerContext ctx, CalculateResponse msg, ByteBuf out) throws Exception { 
        boolean success = msg.isSuccess(); 
        int result = msg.getSum(); 
        out.writeBoolean(success); 
        out.writeInt(result); 
    } 
 
 

  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.

CalculatorService

服务端对应的实现类。

public class CalculatorService implements Calculator { 
 
 
    @Override 
    public CalculateResponse sum(CalculateRequest request) { 
        int sum = request.getOne()+request.getTwo(); 
 
 
        return new CalculateResponse(truesum); 
    } 
 
 

  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.

测试

服务端

启动服务端:

new RpcServer().start(); 
  • 1.

服务端启动日志:

[DEBUG] [2021-10-05 11:53:11.795] [main] [c.g.h.l.i.c.LogFactory.setImplementation] - Logging initialized using 'class com.github.houbb.log.integration.adaptors.stdout.StdOutExImpl' adapter. 
[INFO] [2021-10-05 11:53:11.807] [Thread-0] [c.g.h.r.s.c.RpcServer.run] - RPC 服务开始启动服务端 
十月 05, 2021 11:53:13 上午 io.netty.handler.logging.LoggingHandler channelRegistered 
信息: [id: 0xd399474f] REGISTERED 
十月 05, 2021 11:53:13 上午 io.netty.handler.logging.LoggingHandler bind 
信息: [id: 0xd399474f] BIND: 0.0.0.0/0.0.0.0:9527 
十月 05, 2021 11:53:13 上午 io.netty.handler.logging.LoggingHandler channelActive 
信息: [id: 0xd399474f, L:/0:0:0:0:0:0:0:0:9527] ACTIVE 
[INFO] [2021-10-05 11:53:13.101] [Thread-0] [c.g.h.r.s.c.RpcServer.run] - RPC 服务端启动完成,监听【9527】端口 
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.

客户端

启动客户端:

new RpcClient().start(); 
  • 1.

日志如下:

[DEBUG] [2021-10-05 11:54:12.158] [main] [c.g.h.l.i.c.LogFactory.setImplementation] - Logging initialized using 'class com.github.houbb.log.integration.adaptors.stdout.StdOutExImpl' adapter. 
[INFO] [2021-10-05 11:54:12.164] [Thread-0] [c.g.h.r.c.c.RpcClient.run] - RPC 服务开始启动客户端 
十月 05, 2021 11:54:13 上午 io.netty.handler.logging.LoggingHandler channelRegistered 
信息: [id: 0x4d75c580] REGISTERED 
十月 05, 2021 11:54:13 上午 io.netty.handler.logging.LoggingHandler connect 
信息: [id: 0x4d75c580] CONNECT: /127.0.0.1:9527 
十月 05, 2021 11:54:13 上午 io.netty.handler.logging.LoggingHandler channelActive 
信息: [id: 0x4d75c580, L:/127.0.0.1:54030 - R:/127.0.0.1:9527] ACTIVE 
[INFO] [2021-10-05 11:54:13.403] [Thread-0] [c.g.h.r.c.c.RpcClient.run] - RPC 服务启动客户端完成,监听端口:9527 
十月 05, 2021 11:54:13 上午 io.netty.handler.logging.LoggingHandler write 
信息: [id: 0x4d75c580, L:/127.0.0.1:54030 - R:/127.0.0.1:9527] WRITE: 8B 
         +-------------------------------------------------+ 
         |  0  1  2  3  4  5  6  7  8  9  a  b  c  d  e  f | 
+--------+-------------------------------------------------+----------------+ 
|00000000| 00 00 00 01 00 00 00 02                         |........        | 
+--------+-------------------------------------------------+----------------+ 
十月 05, 2021 11:54:13 上午 io.netty.handler.logging.LoggingHandler flush 
信息: [id: 0x4d75c580, L:/127.0.0.1:54030 - R:/127.0.0.1:9527] FLUSH 
[INFO] [2021-10-05 11:54:13.450] [nioEventLoopGroup-2-1] [c.g.h.r.c.c.RpcClient.channelActive] - [Client] request is :CalculateRequest{one=1, two=2} 
十月 05, 2021 11:54:13 上午 io.netty.handler.logging.LoggingHandler channelRead 
信息: [id: 0x4d75c580, L:/127.0.0.1:54030 - R:/127.0.0.1:9527] READ: 5B 
         +-------------------------------------------------+ 
         |  0  1  2  3  4  5  6  7  8  9  a  b  c  d  e  f | 
+--------+-------------------------------------------------+----------------+ 
|00000000| 01 00 00 00 03                                  |.....           | 
+--------+-------------------------------------------------+----------------+ 
十月 05, 2021 11:54:13 上午 io.netty.handler.logging.LoggingHandler channelReadComplete 
信息: [id: 0x4d75c580, L:/127.0.0.1:54030 - R:/127.0.0.1:9527] READ COMPLETE 
[INFO] [2021-10-05 11:54:13.508] [nioEventLoopGroup-2-1] [c.g.h.r.c.c.RpcClient.channelRead0] - [Client] response is :CalculateResponse{success=truesum=3} 
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.
  • 29.

可以看到,输出了对应的请求参数和响应结果。

当然,此时服务端也有对应的新增日志:

十月 05, 2021 11:54:13 上午 io.netty.handler.logging.LoggingHandler channelRead 
信息: [id: 0xd399474f, L:/0:0:0:0:0:0:0:0:9527] READ: [id: 0xbc9f5927, L:/127.0.0.1:9527 - R:/127.0.0.1:54030] 
十月 05, 2021 11:54:13 上午 io.netty.handler.logging.LoggingHandler channelReadComplete 
信息: [id: 0xd399474f, L:/0:0:0:0:0:0:0:0:9527] READ COMPLETE 
[INFO] [2021-10-05 11:54:13.432] [nioEventLoopGroup-2-1] [c.g.h.r.s.h.RpcServerHandler.channelActive] - [Server] channel {} connected 00e04cfffe360988-00001d34-00000001-2a80d950d8166c0c-bc9f5927 
[INFO] [2021-10-05 11:54:13.495] [nioEventLoopGroup-2-1] [c.g.h.r.s.h.RpcServerHandler.channelRead0] - [Server] receive channel 00e04cfffe360988-00001d34-00000001-2a80d950d8166c0c-bc9f5927 request: CalculateRequest{one=1, two=2} from  
[INFO] [2021-10-05 11:54:13.505] [nioEventLoopGroup-2-1] [c.g.h.r.s.h.RpcServerHandler.channelRead0] - [Server] channel 00e04cfffe360988-00001d34-00000001-2a80d950d8166c0c-bc9f5927 response CalculateResponse{success=truesum=3} 
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.

 

责任编辑:姜华 来源: 今日头条
相关推荐

2021-10-14 08:39:17

Java Netty Java 基础

2021-10-21 08:21:10

Java Reflect Java 基础

2021-10-27 08:10:15

Java 客户端 Java 基础

2021-10-13 08:21:52

Java websocket Java 基础

2021-10-20 08:05:18

Java 序列化 Java 基础

2009-08-21 15:36:41

服务端与客户端

2009-08-21 15:54:40

服务端与客户端

2021-10-29 08:07:30

Java timeout Java 基础

2010-03-18 17:47:07

Java 多客户端通信

2023-11-17 09:13:36

2024-03-06 14:58:52

客户端微服务架构

2009-08-21 15:59:22

服务端与客户端通信

2011-09-09 09:44:23

WCF

2009-08-21 16:14:52

服务端与客户端通信

2023-03-06 08:01:56

MySQLCtrl + C

2010-05-31 10:11:32

瘦客户端

2023-04-03 08:13:05

MySQLCtrl + C

2014-06-01 11:03:13

VDI零客户端

2010-11-19 14:22:04

oracle服务端

2013-05-09 09:33:59

点赞
收藏

51CTO技术栈公众号