Java客户端调用Python服务端
某些大型web服务一般是采用Spring框架编写的,然而深度学习计算框架TensorFlow是采用Python编写的,那么如何让客户端调用Python服务端的计算资源返回结果呢,一种可以采用传统的HTTP Web Service的方式,将服务端封装为Web服务,客户端直接通过JSON请求进行调用并返回结果;另外一种是通过基于HTTP2协议RPC服务,Python将计算服务注册在注册中心,Java客户端通过端口号在注册中心找到对应的服务后,调用服务端的服务实例。
使用Web Service调用
首先Python编写路由,将请求封装为Web服务,然后Java客户端根据路由,发送请求给服务端,服务端收到请求以后执行计算服务并返回结果,Python将返回的计算结果封装为JSON回传给Java客户端,Java客户端对结果进行解析即可。
- 安装依赖
Python: pip install flask
Java在maven中加入相关的包
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-web</artifactId>
<version>5.2.2.RELEASE</version>
</dependency>
- Python服务端采用Flash便携式路由,并调用深度学习服务(这里简单用Hello World代替)
from flask import Flask
app = Flask(__name__)
# 使用<param>传递参数
@app.route('/hello/<param>')
def hello_get(param):
return 'hello %s' % param
if __name__ == '__main__':
app.run()
- Java客户端采用RestTemplate请求URL并返回结果
public class HelloWorldClientByURL {
public static void main(String[] args) {
RestTemplate restTemplate = new RestTemplate();
String url = "http://127.0.0.1:5000/hello/world";
ResponseEntity<String> responseEntity = restTemplate.getForEntity(url, String.class);
String result = responseEntity.getBody();
System.out.println(result);
}
}
- 先开启Python服务端,然后执行Java客户端,这时返回结果
hello world
使用gRPC
服务端直接将Python服务注册到gRPC, Java客户端根据端口号通过gRPC注册中心调用服务端服务。具体过程如下:
Python服务端将服务直接注册到gRPC注册中心,Java客户端通过端口号映射请求相关的服务,gRPC协议层接受服务之后,由应用层对服务进行处理,可以直接调用注册的服务实例,服务计算完成之后返回结果给gRPC,响应回调通知线程,Java客户端接收返回的结果。
- 安装依赖
Python: pipinstall grpcio / pipinstall grpcio-tls
Java在maven中加入相关的依赖:
<dependency>
<groupId>io.grpc</groupId>
<artifactId>grpc-netty-shaded</artifactId>
<version>1.15.0</version>
</dependency>
<dependency>
<groupId>io.grpc</groupId>
<artifactId>grpc-protobuf</artifactId>
<version>1.15.0</version>
</dependency>
<dependency>
<groupId>io.grpc</groupId>
<artifactId>grpc-stub</artifactId>
<version>1.15.0</version>
</dependency>
- Python服务端代码
# python服务端代码
class Greeter(helloworld_pb2_grpc.GreeterServicer):
def SayHello(self, request, context):
return helloworld_pb2.HelloReply(message='Hello, %s!' % request.name)
def serve():
server = grpc.server(futures.ThreadPoolExecutor(max_workers=10))
helloworld_pb2_grpc.add_GreeterServicer_to_server(Greeter(), server)
server.add_insecure_port('[::]:50051')
server.start()
try:
while True:
time.sleep(_ONE_DAY_IN_SECONDS)
except KeyboardInterrupt:
server.stop(0)
if __name__ == '__main__':
serve()
- 编写helloworld.proto,并执行
syntax = "proto3";
package example;
// The greeting service definition.
service Greeter {
// Sends a greeting
rpc SayHello (HelloRequest) returns (HelloReply) {}
}
// The request message containing the user's name.
message HelloRequest {
string name = 1;
}
// The response message containing the greetings
message HelloReply {
string message = 1;
}
在相关目录下执行python -m grpc_tools.protoc -I. --python_out=. --grpc_python_out=. ./helloworld.proto
进行编译
- Java客户端代码
public class HelloWorldClient {
private static final Logger logger = Logger.getLogger(HelloWorldClient.class.getName());
private final ManagedChannel channel;
private final GreeterGrpc.GreeterBlockingStub blockingStub;
/**
* Construct client connecting to HelloWorld server at {@code host:port}.
*/
public HelloWorldClient(String host, int port) {
channel = ManagedChannelBuilder.forAddress(host, port)
.usePlaintext(true)
.build();
blockingStub = GreeterGrpc.newBlockingStub(channel);
}
public void shutdown() throws InterruptedException {
channel.shutdown().awaitTermination(5, TimeUnit.SECONDS);
}
/**
* Say hello to server.
*/
public void greet(String name) {
logger.info("Will try to greet " + name + " ...");
HelloRequest request = HelloRequest.newBuilder().setName(name).build();
HelloReply response;
try {
response = blockingStub.sayHello(request);
} catch (StatusRuntimeException e) {
logger.log(Level.WARNING, "RPC failed: {0}", e.getStatus());
return;
}
logger.info("Greeting: " + response.getMessage());
}
/**
* Greet server. If provided, the first element of {@code args} is the name to use in the
* greeting.
*/
public static void main(String[] args) throws Exception {
HelloWorldClient client = new HelloWorldClient("localhost", 50051);
try {
String user = "world";
if (args.length > 0) {
user = args[0];
}
client.greet(user);
} finally {
client.shutdown();
}
}
}
- 编写helloworld.proto参数与2保持一致
syntax = "proto3";
package example;
option java_package = "com.shgx.grpc";
option java_outer_classname = "HelloWorldServiceProto";
option java_multiple_files = true;
// The greeting service definition.
service Greeter {
// Sends a greeting
rpc SayHello (HelloRequest) returns (HelloReply) {}
}
// The request message containing the user's name.
message HelloRequest {
string name = 1;
}
// The response message containing the greetings
message HelloReply {
string message = 1;
}
- 先启动Python服务端代码,然后启动Java客户端,输出结果
hello world
两种方式对比
采用Web服务进行调用,后端对应用的处理线程采用同步阻塞的模型,阻塞的时间取决对方I/O处理的速度和网络I/O传输的速度,两种不同代码之间通过HTTP请求或者JSON封装进行交互,效率较低。
采用gRPC,有了服务的注册中心,服务切换更新更加轻量化,并且遵循 Netty 的线程分工原则,协议层消息的接收和编解码由Netty 的 I/O(NioEventLoop)线程负责;后续应用层的处理由应用线程负责,防止由于应用处理耗时而阻塞 Netty 的 I/O 线程, 可以通过服务名和方法名调用,直接调用启动的时候注册的服务实例,不需要反射或者JSON编码解码进行调用,性能更优; 不过因为有Netty的线程分工原则,gRPC之间会做频繁的线程切换,如果在一次gRPC调用过程中,做了多次I/O线程到应用线程之间的切换,会导致性能的下降。