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

news2024/11/29 0:39:07

一、线程数据共享和安全 -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 static
    public static ThreadLocal<Object> threadLocal1=new ThreadLocal<>();
    public static ThreadLocal<Object> threadLocal2=new ThreadLocal<>();


    //Task是线程类-内部类/线程
    public static class Task implements Runnable{
        @Override
        public void run() {
            Dog dog=new Dog();
            //给threadLocal1放入一只狗 set dog
            threadLocal1.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: Linran
  Date: 2024/4/23
  Time: 22:16
  Version: 1.0
  To 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 设置图片 url
                img.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/11
                        File 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: Linran
  Date: 2024/4/24
  Time: 10:35
  Version: 1.0
  To 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.coloradmin.cn/o/1802342.html

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

相关文章

量化交易:Miniqmt获取可转债数据和交易python代码

哈喽&#xff0c;大家好&#xff0c;我是木头左&#xff01; 低风险资产除了国债外&#xff0c;还有可转债&#xff0c;兼容有高收益的股性和低风险的债性&#xff0c;号称“下有保底&#xff0c;上不封顶”。 &#x1f50d; 可转债&#xff1a;金融市场的双面娇娃 可转债&am…

web学习笔记(六十五)

目录 1. Hash模式和History模式 2. 导航守卫 3. 路由元信息 4.路由懒加载 1. Hash模式和History模式 Hash模式&#xff08;哈希模式&#xff09;和History模式&#xff08;历史模式&#xff09;是匹配路由的两种模式&#xff0c;一般默认配置Hash模式&#xff0c;可以在in…

软件架构x86 、 x86_64、 arm64、aarch64

看系统信息: 大多数Linux发行版都提供如 uname -a命令 arch命令用于显示当前主机的硬件架构类型。 例如 下面的是Kylin Linux Advanced Server for Kunpeng V10 操作系统 (鲲鹏处理器是华为在2019年1月向业界发布的高性能数据中心处理器 ) 下面这个是 ubuntu 18.04.6 …

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

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

四、利用启发式算法进行特定数据集的残差网络结构搜索【框架+源码】

背景&#xff1a;工作之后干的事情跟算法关联甚少&#xff0c;整理下读书期间的负责和参与的work&#xff0c;再熟悉学习下。 边熟悉边整理喽~ CV Tradictional workCV AI based work机械臂视觉抓取项目机器学习全流程 Pipeline训练平台OCR生产线喷码识别三维重建(SfM)ROS机器人…

SpringBoot Elasticsearch06-以黑马商场为例-黑马程序员学习笔记

黑马商城作为一个电商项目&#xff0c;商品的搜索肯定是访问频率最高的页面之一。目前搜索功能是基于数据库的模糊搜索来实现的&#xff0c;存在很多问题。 首先&#xff0c;查询效率较低。 由于数据库模糊查询不走索引&#xff0c;在数据量较大的时候&#xff0c;查询性能很…

学习笔记——路由网络基础——汇总静态路由

4、汇总静态路由 (1)定义 静态路由汇总&#xff1a;多条静态路由都使用相同的送出接口或下一跳 IP 地址。(将多条路由汇总成一条路由表示) (2)目的 1.减少路由条目数量&#xff0c;减小路由表&#xff0c;加快查表速度 2.增加网络稳定性 (3)路由黑洞以及路由环路的产生…

旧衣回收小程序开发,轻松回收旧衣物

随着环保理念的增强&#xff0c;回收市场得到了快速发展&#xff0c;吸引了不少年轻人进入到市场中创业。除了传统的废品回收外&#xff0c;旧衣回收也受到了大众的重视&#xff0c;市场规模迅速扩大&#xff0c;大众浪费的衣物也获得了归处。 目前旧衣回收的方式主要是线上与…

如何将HTTP升级成HTTPS?既简单又免费的方法!

在当今数字化时代&#xff0c;网络安全已成为用户和企业关注的焦点。HTTPS作为一种更加安全的网络通信协议&#xff0c;正逐渐取代传统的HTTP成为新的标准。对于许多网站管理员和内容创作者来说&#xff0c;如何免费升级到HTTPS是一个值得探讨的问题。本文将详细介绍一些免费的…

Flash均衡算法几个点

Flash均衡保存算法是一种用于调整更改数据时擦除Flash存储器区域大小的高效算法。 在Flash存储器中&#xff0c;写入新数据时需要先将原有数据所在的块进行擦除&#xff0c;然后再进行写入操作。由于Flash存储器的特性&#xff0c;擦除操作比写入操作要慢得多&#xff0c;而且…

Python私教张大鹏 Vue3整合AntDesignVue之Anchor 锚点

用于跳转到页面指定位置。 何时使用 需要展现当前页面上可供跳转的锚点链接&#xff0c;以及快速在锚点之间跳转。 案例&#xff1a;锚点的基本使用 核心代码&#xff1a; <template><a-anchor:items"[{key: part-1,href: #part-1,title: () > h(span, {…

使用Python操作Redis

大家好&#xff0c;在当今的互联网时代&#xff0c;随着数据量和用户量的爆发式增长&#xff0c;对于数据存储和处理的需求也日益增加。Redis作为一种高性能的键值存储数据库&#xff0c;以其快速的读写速度、丰富的数据结构支持和灵活的应用场景而备受青睐。本文将介绍Redis数…

elasticsearch安装与使用(4)-搜索入门

1、创建索引 PUT /hotel {"mappings": {"properties":{"title":{"type": "text"},"city":{"type": "keyword"},"price":{"type":"double"}}} }2、写入文档 …

大语言模型的sft

https://zhuanlan.zhihu.com/p/692892489https://zhuanlan.zhihu.com/p/6928924891.常见的sft的开发流程 a.根据业务场景调整提示词;越详细越好,不要让模型理解歧义,拆分。 b.尝试闭源和开源,以评估LLM能够解决这类场景问题。 c.准备数据,包括多个子任务。 d.训练上线…

产气荚膜梭菌定植与婴儿食物过敏之间的关联

谷禾健康 牛奶蛋白过敏&#xff08;CMPA&#xff09;是婴儿最常见的食物过敏类型之一。粪便病原菌培养显示产气荚膜梭菌阳性率超过30%&#xff0c;明显高于其他细菌。因此推测产气荚膜梭菌定植可能是婴儿牛奶蛋白过敏的发病因素之一。 一项真实世界的研究&#xff0c;杨敏团队从…

2024年跨平台应用解决方法

个人博客:Sekyoro的博客小屋 个人网站:Proanimer的个人网站 很久没有写这类high-level的文章了,本身这类框架就一直层出不穷,但是其中历久弥坚,坚韧不拔的框架又有多少呢? 首先考虑到学习成本以及掌握一些编程语言在工作、学习生态上的价值,给这些东西适用生态划分一下. Reac…

能放在桌面上的倒数提醒软件 桌面提醒软件倒数日

在忙碌的生活中&#xff0c;我们总有许多待办事项和重要的纪念日需要牢记。可是&#xff0c;人的记忆毕竟有限&#xff0c;有时候一不小心就会错过那些重要的时刻。 桌面提醒软件是我日常生活中的得力助手。它小巧轻便&#xff0c;静静地待在我电脑的桌面上&#xff0c;每次打…

【漏洞复现】多客圈子论坛系统 httpGet 任意文件读取漏洞

0x01 产品简介 多客圈子论坛系统是一种面向特定人群或特定话题的社交网络&#xff0c;它提供了用户之间交流、分享、讨论的平台。在这个系统中&#xff0c;用户可以创建、加入不同的圈子&#xff0c;圈子可以是基于兴趣、地域、职业等不同主题的。用户可以在圈子中发帖、评论、…

2024教资认定报名流程,点赞收藏!

2024年要进行教资认定的宝子们提早准备 &#x1f525;教资认定网上报名流程概览 一、进入教资认定网报入口 二、进行实名核验 三、申请网报时间查询 四、个人信息维护 五、认定申请报名 &#x1f525;教资认定所需材料 1⃣️身份证 2⃣️户口本&#xff0f;居住证&#xff0f;学…

17_Vue高级监听器生命周期Vue组件组件通信

文章目录 1. 数据监听器watch2. Vue生命周期3. Vue组件4. Vue组件通信Appendix 1. 数据监听器watch 首先watch需要单独引 import {watch} from vuewatch函数监听ref响应式数据 watch(监听的内容&#xff0c;监听行为)监听行为默认为(newValue,oldValue) let firstname ref…