tensorflow serving java案例

简介: 背景介绍 这篇文章是tensorflow serving java api使用的参考案例,基本上把TFS的核心API的用法都介绍清楚。案例主要分为三部分:动态更新模型:用于在TFS处于runtime时候动态加载模型。

背景介绍

 这篇文章是tensorflow serving java api使用的参考案例,基本上把TFS的核心API的用法都介绍清楚。案例主要分为三部分:

  • 动态更新模型:用于在TFS处于runtime时候动态加载模型。
  • 获取模型状态:用于获取加载的模型的基本信息。
  • 在线模型预测:进行在线预测,分类等操作,着重介绍在线预测。

因为模型的预测需要参考模型内部变量,所以可以先行通过TFS的REST接口获取TF模型的元数据然后才能构建TFS的RPC请求对象


TFS 使用入门

模型源数据获取

curl http://host:port/v1/models/${MODEL_NAME}[/versions/${MODEL_VERSION}]/metadata

说明:


    public static void getModelStatus() {

        // 1、设置访问的RPC协议的host和port
        ManagedChannel channel = ManagedChannelBuilder.forAddress(host, port).usePlaintext().build();

        // 2、构建PredictionServiceBlockingStub对象
        PredictionServiceGrpc.PredictionServiceBlockingStub predictionServiceBlockingStub =
                PredictionServiceGrpc.newBlockingStub(channel);

         // 3、设置待获取的模型
        Model.ModelSpec modelSpec = Model.ModelSpec.newBuilder()
                .setName("wdl_model").build();

        // 4、构建获取元数据的请求
        GetModelMetadata.GetModelMetadataRequest modelMetadataRequest =
                GetModelMetadata.GetModelMetadataRequest.newBuilder()
                        .setModelSpec(modelSpec)
                        .addAllMetadataField(Arrays.asList("signature_def"))
                        .build();
         // 5、获取元数据
        GetModelMetadata.GetModelMetadataResponse getModelMetadataResponse =
                predictionServiceBlockingStub.getModelMetadata(modelMetadataRequest);

        channel.shutdownNow();
    }

说明:

  • Model.ModelSpec.newBuilder绑定需要访问的模型的名字。
  • GetModelMetadataRequest中addAllMetadataField绑定curl命令返回的metadata当中的signature_def 字段。


动态更新模型

    public static void addNewModel() {
        // 1、构建动态更新模型1
        ModelServerConfigOuterClass.ModelConfig modelConfig1 =
                ModelServerConfigOuterClass.ModelConfig.newBuilder()
                        .setBasePath("/models/new_model")
                        .setName("new_model")                      
                        .setModelType(ModelServerConfigOuterClass.ModelType.TENSORFLOW)
                        .build();

        // 2、构建动态更新模型2
        ModelServerConfigOuterClass.ModelConfig modelConfig2 =
                ModelServerConfigOuterClass.ModelConfig.newBuilder()
                        .setBasePath("/models/wdl_model")
                        .setName("wdl_model")
                        .setModelType(ModelServerConfigOuterClass.ModelType.TENSORFLOW)
                        .build();
        
        // 3、合并动态更新模型到ModelConfigList对象中
        ModelServerConfigOuterClass.ModelConfigList modelConfigList =
                ModelServerConfigOuterClass.ModelConfigList.newBuilder()
                        .addConfig(modelConfig1)
                        .addConfig(modelConfig2)
                        .build();

        // 4、添加到ModelConfigList到ModelServerConfig对象当中
        ModelServerConfigOuterClass.ModelServerConfig modelServerConfig = 
                ModelServerConfigOuterClass.ModelServerConfig.newBuilder()
                .setModelConfigList(modelConfigList)
                .build();

        // 5、构建ReloadConfigRequest并绑定ModelServerConfig对象。
        ModelManagement.ReloadConfigRequest reloadConfigRequest =
                ModelManagement.ReloadConfigRequest.newBuilder()
                        .setConfig(modelServerConfig)
                        .build();

        // 6、构建modelServiceBlockingStub访问句柄
        ManagedChannel channel = ManagedChannelBuilder.forAddress(host, port).usePlaintext().build();
        ModelServiceGrpc.ModelServiceBlockingStub modelServiceBlockingStub = 
        ModelServiceGrpc.newBlockingStub(channel);

        ModelManagement.ReloadConfigResponse reloadConfigResponse =
                modelServiceBlockingStub.handleReloadConfigRequest(reloadConfigRequest);

        System.out.println(reloadConfigResponse.getStatus().getErrorMessage());

        channel.shutdownNow();

    }

