当前位置: 首页 > news >正文

将文件使用base64存入数据库并在微信小程序中实现文件下载

文件存储最基础的两个字段是文件内容和文件名称,在数据中FileContent的数据类型为varbinary(max) (这种方式的弊端是不能大量存储文件,会占用数据库的大量内存)

现在将文件通过文件的完整路径,获取文件的二进制流和文件名(包含后缀)

filePath是文件的完整路径

   byte[] FileData = File.ReadAllBytes(filePath);
   // 获取文件名(带扩展名)
   string fileNameWithExtension = Path.GetFileName(filePath);

 

然后使用参数化的方式将byte[]类型数据存入数据库中

现在我们将文件从数据库中读出来

取出文件内容的核心代码逻辑是把取出的二进制数据转为base64类型字符,并取出文件名字(文件的基本信息存储在以下类中)

微信小程序中页面效果图如下:(暂时只包含了文件名称和下载文件的操作)

 

  <!-- 新增文件下载栏 --><view class="file-download-section"><!-- 文件图标占位(根据文件类型显示不同图标) --><view class="file-icon {{fileTypeClass}}"><text class="file-icon-text">{{fileIcon}}</text></view><view class="file-info"><view class="file-name">{{fileName || '未命名文件'}}</view></view><button class="download-btn" bindtap="handleDownload"><text class="download-btn-text">{{isDownloading ? '下载中...' : '下载文件'}}</text></button></view>
/* 文件下载栏样式 */
.file-download-section {display: flex;align-items: center;padding: 24rpx 30rpx;background-color: #ffffff;border-radius: 12rpx;box-shadow: 0 2rpx 10rpx rgba(0, 0, 0, 0.05);margin-bottom: 30rpx;
}/* 文件图标样式 */
.file-icon {width: 80rpx;height: 80rpx;border-radius: 8rpx;background-color: #f0f0f0;display: flex;align-items: center;justify-content: center;margin-right: 20rpx;font-size: 40rpx;
}/* 不同文件类型的图标样式 */
.file-icon.pdf {background-color: #e6f7ff;color: #1890ff;
}
.file-icon.doc {background-color: #f6ffed;color: #52c41a;
}
.file-icon.jpg, .file-icon.png {background-color: #fff3e0;color: #fa8c16;
}
.file-icon.zip {background-color: #fce8e6;color: #ff4d4f;
}/* 文件信息区域样式 */
.file-info {flex: 1;min-width: 0;margin-right: 20rpx;
}.file-name {font-size: 28rpx;font-weight: 500;white-space: nowrap;overflow: hidden;text-overflow: ellipsis;margin-bottom: 6rpx;
}.file-size {font-size: 24rpx;color: #999;
}/* 下载按钮样式 */
.download-btn {min-width: 160rpx;height: 70rpx;line-height: 70rpx;padding: 0;background-color: #1890ff;color: #ffffff;border-radius: 35rpx;font-size: 28rpx;
}.download-btn-text {display: block;
}

接下来需要js文件将base64字符转为可使用的本地文件缓存地址

// Base64 解码实现 (替代浏览器的 atob)function atob(base64) {// Base64 字符表const chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';// 移除空格并验证长度base64 = base64.replace(/\s/g, '');if (base64.length % 4 !== 0) {throw new Error('Invalid base64 string');}let output = '';let i = 0;// 解码逻辑while (i < base64.length) {const enc1 = chars.indexOf(base64.charAt(i++));const enc2 = chars.indexOf(base64.charAt(i++));const enc3 = chars.indexOf(base64.charAt(i++));const enc4 = chars.indexOf(base64.charAt(i++));const chr1 = (enc1 << 2) | (enc2 >> 4);const chr2 = ((enc2 & 15) << 4) | (enc3 >> 2);const chr3 = ((enc3 & 3) << 6) | enc4;output += String.fromCharCode(chr1);if (enc3 !== 64) {output += String.fromCharCode(chr2);}if (enc4 !== 64) {output += String.fromCharCode(chr3);}}return output;}function base64ToArrayBuffer(base64) {const base64Str = base64.replace(/^data:image\/\w+;base64,/, '');const binary = atob(base64Str);const len = binary.length;const buffer = new ArrayBuffer(len);const view = new Uint8Array(buffer);for (let i = 0; i < len; i++) {view[i] = binary.charCodeAt(i);}return buffer;
}function downloadBase64File(base64Str, fileName) {return new Promise((resolve, reject) => {try {const buffer = base64ToArrayBuffer(base64Str);const fs = wx.getFileSystemManager();const tempFilePath = `${wx.env.USER_DATA_PATH}/${fileName || 'download_' + Date.now()}`;console.log(tempFilePath)fs.writeFile({filePath: tempFilePath,data: buffer,success: () => {wx.getFileSystemManager().saveFile({tempFilePath: tempFilePath,success: (res) => resolve(res.savedFilePath),fail: (err) => reject(err)});},fail: (err) => reject(err)});} catch (error) {reject(error);}});
}export { downloadBase64File };

 在这里我们需要将这个文件新建到util文件下

 在微信小程序中通过接口请求后获取到以上的文件信息 

 

 最后实现方法(以上参数中的DocumentContent、FileName分别赋值给data中的fileContent、fileName)

