图片服务器项目
项目源码
项目源码地址:https://gitee.com/StoneLib/picture-server
项目简介
该项目基于 web 下的 servlet 开发,具有简单的图片的上传、图片的查看和删除的功能,用户可以在这里上传自己的照片,进行存储与观看
项目描述
- 使用 JavaScript 编辑前端页面进行 Ajax 请求,简单使用 CSS 美化界面;
- 使用 Servlet、JSON 完成前后端的交互和分离;
- 简单的 Web 服务器设计开发能力(Servlet)
- 使用数据库(Mysql)JDBC 操作 MySQL
- 数据库设计(根据实际场景设计数据库表结构)
- 前后端交互的 API 的设计(基于HTTP协议)
- 学习测试 HTTP 服务器,Postman
- 使用 HTML、CSS、JavaScript技术构造一个简单的网页
服务器设计
服务器设计
- MySQL 本质上也是一个服务器程序
- 数据库中存储的图片的属性(元信息);图片正文,以文件的形式直接存在磁盘上;数据库中就记录一个 path 对应到磁盘上的文件
- 校验和:通过一个更短的字符串,来验证整体数据是否正确,短的字符串是根据原串内容通过一定的规则来计算出来的。
- md5:图片的md5校验和 —— 字符串哈希算法
服务器 API 设计(前后端交互接口设计)
JSON:一种数据组织的格式,格式为键值对的结构。此处有使用 JSON 完成数据的序列化,方便进行网络传输
// JSON 只是一个数据格式,和编程语言无关(JavaScript)
正式开始设计前后端交互 API:
- 新增图片
请求:
POST/image
Content-Type:multipart/form-data
//正文内容 包含图片自身的一些信息图片正文的二进制内容响应:
HTTP /1.1 200 OK
{
// 上传成功
“ok":true
// 上传失败
"ok":false
"reason":"具体失败原因"
}
- 查看所有图片的属性
请求:GET/image
响应:
HTTP /1.1 200 OK
[ // 响应成功
{
imageId: 1,
imageName: "a.png",
size: 10,
uploadTime: "20220815",
path: "./image/a.png",
md5:"dq2ew4"
},
{ ........ }
]
响应失败:
HTTP /1.1 200 OK
[// 上传成功
“ok":true
// 上传失败
"ok":false
"reason":"具体失败原因"
]
1.API 具体的设定有很多方式,可以用200表示成功,404表示失败;
2.也可以使用 body 中的 OK 字段 ,true 表示成功,false 表示失败;
3.还可以使用 [] 有内容表示成功,为空表示失败。
- 查看指定图片属性
请求:
GET / image?imageId = [具体的数值]
响应:
HTTP/1.1 200 OK
{ // 响应成功imageId: 1,
imageName: "a.png",
size: 10,
uploadTime: "20220815",
path: "./image/a.png",
md5:"dq2ew4"
}
HTTP/1.1 200 OK
{ // 响应失败
"ok": false,
"reason":"具体出错的原因"
}
- 删除指定图片属性
请求:
DELETE/image?imageId = [具体图片的ID]
响应:
HTTP/1.1 200 OK
{ // 响应成功
"ok": true,
}
HTTP/1.1 200 OK
{ // 响应失败
"ok": false,
"reason":"具体出错的原因"
}
服务器实现代码的时候就可以判定方法,如果是 DELETE 方法,就执行删除操作
删除也不一定非得用 DELETE 方法,
例如:GET / image?imageId=xxx&delete=1
- 查看指定图片的内容
请求:
GET / imageShow?imageId = [具体图片的id]
响应:
HTTP/1.1 200 OK
// 响应成功
Content-Type: image/png [图片的二进制内容]
HTTP/1.1 200 OK
{ // 响应失败
ok: false,
reason:"具体出错的原因"
}
数据库操作
1.先创建 DBUtil 封装一下获取数据库连接的过程, dao 数据访问层:这里的类围绕着数据操作展开
package dao;import java.sql.Connection;
import com.mysql.jdbc.jdbc2.optional.MysqlDataSource;import javax.sql.DataSource;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;public class DBUtil {//3306后面跟的是数据库的名字private static final String URL = "jdbc:mysql://127.0.0.1:3306/数据库?characterEncoding=utf8&useSSL=false";private static final String USERNAME = "root";private static final String PASSWORD = "数据库密码";private static DataSource dataSource = null;public static DataSource getDataSource(){// 通过这个方法来创建DataSource的实例// 保证线程安全:// 1.先加锁 2.二次判断 3.使用volatile关键字if(dataSource == null){synchronized (DBUtil.class){if(dataSource == null){dataSource = new MysqlDataSource();MysqlDataSource tempDataSource = (MysqlDataSource) dataSource;tempDataSource.setURL(URL);tempDataSource.setUser(USERNAME);tempDataSource.setPassword(PASSWORD);}}}return dataSource;}public static Connection getConnection(){try {return getDataSource().getConnection();} catch (SQLException e) {e.printStackTrace();}return null;}public static void close(Connection connection, PreparedStatement statement, ResultSet resultSet){//注意关闭顺序,先连接的后关闭try {if(resultSet != null){resultSet.close();}if(statement != null){statement.close();}if(connection != null){connection.close();}} catch (SQLException e) {e.printStackTrace();}}
}
2.建立数据库中的表
create table image_table (imageId int not null primary key auto_increment,
imageName varchar(50),
size int,
uploadTime varchar(50),
contentType varchar(50),
path varchar(1024),
md5 varchar(1024));
代码实现
封装数据库操作(DAO层)
- DBUtil 封装获取数据库连接的操作,上述已经给出
- Image 对应到一个图片对象(包括图片的相关属性)
public class Image {private int imageId;private String imageName;private int size;private String uploadTime;private String contentType;private String path;private String md5;// 再利用 IDEA 快捷键 alt+enter+insert 对上述属性进行 set 和 get 方法的创建
}
- ImageDao 是 Image 对象的管理器,借助这个类完成 Image 对象的增删查
- 增加图片
public void insert(Image image) {// 1.获取数据库连接Connection connection = DBUtil.getConnection();// 2.创建片拼接SQL语句PreparedStatement statement = null;try {String sql = "insert into image_table values(null, ?, ?, ?, ?, ?, ?)";statement = connection.prepareStatement(sql);statement.setString(1,image.getImageName());statement.setInt(2,image.getSize());statement.setString(3,image.getUploadTime());statement.setString(4,image.getContentType());statement.setString(5,image.getPath());statement.setString(6,image.getMd5());// 3.执行SQL语句int ret = statement.executeUpdate();if(ret != 1){//程序出现问题,抛出一个异常throw new ImageServerException("插入数据库错误");}} catch (SQLException | ImageServerException e) {e.printStackTrace();} finally {// 4.关闭连接和statement对象DBUtil.close(connection,statement,null);}}
- 查看图片
// 查看所有图片
public List<Image> selectAll() {List<Image> images = new ArrayList<>();// 1.获取数据库连接Connection connection = DBUtil.getConnection();// 2.构造 SQL 语句String sql = "select * from image_table";PreparedStatement statement = null;ResultSet resultSet = null;// 3.执行 SQL 语句try {statement = connection.prepareStatement(sql);resultSet = statement.executeQuery();// 4.处理结果集while (resultSet.next()) {Image image = new Image();image.setImageId(resultSet.getInt("imageId"));image.setImageName(resultSet.getString("imageName"));image.setSize(resultSet.getInt("size"));image.setUploadTime(resultSet.getString("uploadTime"));image.setContentType(resultSet.getString("contentType"));image.setPath(resultSet.getString("path"));image.setMd5(resultSet.getString("md5"));images.add(image);}return images;} catch (SQLException e) {e.printStackTrace();} finally {// 5.关闭连接和 statement 对象DBUtil.close(connection,statement,resultSet);}return null;}// 根据 imageId 查看指定图片public Image selectOne(int imageId) {// 1.获取数据库连接Connection connection = DBUtil.getConnection();// 2.构造 SQL 语句String sql = "select * from image_table where imageId = ?";PreparedStatement statement = null;ResultSet resultSet = null;// 3.执行 SQL 语句try {statement = connection.prepareStatement(sql);statement.setInt(1,imageId);resultSet = statement.executeQuery();// 4.处理结果集if (resultSet.next()) {Image image = new Image();image.setImageId(resultSet.getInt("imageId"));image.setImageName(resultSet.getString("imageName"));image.setSize(resultSet.getInt("size"));image.setUploadTime(resultSet.getString("uploadTime"));image.setContentType(resultSet.getString("contentType"));image.setPath(resultSet.getString("path"));image.setMd5(resultSet.getString("md5"));return image;}} catch (SQLException e) {e.printStackTrace();} finally {// 5.关闭连接和 statement 对象DBUtil.close(connection,statement,resultSet);}return null;}
- 删除图片
public void delete(int imageId) {//1.获取数据库连接Connection connection = DBUtil.getConnection();// 2.创建并片接sql语句// ?作为占位符String sql = "delete from image_table where imageId = ?";// 3.执行sql语句PreparedStatement statement = null;try {statement = connection.prepareStatement(sql);statement.setInt(1,imageId);int ret = statement.executeUpdate();if(ret != 1){throw new ImageServerException("删除数据库操作失败");}} catch (SQLException | ImageServerException e) {e.printStackTrace();}finally {// 4.关闭连接DBUtil.close(connection,statement,null);}}
- 根据 MD5 进行查看,此方法和 根据指定 imageId 查看的方法油异曲同工之处
ImageDao 有一个 selectAll 方法,查出所有数据库中的数据。如果数据库中内容只有几千条,这样查找可以;但若是数据库有几亿条数据,这样查就会非常低效,很可能直接把数据库或者应用程序运行崩溃。
更科学的方法是指定一些其他筛选的条件(分页)
单元测试- 把一个类/方法看作是一个单元进行测试
- 一旦出现问题,就能及时发现,BUG 发现地越早,解决成本就越低
- 软件开发的核心任务 —— 管理软件的复杂程度(不要让软件变得更复杂)
- 封装是一种有效的管理复杂程度的手段,让类的使用者不需要关注类具体实现细节(还是需要关注这个对象是什么类型)
- 多态是封装的更进一步,使用者不仅不需要知道类的具体实现功能,也不需要关注这个对象的类型是什么
基于 Servlet 搭建服务器
- 安装 Servlet,根据 maven 进行安装
- 创建一个类,继承 HttpServlet 父类,并且重写这个父类中的一些重要方法
- HttpServlet 中提供的 doXXX 系列,和 HTTP 协议的方法是一一对应的
- doGet 方法 —— 查看图片
@Overrideprotected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {// 考虑到查看所有图片和查看指定图片属性// 通过是否 URL 中带有 imageId 参数进行区分// 存在 imageId 查看指定图片属性,否咋查看所有图片属性// 例如:URL / image?imageId=100// imageId 的值就是 100// 如果 URL 中不存在 imageId 那么返回 nullString imageId = req.getParameter("imageId");if (imageId == null || imageId.equals("")) {// 查看所有图片属性selectAll(req, resp);} else {// 查看指定图片selectOne(imageId,resp);}}private void selectAll(HttpServletRequest req, HttpServletResponse resp) throws IOException {resp.setContentType("application/json;charset=utf-8");// 1.创建一个 ImageDao 对象,并查找数据ImageDao imageDao = new ImageDao();List<Image> images = imageDao.selectAll();// 2.将查找到的结果转换成Json格式的字符串,然后写回给resp对象Gson gson = new GsonBuilder().create();// jsonData 就是一个 json 格式的字符串,和之前约定的格式是一样的// 重点体会下列代码,是方法的核心,gson 帮我们完成了大量的格式转换工作// 只要把之前的相关字段约定成统一的命名,下面的操作就可一步到位String jsonData = gson.toJson(images);resp.getWriter().write(jsonData);}private void selectOne(String imageId, HttpServletResponse resp) throws IOException {resp.setContentType("application/json;charset=utf-8");// 1.创建一个 ImageDao 对象,并查找数据ImageDao imageDao = new ImageDao();Image image = imageDao.selectOne(Integer.parseInt(imageId));// 2.将查找到的结果转换成Json格式的字符串,然后写回给resp对象Gson gson = new GsonBuilder().create();String jsonData = gson.toJson(image);resp.getWriter().write(jsonData);}
- doPost 方法 —— 上传图片
@Overrideprotected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {// 1.获取图片的属性信息,并且写入数据库// a.创建factory对象和upload对象FileItemFactory factory = new DiskFileItemFactory();ServletFileUpload upload = new ServletFileUpload(factory);// b.通过upload对象进一步解析请求,解析HTTP请求中body的内容// FileItem代表一个文件对象,HTTP支持一个请求同时上传多个文件List<FileItem> items = null;try {items = upload.parseRequest(req);} catch (FileUploadException e) {// 出现异常说明解析出错e.printStackTrace();// 告诉客户端出现的具体的错误resp.setContentType("application/json; charset=utf-8");resp.getWriter().write("{\"ok\":false,\"reason\":\"请求解析失败\"}");return;}// c.将FileItem中的属性提取出来,转换成Image对象,存入数据库中// 当前只考虑一张图片的情况FileItem fileItem = items.get(0);Image image = new Image();image.setImageName(fileItem.getName());image.setSize((int) fileItem.getSize());// 手动获取时间,并转成格式化日期SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyyMMdd");image.setUploadTime(simpleDateFormat.format(new Date()));image.setContentType(fileItem.getContentType());// MD5 暂时不用要求image.setMd5(DigestUtils.md5Hex(fileItem.get()));// 构造一个路径来进行保存image.setPath("./image/" + image.getMd5());// 存到数据库ImageDao imageDao = new ImageDao();// 看看数据库中是否存在相同的MD5的图片,不存在,返回 nullImage existImage = imageDao.selectByMd5(image.getMd5());imageDao.insert(image);// 2.获取图片的内容信息,并且写入磁盘if (existImage == null) {File file = new File(image.getPath());try {fileItem.write(file);} catch (Exception e) {e.printStackTrace();resp.setContentType("application/json; charset=utf-8");resp.getWriter().write("{\"ok\":false,\"reason\":\"写磁盘失败\"}");return;}}// 3.给客户端返回一个结果数据/*resp.setContentType("application/json; charset=utf-8");resp.getWriter().write("{\"ok\":true}");*///上传之后可以直接看到新上传的图片resp.sendRedirect("images.html");}
- doDelete方法 —— 删除图片
@Overrideprotected void doDelete(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {resp.setContentType("application/json; charset=utf-8");// 1.先获取到请求中的 imageIdString imageId = req.getParameter("imageId");if (imageId == null || imageId.equals("")) {resp.setStatus(200);resp.getWriter().write("{ \"ok\": false, \"reason\": \"请求解析失败\"}");return;}// 2.创建 ImageDao 对象,查看到该图片对象对应的相关属性(为了知道这个图片对应文件路径)ImageDao imageDao = new ImageDao();Image image = imageDao.selectOne(Integer.parseInt(imageId));if (image == null) {// 此时请求中传入的 id 在数据库中不存在resp.setStatus(200);resp.getWriter().write("{ \"ok\": false, \"reason\": \"imageId 在数据库中不存在\"}");return;}// 3.删除数据库中的记录imageDao.delete(Integer.parseInt(imageId));// 4.删除本地磁盘文件File file = new File(image.getPath());file.delete();resp.setStatus(200);resp.getWriter().write("{ \"ok\": true }");}
- ImageServletShow
标签
告诉 Tomcat 当前这个 Servlet 对应到代码中的哪个类
标签
告诉 Tomcat 当前这个 Servlet 对应到的 URL 的 path 是什么
HTTP 服务器
- 启动 HTTP 服务器,直接基于 socket,伪代码如下
serverStartSocket socketsocket.bind(服务器IP + 端口号)while(true) {Socket newSocket = socket.accept();// 读取数据(客户端发来数据,读到的数据相当于一份完整的HTTP请求)Byte[] inputBuffer = newSocket.read();// 解析读到的 HTTP 请求格式的数据HttpServletRequest req = parse(inputBuffer);// 调用自己编写的代码(ImageServlet)// 根据 url 中的 path 决定创建哪个对象// h 对象本质上是一个 ImageServlet// 根据 web.xmlHttpServlet h = build(req.getUrl)if(req.getMethod().equals("GET")) {h.doGet(req,resp);} else if(req.getMethod().equals("GET")) {h.doGet(req,resp);} else if() {}// 执行完上述方法之后,再把 resp 对象转为字符串,写回到 socket 中newSocket.write(resp.toString());}
- Servlet 可以理解成一种编程框架
当前已经有许多现成的代码,只有一少部分需要用户进行编写,完成整个工作 - Servlet 需要做的核心工作,创建一个 Servlet 类,完成如何 根据 HTTP 请求,计算生成 HTTP 响应
前端页面
1.关于前端代码的知识
- HTML:网页的骨架
- CSS:描述网页上组件的样式(颜色、位置、大小、字体、背景)
- JavaScript:描述前端页面上的一些动作(和用户具体交互的行为)
- 关于这些的教程:w3school 在线教程
- ul 表示无序列表
- li 嵌套在 ul 内部,列表中具体的某个条目
2.为了实现页面的上传 - 把搜索框改成了文件上传按钮
- 新增了一个提交按钮
- 修改了 form 属性,新增 action,method,enctype
3.测试 HTML 显示效果的时候,有两种方式 - 在 IDEA 直接点击浏览器图标(只是在本地进行测试)
- 部署到服务器上,通过浏览器远程访问服务器
4.浏览器看到的页面内容可能和源代码中的内容有一定差别,这些都是通过 JS 动态渲染的
5.把网页上显示的这些预览图片替换成服务器上保存的图片,img 标签中 src 改成服务器上存储的图片的 url 就可以
6.需要获取到服务器上所有的图片的 url(ImageServlet),需要通过 JS 先获取所有照片的属性,再分别加载每一个图片,使用 JS 完成
7.此处引入 Vue JS 的框架来帮助我们更方便编写代码 —— 主流的前端框架 Vue、React(Facebook)、Auglar(Google)
8.Vue 中的命令 - v-for:循环访问一个数据
- v-bind:把数据绑定到 html 标签上的某个属性
- v-on:绑定某种事件的处理函数
- 如果是在标签内部使用 Vue 对象中的数据,就需要使用插值表达式
- 如果是在 标签属性 中使用 Vue 对象中的数据,不需要使用插值表达式,但是需要搭配 Vue 的命令
9.接下来通过浏览器中 JS 代码请求服务器,获取到服务器上都有哪些图片,把这个数据作为 Vue 渲染的依据(Vue 对象中 images 数组)
10.原来页面的渲染过程 - 先加载图片
- 再根据图片大小设定图片的位置,设定显示图片的空间大小
11.现在使用 ajax 进行渲染 - 页面尝试获取图片大小,并设定显示图片的空间(当前图片还没获取到,也不知图片大小)
- 通过 ajax 获取图片内容
- 改进上传操作,上传成功之后,自动掉转到主页 index.html,使用 HTTP 重定向(302可以完成重定向)
本文来自互联网用户投稿,文章观点仅代表作者本人,不代表本站立场,不承担相关法律责任。如若转载,请注明出处。 如若内容造成侵权/违法违规/事实不符,请点击【内容举报】进行投诉反馈!