说明:

  • 动态更新模型是一个全量的模型加载,在发布A模型后想动态发布B模型需要同时传递模型A和B的信息。
  • 再次强调,需要全量更新,全量更新,全量更新!!!


在线模型预测

    public static void doPredict() throws Exception {

        // 1、构建feature
        Map<String, Feature> featureMap = new HashMap<>();
        featureMap.put("match_type", feature(""));
        featureMap.put("position", feature(0.0f));
        featureMap.put("brand_prefer_1d", feature(0.0f));
        featureMap.put("brand_prefer_1m", feature(0.0f));
        featureMap.put("brand_prefer_1w", feature(0.0f));
        featureMap.put("brand_prefer_2w", feature(0.0f));
        featureMap.put("browse_norm_score_1d", feature(0.0f));
        featureMap.put("browse_norm_score_1w", feature(0.0f));
        featureMap.put("browse_norm_score_2w", feature(0.0f));
        featureMap.put("buy_norm_score_1d", feature(0.0f));
        featureMap.put("buy_norm_score_1w", feature(0.0f));
        featureMap.put("buy_norm_score_2w", feature(0.0f));
        featureMap.put("cate1_prefer_1d", feature(0.0f));
        featureMap.put("cate1_prefer_2d", feature(0.0f));
        featureMap.put("cate1_prefer_1m", feature(0.0f));
        featureMap.put("cate1_prefer_1w", feature(0.0f));
        featureMap.put("cate1_prefer_2w", feature(0.0f));
        featureMap.put("cate2_prefer_1d", feature(0.0f));
        featureMap.put("cate2_prefer_1m", feature(0.0f));
        featureMap.put("cate2_prefer_1w", feature(0.0f));
        featureMap.put("cate2_prefer_2w", feature(0.0f));
        featureMap.put("cid_prefer_1d", feature(0.0f));
        featureMap.put("cid_prefer_1m", feature(0.0f));
        featureMap.put("cid_prefer_1w", feature(0.0f));
        featureMap.put("cid_prefer_2w", feature(0.0f));
        featureMap.put("user_buy_rate_1d", feature(0.0f));
        featureMap.put("user_buy_rate_2w", feature(0.0f));
        featureMap.put("user_click_rate_1d", feature(0.0f));
        featureMap.put("user_click_rate_1w", feature(0.0f));

        Features features = Features.newBuilder().putAllFeature(featureMap).build();
        Example example = Example.newBuilder().setFeatures(features).build();

        // 2、构建Predict请求
        Predict.PredictRequest.Builder predictRequestBuilder = Predict.PredictRequest.newBuilder();

        // 3、构建模型请求维度ModelSpec,绑定模型名和预测的签名
        Model.ModelSpec.Builder modelSpecBuilder = Model.ModelSpec.newBuilder();
        modelSpecBuilder.setName("wdl_model");
        modelSpecBuilder.setSignatureName("predict");
        predictRequestBuilder.setModelSpec(modelSpecBuilder);

        // 4、构建预测请求的维度信息DIM对象
        TensorShapeProto.Dim dim = TensorShapeProto.Dim.newBuilder().setSize(300).build();
        TensorShapeProto shapeProto = TensorShapeProto.newBuilder().addDim(dim).build();
        TensorProto.Builder tensor = TensorProto.newBuilder();
        tensor.setTensorShape(shapeProto);
        tensor.setDtype(DataType.DT_STRING);

        // 5、批量绑定预测请求的数据
        for (int i=0; i<300; i++) {
            tensor.addStringVal(example.toByteString());
        }
        predictRequestBuilder.putInputs("examples", tensor.build());

        // 6、构建PredictionServiceBlockingStub对象准备预测
        ManagedChannel channel = ManagedChannelBuilder.forAddress(host, port).usePlaintext().build();
        PredictionServiceGrpc.PredictionServiceBlockingStub predictionServiceBlockingStub = 
            PredictionServiceGrpc.newBlockingStub(channel);
        
        // 7、执行预测
        Predict.PredictResponse predictResponse = 
         predictionServiceBlockingStub.predict(predictRequestBuilder.build());
      
        // 8、解析请求结果
        List<Float> floatList = predictResponse
         .getOutputsOrThrow("probabilities")
         .getFloatValList();
    }

