XML技术上传文件XML技术

原创 2005年01月03日 02:08:00
引言 

为了在HTML网页中获得上传功能,在客户端我们可以使用如下格式的FORM: 

< FORM NAME="myForm" ACTION="TargetURL.asp" ENCTYPE="multipart/form-data"METHOD="post" > 
< INPUT TYPE="file" NAME="myFile" > 
< INPUT TYPE="submit" value="Upload File" > 
< /FORM > 


  这种方案在客户端和服务器端的使用都有很多限制。首先,我们必须使用POST方法,因为GET方法无法处理这样的表单数据。并且,没有什么方法可以在不使用表单的情况下引发一个POST动作。把数据发送给表单处理程序后,浏览器将会把处理程序作为新页面加载,然后使用者会看到一个不讨人喜欢的页面转换过程。 

  ENCTYPE属性为表单定义了MIME编码方式,上传文件的表单的ENCTYPE属性必须使用“multipart/form-data”。把这个属性设置为“multipart/form-data”就创建了一个与传统结构不同的POST缓冲区(复合结构),ASP的Request对象无法访问这样的表单内容。所以,我们可以使用Request.binaryRead方法来访问这些数据,但是无法使用脚本语言来完成这一切。Request.binaryRead方法返回一个VTarray型数据(只包含无符号一字节字符的Variant型数组)。但是脚本语言只能处理Variant型数据。为了解决这个问题,只能使用专用的ASP上传组件,或者ISAPI扩展程序,比如CPSHOST.DLL。这是设计上的限制。 

新的上传方案 

需要按照如下步骤操作。 
客户端: 

使用MSXML 3.0创建一个XML文档 
创建一个针对二进制内容的XML节点 
使用ADO Stream object将上传的文件数据放入该节点 
使用XMLHTTP对象把这个XML文档发送给Web服务器 



服务器端: 

  从Request对象中读出XML文档读出二进制节点中的数据并且存储到服务器上的文件中。当然,我们也可以将其存储到数据库的BLOB型字段中。 
  在解释这段代码之前,我们可以对这个方案进行一些思考。 

