Base64 上传到阿里云 OSS

Canvas 截图是 Base64 格式,我要把它上传到阿里云 OSS。但 OSS 的 FormData 上传不支持 Base64。

第一步:Base64 转 Blob
javascript 复制代码
const uploadBase64ToOss = async (base64Str, directory = "files") => {
  // 解析 base64
  const arr = base64Str.split(",");
  const mime = arr[0].match(/:(.*?);/)[1]; // e.g. image/png
  const bstr = atob(arr[1]);
  const n = bstr.length;
  const u8arr = new Uint8Array(n);
  
  while (n--) {
    u8arr[n] = bstr.charCodeAt(n);
  }
  
  // 构造 Blob
  const blob = new Blob([u8arr], { type: mime });
  
  // ... 后续上传逻辑
};

atob 把 Base64 字符串转成二进制字符串,然后逐个字符转成 Uint8Array

第二步:构造 FormData
javascript 复制代码
// 获取 OSS 临时签名
const signData = await getSign(directory);

// 时间戳文件名
const timestamp = new Date().getTime();
const fileExt = mime.split("/")[1] || "bin"; // e.g. png
const fileName = `${timestamp}.${fileExt}`;

// OSS 表单数据
const formData = new FormData();
formData.append("OSSAccessKeyId", signData.data.accessId);
formData.append("policy", signData.data.policy);
formData.append("signature", signData.data.signature);
formData.append("key", `${directory}/${fileName}`);
formData.append("success_action_status", "200");
formData.append("file", blob, fileName);
第三步:用 fetch 上传
javascript 复制代码
const response = await fetch("https://<bucket-name>.<region>.aliyuncs.com", {
  method: "POST",
  body: formData,
});

if (response.status === 200) {
  return `https://<bucket-name>.<region>.aliyuncs.com/${directory}/${fileName}`;
} else {
  throw new Error("上传失败");
}
为什么不用 XMLHttpRequest

之前上传文件用的是 XMLHttpRequest,但 Base64 转 Blob 后,用 fetch 更简洁:

javascript 复制代码
// ❌ XMLHttpRequest
const xhr = new XMLHttpRequest();
xhr.open('POST', url);
xhr.onload = () => resolve(xhr.status === 200 ? url : null);
xhr.onerror = reject;
xhr.send(formData);

// ✅ fetch
const response = await fetch(url, { method: 'POST', body: formData });
return response.status === 200 ? url : null;
几个踩坑总结
  1. Base64 要去掉前缀split(",")[1] 只取数据部分
  2. MIME type 从 Base64 提取 :用正则 /:(.*?);/
  3. 文件扩展名从 MIME 提取mime.split("/")[1]
  4. Blob 构造要指定 typenew Blob([u8arr], { type: mime })
  5. fetch 比 XHR 简洁:异步代码更易读
相关推荐
主机哥哥15 小时前
2026年阿里云618活动优惠政策详细解读
阿里云
Database_Cool_19 小时前
AI 时代的数据仓库:阿里云 AnalyticDB MySQL 向量检索 + SQL 分析一体化实战
数据仓库·人工智能·mysql·阿里云
Database_Cool_19 小时前
Doris vs 阿里云 AnalyticDB MySQL vs ClickHouse:3 大 OLAP 产品 2026 深度对比
数据库·mysql·阿里云
Database_Cool_20 小时前
PB 级数据实时分析:阿里云 AnalyticDB MySQL Serverless 弹性架构深度解析
阿里云·架构·云计算
Database_Cool_20 小时前
数据仓库弹性扩缩容实践:阿里云 AnalyticDB MySQL 按需付费方案详解
数据仓库·mysql·阿里云
中草药z20 小时前
【RAG】工程化实战:全链路原理复盘 + 方案选型 + 实战高阶玩法
java·深度学习·机器学习·阿里云·rag·springai
王码码20351 天前
多台服务器怎么统一看状态?Beszel 轻量监控,搭起来不费事
运维·服务器·后端·安全·阿里云·接口·web
Mixtral1 天前
职场录音转写工具投入产出比实测:随身鹿、通义听悟、阿里云与Trint该怎么选?
阿里云·云计算
Linlingu1 天前
OpenClaw接入阿里云百炼模型配置教程(完整可落地)
人工智能·阿里云·云计算·办公自动化·数字员工·小龙虾
Database_Cool_2 天前
AnalyticDB MySQL vs StarRocks/ByteHouse:云数仓选型指南——全托管 vs 自建方案
数据库·数据仓库·mysql·阿里云