说明:

  • TFS的RPC请求过程中设置的参数需要考虑TF模型的数据结构。
  • TFS的RPC请求有同步和异步两种方式,上述只展示同步方式。


TF模型结构

{
    "model_spec": {
        "name": "wdl_model",
        "signature_name": "",
        "version": "4"
    },
    "metadata": {
        "signature_def": {
            "signature_def": {
                "predict": {
                    "inputs": {
                        "examples": {
                            "dtype": "DT_STRING",
                            "tensor_shape": {
                                "dim": [{
                                    "size": "-1",
                                    "name": ""
                                }],
                                "unknown_rank": false
                            },
                            "name": "input_example_tensor:0"
                        }
                    },
                    "outputs": {
                        "logistic": {
                            "dtype": "DT_FLOAT",
                            "tensor_shape": {
                                "dim": [{
                                        "size": "-1",
                                        "name": ""
                                    },
                                    {
                                        "size": "1",
                                        "name": ""
                                    }
                                ],
                                "unknown_rank": false
                            },
                            "name": "head/predictions/logistic:0"
                        },
                        "class_ids": {
                            "dtype": "DT_INT64",
                            "tensor_shape": {
                                "dim": [{
                                        "size": "-1",
                                        "name": ""
                                    },
                                    {
                                        "size": "1",
                                        "name": ""
                                    }
                                ],
                                "unknown_rank": false
                            },
                            "name": "head/predictions/ExpandDims:0"
                        },
                        "probabilities": {
                            "dtype": "DT_FLOAT",
                            "tensor_shape": {
                                "dim": [{
                                        "size": "-1",
                                        "name": ""
                                    },
                                    {
                                        "size": "2",
                                        "name": ""
                                    }
                                ],
                                "unknown_rank": false
                            },
                            "name": "head/predictions/probabilities:0"
                        },
                        "classes": {
                            "dtype": "DT_STRING",
                            "tensor_shape": {
                                "dim": [{
                                        "size": "-1",
                                        "name": ""
                                    },
                                    {
                                        "size": "1",
                                        "name": ""
                                    }
                                ],
                                "unknown_rank": false
                            },
                            "name": "head/predictions/str_classes:0"
                        },
                        "logits": {
                            "dtype": "DT_FLOAT",
                            "tensor_shape": {
                                "dim": [{
                                        "size": "-1",
                                        "name": ""
                                    },
                                    {
                                        "size": "1",
                                        "name": ""
                                    }
                                ],
                                "unknown_rank": false
                            },
                            "name": "add:0"
                        }
                    },
                    "method_name": "tensorflow/serving/predict"
                },
                "classification": {
                    "inputs": {
                        "inputs": {
                            "dtype": "DT_STRING",
                            "tensor_shape": {
                                "dim": [{
                                    "size": "-1",
                                    "name": ""
                                }],
                                "unknown_rank": false
                            },
                            "name": "input_example_tensor:0"
                        }
                    },
                    "outputs": {
                        "classes": {
                            "dtype": "DT_STRING",
                            "tensor_shape": {
                                "dim": [{
                                        "size": "-1",
                                        "name": ""
                                    },
                                    {
                                        "size": "2",
                                        "name": ""
                                    }
                                ],
                                "unknown_rank": false
                            },
                            "name": "head/Tile:0"
                        },
                        "scores": {
                            "dtype": "DT_FLOAT",
                            "tensor_shape": {
                                "dim": [{
                                        "size": "-1",
                                        "name": ""
                                    },
                                    {
                                        "size": "2",
                                        "name": ""
                                    }
                                ],
                                "unknown_rank": false
                            },
                            "name": "head/predictions/probabilities:0"
                        }
                    },
                    "method_name": "tensorflow/serving/classify"
                },
                "regression": {
                    "inputs": {
                        "inputs": {
                            "dtype": "DT_STRING",
                            "tensor_shape": {
                                "dim": [{
                                    "size": "-1",
                                    "name": ""
                                }],
                                "unknown_rank": false
                            },
                            "name": "input_example_tensor:0"
                        }
                    },
                    "outputs": {
                        "outputs": {
                            "dtype": "DT_FLOAT",
                            "tensor_shape": {
                                "dim": [{
                                        "size": "-1",
                                        "name": ""
                                    },
                                    {
                                        "size": "1",
                                        "name": ""
                                    }
                                ],
                                "unknown_rank": false
                            },
                            "name": "head/predictions/logistic:0"
                        }
                    },
                    "method_name": "tensorflow/serving/regress"
                },
                "serving_default": {
                    "inputs": {
                        "inputs": {
                            "dtype": "DT_STRING",
                            "tensor_shape": {
                                "dim": [{
                                    "size": "-1",
                                    "name": ""
                                }],
                                "unknown_rank": false
                            },
                            "name": "input_example_tensor:0"
                        }
                    },
                    "outputs": {
                        "classes": {
                            "dtype": "DT_STRING",
                            "tensor_shape": {
                                "dim": [{
                                        "size": "-1",
                                        "name": ""
                                    },
                                    {
                                        "size": "2",
                                        "name": ""
                                    }
                                ],
                                "unknown_rank": false
                            },
                            "name": "head/Tile:0"
                        },
                        "scores": {
                            "dtype": "DT_FLOAT",
                            "tensor_shape": {
                                "dim": [{
                                        "size": "-1",
                                        "name": ""
                                    },
                                    {
                                        "size": "2",
                                        "name": ""
                                    }
                                ],
                                "unknown_rank": false
                            },
                            "name": "head/predictions/probabilities:0"
                        }
                    },
                    "method_name": "tensorflow/serving/classify"
                }
            }
        }
    }
}
目录
相关文章
|
16天前
|
存储 Java
java用base64编码案例
Java Base64编码示例:导入`java.util.Base64`,设置字符串`originalString`,使用`Base64.getEncoder().encodeToString()`编码并存储到`encodedString`,打印编码后字符串。解码用`Base64.getDecoder().decode()`。
20 0
|
1月前
|
安全 Java API
精通 Java 后台开发:案例分析与实践
【4月更文挑战第5天】本文旨在帮助读者掌握 Java 后台开发,通过电子商务系统案例探讨数据库设计、RESTful API、安全性和性能优化。使用 Spring 框架简化开发,Spring Security 保障安全,缓存技术提升性能。实践部分强调版本控制、单元测试、CI/CD 和代码规范的重要性,助力开发者提升技能,应对挑战。
|
2月前
使用ueditor实现多图片上传案例——实体类(Shopping.java)
使用ueditor实现多图片上传案例——实体类(Shopping.java)
20 0
|
2月前
Mybatis+mysql动态分页查询数据案例——分页工具类(Page.java)
Mybatis+mysql动态分页查询数据案例——分页工具类(Page.java)
28 1
|
1月前
|
SQL 设计模式 安全
Java单例模式几种写法以及代码案例拿来直接使用
Java单例模式几种写法以及代码案例拿来直接使用
34 0
|
2月前
Mybatis+mysql动态分页查询数据案例——工具类(MybatisUtil.java)
Mybatis+mysql动态分页查询数据案例——工具类(MybatisUtil.java)
18 1
|
2月前
|
Java 数据库连接 mybatis
mybatis简单案例源码详细【注释全面】——实体层(User.java)
mybatis简单案例源码详细【注释全面】——实体层(User.java)
14 0
|
4天前
|
设计模式 消息中间件 安全
【Java多线程】关于多线程的一些案例 —— 单例模式中的饿汉模式和懒汉模式以及阻塞队列
【Java多线程】关于多线程的一些案例 —— 单例模式中的饿汉模式和懒汉模式以及阻塞队列
9 0
|
12天前
|
Java 关系型数据库 测试技术
Java代码一键生成数据库文档(案例详解)
Screw是一个自动化数据库文档生成工具,能根据数据库表结构快速生成简洁、多格式(HTML、Word、Markdown)的文档,支持MySQL、MariaDB等多数据库。它使用Freemarker模板,允许用户自定义样式。依赖包括HikariCP数据库连接池和对应JDBC驱动。通过在Java代码或Maven插件中配置,可方便生成文档。示例代码展示了如何在测试用例中使用Screw。文档效果依赖于数据库中的表和字段注释。
|
14天前
|
Java
【专栏】Java 8 的 Streams 提供了一种处理数据集合的新方式,增强了代码的可读性和可维护性
【4月更文挑战第28天】Java 8 的 Streams 提供了一种处理数据集合的新方式,增强了代码的可读性和可维护性。本文介绍了 Streams 的基本概念,如从数据源创建 Stream,以及中间和终端操作。通过过滤、映射、归并、排序、分组等案例,展示了 Streams 的使用,包括并行 Streams 提高效率。学习 Streams 可以提升代码质量和效率,文章鼓励读者在实际开发中探索更多 Streams 功能。