对XML的思考 

  XML格式支持很多数据类型,比如numeric, float, character等等。很多作者将XML定义为ASCII格式,但是我们不能忽视,XML技术还可以使用“bin.base64”数据类型来描述二进制信息。这个特性在MS XML3.0解析器重得到完全的支持,但是目前还需要一些特别设置。该对象提供一些可以对二进制数据进行完全控制的属性: 

  obj_node.dataType - 该可读写的属性定义了特定节点的数据类型。MSXML解析器支持更多的数据类型(参见MSDN:url.gifhttp://msdn.microsoft.com/library/psdk/xmlsdk/xmls3z1v.htm) 
对于二进制数据,我们可以使用“bin.base64”类型。 

  obj_node.nodeTypedvalue - 该可读写属性包含了按照制定类型表示的指定节点的数据。 
我们可以创建一个包含多个bin.base64类型节点的XML文档,节点中包含上传的文件。这点特性可以使用一个POST一次上传多个文件。 

  我们可以使用XMLHttpRequest对象和POST方法发送一个XML文档给Web服务器。该对象为HTTP服务器提供了客户端协议支持,允许在Web服务器上发送和接受MS XMLDOM对象。XMLHttpRequest是Internet Explorer 5内置的COM对象(不需要定制安装),并且发送完毕后无需转换页面。 


对ADO Stream对象的思考 

  我们可以在客户端创建一个包含一个或者多个二进制节点的XML文档。我们还必须把文件内容填入节点中。但是很不幸,脚本语言不能访问本地文件系统,并且scripting.FileSystem对象(是Win32系统的内置对象)到目前为止还不能访问二进制文件。这是设计上的限制。所以我们需要另外找一个可以提供对本地二进制文件的访问的COM对象。 

  ADO Stream对象(MDAC 2.5中的组件)提供了读、写和管理二进制流数据的手段。字节流的内容可以是文本,或者二进制数据,并且没有容量上的限制。在ADO 2.5中,Microsoft对Stream对象的介绍不属于ADO对象结构的任何一层,所以,我们无需捆绑即可使用该对象。 

  本文中使用Stream对象来访问文件内容,再把内容存入XML节点。 

客户端 

以下示例代码使用Stream和MSXML对象完成文件上传动作。 

< HTML > 
< HEAD >< TITLE >File Send< /TITLE >< /HEAD > 
< BODY > 
< INPUT id=btn_send name="btn_send" type=button value="FILE SEND" > 
< DIV id=div_message >Ready 


< /BODY > 
< /HTML > 

script LANGUAGE=Javascript > 

// 上传函数 
function btn_send.onclick() 

// 创建 ADO-stream 对象 
var ado_stream = new ActiveXObject("ADODB.Stream"); 

// 创建包含默认头信息和根节点的 XML文档 
var xml_dom = new ActiveXObject("MSXML2.DOMdocument"); 
xml_dom.loadXML(   ); 
// 指定数据类型 
xml_dom.documentElement.setAttribute("xmlns:dt""urn:schemas-microsoft-com:datatypes"); 

// 创建一个新节点,设置其为二进制数据节点 
var l_node1 = xml_dom.createElement("file1"); 
l_node1.dataType = "bin.base64"
// 打开Stream对象,读源文件 
ado_stream.Type = 1; // 1=adTypeBinary 
ado_stream.Open(); 
ado_stream.LoadFromFile("c://tmp//myfile.doc"); 
// 将文件内容存入XML节点 
l_node1.nodeTypedvalue = ado_stream.Read(-1); // -1=adReadAll 
ado_stream.Close(); 
xml_dom.documentElement.appendChild(l_node1); 

// 可以创建多个二进制节点,一次上传多个文件 

// 把XML文档发送到Web服务器 
var xmlhttp = new ActiveXObject("Microsoft.XMLHTTP"); 
xmlhttp.open("POST","./file_recieve.asp",false); 
xmlhttp.send(xml_dom); 
// 显示服务器返回的信息 
div_message.innerHTML = xmlhttp.ResponseText; 

< /script > 


服务器端 

以下代码使用相同的对象提供服务器端的上传处理功能。 


< %@ LANGUAGE=VBscript% > 
< % Option Explicit 
Response.Expires = 0 

 定义变量和对象。 
dim ado_stream 
dim xml_dom 
dim xml_file1 

 创建 Stream 对象 
set ado_stream = Server.CreateObject("ADODB.Stream"
 从Request对象创建 XMLDOM对象 
set xml_dom = Server.CreateObject("MSXML2.DOMdocument"
xml_dom.load(request) 
 读出包含二进制数据的节点 
set xml_file1 = xml_dom.selectSingleNode("root/file1"

 打开Stream对象,把数据存入其中 
ado_stream.Type = 1   1=adTypeBinary 
ado_stream.open 
ado_stream.Write xml_file1.nodeTypedvalue 
 文件存盘 
ado_stream.SaveToFile "c:/tmp/upload1.doc",2   2=adSaveCreateOverWrite 
ado_stream.close 

 销毁对象 
set ado_stream = Nothing 
set xml_dom = Nothing 
 向浏览器返回信息 
Response.Write "Upload successful!" 
% > 


也可以使用Stream对象把数据放到数据库的BLOB型字段中。 

使用该方法的益处 

不引起页面转换。 
不需要专用组件。 
可同时上传多个文件。 

  这段程序是纯脚本写成的,可以很容易的插入到其他代码中,而不需要任何HTML对象的配合。还可以把这个逻辑在任何支持COM标准的语言中实现。 

系统安全考虑 

  该方法只能使用于内部网络,因为它需要IE5的安全级别设置为“低”。必须: 

  允许脚本和ActiveX对象。该设置允许浏览器执行类似 "myobj = new activexobject(...)"的 Jscript语句; 

  必须允许穿越域访问数据源。这个设置允许在客户端使用Stream对象。还必须在服务器和客户端都安装MS XML DOM 3.0 和MDAC 2.5 。 

学习J2EE的路途(二)纯干货

继续上次的学习,将新的一些知识点进行总结如下: 1:WEB开发中的四个域对象(在笔试和面试中比较常见,要熟悉它们四个的区别和对应的域范围,每个对象得生命周期,及其适用的情况)   (1)PageCon...
  • Cs_hnu_scw
  • Cs_hnu_scw
  • 2017年08月03日 13:58
  • 174

xml有哪些解析技术?区别是什么?

1.DOM生成和解析XML文档 为 XML 文档的已解析版本定义了一组接口。解析器读入整个文档,然后构建一个驻留内存的树结构,然后代码就可以使用 DOM 接口来操作这个树结构。优点:整个文档树在内...
  • lingzhm
  • lingzhm
  • 2015年09月10日 21:27
  • 8878

四种xml的解析方式

大致可以分为两种:一种是基于树结构处理的Dom解析,另外一种是基于事件模型的SAX解析 一、介绍及优缺点分析 1. DOM(Document Object Model)       ...
  • HUXU981598436
  • HUXU981598436
  • 2016年06月10日 17:49
  • 3809

Java解析XML的几种技术

很庆幸在交谈的这近一个小时内每句话都没白讲白听,对面的技术官让我很钦佩。总得来说,对方比较看重实战经验,几乎所有的问题都是从你自己的阐述中提取出来的。说实话,我在之前做过很多Java的功课,但实际上根...
  • u013485804
  • u013485804
  • 2014年01月31日 10:49
  • 518

Xml方面xml有哪些解析技术?区别是什么?你在项目中用到了xml技术的哪些方面?如何实现的?

原文地址:http://blog.sina.com.cn/s/blog_49d3ec2f0102dvm4.html作者:LuChao 1、xml有哪些解析技术?区别是什么? 2、你在项...
  • xiong_1234
  • xiong_1234
  • 2014年07月28日 13:10
  • 1252

递归算法、File类、输入输出流、XML解析

递归算法定义递归算法是把问题转化为规模缩小了的同类问题的子问题。然后递归调用函数(或过程)来表示问题的解。 一个过程(或函数)直接或间接调用自己本身,这种过程(或函数)叫递归过程(或函数).简单代码...
  • aiynmimi
  • aiynmimi
  • 2015年07月23日 20:43
  • 658

解析XML的几种方式

XML现在已经成为一种通用的数据交换格式,它的平台无关性,语言无关性,系统无关性,给数据集成与交互带来了极大的方便。对于XML本身的语法知识与技术细节,需要阅读相关的技术文献,这里面包括的内容有DOM...
  • yuan16423276
  • yuan16423276
  • 2013年12月17日 23:29
  • 31371

SpringMVC实现文件上传并解析XML

1.配置文件spring-mvc.xml文件 ...
  • QCIWYY
  • QCIWYY
  • 2017年01月04日 11:34
  • 2151

java解析xml的几种方式

【目录】 一、【基础知识——扫盲】 二、【DOM、SAX、JDOM、DOM4j简单使用介绍】 三、【性能测试】 四、【对比】 五、【小插曲XPath】 六、【补充】...
  • u012868901
  • u012868901
  • 2016年05月26日 15:12
  • 2889

JAVA 异步ajax实现xls 文件上传 并且解析xls

html: var upload = function(){ $.ajaxFileUpload({ url: 'map/upload', secureuri: f...
  • qilin001cs
  • qilin001cs
  • 2016年07月20日 10:09
  • 1247
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:XML技术上传文件XML技术
举报原因:
原因补充:

(最多只允许输入30个字)