javaweb学习(day14-ThreadLocal文件上传下载)

一、线程数据共享和安全 -ThreadLocal

1 什么是 ThreadLocal

  • ThreadLocal 的作用,可以实现在同一个线程数据共享, 从而解决多线程数据安全问题.
  • ThreadLocal 可以给当前线程关联一个数据(普通变量、对象、数组)set 方法 [源码!]
  • ThreadLocal 可以像 Map 一样存取数据,key 为当前线程, get 方法
  • 每一个 ThreadLocal 对象,只能为当前线程关联一个数据,如果要为当前线程关联多个数据,就需要使用多个 ThreadLocal 对象实例
  • 每个 ThreadLocal 对象实例定义的时候,一般为 static 类型
  • ThreadLocal 中保存数据,在线程销毁后,会自动释放

2 快速入门 ThreadLocal 

演示 ThreadLocal (作用:在一个线程中, 共享数据(线程安全))的使用 

创建Dog,Pig类

创建T2DAO类

package com.hspedu.threadlocal;public class T2DAO {public void update(){Object o = T1.threadLocal1.get();String name =Thread.currentThread().getName();//获取当前线程名System.out.println("在TDAO的update()线程是="+name+" 取出的dog= "+o);}
}

 创建T1Service类

package com.hspedu.threadlocal;public class T1Service {public void update(){//取出ThreadLocal对象关联的对象Object o = T1.threadLocal1.get();String name=Thread.currentThread().getName();System.out.println("在T1Service的update()线程name= "+name+" dog= "+o);//调用了dao-update方法new T2DAO().update();}
}
ThreadLocalTest
package com.hspedu.threadlocal;public class T1 {//创建ThreadLocal对象,做成一个public staticpublic static ThreadLocal<Object> threadLocal1=new ThreadLocal<>();public static ThreadLocal<Object> threadLocal2=new ThreadLocal<>();//Task是线程类-内部类/线程public static class Task implements Runnable{@Overridepublic void run() {Dog dog=new Dog();//给threadLocal1放入一只狗 set dogthreadLocal1.set(dog);//用同一个threadlocal会替换哦System.out.println("Task放入了 dog= "+dog);Pig pig=new Pig();threadLocal2.set(pig);System.out.println("在run 方法中 线程="+Thread.currentThread().getName());new T1Service().update();}}public static void main(String[] args) {new Thread(new Task()).start();}
}

 二、文件上传

1 基本介绍

  • 文件的上传和下载,是常见的功能。
  • 后面项目就使用了文件上传下载。
  • 如果是传输大文件,一般用专门工具或者插件
  • 文件上传下载需要使用到两个包 , 需要导入
  • 说明:

2 文件上传示意图

 

 3 应用实例

load.jsp

<%--User: LinranDate: 2024/4/23Time: 22:16Version: 1.0To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>Title</title>
<%--    指定了base标签 web工程/--%><base href="<%=request.getContextPath()+"/"%>>"><style type="text/css">input[type="submit"] {outline: none;border-radius: 5px;cursor: pointer;background-color: #31B0D5;border: none;width: 70px;height: 35px;font-size: 20px;}img {border-radius: 50%;}form {position: relative;width: 200px;height: 200px;}input[type="file"] {position: absolute;left: 0;top: 0;height: 200px;opacity: 0;cursor: pointer;}</style><script type="text/javascript">function prev(event) {
//获取展示图片的区域var img = document.getElementById("prevView");
//获取文件对象let file = event.files[0];
//获取文件阅读器let reader = new FileReader();reader.readAsDataURL(file);reader.onload = function () {
//给 img 的 src 设置图片 urlimg.setAttribute("src", this.result);}}</script>
</head>
<body>
<!-- 表单的 enctype 属性要设置为 multipart/form-data -->
<form action="fileUploadServlet" method="post" enctype="multipart/form-data">家居图: <img src="2.jpg" alt="" width="200" height="200" id="prevView"><input type="file" name="pic" id="" value="2xxx.jpg" onchange="prev(this)"/>家居名: <input type="text" name="name"><br/><input type="submit" value="上传"/>
</form>
</body>
</html>
FileUploadServlet
package com.hspedu.servlet;import com.hspedu.utils.WebUtils;
import org.apache.commons.fileupload.FileItem;
import org.apache.commons.fileupload.FileUploadException;
import org.apache.commons.fileupload.disk.DiskFileItemFactory;
import org.apache.commons.fileupload.servlet.ServletFileUpload;import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.File;
import java.io.IOException;
import java.util.List;
import java.util.UUID;public class FileUploadServlet extends HttpServlet {protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {System.out.println("FileUploadServlet被调用");//1.判断是不是文件表单(enctype=multipart/form-data)if(ServletFileUpload.isMultipartContent(request)){System.out.println("ok");//创建DiskFileItemFactory对象,用于构建一个解析上传数据的工具对象DiskFileItemFactory diskFileItemFactory = new DiskFileItemFactory();//创建一个解析上传数据的对象ServletFileUpload servletFileUpload=new ServletFileUpload(diskFileItemFactory);//解决接收到的文件名是中文乱码的问题servletFileUpload.setHeaderEncoding("utf-8");//关键地方,servletFileUpload对象可以把表单提交的数据text/文件//将其封装到<FileItem>文件项中try {List<FileItem> list = servletFileUpload.parseRequest(request);System.out.println("list==>"+list);//遍历,并分别处理for(FileItem fileItem:list){//System.out.println("fileItem= "+fileItem);//判断是不是一个文件if(fileItem.isFormField()){//这是一个文本控件//如果为真就是文本
//                        String name = fileItem.getName();
//                        System.out.println("name= "+name);String name = fileItem.getString("utf-8");System.out.println("家具名= "+name);}else {//说明是一个文件//获取文件的名字String name = fileItem.getName();System.out.println("上传的文件名= "+name);//上传到服务器的temp下的文件保存到指定的目录//1.指定一个目录,就是我们网站工作目录下String filePath="/upload/";//2。获取完整目录,这个目录是和项目运行的环境绑定的String realPath = request.getServletContext().getRealPath(filePath);System.out.println("fileRealPath= "+realPath);//3 创建上传的目录//写一个工具类,可以返回/日期/2024/11/11File file=new File(realPath+ WebUtils.getYearMonthDay());if(!file.exists()){//不存在就创建file.mkdirs();//创建}//4 将文件拷贝到realpath//构建一个完整的路径名+文件名//为了防止重名替换,我们对上传的文件名进行处理,在前面增加前缀,保证是唯一即可name=UUID.randomUUID().toString()+"_"+name;String final_name=file+name;fileItem.write(new File(final_name));//5 提示信息response.setContentType("text/html;charset=utf-8");response.getWriter().write("上传成功");}}} catch (Exception e) {e.printStackTrace();}}else {System.out.println("不是文件表单");}}protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {doPost(request,response);}
}
WebUtils
package com.hspedu.utils;import java.time.LocalDateTime;public class WebUtils {public static String getYearMonthDay(){//如何得到当前的日期LocalDateTime ldt=LocalDateTime.now();int year=ldt.getYear();int month=ldt.getMonthValue();int dayofMonth=ldt.getDayOfMonth();String YearMonthDay=year+"/"+month+"/"+dayofMonth+"/";return YearMonthDay;}
}

文件上传注意事项和细节

  • 一个完美的文件上传,要考虑的因素很多,比如断点续传、控制图片大小,尺寸,分片上传,防止恶意上传等,在项目中,可以考虑使用 WebUploader 组件(百度开发) http://fex.baidu.com/webuploader/doc/index.html
  • 文件上传功能,在项目中建议有限制的使用,一般用在头像、证明、合同、产品展示等,如果不加限制,会造成服务器空间被大量占用 [比如 b 站评论,就不能传图片,微信发 1 次朋友圈最多 9 张图等..]
  • 如果将文件都上传到一个目录下,当上传文件很多时,会造成访问文件速度变慢,因此可以将文件上传到不同目录 比如 一天上传的文件,统一放到一个文件夹 年月日, 比如21001010 文件夹
  • 文件上传,创建 web/upload 的文件夹,在 tomcat 启动时,没有在 out 目录下 创建 对应的 upload 文件夹, 原因是 tomcat 对应空目录是不会在 out 下创建相应目录的,所以,只需在 upload 目录下,放一个文件即可, 这个是 Idea + Tomcat 的问题, 实际开发不会存.

三、文件下载 

1 文件下载的原理分析图 

应用实例 

<%--User: LinranDate: 2024/4/24Time: 10:35Version: 1.0To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>文件下载</title><base href="<%=request.getContextPath()+"/"%>>">
</head>
<body>
<h1>文件下载</h1>
<a href="fileDownloadServlet?name=1.png">点击下载小狗图片</a><br/><br/>
<a href="fileDownloadServlet?name=韩顺平零基础java笔记.pdf">点击下载 韩顺平零基础 Java 笔记.pdf</a><br/><br/>
</body>
</html>
package com.hspedu.servlet;import org.apache.commons.io.IOUtils;
import sun.misc.BASE64Encoder;import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.InputStream;
import java.net.URLEncoder;public class FileDownloadServlet extends HttpServlet {protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {System.out.println("FileDownloadServlet被调用");//1. 先准备要下载的文件[假定这些文件是公共的资源]//重要:当我们的tomcat启动后在我们的工作目录下有我们需要的文件夹request.setCharacterEncoding("utf-8");//2.获取到要下载的文件的名字String downloadFileName = request.getParameter("name");//System.out.println("downloadFileName= "+downloadFileName);//3 给http响应,设置响应头Content-Type就是文件的MiME//通过servletContext来获取ServletContext servletContext=request.getServletContext();String downloadPath="/download/";String downloadFullPath=downloadPath+downloadFileName;String mimeType = servletContext.getMimeType(downloadFullPath);//System.out.println(mimeType);response.setContentType(mimeType);//4 给http响应,设置响应头Content-Disposition//这里考虑的细节比较多,比如不同的浏览器写法不一样,考虑编码//ff是文件名中文需要base64,而ie/chrome是URL编码//这里我们不需要同学背下来,只需要知道原理//(1)如果是Firefox 则中文编码需要 base64//(2)Content-Disposition 是指定下载的数据的展示形式 , 如果attachment 则使用文件下载方式//(3)如果是其他(主流ie/chrome) 中文编码使用URL编码if (request.getHeader("User-Agent").contains("Firefox")) {// 火狐 Base64编码response.setHeader("Content-Disposition", "attachment; filename==?UTF-8?B?" +new BASE64Encoder().encode(downloadFileName.getBytes("UTF-8")) + "?=");} else {// 其他(主流ie/chrome)使用URL编码操作response.setHeader("Content-Disposition", "attachment; filename=" +URLEncoder.encode(downloadFileName, "UTF-8"));}//读取下载的文件数据,返回给客户端/浏览器//(1)创建一个和要下载的文件,关联的输入流InputStream resourceAsStream=servletContext.getResourceAsStream(downloadFullPath);//(2)得到返回数据的输出流ServletOutputStream outputStream = response.getOutputStream();//(3)返回数据,使用工具类将输入流关联的文件拷贝到输出流并返回给客户端IOUtils.copy(resourceAsStream,outputStream);}protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {doPost(request,response);}
}

注意事项和细节

文件下载,比较麻烦的就是文件名中文处理

因此,在代码中,针对不同浏览器做了处理

对于网站的文件,很多文件使用另存为即可下载,对于大文件 ( 文档,视频 ) ,会使用专 业的下载工具 ( 迅雷、百度,腾讯,华为网盘等 )
对于不同的浏览器 , 在把文件下载完毕后,处理的方式不一样 , 有些是直接打开文件 , 有些是将文件下载到 本地 / 下载目录

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/850013.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Vue的基础知识:v-model的原理,由:value与@input合写。

原理&#xff1a;v-model本质上是一个语法糖&#xff0c;比如应用在输入框上&#xff0c;就是value属性和input事件的合写。&#xff08;补充说明&#xff1a;语法糖就是语法的简写&#xff09; 作用&#xff1a;提供数据的双向绑定 1.数据变&#xff0c;视图&#xff08;也就…

[数据集][目标检测]叶子计数检测数据集VOC+YOLO格式240张1类别

数据集格式&#xff1a;Pascal VOC格式YOLO格式(不包含分割路径的txt文件&#xff0c;仅仅包含jpg图片以及对应的VOC格式xml文件和yolo格式txt文件) 图片数量(jpg文件个数)&#xff1a;240 标注数量(xml文件个数)&#xff1a;240 标注数量(txt文件个数)&#xff1a;240 标注类别…

2024年谷歌SEO如何快速出排名效果抢占首页制高点?

2024年&#xff0c;随着谷歌搜索引擎算法的不断更新&#xff0c;SEO策略也需要与时俱进才能快速出排名。本文将结合谷歌最新SEO趋势&#xff0c;平哥SEO分享一些实操性的快速排名技巧&#xff0c;帮助你在竞争激烈的搜索结果中脱颖而出。 额外话题&#xff1a;就是通过微信公众…

Java Web学习笔记25——Vue组件库Element

什么是Element&#xff1f; Element: 是饿了么团队研发的&#xff0c;一套为开发者、设计师和产品经理准备的基于Vue2.0的桌面端组件库。 组件&#xff1a;组成网页的部件&#xff0c;例如&#xff1a;超链接、按钮、图片、表格、表单、分页条等等。 官网&#xff1a;https:…

深拷贝、浅拷贝、引用拷贝

深拷贝和浅拷贝的区别 1. 引用拷贝2. 对象拷贝 1. 引用拷贝 两个对象指向同一个地址值。 创建一个指向对象的引用变量的拷贝Teacher teacher new Teacher("Taylor",26); Teacher otherteacher teacher; System.out.println(teacher); System.out.println(otherte…

前端多人项目开发中,如何保证CSS样式不冲突?

在前端项目开发中&#xff0c;例如突然来了一个大项目&#xff0c;很可能就需要多人一起开发&#xff0c;领导说了&#xff0c;要快&#xff0c;要快&#xff0c;要快&#xff0c;你们给我快。然后下面大伙就一拥而上&#xff0c;干着干着发现&#xff0c;一更新代码&#xff0…

【AI论文与新生技术】Follow-Your-Emoji:精细可控且富有表现力的自由式人像动画技术

我们提出了 Follow-Your-Emoji&#xff0c;这是一种基于扩散的肖像动画框架&#xff0c;它使用目标地标序列对参考肖像进行动画处理。肖像动画的主要挑战是保留参考肖像的身份并将目标表情转移到该肖像&#xff0c;同时保持时间一致性和保真度。为了应对这些挑战&#xff0c;Fo…

JFinal学习07 控制器——接收数据之getBean()和getModel()

JFinal学习07 控制器——接收数据之getBean()和getModel() 视频来源https://www.bilibili.com/video/BV1Bt411H7J9/?spm_id_from333.337.search-card.all.click 文章目录 JFinal学习07 控制器——接收数据之getBean()和getModel()一、接收数据的类型二、getBean()和getModel()…

GDPU JavaWeb Ajax请求

异步请求可以提升用户体验并优化页面性能。 ajax登录 实现ajax异步登录。 注意&#xff0c;ajax用到了jQuery库&#xff0c;先下载好相应的js库&#xff0c;然后复制导入到工程的web目录下&#xff0c;最好与你的前端页面同一层级。然后编写时路径一定要找准&#xff0c;“pag…

转让北京公司带旅行许可证流程和要求

旅行社是开展旅游服务业务的专项经济主体&#xff0c;旅行社开展相关业务必须持有旅行社业务经营许可证。该资质又分为国内旅行社经营许可证和出境旅行社经营许可证。主要区别在于能否开展出境旅游业务&#xff0c;下面老耿带大家了解&#xff0c;新成立国内旅行社要求及出境旅…

python-windows10普通笔记本跑bert mrpc数据样例0.1.001

python-windows10普通笔记本跑bert mrpc数据样例0.1.000 背景参考章节获取数据下载bert模型下载bert代码windows10的cpu执行结果注意事项TODOLIST背景 看了介绍说可以在gpu或者tpu上去微调,当前没环境,所以先在windows10上跑一跑,看是否能顺利进行,目标就是训练的过程中没…

js--hasOwnProperty()讲解与使用

@TOC 前言 hasOwnProperty(propertyName)方法 是用来检测属性是否为对象的自有属性 object.hasOwnProperty(propertyName) // true/false 讲解 hasOwnProperty() 方法是 Object 的原型方法(也称实例方法),它定义在 Object.prototype 对象之上,所有 Object 的实例对象都会继…

6.7 输入输出流

输入&#xff1a;将数据放到程序&#xff08;内存&#xff09;中 输出&#xff1a;将数据从程序&#xff08;内存&#xff09;放到设备中 C的输入输出分为3种形式&#xff1a; 从键盘屏幕中输入输出&#xff0c;称为标准IO 对于磁盘进行标准输入输出&#xff0c;称为文件IO…

第5章 if语句

第5章 if语句 5.1 示例5.2 条件测试5.2.1 检查是否相等5.2.2 检查是否相等时忽略大小写5.2.3 检查是否不相等5.2.4 数值比较5.2.5 检查多个条件5.2.6 检查特定值是否包含在列表中5.2.7 检查特定值是否不包含在列表中5.2.8 布尔表达式 5.3 if 语句5.3.1 简单的if 语句5.3.2 if-e…

硕思闪客精灵(shankejingling)软件最新版下载及详细安装教程

闪客精灵&#xff08;Sothink SWF Decompiler&#xff09;是一款先进的SWF反编译软件&#xff0c;它不但能捕捉、反编译、查看和提取Shock Wave Flash影片&#xff08;.swf和.exe格式文件&#xff09;&#xff0c;而且可以将SWF格式文件转化为FLA格式文件。它能反编译Flash的所…

四天工作制,比你想象的更近了一点

原文&#xff1a;Andrew Keshner - 2024.05.30 软件公司、大型企业甚至警察部门都在试验这一看似遥不可及的概念。 教育软件公司 Kuali 的会议精简&#xff0c;除非绝对必要&#xff0c;员工尽量避免安排会议。即使有会议&#xff0c;也鼓励员工跳过与自己工作无关的部分。在…

从技术到产品:以客户为中心的产品研发之路

一、引言 在快速发展的商业环境中&#xff0c;产品作为连接企业与市场的桥梁&#xff0c;其重要性不言而喻。从摸着石头过河搞产品&#xff0c;到广泛传播NPDP&#xff08;新产品开发流程&#xff09;理念&#xff0c;产品研发的道路经历了从直觉驱动到系统思维的转变。本文将…

ArcGIS for js 4.x 加载图层

二维&#xff1a; 1、创建vue项目 npm create vitelatest 2、安装ArcGIS JS API依赖包 npm install arcgis/core 3、引入ArcGIS API for JavaScript模块 <script setup> import "arcgis/core/assets/esri/themes/light/main.css"; import Map from arcgis…

哈希桶封装unordered_map、unordered_set

哈希桶源代码 我们将由下列的哈希桶来模拟封装STL库中的unordered_map和unordered_set 注意&#xff1a;为了实现封装unordered_map和unordered_set&#xff0c;我们需要对下列源码进行优化。 //哈希桶 namespace hashbucket {template<class K,class V>struct HashNo…

高考作文:时光之河,逐梦前行

时光之河&#xff0c;奔流不息&#xff0c;如同我们的人生旅途&#xff0c;充满了未知与挑战。站在2024年的高考门槛前&#xff0c;我们回望过去&#xff0c;展望未来&#xff0c;心中充满了期待与憧憬。 首先&#xff0c;让我们回顾一下这条时光之河中的点滴。过去的岁月里&am…