百度图片识别自定义实现(替代AipOcr)

package com.visy.service;

import com.baidu.aip.util.Base64Util;
import com.baomidou.mybatisplus.core.toolkit.StringUtils;
import com.visy.exception.ErrorEnum;
import com.visy.exception.BusinessException;
import com.visy.config.BaiduConfig;
import lombok.AllArgsConstructor;
import lombok.Getter;
import lombok.extern.slf4j.Slf4j;
import org.json.JSONObject;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.*;
import org.springframework.stereotype.Service;
import org.springframework.util.LinkedMultiValueMap;
import org.springframework.util.MultiValueMap;
import org.springframework.web.client.RestTemplate;
import org.springframework.web.util.UriComponentsBuilder;

import java.util.Date;
import java.util.HashMap;
import java.util.Map;
import java.util.Objects;

/**
 * 百度图片识别自定义实现(替代AipOcr)
 * @author visy.wang
 * @date 2022/1/19 9:24
 */
@Slf4j
@Service
public class AipOcrClient {
    private static long expireTime = 0L; //token过期时间戳
    private static String accessToken = null; //token值

    @Autowired
    private BaiduConfig baiduConfig; //百度认证信息,配置自己的apiKey和secretKey
    @Autowired
    private RestTemplate restTemplate;

    //Url管理枚举
    @Getter
    @AllArgsConstructor
    enum UrlKey {
        ACCESS_TOKEN("https://aip.baidubce.com/oauth/2.0/token", "认证地址"),
        DRIVING_LICENSE("https://aip.baidubce.com/rest/2.0/ocr/v1/driving_license", "驾驶证识别地址"),
        VEHICLE_LICENSE("https://aip.baidubce.com/rest/2.0/ocr/v1/vehicle_license", "行驶证识别地址");

        private final String url;
        private final String desc;
    }


    /**
     * 驾驶证识别
     * @param image 图片(二进制)
     * @param options 其他参数
     * @return 识别结果
     */
    public JSONObject drivingLicense(byte[] image, HashMap<String,String> options){
        options.put("image", Base64Util.encode(image));
        return request(UrlKey.DRIVING_LICENSE, options);
    }

    /**
     * 驾驶证识别
     * @param imageUrl 图片地址
     * @param options 其他参数
     * @return 识别结果
     */
    public JSONObject drivingLicense(String imageUrl, HashMap<String,String> options){
        options.put("url", imageUrl);
        return request(UrlKey.DRIVING_LICENSE, options);
    }

    /**
     * 行驶证识别
     * @param image 图片(二进制)
     * @param options 其他参数
     * @return 识别结果
     */
    public JSONObject vehicleLicense(byte[] image, HashMap<String,String> options){
        options.put("image", Base64Util.encode(image));
        return request(UrlKey.VEHICLE_LICENSE, options);
    }

    /**
     * 行驶证识别
     * @param imageUrl 图片地址
     * @param options 其他参数
     * @return 识别结果
     */
    public JSONObject vehicleLicense(String imageUrl, HashMap<String,String> options){
        options.put("url", imageUrl);
        return request(UrlKey.VEHICLE_LICENSE, options);
    }

    //带Token过期检查的request,暂不需要,现已计算过期时间并在失效前重新获取
    private JSONObject requestWithTokenCheck(UrlKey urlKey, Map<String,String> options){
        JSONObject res = request(urlKey, options);
        if(res.has("error_code")){
            int errCode = res.getInt("error_code");
            if(errCode==110 || errCode==111){
                accessToken = null; //Token失效,清除并重新获取
                return request(urlKey, options);
            }
        }
        return res;
    }

    /**
     * 请求
     * @param urlKey url地址枚举
     * @param options 其他参数
     * @return 识别结果
     */
    private JSONObject request(UrlKey urlKey, Map<String,String> options){
        HttpHeaders headers = new HttpHeaders();
        headers.setContentType(MediaType.APPLICATION_FORM_URLENCODED);

        String url = urlKey.getUrl();
        MultiValueMap<String, String> queryStr = getQueryStr(urlKey);
        UriComponentsBuilder uriBuilder = UriComponentsBuilder.fromHttpUrl(url).queryParams(queryStr);

        MultiValueMap<String, Object> formData = new LinkedMultiValueMap<>();
        if(Objects.nonNull(options) && !options.isEmpty()){
            options.forEach(formData::add);
        }

        HttpEntity<MultiValueMap<String, Object>> request = new HttpEntity<>(formData, headers);

        ResponseEntity<String> res = restTemplate.postForEntity(uriBuilder.toUriString(), request, String.class);
        if(!HttpStatus.OK.equals(res.getStatusCode())){
            throw buzError("请求失败:httpStatus=" + res.getStatusCode());
        }

        String resBody = res.getBody();

        if(StringUtils.isBlank(resBody)){
            throw buzError("响应体为空");
        }

        return new JSONObject(resBody);
    }

