OSS下载中文名编码错误

news2024/9/24 23:24:13

        最近工作中有个需求,是将客户支付的银行回执单上按照客户姓名上传到oss,然后将oss地址反显到pc后台,供客户自己查看下载。

        开始的时候感觉很简单,设计思路是根据客户支付单单号,查询数据库中是否存在该支付单的回执单,若存在直接返回,若不返回则通过拉卡拉API申请回执单,然后将结果保存到数据库(因不是频繁操作需求所以没有做缓存处理)。

private static LklSOAClient soaClient;

/**

* 初始化请求

*/

private static void iniLklSOAClient() {
    try {
        soaClient = new LklSOAClient();
        //应用号,钱账通提供的应用ID
        soaClient.setAppId(LaKaLaConstant.appId);
        //版本号,当前版本为 3.0
        soaClient.setVersion(LaKaLaConstant.version);
        //商户DSA私钥,钱账通提供的DSA私钥
        log.info("privateKeyPwd={}", LaKaLaConstant.privateKeyPwd);
        soaClient.setPrivateKey(LklDSAUtil.loadPrivateKey(LaKaLaConstant.private        KeyUrl, LaKaLaConstant.privateKeyPwd));
        //钱账通DSA公钥,钱账通提供的DSA公钥
        soaClient.setPublicKey(LklDSAUtil.loadPublicKey(LaKaLaConstant.publicKey        Url));
        //钱账通接口地址
        soaClient.setServerUrl(LaKaLaConstant.soaServerUrl);
    } catch (Exception e) {
        log.error(e.toString(), e);
    }
}
/**
 * 根据订单号查询电子回单文件列表
 * @param orderNo
 * @return
 */
public static JSONObject receiptQuery(String orderNo){
    JSONObject jsonParam = new JSONObject();
    jsonParam.put("order_no", orderNo);//订单单号必填
    JSONObject response = null;
    String service = "order.receipt.query";
    try {
        log.info("拉卡拉回单查询接口请求参数:{}", JSONUtil.format(jsonParam));
        response = soaClient.request(service, jsonParam);
        log.info("拉卡拉回单查询接口返回:{}", JSONUtil.format(response));
    } catch (Exception e) {
        log.error(e.toString(), e);
        return null;
    } finally {
        //此处保存请求日志信息
    }
    //此处返回请求结果    
    return Objects.requireNonNull(response).getJSONObject("response");
}

以上是关于获取电子回单的请求,截止到这里还没有问题

下面我们进行oss文件上传

/**
 * 上传到oss
 * @param path
 * @param fileName
 * @return
 */
public String fileUploadByFileNo(String path,String fileName){
    log.info("文件在服务器地址:{}",path);
    File file = new File(path);
    if (cn.hutool.core.io.FileUtil.isEmpty(file)) {
        log.error("文件不存在,filePath = {}", path);
        throw new RuntimeException();
    }
    log.info("文件长度:{}",file.length());
    // 上传到oss
    FileInputStream fileInputStream = null;
    String ossPath ;
    try {
        fileInputStream = new FileInputStream(file);
        //上传oss工具类方法
        ossPath = OSSUtil.uploadFile2OSS(fileInputStream, fileName);
        if (StringUtils.isEmpty(ossPath)) {
            log.error("oss文件上传失败,fileName={}",fileName);
            throw new RuntimeException();
        }
    } catch (Exception e) {
        log.error("oss文件流异常:",e);
        throw new RuntimeException();
    } finally {
        IoUtil.close(fileInputStream);
    }
    return ossPath;
}

/**
 * 上传到OSS服务器  如果同名文件会覆盖服务器上的
 *
 * @param inStream 文件流
 * @param fileName 文件名称 包括后缀名
 * @return 出错返回"" ,唯一MD5数字签名
 */