下载文件核心代码:

getBase64FromServer(){return this.data.fileContent;},handleDownload() {const base64Data = this.getBase64FromServer();this.downloadFile(base64Data, this.data.fileName);},///下载文件(逻辑代码)downloadFile(base64Str, fileName) {wx.showLoading({title: '下载中...'});downloadBase64File(base64Str, fileName).then(savedFilePath => {console.log("缓存成功路径:" + savedFilePath)// 打开文件供用户保存wx.openDocument({filePath: savedFilePath,showMenu: true,success: () => {wx.showToast({title: '下载成功',})},fail: (err) => {wx.showToast({title: '打开文件失败',icon: 'none'});console.error(err);}});wx.hideLoading();}).catch(err => {wx.hideLoading();wx.showToast({title: '保存失败',icon: 'none'});console.error('下载错误:', err);});},

最后测试点击下载文件按钮可以预览文件

 

可以点击右上角三个点选择下载到手机上(可以手机的文件管理的文档中找到)

http://www.dtcms.com/a/264545.html

相关文章:

  • springboot 显示打印加载bean耗时工具类
  • iOS App无源码安全加固实战:如何对成品IPA实现结构混淆与资源保护
  • python中两种策略模式的实现
  • 2-RuoYi-UI管理平台的启动
  • 大语言模型随意猜测网址引发网络安全危机
  • 零信任安全:重塑网络安全架构的革命性理念
  • 【Unity3D实现加载在线地图——WebGL】
  • 【ABAP】 从无到有 新建一个Webdynpro程序
  • 同一水平的 RISC-V 架构的 MCU,和 ARM 架构的 MCU 相比,运行速度如何?
  • Tomcat log日志解析
  • 【Linux】文件权限以及特殊权限(SUID、SGID)
  • Highcharts 安装使用教程
  • Flutter Widget Preview 功能已合并到 master,提前在体验毛坯的预览支持
  • flutter flutter_vlc_player播放视频设置循环播放失效、初始化后获取不到视频宽高
  • 机器学习:集成学习方法之随机森林(Random Forest)
  • AWS RDS Aurora全局数据库转区域数据库实战指南:无缝迁移零停机
  • Windows VMWare Centos Docker部署Springboot 应用实现文件上传返回文件http链接
  • php上传或者压缩图片后图片出现倒转或者反转的问题
  • Hyper-YOLO: When Visual Object Detection Meets Hypergraph Computation
  • 在Ubuntu上多网卡配置HTTP-HTTPS代理服务器
  • c语言中的函数II
  • 今日学习:音视频领域入门文章参考(待完善)
  • 数据结构:数组(Array)
  • 文心快码答用户问|Comate AI IDE专场
  • 文心4.5开源模型部署实践
  • 使用Vue3实现输入emoji 表情包
  • 阿里云AppFlow AI助手打造智能搜索摘要新体验
  • 基于开源链动2+1模式AI智能名片S2B2C商城小程序的场景零售创新研究
  • 【Unity】MiniGame编辑器小游戏(八)三国华容道【HuarongRoad】
  • Active-Prompt:让AI更智能地学习推理的革命性技术