关于前端form提交后端返回文件流触发浏览器下载(并发控制)

前文介绍过这种文件下载方式,它有好处也有不足的地方。好处是可以充分利用浏览器自身的资源调度优势,开发只管提交不用关注下载细节,用户可以在浏览器下载任务中看到下载的状态。不足之处是只能下载到浏览器配置的默认下载位置,而且JS里不能掌握下载状态。

更为关键的是如果一次下载的文件较多的话,提交了N个请求,这样浏览器会创建N个标签页,虽然浏览器自己会调度,下载完后自行关闭这些标签页,但是会带来密集恐惧,那有没有办法控制下浏览器下载的任务数(虽然处于下载中的最多6个任务,但是等待的任务也是任务啊)

既然文件下载提交后,无法从浏览器得到任务下载状态,那么从服务器呢,这个方案是可行的。

前端任务提交后,后端服务器读取本地文件流通过pipeline对接到response,response设置头参数,pipeline完成后回调sse消息发送到前端通知文件下载完成。

javascript 复制代码
app.get('/events', (req,res)=>{
	res.writeHead(200,  { 'Content-Type': 'text/event-stream', 'Connection': 'keep-alive', 'Cache-Control': 'no-cache' });
	res.write(`data: ${JSON.stringify({"msg":"消息服务正常"})}\n\n`);
	let clientid=stringRandom(32, { letters: 'ABCDEF' });
	clients.push({ "clientid":clientid, "uid":req.session.user.userid, "res":res });
	req.on('close', () => { clients = clients.filter(item => (item.clientid != clientid)); });
});

function sendssemsg(jvar)
	clients.forEach(client=>{ if (client.uid==jvar.uid) { client.res.write(`data: ${JSON.stringify(jvar.msg)}\n\n`) } });
}

...
app.post("/getfile",express.urlencoded({ extended: false }),(req,res)=>{
...
	res.set({ "Content-Type": "application/octet-stream",  "Content-Disposition": "attachment;filename* = UTF-8''"+fixedEncodeURIComponent(filename.substr(filename.lastIndexOf("/")+1)),"Content-Length": stats.size });
	pipeline(fs.createReadStream(filepath), res, (err) => { 
		if (err) console.log("下载出错") 
		else sendssemsg({"uid":userid,"msg":filename+" 下载完成"})
		});
	...

通过sse发送下载结果,效率最好

前端收到相应的sse消息,就可以处理下载任务队列操作了

javascript 复制代码
	function downloadfile(filename) {
		let form=$("#formp");
		$("input[name=filename]").attr("value",filename);
		form.submit();
		}
	
	var indownloading=0;
	const evtSource = new EventSource("/events");

	evtSource.addEventListener('message', function(event) {
		//let jvar=JSON.parse(event.data);
		indownloading-=1;
		if (waitinglist.length>0) {
			indownloading+=1;
			downloadfile(waitinglist.shift());
			}
		})

	let indownloading=0;
	let taskLimit=navigator.hardwareConcurrency;
	while ((indownloading<tasklimit)&&(waitinglist.length>0)) {
		indownloading+=1;
		downloadfile(waitinglist.shift());
		}
html 复制代码
<form name="formp" id="formp" action="/getfile" method="post" target="_blank" rel="noopener noreferrer">
<input name="filename" value="">
</form>

这样效果就是下载任务控制在CPU核数,浏览器上看到的下载标签也不会太多,视觉观感上对用户更友好一些。

相关推荐
m0_748250744 分钟前
高性能Web网关:OpenResty 基础讲解
前端·openresty
前端没钱29 分钟前
从 Vue 迈向 React:平滑过渡与关键注意点全解析
前端·vue.js·react.js
汪洪墩34 分钟前
【Mars3d】设置backgroundImage、map.scene.skyBox、backgroundImage来回切换
开发语言·javascript·python·ecmascript·webgl·cesium
NoneCoder34 分钟前
CSS系列(29)-- Scroll Snap详解
前端·css
无言非影38 分钟前
vtie项目中使用到了TailwindCSS,如何打包成一个单独的CSS文件(优化、压缩)
前端·css
.生产的驴1 小时前
SpringBoot 对接第三方登录 手机号登录 手机号验证 微信小程序登录 结合Redis SaToken
java·spring boot·redis·后端·缓存·微信小程序·maven
我曾经是个程序员1 小时前
鸿蒙学习记录
开发语言·前端·javascript
顽疲1 小时前
springboot vue 会员收银系统 含源码 开发流程
vue.js·spring boot·后端
机器之心1 小时前
AAAI 2025|时间序列演进也是种扩散过程?基于移动自回归的时序扩散预测模型
人工智能·后端
羊小猪~~1 小时前
前端入门之VUE--ajax、vuex、router,最后的前端总结
前端·javascript·css·vue.js·vscode·ajax·html5