    private MultiValueMap<String, String> getQueryStr(UrlKey urlKey){
        MultiValueMap<String, String> queryStr = new LinkedMultiValueMap<>();
        if(UrlKey.ACCESS_TOKEN.equals(urlKey)){
            queryStr.add("grant_type", "client_credentials");
            queryStr.add("client_id", baiduConfig.getApiKey());
            queryStr.add("client_secret", baiduConfig.getSecretKey());
        }else{
            queryStr.add("access_token", getAccessToken()); //添加认证信息
        }
        return queryStr;
    }

    /**
     * 获取AccessToken
     * @return AccessToken
     */
    private String getAccessToken(){
        long now = System.currentTimeMillis();

        if(StringUtils.isNotBlank(accessToken) && now<expireTime){
            //有缓存并在有效期内,直接使用
            return accessToken;
        }

        try{
            return getAccessToken(now);
        }catch (Exception e){
            throw buzError("AccessToken获取失败:" + e.getMessage());
        }
    }

    private String getAccessToken(long now){
        log.info("正在重新获取AccessToken...");

        JSONObject body = request(UrlKey.ACCESS_TOKEN, null);

        if(body.has("error")){
            throw buzError(body.getString("error_description"));
        }

        int tokenExpires = body.getInt("expires_in"); //有效期,单位秒,默认30天
        expireTime = now + (tokenExpires-86400L)*1000L; //提前1天失效
        accessToken = body.getString("access_token"); //缓存token
        log.info("认证成功,有效期至:{}", new Date(expireTime));

        return accessToken;
    }

    /**
     * 抛出业务异常
     * @param errMsg 错误信息
     * @return 业务异常
     */
    private BusinessException buzError(String errMsg){
        //BusinessException: 自定义业务异常
        return new BusinessException(ErrorEnum.BUSINESS_ERR.getCode(), errMsg);
    }
}

package com.visy.config;

import lombok.Data;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Configuration;

/**
 * @author visy.wang
 * @date 2022/1/19 9:28
 */
@Data
@Configuration
@ConfigurationProperties(prefix = "baidu")
public class BaiduConfig {
    private String appId;
    private String apiKey;
    private String secretKey;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
实现微信小程序中的百度图片识别,你需要完成以下步骤: 1. 在百度AI开放平台申请一个账号,并创建一个图片识别应用。 2. 在微信小程序中引入百度AI的JavaScript SDK。 3. 在小程序页面中编写相应的代码,调用百度AI的图片识别API实现图片识别的功能。例如,你可以在小程序的某个页面上添加一个按钮,当用户点击按钮后,可以让用户上传一张图片,然后通过百度AI的图片识别API进行识别,并将结果显示在页面上。 下面是一个简单的实现图片识别的代码示例: 1. 在小程序的页面中引入百度AI的JavaScript SDK: ``` import BaiduAI from 'baiduai.js'; const baiduAI = new BaiduAI({ appId: 'your_app_id', apiKey: 'your_api_key', secretKey: 'your_secret_key' }); ``` 2. 在按钮的点击事件中,调用百度AI的图片识别API实现图片识别: ``` onImageUpload: function() { wx.chooseImage({ success: function(res) { wx.showLoading({ title: '正在上传图片...', }) wx.uploadFile({ url: 'https://your_api_service_url', filePath: res.tempFilePaths[0], name: 'image', success: function(res) { const result = JSON.parse(res.data).result; baiduAI.imageClassify(result.image, result.imageType).then((res) => { wx.hideLoading(); wx.showToast({ title: '识别成功!', }) console.log(res); }).catch((err) => { wx.hideLoading(); wx.showToast({ title: '识别失败!', icon: 'none' }) console.log(err); }); } }) } }) } ``` 在这个示例中,我们首先使用微信的chooseImage API让用户选择一张图片,然后使用微信的uploadFile API图片上传到自己的API服务中。在API服务中,我们调用百度AI的图片识别API,获取识别结果。最后,我们将识别结果显示在页面上。 需要注意的是,在使用百度AI的API时,需要传入正确的参数和密钥,以确保API能够正常工作。同时,需要注意百度AI的API调用次数和频率限制,不要超过限制。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值