用javascript实现。
其实获得的只是本地文件名,你要参考smartupload组件看看如何在这种方式下上载文件
其实获得的只是本地文件名,你要参考smartupload组件看看如何在这种方式下上载文件
解决方案 »
- JSP页面设计问题……如何只刷新部分页面???
- 最近公司做个java对Excel导入导出数据
- 两个jsp依次执行的问题!
- 如果单是学jsp java中的哪些不需要细看
- 在JAVABEAN中如何创建HttpServletRequest实例,绝对急
- nutch 如何动态更新 索引库
- +++++++++++++++++++++++一个字闷!!!!++++++++++++++++++++++++++++++++++++++++++++++++++++++++
- 一个jdbc驱动的问题,大家帮我一下啊!!
- 呵呵,今天换了一家新地方,感觉比较满意,散分。(内空)
- 寻找weblogic
- 能通过forward把request信息转发给另一个网站的页面吗?
- 高分求救:eclipse的jvm突然关闭,然后整个eclipse也关闭。急!!!
本文讲解了一个使用XML技术上传文件的例子,使用该方法没有传统方法中的种种限制。 这个例子讲述了如何使用MSXML3.0和ADO Stream对象来实现这种新的上传方法。好处有很多,比如,不需要专用的上传组件。
引言
为了在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:http://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</DIV>
</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 version="1.0" ?> <root/>');
// 指定数据类型
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 。
一个表单可以同时上传多文件的啊,如果你说的表指form的话,至少fileupload可以如果要实现1~n多文件上传的话,也简单啊
类似那种粘贴的,上传完一个再上传一个,把先前的文件名一个个纪录到session里面,最后再一起处理罢了楼上的办法是没有办法的办法
小弟现在也遇到这个问题了,
我用jspsmartupload做上传的时候,感觉不满意。
/*初始可见部分*/
<table class="form" width="100%">
<tr>
<td><br>
<input type="button" name="add" value="添加附件" onClick="addPlan()">
<input type="button" name="del" value="删除附件" onClick="delPlan()">
<br>
<table class="form" id="policyPlanTable">
<tr> </tr> <tr class="row0">
<td width="15%"><input type="checkbox" name="planIndex" value='0'></td>
<td width="85%"><input type="file" name="filename" value='0'></td>
</tr> </table></td>
</tr></table>
/*隐藏的表单*/
<table id="hiddenRow" style="display:none">
<tr class="row">
<td><input type="checkbox" name="planIndex" value=""></td>
<td><input type="file" name="filename" value=""></td></tr></table>
<SCRIPT LANGUAGE="JavaScript">
/**
* InsertRow() 在当前位置插入一条记录
* tableId 表格ID
* hideRowId 用来给新增的行赋初值的隐藏行
*/
function InsertRow(tableId, hideRowId, rowIndex) {
var oInput = window.event.srcElement;
var oTd = oInput.parentElement;
var oTr = oTd.parentElement;
var rowid = oTr.rowIndex;
var oTable = document.getElementById(tableId);
if (oTable == null) {
alert("Table [" + tableId + "] not found");
return;
}
var hideRow = document.getElementById(hideRowId);
if(hideRow == null) {
alert("Hidden row [" + hideRowId + "] not found");
return;
}
//新增一行;
var oRow;
if (rowIndex != null){
oRow = oTable.insertRow(rowIndex+1);
} else {
oRow = oTable.insertRow();
}
//赋值;
for (i = 0;i < hideRow.cells.length;i++){
var oCell = oRow.insertCell(i);
oCell.innerHTML = hideRow.cells(i).innerHTML;
oCell.colSpan = hideRow.cells(i).colSpan;
oCell.rowSpan = hideRow.cells(i).rowSpan;
oCell.vAlign = hideRow.cells(i).vAlign;
oCell.align = hideRow.cells(i).align;
oCell.bgColor = hideRow.cells(i).bgColor;
oCell.style.cssText = hideRow.cells(i).style.cssText;
}
}
/**
* DeleteRow() 删除选中记录;
* tableId 表格ID
* hideRowId 用来给新增的行赋初值的隐藏行
*/
function DeleteRow(tableId, hideRowId, rowIndex){
var oInput = window.event.srcElement;
var oTd = oInput.parentElement;
var oTr = oTd.parentElement;
var rowid = oTr.rowIndex;;
if (rowIndex !=null) {
rowid = rowIndex;
}
var oTable = document.getElementById(tableId);
var oHideRow = document.getElementById(hideRowId);
if(oHideRow == null) {
return;
}
var oRow = oTable.rows[rowid];
if((oTable.rows[rowid+1] != null || oTable.rows[rowid-1] != null)
&& oRow.cells.length == oHideRow.cells.length) {
oTable.deleteRow(rowid);
} else {
var oNextRow = oTable.rows[rowid+1];
if(oNextRow != null && oNextRow.cells.length == oHideRow.cells.length) {
//下一行与隐藏行单元数目相同,将下一行内容复制到当前行,然后删除下一行;
for (i = 0;i < oNextRow.cells.length ;i++){
var oCell = oRow.cells[i + oRow.cells.length - oNextRow.cells.length];
oCell.innerHTML = oNextRow.cells(i).innerHTML;
}
oTable.deleteRow(rowid + 1);
} else {
//当前行已经是最后一行,清除内容即可
for (i = 0;i < oHideRow.cells.length ;i++){
var oCell = oRow.cells[i + oRow.cells.length - oHideRow.cells.length];
oCell.innerHTML = oHideRow.cells(i).innerHTML;
}
}
}
}
/**
* 添加一个附件
*/
function addPlan(){
InsertRow("policyPlanTable", "hiddenRow");
}
/**
* 删除一个附件
*/
function delPlan() {
if (document.actionform.planIndex.length==1) {
alert("至少要有一个险种");
return false;
}
for (i=document.actionform.planIndex.length-1; i>=0; i--){
if (document.actionform.planIndex[i].checked == true){
DeleteRow("policyPlanTable", "hiddenRow", i+1);
}
}
}//-->
</SCRIPT>