热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

HTML5WebSocket实例(三)-文件上传处理

一、后台处理:publicvoidProcessRequest(HttpContextcontext){处理WebSocket请

一、后台处理:

        public void ProcessRequest(HttpContext context)
{
//处理WebSocket 请求
context.AcceptWebSocketRequest(DoWork);
}
///
/// 委托处理函数定义
///

/// 当前WebSocket上下文
///
public async Task DoWork(AspNetWebSocketContext context)
{
//1.获取当前WebSocket 对象
WebSocket socket = context.WebSocket;
string filename = "";
//2.监视相应
while (true)
{
/*
* 此处缓存数组指定读取客户端数据的长度
* 如果客户端发送数据超过当前缓存区,则会读取多次
*/
ArraySegment buffer = new ArraySegment(new byte[1024 * 256]);
//接收客户端信息
CancellationToken token;
WebSocketReceiveResult result = await socket.ReceiveAsync(buffer, token);
if (socket.State == WebSocketState.Open)
{
//判断是否已经到了最后
int curLength = Math.Min(buffer.Array.Length, result.Count);
//判断用户传入的类型进行处理
if (result.MessageType == WebSocketMessageType.Text)
{
string msg = Encoding.UTF8.GetString(buffer.Array, 0, curLength);
filename = msg;
buffer = new ArraySegment(Encoding.UTF8.GetBytes("接收文件名成功:" + filename));
await socket.SendAsync(buffer, WebSocketMessageType.Text, true, CancellationToken.None);
}
else if (result.MessageType == WebSocketMessageType.Binary)
{
//创建并保存文件,如果上传成功,返回当前接收到的文件大小
string msg = SaveFile(filename, buffer, curLength);
buffer = new ArraySegment(Encoding.UTF8.GetBytes(curLength.ToString()));
await socket.SendAsync(buffer, WebSocketMessageType.Text, true, CancellationToken.None);
}
}
else { break; }
}
}
///
/// 追加二进制数据到文件
///

public string SaveFile(string file, ArraySegment buffer, int Length)
{
//去除文件名中的前后空格
file = file.Trim();
string fullname = @"F:\Javascript_Solution\H5Solition\UploadWebForm\content\" + file;
try
{
FileStream fs = new FileStream(fullname, FileMode.Append, FileAccess.Write);
try
{
byte[] result = buffer.ToArray();
fs.Write(result, 0, Length);
}
finally
{
fs.Close();
}
return "保存文件成功";
}
catch (Exception ex)
{
return ex.Message;
}
}

二,前台处理

HTML:



分段读取文件:










JS:

/*
* 测试WebSocket上传
* 本地浏览器上传速度测试单个文件,上传速度IE>FF>Google(Google浏览器慢相当多,原因:点击查看)
*/
var fileBox = document.getElementById('file');
var reader = null; //读取操作对象
var step = 1024 * 256; //每次读取文件大小 ,字节数
var cuLoaded = 0; //当前已经读取总数
var file = null; //当前读取的文件对象
var enableRead = true;//标识是否可以读取文件
var total = 0; //记录当前文件总字节数
var startTime = null; //标识开始上传时间
fileBox.Onchange= function () {
//获取文件对象
file = this.files[0];
total = file.size;
console.info("文件大小:" + file.size);
if (ws == null) {
if (window.confirm('建立与服务器链接失败,确定重试链接吗')) {
createSocket(function () {
bindReader();
});
}
return;
}
bindReader();
}
//绑定reader
function bindReader() {
cuLoaded = 0;
startTime = new Date();
enableRead = true;
reader = new FileReader();
//读取一段成功
reader.Onload= function (e) {
console.info('读取总数:' + e.loaded);
if (enableRead == false)
return false;
//根据当前缓冲区来控制客户端读取速度
if (ws.bufferedAmount > step * 10) {
setTimeout(function () {
//继续读取
console.log('--------------》进入等待');
loadSuccess(e.loaded);
}, 3);
} else {
//继续读取
loadSuccess(e.loaded);
}
}
//开始读取
readBlob();
}
//读取文件成功处理
function loadSuccess(loaded) {
//将分段数据上传到服务器
var blob = reader.result;
//使用WebSocket 服务器发送数据
if (cuLoaded == 0) //发送文件名
ws.send(file.name);
ws.send(blob);
//如果没有读完,继续
cuLoaded += loaded;
if (cuLoaded readBlob();
} else {
console.log('总共上传:' + cuLoaded + ',总共用时:' + (new Date().getTime() - startTime.getTime()) / 1000);
}
//显示结果进度
var percent = (cuLoaded / total) * 100;
document.getElementById('Status').innerText = percent;
document.getElementById('progressOne').value = percent;
}
//指定开始位置,分块读取文件
function readBlob() {
//指定开始位置和结束位置读取文件
var blob = file.slice(cuLoaded, cuLoaded + step);
reader.readAsArrayBuffer(blob);
}
//中止
function stop() {
//中止读取操作
console.info('中止,cuLoaded:' + cuLoaded);
enableRead = false;
reader.abort();
}
//继续
function containue() {
console.info('继续,cuLoaded:' + cuLoaded);
enableRead = true;
readBlob();
}
var ws = null;
//创建和服务器的WebSocket 链接
function createSocket(onSuccess) {
var url = 'ws://localhost:55373/ashx/upload3.ashx';
ws = new WebSocket(url);
ws.Onopen= function () {
console.log('connected成功');
if (onSuccess)
onSuccess();
}
ws.Onmessage= function (e) {
var data = e.data;
if (isNaN(data) == false) {
//console.log('当前上传成功:' + data);
} else {
console.info(data);
}
}
ws.Onclose= function (e) {
//中止客户端读取
stop();
console.log('链接断开');
}
ws.Onerror= function (e) {
//中止客户端读取
stop();
console.info(e);
console.log('传输中发生异常');
}
}
//页面加载完建立链接
createSocket();

更多实例:

分段上传七:http://www.cnblogs.com/tianma3798/p/5852475.html

分段上传八:http://www.cnblogs.com/tianma3798/p/5852527.html

更多Demo源代码:http://git.oschina.net/tiama3798/HTML5Demo/tree/WebSocket/


推荐阅读
  • 在重复造轮子的情况下用ProxyServlet反向代理来减少工作量
    像不少公司内部不同团队都会自己研发自己工具产品,当各个产品逐渐成熟,到达了一定的发展瓶颈,同时每个产品都有着自己的入口,用户 ... [详细]
  • Spring源码解密之默认标签的解析方式分析
    本文分析了Spring源码解密中默认标签的解析方式。通过对命名空间的判断,区分默认命名空间和自定义命名空间,并采用不同的解析方式。其中,bean标签的解析最为复杂和重要。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • 在Android开发中,使用Picasso库可以实现对网络图片的等比例缩放。本文介绍了使用Picasso库进行图片缩放的方法,并提供了具体的代码实现。通过获取图片的宽高,计算目标宽度和高度,并创建新图实现等比例缩放。 ... [详细]
  • 本文介绍了如何使用php限制数据库插入的条数并显示每次插入数据库之间的数据数目,以及避免重复提交的方法。同时还介绍了如何限制某一个数据库用户的并发连接数,以及设置数据库的连接数和连接超时时间的方法。最后提供了一些关于浏览器在线用户数和数据库连接数量比例的参考值。 ... [详细]
  • t-io 2.0.0发布-法网天眼第一版的回顾和更新说明
    本文回顾了t-io 1.x版本的工程结构和性能数据,并介绍了t-io在码云上的成绩和用户反馈。同时,还提到了@openSeLi同学发布的t-io 30W长连接并发压力测试报告。最后,详细介绍了t-io 2.0.0版本的更新内容,包括更简洁的使用方式和内置的httpsession功能。 ... [详细]
  • 如何查询zone下的表的信息
    本文介绍了如何通过TcaplusDB知识库查询zone下的表的信息。包括请求地址、GET请求参数说明、返回参数说明等内容。通过curl方法发起请求,并提供了请求示例。 ... [详细]
  • 图像因存在错误而无法显示 ... [详细]
  • Android自定义控件绘图篇之Paint函数大汇总
    本文介绍了Android自定义控件绘图篇中的Paint函数大汇总,包括重置画笔、设置颜色、设置透明度、设置样式、设置宽度、设置抗锯齿等功能。通过学习这些函数,可以更好地掌握Paint的用法。 ... [详细]
  • Asp.net Mvc Framework 七 (Filter及其执行顺序) 的应用示例
    本文介绍了在Asp.net Mvc中应用Filter功能进行登录判断、用户权限控制、输出缓存、防盗链、防蜘蛛、本地化设置等操作的示例,并解释了Filter的执行顺序。通过示例代码,详细说明了如何使用Filter来实现这些功能。 ... [详细]
  • 本文介绍了Python语言程序设计中文件和数据格式化的操作,包括使用np.savetext保存文本文件,对文本文件和二进制文件进行统一的操作步骤,以及使用Numpy模块进行数据可视化编程的指南。同时还提供了一些关于Python的测试题。 ... [详细]
  • Spring特性实现接口多类的动态调用详解
    本文详细介绍了如何使用Spring特性实现接口多类的动态调用。通过对Spring IoC容器的基础类BeanFactory和ApplicationContext的介绍,以及getBeansOfType方法的应用,解决了在实际工作中遇到的接口及多个实现类的问题。同时,文章还提到了SPI使用的不便之处,并介绍了借助ApplicationContext实现需求的方法。阅读本文,你将了解到Spring特性的实现原理和实际应用方式。 ... [详细]
  • 1,关于死锁的理解死锁,我们可以简单的理解为是两个线程同时使用同一资源,两个线程又得不到相应的资源而造成永无相互等待的情况。 2,模拟死锁背景介绍:我们创建一个朋友 ... [详细]
  • JVM 学习总结(三)——对象存活判定算法的两种实现
    本文介绍了垃圾收集器在回收堆内存前确定对象存活的两种算法:引用计数算法和可达性分析算法。引用计数算法通过计数器判定对象是否存活,虽然简单高效,但无法解决循环引用的问题;可达性分析算法通过判断对象是否可达来确定存活对象,是主流的Java虚拟机内存管理算法。 ... [详细]
  • 全面介绍Windows内存管理机制及C++内存分配实例(四):内存映射文件
    本文旨在全面介绍Windows内存管理机制及C++内存分配实例中的内存映射文件。通过对内存映射文件的使用场合和与虚拟内存的区别进行解析,帮助读者更好地理解操作系统的内存管理机制。同时,本文还提供了相关章节的链接,方便读者深入学习Windows内存管理及C++内存分配实例的其他内容。 ... [详细]
author-avatar
不言不语都可以温柔_631
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有