当前位置: 首页 > 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.lryc.cn/news/578933.html

相关文章:

  • 修复手机液晶面板显性横向线性不良定位及相关液晶线路激光修复原理
  • 【安全工具】SQLMap 使用详解:从基础到高级技巧
  • 【深度学习机器学习】Epoch 在深度学习实战中的合理设置指南
  • cmake find_package
  • Minio安装配置,桶权限设置,nginx代理 https minio
  • JAVA学习-练习试用Java实现“人脸识别:使用OpenCV+Java实现人脸检测与识别”
  • 【论文阅读】DeepEyes: Incentivizing “Thinking with Images” via Reinforcement Learning
  • STM32之光敏电阻传感器模块
  • uniapp 滚动tab
  • WPF控件大全:核心属性详解
  • Android-EDLA 解决 GtsMediaRouterTestCases 存在 fail
  • 移动公司Linux运维工程师招聘笔试题
  • 深入解析外观模式(Facade Pattern):简化复杂系统的优雅设计
  • STM32F103RET6 介绍
  • 机器学习:集成学习方法之随机森林(Random Forest)
  • 基于多种机器学习的江苏省二手房价格预测系统的设计与实现【城市全国可换】
  • 【QT】ROS2 Humble联合使用QT教程
  • Qt designer坑-布局内子控件的顺序错乱
  • ABAP+记录一个BDC的BUG修改过程
  • transformers==4.42.0会有一个BUG
  • pdf 合并 python实现(已解决)
  • Python 数据分析与可视化 Day 14 - 建模复盘 + 多模型评估对比(逻辑回归 vs 决策树)
  • 文档全文搜索引擎:AnyTXT Searcher
  • 【RHCSA-Linux考试题目笔记(自用)】servera的题目
  • 2025-6GESP六级编程题分析
  • 机器学习在计算机视觉中的应用
  • 中科米堆三维扫描仪3D扫描压力阀抄数设计
  • 3D扫描:开启数字化世界的多面钥匙
  • 【强化学习】深度解析 GRPO:从原理到实践的全攻略
  • ESP32-S3开发板LVGL图形界面开发实战教程