- 浏览: 200402 次
文章分类
最新评论
-
code_xiaoke:
session可是有30分钟有效期的还有如果在分布式的环境下 ...
Java Web 用户登陆示例代码 -
xul0038:
http://www.baidu.com
Java Web 用户登陆示例代码 -
16866:
非常棒,配置信息呢
Nginx负载均衡 -
开发小菜:
什么意思,没明白?能不能写一个例子
JS 实现DIV随浏览器窗口大小变化
概 述
文件上传和下载是 Web 应用中的一个常见功能,相信各位或多或少都曾写过这方面相关的代码。但本座看过不少人在实现上传或下载功能时总是不知不觉间与程序的业务逻辑纠缠在一起, 因此,当其他地方要用到这些功能时则无可避免地 Copy / Pase,然后再进行修改。这样丑陋不堪的做法导致非常容易出错不说,更大的问题是严重浪费时间不断做重复类似的工作,这是本座绝不能容忍的。哎,人生苦 短啊,浪费时间在这些重复工作身上实在是不值得,何不把这些时间省出来打几盘罗马或者踢一场球?为此,本座利用一些闲暇之时光编写了一个通用的文件上传和 文件下载组件,实现方法纯粹是基于 JSP,没有太高的技术难度,总之老少咸宜 ^_^。现把设计的思路和实现的方法向各位娓娓道来,希望能起到抛砖引玉的效果,激发大家的创造性思维。
任何公共组件的设计都必须考虑下面两个问题:
一、如何才能重用?
答:首先,重用的组件必须有用,也就是说,是功能完备的。但另一 方面,如果组件负责的职能太多也会影响重用。试想,一个文件上传组件同时还要负 责插入数据库记录,一个文件下载组件还要负责解析下载请求并查找要下载的文件那可真是太悲哀了,叫别人如何重用?也就是说,重用组件必须是功能完备并职责 单一的,绝对不能越界参与应用业务逻辑处理,不在其位不谋其政。
另外,要重用的组件绝不能反向依赖于上层使用者。通常,重用组件 以接口或类的形式提供给上层使用者调用,并放在单独的包中。也就是说,上层使用 者所在的包依赖于组件所在的包,如果组件再反向依赖于上层使用者,则两个包之间就存在依赖环,严重违反了面向对象设计原则中的无环依赖原则。试想,通用的 文件上传或下载组件如果需要读取一个在应用程序的某个地方定义的字符串常量来确定文件上传或下载的目录,那是囧了。
本文件上传和下载组件在设计时充分考虑到上述问题,只负责单纯的上传和下载工作,所需要的外部信息均通过相应方法进行设置,不会依赖于任何使用者。
二、组件的可用性如何?
答:这是一个相当有深度的问题 ^_^ 所谓可用性通俗来说就是好不好用,使用起来是否方便。作为公共组件来说,可用性是一个十分重要的质量指标。如果组件十分复杂难用,倒不如自己花点时间自己 写一个来得舒坦。本文件上传和下载组件在设计的过程中十分注重可用性目标,操作组件的代码行数不超过 10 行,只需几个步骤:
- 生成组件实例
- 设置实例属性
- 调用上传/下载方法
- 处理调用结果
水吹得已经够多了,下面让我们来看看文件上传和下载组件分别是如何实现的。
文件上传
文件上传操作通常会附加一些限制,如:文件类型、上传文件总大 小、每个文件的最大大小等。除此以外,作为一个通用组件还需要考虑更多的问题, 如:支持自定义文件保存目录、支持相对路径和绝对路径、支持自定义保存的文件的文件名称、支持上传进度反馈和上传失败清理等。另外,本座也不想重新造车 轮,本组件是基于 Commons File Upload 实现,省却了本座大量的工作 ^_^ 下面先从一个具体的使用例子讲起:
- 上传请求界面及代码
<form action="checkupload.action" method="post" enctype="multipart/form-data"> First Name: <input type="text" name="firstName" value="丑"> <br> Last Name: <input type="text" name="lastName" value="怪兽"> <br> Birthday: <input type="text" name="birthday" value="1978-11-03"> <br> Gender: 男 <input type="radio"" name="gender" value="false"> 女 <input type="radio"" name="gender" value="true" checked="checked"> <br> Working age: <select name="workingAge"> <option value="-1">-请选择-</option> <option value="3">三年</option> <option value="5" selected="selected">五年</option> <option value="10">十年</option> <option value="20">二十年</option> </select> <br> Interest: 游泳 <input type="checkbox" name="interest" value="1" checked="checked"> 打球 <input type="checkbox" name="interest" value="2" checked="checked"> 下棋 <input type="checkbox" name="interest" value="3"> 打麻将 <input type="checkbox" name="interest" value="4"> 看书 <input type="checkbox" name="interest" value="5" checked="checked"> <br> Photo 1.1: <input type="file"" name="photo-1"> <br> Photo 1.2: <input type="file"" name="photo-1"> <br> Photo 2.1: <input type="file"" name="photo-2"> <br> <br> <input type="submit" value="确 定"> <input type="reset" value="重 置"> </form> |
从上面的 HTML 代码可以看出,表单有 6 个普通域和 3 个文件域,其中前两个文件域的 name 属性相同。
- 上传处理代码
import com.bruce.util.BeanHelper; import com.bruce.util.Logger; import com.bruce.util.http.FileUploader; import com.bruce.util.http.FileUploader.FileInfo; import static com.bruce.util.http.FileUploader.*; @SuppressWarnings("unused") public class CheckUpload extends ActionSupport { // 上传路径 private static final String UPLOAD_PATH = "upload"; // 可接受的文件类型 private static final String[] ACCEPT_TYPES = {"txt", "pdf", "doc", ".Jpg", "*.zip", "*.RAR"}; // 总上传文件大小限制 private static final long MAX_SIZE = 1024 * 1024 * 100; // 单个传文件大小限制 private static final long MAX_FILE_SIZE = 1024 * 1024 * 10; @Override public String execute() { // 创建 FileUploader 对象 FileUploader fu = new FileUploader(UPLOAD_PATH, ACCEPT_TYPES, MAX_SIZE, MAX_FILE_SIZE); // 根据实际情况设置对象属性(可选) /* fu.setFileNameGenerator(new FileNameGenerator() { @Override public String generate(FileItem item, String suffix) { return String.format("%d_%d", item.hashCode(), item.get().hashCode()); } }); fu.setServletProgressListener(new ProgressListener() { @Override public void update(long pBytesRead, long pContentLength, int pItems) { System.out.printf("%d: length -> %d, read -> %d.\n", pItems, pContentLength, pBytesRead); } }); */ // 执行上传并获取操作结果 Result result = fu.upload(getRequest(), getResponse()); // 检查操作结果 if(result != FileUploader.Result.SUCCESS) { // 设置 request attribute setRequestAttribute("error", fu.getCause()); // 记录日志 Logger.exception(fu.getCause(), "upload file fail", Level.ERROR, false); return ERROR; } // 通过非文件表单域创建 Form Bean Persion persion = BeanHelper.createBean(Persion.class, fu.getParamFields()); // 图片保存路径的列表 List<String> photos = new ArrayList<String>(); /* 轮询文件表单域,填充 photos */ Set<String> keys = fu.getFileFields().keySet(); for(String key : keys) { FileInfo[] ffs = fu.getFileFields().get(key); for(FileInfo ff : ffs) { photos.add(String.format("(%s) %s%s%s", key, fu.getSavePath(), File.separator, ff.getSaveFile().getName())); } } // 设置 Form Bean 的 photos 属性 persion.setPhotos(photos); // 设置 request attribute setRequestAttribute("persion", persion); return SUCCESS; } } |
public class Persion { private String firstName; private String lastName; private Date birthday; private boolean gender; private int workingAge; private int[] interest; private List<String> photos; } |
public static class FileInfo { private String uploadFileName; private File saveFile; } |
分析下上面的 Java 代码,本例先根据保存目录、文件大小限制和文件类型限制创建一个 FileUploader 对象,然后调用该对象的 upload() 方法执行上传并返回操作结果,如果上传成功则 通过 getParamFields() 方法获取所有非文件表单域内容,并交由 BeanHelper 进行解析创建 Form Bean,再调用 getFileFields() 方法获取所有文件表单域的 FileInfo(FileInfo 包含上传文件的原始名称和被保存文件的 File 对象),最后完成 Form Bean 所有字段的填充并把 Form Bean 设置为 request 属性。
- 上传结果界面及代码
<table border="1"> <caption>Persion Attributs</caption> <tr><td>Name</td><td><c:out value="${persion.firstName} ${persion.lastName}"/> </td></tr> <tr><td>Brithday</td><td><c:out value="${persion.birthday}"/> </td></tr> <tr><td>Gender</td><td><c:out value="${persion.gender}"/> </td></tr> <tr><td>Working Age</td><td><c:out value="${persion.workingAge}"/> </td></tr> <tr><td>Interest</td><td><c:forEach var="its" items="${persion.interest}"> <c:out value="${its}" /> </c:forEach> </td></tr> <tr><td>Photos</td><td><c:forEach var="p" items="${persion.photos}"> <c:out value="${p}" /><br> </c:forEach> </td></tr> </table> |
从上面的处理结果可以看出,文件上传组件 FileUploader 正确地处理了表单的所有文件域和非文件域名,并且,整个文件上传操作过程非常简单,无需用户过多参与。下面我们来详细看看组件的主要实现代码:
/** 文件上传器 */ public class FileUploader { /** 不限制文件上传总大小的 Size Max 常量 */ public static final long NO_LIMIT_SIZE_MAX = -1; /** 不限制文件上传单个文件大小的 File Size Max 常量 */ public static final long NO_LIMIT_FILE_SIZE_MAX = -1; /** 默认的写文件阀值 */ public static final int DEFAULT_SIZE_THRESHOLD = DiskFileItemFactory.DEFAULT_SIZE_THRESHOLD; /** 默认的文件名生成器 */ public static final FileNameGenerator DEFAULT_FILE_NAME_GENERATOR = new CommonFileNameGenerator(); /** 设置上传文件的保存路径(不包含文件名) * * 文件路径,可能是绝对路径或相对路径 * 1) 绝对路径:以根目录符开始(如:'/'、'D:\'),是服务器文件系统的路径 * 2) 相对路径:不以根目录符开始,是相对于 WEB 应用程序 Context 的路径,(如:mydir 是指 * '${WEB-APP-DIR}/mydir') * 3) 规则:上传文件前会检查该路径是否存在,如果不存在则会尝试生成该路径,如果生成失败则 * 上传失败并返回 {@link Result#INVALID_SAVE_PATH} * */ private String savePath; /** 文件上传的总文件大小限制 */ private long sizeMax = NO_LIMIT_SIZE_MAX; /** 文件上传的单个文件大小限制 */ private long fileSizeMax = NO_LIMIT_FILE_SIZE_MAX; /** 可接受的上传文件类型集合,默认:不限制 */ private Set acceptTypes = new LStrSet(); /** 非文件表单域的映射 */ private Map paramFields = new HashMap(); /** 文件表单域的映射 */ private Map fileFields = new HashMap(); /** 文件名生成器 */ private FileNameGenerator fileNameGenerator = DEFAULT_FILE_NAME_GENERATOR; // commons file upload 相关属性 private int factorySizeThreshold = DEFAULT_SIZE_THRESHOLD; private String factoryRepository; private FileCleaningTracker factoryCleaningTracker; private String servletHeaderencoding; private ProgressListener servletProgressListener; /** 文件上传失败的原因(文件上传失败时使用) */ private Throwable cause; /** 执行上传 * * @param request : {@link HttpServletRequest} 对象 * @param response : {@link HttpServletResponse} 对象 * * @return : 成功:返回 {@link Result#SUCCESS} ,失败:返回其他结果, * 失败原因通过 {@link FileUploader#getCause()} 获取 * */ @SuppressWarnings("unchecked") public Result upload(HttpServletRequest request, HttpServletResponse response) { reset(); // 获取上传目录绝对路径 String absolutePath = getAbsoluteSavePath(request); if(absolutePath == null) { cause = new FileNotFoundException(String.format("path '%s' not found or is not directory", savePath)); return Result.INVALID_SAVE_PATH; } ServletFileUpload sfu = getFileUploadComponent(); List fiis = new ArrayList(); List items = null; Result result = Result.SUCCESS; // 获取文件名生成器 String encoding = servletHeaderencoding != null ? servletHeaderencoding : request.getCharacterEncoding(); FileNameGenerator fnGenerator = fileNameGenerator != null ? fileNameGenerator : DEFAULT_FILE_NAME_GENERATOR; try { // 执行上传操作 items = (List)sfu.parseRequest(request); } catch (FileUploadException e) { cause = e; if(e instanceof FileSizeLimitExceededException) result = Result.FILE_SIZE_EXCEEDED; else if(e instanceof SizeLimitExceededException) result = Result.SIZE_EXCEEDED; else if(e instanceof InvalidContentTypeException) result = Result.INVALID_CONTENT_TYPE; else if(e instanceof IOFileUploadException) result = Result.FILE_UPLOAD_IO_EXCEPTION; else result = Result.OTHER_PARSE_REQUEST_EXCEPTION; } if(result == Result.SUCCESS) { // 解析所有表单域 result = parseFileItems(items, fnGenerator, absolutePath, encoding, fiis); if(result == Result.SUCCESS) // 保存文件 result = writeFiles(fiis); } return result; } // 解析所有表单域 private Result parseFileItems(List items, FileNameGenerator fnGenerator, String absolutePath, String encoding, List fiis) { for(FileItem item : items) { if(item.isFormField()) // 解析非文件表单域 parseFormField(item, encoding); else { if(item.getSize() == 0) continue; // 解析文件表单域 Result result = parseFileField(item, absolutePath, fnGenerator, fiis); if(result != Result.SUCCESS) { reset(); cause = new InvalidParameterException(String.format("file '%s' not accepted", item.getName())); return result; } } } return Result.SUCCESS; } // 解析文件表单域 private Result parseFileField(FileItem item, String absolutePath, FileNameGenerator fnGenerator, List fiis) { String suffix = null; String uploadFileName = item.getName(); boolean isAcceptType = acceptTypes.isEmpty(); if(!isAcceptType) { suffix = null; int stuffPos = uploadFileName.lastIndexOf("."); if(stuffPos != -1) { suffix = uploadFileName.substring(stuffPos, uploadFileName.length()).toLowerCase(); isAcceptType = acceptTypes.contains(suffix); } } if(!isAcceptType) return Result.INVALID_FILE_TYPE; // 通过文件名生成器获取文件名 String saveFileName = fnGenerator.generate(item, suffix); if(!saveFileName.endsWith(suffix)) saveFileName += suffix; String fullFileName = absolutePath + File.separator + saveFileName; File saveFile = new File(fullFileName); FileInfo info = new FileInfo(uploadFileName, saveFile); // 添加表单域文件信息 fiis.add(new FileItemInfo(item, saveFile)); addFileField(item.getFieldName(), info); return Result.SUCCESS; } private void parseFormField(FileItem item, String encoding) { String name = item.getFieldName(); String value = item.getString(); // 字符串编码转换 if(!GeneralHelper.isStrEmpty(value) && encoding != null) { try { value = new String(value.getBytes("ISO-8859-1"), encoding); } catch(UnsupportedEncodingException e) { } } // 添加表单域名/值映射 addParamField(name, value); } /** 文件名生成器接口 * * 每次保存一个上传文件前都需要调用该接口的 {@link FileNameGenerator#generate} 方法生成要保存的文件名 * */ public static interface FileNameGenerator { /** 文件名生成方法 * * @param item : 上传文件对应的 {@link FileItem} 对象 * @param suffix : 上传文件的后缀名 * */ String generate(FileItem item, String suffix); } /** 默认通用文件名生成器 * * 实现 {@link FileNameGenerator} 接口,根据序列值和时间生成唯一文件名 * */ public static class CommonFileNameGenerator implements FileNameGenerator { private static final int MAX_SERIAL = 999999; private static final AtomicInteger atomic = new AtomicInteger(); private static int getNextInteger() { int value = atomic.incrementAndGet(); if(value >= MAX_SERIAL) atomic.set(0); return value; } /** 根据序列值和时间生成 'XXXXXX_YYYYYYYYYYYYY' 格式的唯一文件名 */ @Override public String generate(FileItem item, String suffix) { int serial = getNextInteger(); long millsec = System.currentTimeMillis(); return String.format("%06d_%013d", serial, millsec); } } /** 上传文件信息结构体 */ public static class FileInfo { private String uploadFileName; private File saveFile; // getters and setters ... } private class FileItemInfo { FileItem item; File file; // getters and setters ... } /** 文件上传结果枚举值 */ public static enum Result { /** 成功 */ SUCCESS, /** 失败:文件总大小超过限制 */ SIZE_EXCEEDED, /** 失败:单个文件大小超过限制 */ FILE_SIZE_EXCEEDED, /** 失败:请求表单类型不正确 */ INVALID_CONTENT_TYPE, /** 失败:文件上传 IO 错误 */ FILE_UPLOAD_IO_EXCEPTION, /** 失败:解析上传请求其他异常 */ OTHER_PARSE_REQUEST_EXCEPTION, /** 失败:文件类型不正确 */ INVALID_FILE_TYPE, /** 失败:文件写入失败 */ WRITE_FILE_FAIL, /** 失败:文件的保存路径不正确 */ INVALID_SAVE_PATH; } } |
具体的实现细节就不多描述了,大家可以下载附件慢慢研究。这里只说明两点:
1、应用可以实现自己的 FileNameGenerator 类替代默认的文件名生成器。
2、上传操作通过 FileUploader.Result 返回结果,并没有采用抛出异常的方式,因为本座认为在这里采用异常方式报告结果其实并不方便使用;另一方面,程序可以通过 getCause() 获取详细的错误信息。
文件下载
相对于文件上传,文件下载则简单很多,主要实现流程是根据文件名找到实际文件,并利用 Java 的相关类对 I/O 流进行读写。下面先看看一个使用示例:
import static com.bruce.util.http.FileDownloader.*; public class TestDownload extends ActionSupport { // 绝对路径 private static final String ABSOLUTE_PATH = "/Server/apache-tomcat-6.0.32/webapps/portal/download/下载测试 - 文本文件.txt"; // 相对路径 private static final String RELATE_PATH = "download/下载测试 - 项目框架.jpg"; @Override public String execute() { int type = getIntParam("type", 1); String filePath = (type == 1 ? ABSOLUTE_PATH : RELATE_PATH); // 创建 FileDownloader 对象 FileDownloader fdl = new FileDownloader(filePath); // 执行下载 Result result = fdl.downLoad(getRequest(), getResponse()); // 检查下载结果 if(result != Result.SUCCESS) { // 记录日志 Logger.exception(fdl.getCause(), String.format("download file '%s' fail", fdl.getFilePath()), Level.ERROR, false); } return NONE; } } |
从这个示例可以看出,文件下载组件的使用方法更简单,因为它不需要对下载结果进行很多处理。可以看出该组件也支持相对路径和绝对路径。下面我们来详细看看组件的主要实现代码:
/** 文件下载器 */ public class FileDownloader { /** 默认字节交换缓冲区大小 */ public static final int DEFAULT_BUFFER_SIZE = 1024 * 4; /** 下载文件的默认 Mime Type */ public static final String DEFAULT_CONTENT_TYPE = "application/force-download"; /** 设置下载文件的路径(包含文件名) * * filePath : 文件路径,可能是绝对路径或相对路径 * 1) 绝对路径:以根目录符开始(如:'/'、'D:\'),是服务器文件系统的路径 * 2) 相对路径:不以根目录符开始,是相对于 WEB 应用程序 Context 的路径,(如:mydir/myfile 是指 '${WEB-APP-DIR}/mydir/myfile') */ private String filePath; /** 显示在浏览器的下载对话框中的文件名称,默认与 filePath 参数中的文件名一致 */ private String saveFileName; /** 下载文件的 Mime Type,默认:{@link FileDownloader#DEFAULT_CONTENT_TYPE} */ private String contentType = DEFAULT_CONTENT_TYPE; /** 字节缓冲区大小,默认:{@link FileDownloader#DEFAULT_CONTENT_TYPE} */ private int bufferSize = DEFAULT_BUFFER_SIZE; /** 获取文件下载失败的原因(文件下载失败时使用) */ private Throwable cause; /** 执行下载 * * @param request : {@link HttpServletRequest} 对象 * @param response : {@link HttpServletResponse} 对象 * * @return : 成功:返回 {@link Result#SUCCESS} ,失败:返回其他结果, * 失败原因通过 {@link FileDownloader#getCause()} 获取 * */ public Result downLoad(HttpServletRequest request, HttpServletResponse response) { reset(); try { // 获取要下载的文件的 File 对象 File file = getFile(request); // 执行下载操作 downLoadFile(request, response, file); } catch(Exception e) { cause = e; if(e instanceof FileNotFoundException) return Result.FILE_NOT_FOUND; if(e instanceof IOException) return Result.READ_WRITE_FAIL; return Result.UNKNOWN_EXCEPTION; } return Result.SUCCESS; } // 执行下载操作 private void downLoadFile(HttpServletRequest request, HttpServletResponse response, File file) throws IOException { String fileName = new String(saveFileName.getBytes(), "ISO-8859-1"); // 解析 HTTP 请求头,获取文件的读取范围 Range range = parseDownloadRange(request); int length = (int)file.length(); int begin = 0; int end = length - 1; // 设置 HTTP 响应头 response.setContentType(contentType); response.setContentLength(length); response.setHeader("Content-Disposition", "attachment;filename=" + fileName); // 确定文件的读取范围(用于断点续传) if(range != null) { if(range.getBegin() != null) { begin = range.getBegin(); if(range.getEnd() != null) end = range.getEnd(); } else { if(range.getEnd() != null) begin = end + range.getEnd() + 1; } String contentRange = String.format("bytes %d-%d/%d", begin, end, length); response.setHeader("Accept-Ranges", "bytes"); response.setHeader("Content-Range", contentRange); response.setStatus(HttpServletResponse.SC_PARTIAL_CONTENT); } // 实际执行下载操作 doDownloadFile(response, file, begin, end); } // 实际执行下载操作 private void doDownloadFile(HttpServletResponse response, File file, int begin, int end) throws IOException { InputStream is = null; OutputStream os = null; try { byte[] b = new byte[bufferSize]; is = new BufferedInputStream(new FileInputStream(file)); os = new BufferedOutputStream(response.getOutputStream()); // 跳过已下载的文件内容 is.skip(begin); // I/O 读写 for(int i, left = end - begin + 1; left > 0 && ((i = is.read(b, 0, Math.min(b.length, left))) != -1); left -= i) os.write(b, 0, i); os.flush(); } finally { if(is != null) {try{is.close();} catch(IOException e) {}} if(os != null) {try{os.close();} catch(IOException e) {}} } } /** 文件下载结果枚举值 */ public static enum Result { /** 成功 */ SUCCESS, /** 失败:文件不存在 */ FILE_NOT_FOUND, /** 失败:读写操作失败 */ READ_WRITE_FAIL, /** 失败:未知异常 */ UNKNOWN_EXCEPTION; } } |
实现文件下载的代码也相当简单,另外,本组件支持断点续传。大家可以下载附件慢慢研究。
发表评论
-
java实现动态切换上网IP (ADSL拨号上网) java开发
2013-04-24 10:06 1300动态切换IP的实现主是也由Windows的rasdial命令提 ... -
JAVA字符串处理函数
2013-04-12 09:21 1150Java中的字符串也是一连串的字符。但是与许多其他的计算机语 ... -
(转)Lucene打分规则与Similarity模块详解
2013-02-06 14:08 1208搜索排序结果的控制 Lu ... -
Compass将lucene、Spring、Hibernate三者结合
2013-02-01 11:02 1679版权声明:转载时请以超链接形式标明文章原始出处和作者信息及本声 ... -
Lucene3.0详解
2013-02-01 10:57 1416★第一部分:概述 1. 我 ... -
Java Web 用户登陆示例代码
2013-02-01 09:56 58115实现功能: 1、用户登陆、注销 2、利用session记 ... -
Java对数函数及Java对数运算
2013-02-01 09:47 6799Java对数函数的计算方法非常有问题,然而在API中却有惊人 ... -
Lucene为不同字段指定不同分词器(转)
2013-01-31 17:34 3474在lucene使用过程中,如 ... -
域名管理与解析原理 — 《Java邮件开发详解》读书笔记
2013-01-31 14:56 1722一 基本概念 1. 域名:域名是由圆点分开一串单词或缩写组 ... -
优秀的Java工程师需要掌握的10项技能
2013-01-31 14:04 1861编程专业相对于计算机领域其他专业来讲,是一门比较难以修炼的专业 ... -
Web开发入门不得不看
2013-01-28 17:31 1036如今,各种互联网的Web ... -
MVC框架的映射和解耦
2013-01-25 21:37 835最近在写一个业务上用到的框架,回想起接触过的一些MVC框架, ... -
JAVA发送EMAIL的例子
2013-07-09 09:44 907import javax.mail.*; ... -
SSH + Lucene + 分页 + 排序 + 高亮 模拟简单新闻网站搜索引擎
2012-11-19 09:55 1387前两天看到了一个中国新闻网,这个网站的搜索form的actio ... -
Lucene多字段搜索
2012-11-19 09:53 1041最近在学习Lucene的过程中遇到了需要多域搜索并排序的问题, ... -
lucene之sort
2012-11-16 15:06 1094package cn.zqh.lucene.sort; im ... -
Nginx负载均衡
2012-11-16 11:45 7667最近迷上了Nginx,真实麻雀虽小,五脏俱全..功能实在强大. ... -
Lucene相关度排序的调整
2012-11-16 11:38 1730Lucene的搜索结果默认按 ... -
HashSet重复元素判断
2012-10-15 16:37 9139HashSet不能添加重复的元素,当调用add(Object) ... -
JAVA提高教程(2)-认识Set集合之HashSet
2012-10-09 09:44 974集合在Java里面的作用非 ...
相关推荐
发布于2012-5-7文件上传和下载是Web应用中的一个常见功能,相信各位或多或少都曾写过...为此,本座利用一些闲暇之时光编写了一个通用的文件上传和文件下载组件,实现方法纯粹是基于JSP,没有太高的技术难度,总之老少咸
这个项目提供了一个通用的文件上传和下载组件,它以Maven项目的形式存在,意味着它依赖于Maven的构建系统来管理项目依赖。IDEA作为开发工具,是JetBrains公司出品的一款强大的Java集成开发环境,为开发人员提供了...
JSPSmartUpload是一个针对JSP的文件上传组件,它允许开发者在JSP页面中轻松实现文件上传功能。这个库提供了一系列的类和方法,可以方便地处理多文件上传、文件大小限制、文件类型检查等问题。通过创建一个...
在这个压缩包中,包含的“commons”系列jar文件就是用来支持文件上传和下载的核心组件。 1. **Apache Commons FileUpload**: 这个库专门用于处理HTTP请求中的多部分数据,包括文件上传。在Servlet或Filter中,我们...
2、采用链式调用模式,只需要简单的配置当前运行采用哪个厂商,即可灵活的传输文件到对应云文件服务器中 3、支持OSS、COS、OBS、Minio等,如果是单机应用直接配置成本地即可 4、适用于对文件依赖比较大的应用
"java文件上传的用到的jar包"就是这样一个集合,它包含了实现文件上传功能所需的各种组件。 1. **Apache Commons FileUpload**:这是最常用的文件上传库之一,提供了处理HTTP请求中的多部分数据(multipart/form-...
8. **源码结构分析**:"spring boot+mvc+mybatis(通用mapper)+druid+jsp+bootstrap实现后台权限管理系统源码.docx"可能是项目的主要代码文档,详细描述了源码结构和实现细节。"demo-boot"可能是一个演示性的Spring ...
标题中的“仿造126,163,QQ邮箱实现的多文件无刷新上传组件”指的是该资源是模仿了126、163和QQ邮箱的文件上传功能,特别是其多文件同时上传并且在页面不刷新的情况下完成的特性。这种技术在现代Web应用中非常常见,...
总结来说,在Oracle EBS中实现文件上传和下载功能需要结合前端Form设计、后端Java处理以及数据库或文件系统的存储策略。通过合理的设计和编程,可以有效地实现安全、高效的文件管理功能,满足企业的业务需求。
在本项目中,"springboot+mvc+mybatis(通用mapper)+druid+jsp+bootstrap实现权限管理文件上传下载多数据源切换" 是一个综合性的Web应用开发实践,它涵盖了多个核心技术和工具,用于构建一个功能强大的后台系统。...
这里我们探讨的主题是“spring文件上传代码”,这涉及到Spring MVC如何处理文件上传请求,以及如何实现通用的Excel导入功能。我们将从Java微服务的视角出发,讨论相关依赖包和关键组件。 1. **文件上传**: - ...
对于Blob类型的扩展,可能包括了更便捷的读写方法,如直接从文件系统上传或下载Blob对象,或者提供了流式处理,以处理大量二进制数据,而不会消耗大量内存。对于Clob类型,可能会有新的文本处理功能,如格式化、搜索...
在Java编程语言中,大批量上传和下载是常见的任务,特别是在处理大量数据或者文件交互的场景下。这个压缩包文件包含了一些关键的类,可以帮助我们理解如何在Java中实现这样的功能。下面将逐一解析这些类的功能及其在...
在Java编程中,经常需要处理文件的上传与下载。本文档将详细介绍如何使用Java实现基于HTTP协议的本地文件下载功能。主要聚焦于通过文件名及文件的全路径来实现本地文件的下载。 #### 二、关键概念与技术点 ##### ...
可以看到一些关键文件,如`menu.js`用于管理后台的菜单导航,`main.js`是应用的入口文件,`Editor.vue`可能是用于富文本编辑的组件,`BreadCrumbs.vue`则是面包屑导航组件,而`MPUtil.java`可能是包含一些通用工具...
总的来说,"文件上传ActiveX控件"是一个利用Delphi开发的、能够实现多文件上传到Web服务器的组件。虽然在当前的Web开发趋势中,ActiveX控件的应用越来越少,但了解其工作原理和历史对于理解Web技术的发展历程仍然是...
总的来说,这个项目为初学者提供了学习Java网络编程和实现简单即时通讯应用的宝贵实践机会,同时也对有经验的开发者来说,是一个理解局域网通信机制和提升编程技能的好例子。通过阅读和理解这些源代码,你可以深入...