public static String uploadFile2OSS(InputStream inStream, String fileName){
    OSSClient ossClient = OSSUtil.getOSSClient();
    String ret = "";
    try {
        //创建上传Object的Metadata
        ObjectMetadata objectMetadata = new ObjectMetadata();
        objectMetadata.setContentLength(inStream.available());
        objectMetadata.setCacheControl("no-cache");
        objectMetadata.setHeader("Pragma","no-cache");
        //通过文件名判断并获取文件的contentType
objectMetadata.setContentType(getContentType(fileNam        e.substring        (fileName.lastIndexOf("."))));
        objectMetadata.setContentDisposition("inline;filename=" + fileName);
        //上传文件
        PutObjectResult putResult = ossClient.putObject(bucketName, filedir + fileName, inStream, objectMetadata);
        //文件在oss地址
        ret = getUrl(fileName),StandardCharsets.UTF_8.toString();
    } catch (IOException e) {
        logger.error(e.getMessage(), e);
    } finally {
        try {
            if (inStream != null) {
                inStream.close();
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
    return ret;
}
/**
 * 获取文件URL
 *
 * @Title getUrl
 * @param @param
 *            key
 * @param @return
 * @return URL
 */
public static String getUrl(String key) throws UnsupportedEncodingException {
    if (key.indexOf("http") == -1) {
        OSSClient server = new OSSClient(endpoint, accessKeyId, accessKeySecret);// 连接oss云存储服务器
        // 设置URL过期时间为10年 3600l* 1000*24*365*10
        Date expirations = new Date(new Date().getTime() + 1000 * 60 * 5);// url超时时间
        // 生成URL
        URL url = server.generatePresignedUrl(bucketName,filedir + key, expirations);
        String strUrl = String.valueOf(url);
        // 关闭client
        server.shutdown();
        return splitUrl(strUrl);
    } else {
        return splitUrl(key);
    }
}
/**
 * 切割url
 *
 * @param url
 * @return
 */
public static String splitUrl(String url) {
    if (url.indexOf("?Expires") != -1) {
        return url.split("Expires")[0].substring(0,url.split("Expires")[0].length() -1 );
    }
    return url;
}

根据fileName获取URL时就会发现获取到地址不是这种

https://img-video.oss-cn-beijing.aliyuncs.com/测试中.pdf

而是一下这种

https://img-video.oss-cn-beijing.aliyuncs.com/%E5%BC%A0%E5%AE%81.pdf

开始的时候以为是自己没有指定上传文件编码导致的,但是在oss后台查看发现已经是中文名称路径了,所以就在上传时指定了编码格式为UTF-8

objectMetadata.setContentEncoding(StandardCharsets.UTF_8.toString());

后来发现并没有用,就去翻了一下oss源码,发现他们在上传时就已经指定了编码为UTF-8

源码路径com.aliyun.oss.internal.OSSOperation#createDefaultContext(com.aliyun.oss.HttpMethod, java.lang.String, java.lang.String)

那就只能是下载url时有问题了,继续查看源码发现文件下载时,又用UTF-8进行了编码

源码地址:

com.aliyun.oss.OSSClient#generatePresignedUrl(com.aliyun.oss.model.GeneratePresignedUrlRequest)

方法中在做http请求时

HttpUtil.paramToQueryString(params, "utf-8");

继续查看HttpUtil.paramToQueryString会发现他对url做了处理,如下

public static String urlEncode(String value, String encoding) {
    if (value == null) {
        return "";
    } else {
        try {
            String encoded = URLEncoder.encode(value, encoding);
            return encoded.replace("+", "%20").replace("*", "%2A").replace("~", "%7E").replace("/", "%2F");
        } catch (UnsupportedEncodingException var3) {
            throw new IllegalArgumentException(OSSUtils.OSS_RESOURCE_MANAGER.getString("FailedToEncodeUri"), var3);
        }
    }
}
截止这里就会明白了,因为在获取文件url时进行了URLEncoder.encode(value, encoding)所以获取的结果地址不是中文名称。

所以我们只需要在获取文件url地址后进行再进行编译即可,如下

ret = URLDecoder.decode(getUrl(fileName),StandardCharsets.UTF_8.toString());

会发现结果ret结果变成了我们想要的中文名称,如下

https://img-video.oss-cn-beijing.aliyuncs.com/测试中.pdf

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/430743.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Java生成二维码之Graphics2D自定义码眼形状

Java 2D API 提供了几个类来定义常见的几何对象,例如点、直线、曲线和矩形。这些几何类是 java.awt.geom包的一部分。通过熟练使用Graphics2D类,可以绘制出任意类型的图形。 官网教程地址:https://docs.oracle.com/javase/tutorial/2d/geome…

【虹科案例】固态量子发射器——虹科数字化仪用于控制钻石色心中的脉冲序列

前言 钻石的色心是晶格中的缺陷,其中碳原子被不同种类的原子取代,相邻的晶格位置是空的。由于其明亮的单光子发射和光学可访问的自旋,色心可以成为未来量子信息处理和量子网络的有前途的固态量子发射器。 实现自旋量子比特和相干光子纠缠的两…

基于RK3568的Linux驱动开发—— GPIO知识点(二)

authordaisy.skye的博客_CSDN博客-嵌入式,Qt,Linux领域博主系列基于RK3568的Linux驱动开发——GPIO知识点(一)_daisy.skye的博客-CSDN博客 查看goio使用情况 cat /sys/kernel/debug/gpio 1|rk3568_r:# cat /sys/kernel/debug/gpio gpiochip0: GPIOs 0-3…

English Learning - L2-14 英音地道语音语调 重音技巧 2023.04.10 周一

English Learning - L2-14 英音地道语音语调 重音技巧 2023.04.10 周一课前热身重音日常表达节奏单词全部重读的句子间隔时间非重读单词代词和缩约词助动词声临其境语调预习课前热身 学习目标 重音 重弱突出,重音突出核心表达的意思 重音是落在重读单词上&#x…

Vue3简介

1.Vue3简介 2020年9月18日,Vue.js发布3.0版本,代号:One Piece(海贼王)耗时2年多、[2600次提交](https://github.com/vuejs/vue-next/graphs/commit-activity)、[30个RFC](https://github.com/vuejs/rfcs/tree/master/…

机器学习 | 实验四:正则化

⭐对应笔记:正则化 📚描述 在这个练习中,你将实现正则化的线性回归和正则化的逻辑回归。 📚数据 这个数据包包含两组数据,一组用于线性回归,另一个用于逻辑回归。还包含一个名为"map_feature"…

Win11快速打开便签和使用技巧分享

Win11快速打开便签和使用技巧分享。Win11系统中为用户提供了一个非常实用的系统组件,就是便签功能,使用这个功能可以帮助我们便捷的进行一些重要内容的记录。那么如何去开启开启这个程序来使用呢?来看看以下的详情分享吧。 详细分享&#xff…

docker介绍与安装

目录 Docker docker概述 容器化优点 虚拟化架构 docker与虚拟机区别 docker三大核心概念 docker运行的原理 Docker安装 查看 docker 版本信息 docker 信息查看 Docker docker概述 Docker是一个开源的应用容器引擎,基于go语言开发并遵循了apache2.0协议开…

Perceiver Perceiver IO: 人工智能的多功能工具

如今人工智能系统使用的大多数架构都是专业的。2D 残差网络可能是处理图像的一个很好的选择,但它最多只能用于其他类型的数据,比如自动驾驶汽车中使用的激光雷达信号或机器人中使用的 torques。此外,标准架构在设计时通常只考虑一项任务&…

Seal AppManager发布:基于平台工程理念的全新应用部署管理体验

4月12日,数澈软件Seal(以下简称“Seal”)宣布推出新一代应用统一部署管理平台 Seal AppManager,采用平台工程的理念,降低基础设施操作的复杂度为研发和运维团队提供易用、一致的应用管理和部署体验,进而提升…

SpringMVC使用介绍-快速入门

文章目录SpringMVCSpringMVC快速入门bean加载控制SpringMVC SpringMVC快速入门 SpringMVC是一种基于Java实现MVC模型的轻量级Web框架 优点 使用简单,开发便捷(相比于Servlet) 灵活性强 使用SpringMVC技术开发web程序流程: 1.创建web工程&a…

[牛客复盘] 牛客小白月赛70 20230407

[牛客复盘] 牛客小白月赛70 20230407 一、本周周赛总结A、 小d和答案修改2. 思路分析3. 代码实现B、小d和图片压缩1. 题目描述2. 思路分析3. 代码实现C、小d和超级泡泡堂1. 题目描述2. 思路分析3. 代码实现D、小d和孤独的区间1. 题目描述2. 思路分析3. 代码实现E、小d的博弈1. …

C语言内存函数介绍以及实现

目录 前言 一:内存拷贝函数 (1)memcpy( )函数 (2)memove( )函数 二:内存比较函数 三:内存设置函数 前言 本文介绍的函数的函数声明都在头文件string.h中。 一:内存拷贝函数 (1)memcpy( )函数 函数声明:void* memcpy(void* dest,const void* src,size_t num) 作用…

【python】制作一个简单的界面,有手就行的界面~

目录前言准备工作试手小案例开始我们今天的案例教学尾语 💝前言 嗨喽~大家好呀,这里是魔王呐 ❤ ~! ttkbootstrap 是一个基于 tkinter 的界面美化库, 使用这个工具可以开发出类似前端 bootstrap 风格的 tkinter 桌面程序。 ttkbootstrap 不…

8:00面试,8:05就出来了 ,问的实在是太变态了···

从外包出来,没想到算法死在另一家厂子。 自从加入这家公司,每天都在加班,钱倒是给的不少,所以也就忍了。没想到12月一纸通知,所有人不许加班,薪资直降30%,顿时有吃不起饭的赶脚。 好在有个兄弟…

Yolov5_DeepSort_Pytorch:基于 Yolov5 + Deep Sort 的实时多目标跟踪器

Yolov5_DeepSort_Pytorch:基于 Yolov5 Deep Sort 的实时多目标跟踪器 原创 视界君 Python视界 昨天 Python视界分享 原文地址:Yolov5_DeepSort_Pytorch:基于 Yolov5 Deep Sort 的实时多目标跟踪器 简介 该存储库包含一个两阶段跟踪器。…

linux主机设置主机间免密登录

举例:想要在A主机免密登录到B主机; 此文案前提是linux都安装了ssh服务,可以使用systemctl status sshd 查看ssh状态 1、使用任意用户在A主机上执行ssh-keygen -t rsa 所有提示均按回车默认,会在当前目录生成.ssh文件夹&#xff0…

实战-高并发下的读/写

文章目录高并发下的读/写高并发读业务场景高并发写业务场景同时高并发读和高并发写业务场景高并发读策略一:加缓存/读副本方案一:本地缓存/集中式缓存方案二:数据库层面的改变,Master/Slave,使用主从完成读写分离方案三…

MySQL学习笔记:count(1)、count(*)、count(字段)的区别

关于数据库中行数统计,无论是MySQL还是Oracle,都有一个函数可以使用,那就是COUNT()。 但是,就是这个常用的COUNT函数,却暗藏着很多玄机,尤其是在面试的时候,一不小心就会…

JUC多并发编程 CompletableFuture

Future 接口理论 Future 接口(FutureTask 实现类): 定义了操作异步任务执行一些方法,如获取异步任务的执行结果、取消任务的执行、判断任务是否被取消、判断任务执行是否完毕等 方法图: 类图: 代码示例: import ja…