Java后端入门之书城项目

下载地址:
https://wws.lanzoui.com/i7G2Gp79bli
几个通用的base工具类

DAO的工具类——BaseDao

package com.xzh.dao.impl;

import com.xzh.utils.JdbcUtils;
import org.apache.commons.dbutils.QueryRunner;
import org.apache.commons.dbutils.handlers.BeanHandler;
import org.apache.commons.dbutils.handlers.BeanListHandler;
import org.apache.commons.dbutils.handlers.ScalarHandler;

import java.sql.Connection;
import java.sql.SQLException;
import java.util.List;

/**
 * DAO的父类主要用于继承 里面写的查询功能
 * 工具类
 * @Author Lin
 * @CreateTime 2021/5/7 21:09
 */
public abstract class BaseDao {
    //DBUtils操作数据库
    private QueryRunner queryRunner = new QueryRunner();

    public int update(String s, Object... objects) {
        Connection connection = JdbcUtils.getConnection();

        try {
            return queryRunner.update(connection, s, objects);
        } catch (SQLException throwables) {
            throwables.printStackTrace();
            //这里必须往上抛 如果不往上抛的话 外面不能捕获异常
            // 就不能判断到底是提交还是回滚
            throw new RuntimeException(throwables);
        }

    }

    /**
     * @param type
     * @param sql
     * @param args
     * @param <T>
     * @return
     */
    public < T > T queryForOne(Class< T > type, String sql, Object... args) {
        Connection connection = JdbcUtils.getConnection();
        try {
            return queryRunner.query(connection, sql, new BeanHandler< T >(type), args);
        } catch (SQLException throwables) {
            throwables.printStackTrace();
            //这里必须往上抛 如果不往上抛的话 外面不能捕获异常
            // 就不能判断到底是提交还是回滚
            throw new RuntimeException(throwables);
        }
    }

    /**
     * @param type
     * @param sql
     * @param args
     * @param <T>
     * @return
     */
    public < T > List< T > queryForList(Class< T > type, String sql, Object... args) {
        Connection connection = JdbcUtils.getConnection();
        try {
            return queryRunner.query(connection, sql, new BeanListHandler< T >(type), args);
        } catch (SQLException throwables) {
            throwables.printStackTrace();
            //这里必须往上抛 如果不往上抛的话 外面不能捕获异常
            // 就不能判断到底是提交还是回滚
            throw new RuntimeException(throwables);
        }
    }

    /**
     * 查询一行一列的东西
     *
     * @param sql
     * @param args
     * @return
     */
    public Object queryForSingleValue(String sql, Object... args) {
        Connection connection = JdbcUtils.getConnection();
        try {
            return queryRunner.query(connection, sql, new ScalarHandler(), args);
        } catch (SQLException throwables) {
            throwables.printStackTrace();
            //这里必须往上抛 如果不往上抛的话 外面不能捕获异常
            // 就不能判断到底是提交还是回滚
            throw new RuntimeException(throwables);
        }
    }

    public int queryId(String sql,Object username){
        Connection connection = JdbcUtils.getConnection();
        try {

            return (int)queryRunner.query(connection, sql, new ScalarHandler(), username);
        } catch (SQLException throwables) {
            throwables.printStackTrace();
            //这里必须往上抛 如果不往上抛的话 外面不能捕获异常
            // 就不能判断到底是提交还是回滚
            throw new RuntimeException(throwables);

        }
    }

}

Jdbc工具类——JdbcUtils

package com.xzh.utils;

import com.alibaba.druid.pool.DruidDataSource;
import com.alibaba.druid.pool.DruidDataSourceFactory;

import java.io.InputStream;
import java.sql.Connection;
import java.sql.SQLException;
import java.util.Properties;

/**
 * 数据库连接工具类
 * 后面调用需要手动提交
 * 提交调用  commitAndClose()
 * 关闭连接  rollbackAndClose()
 * @Author Lin
 * @CreateTime 2021/5/7 19:14
 */
public class JdbcUtils {
    //创建一个ThreadLocal类 里面值得类为Connection类
    private static ThreadLocal< Connection > conns = new ThreadLocal<>();
    private static DruidDataSource dataSource;

    static {
        try {
            Properties properties = new Properties();
            //读取配置文件
            InputStream resourceAsStream = JdbcUtils.class.getClassLoader().getResourceAsStream("jdbc.properties");
            //加载数据
            properties.load(resourceAsStream);
            //获取链接
            dataSource = (DruidDataSource) DruidDataSourceFactory.createDataSource(properties);
        } catch (Exception e) {
            e.printStackTrace();
        }

    }


    /**
     * 获取链接
     *
     * @return //返回null说明获取连接失败
     */
    public static Connection getConnection() {
        //从ThreadLocal中得到链接
        Connection conn = conns.get();
        //若果没有就创建一个链接
        if (conn == null) {
            try {
                //从datasource中获取一个链接
                conn = dataSource.getConnection();
                //设置为手动提交
                conn.setAutoCommit(false);
                //保存链接到ThreadLocal中
                conns.set(conn);
            } catch (SQLException throwables) {
                throwables.printStackTrace();
            }
        }

        return conn;
    }

    /**
     * 事务回滚
     */
    public static void commitAndClose() {
        //从线程中得到一个数值
        Connection connection = conns.get();
        if (connection != null) {//如果不等于空就说名使用过这个链接
            try {
                connection.commit();//事务提交
            } catch (SQLException throwables) {
                throwables.printStackTrace();
            } finally {
                try {
                    connection.close();//关闭连接
                } catch (SQLException throwables) {
                    throwables.printStackTrace();
                }
            }

        }

        conns.remove();//必须加上这个因为tomcat底层使用了线程池技术
    }


    /**
     * 事务回滚
     */
    public static void rollbackAndClose() {
        //从线程中得到一个数值
        Connection connection = conns.get();
        if (connection != null) {//如果不等于空就说名使用过这个链接
            try {
                connection.rollback();//事务回滚
            } catch (SQLException throwables) {
                throwables.printStackTrace();
            } finally {
                try {
                    connection.close();//关闭连接
                } catch (SQLException throwables) {
                    throwables.printStackTrace();
                }
            }

        }

        conns.remove();//必须加上这个因为tomcat底层使用了线程池技术
    }


}

Web数据保存到javabean工具类——WebUtils

package com.xzh.utils;

import org.apache.commons.beanutils.BeanUtils;

import java.lang.reflect.InvocationTargetException;
import java.util.Map;

/**
 * 工具类
 * parseInt 把数字字符串 转换成整数
 * copyParamToBean 把map的数据根据key保存到javabean中
 * @Author Lin
 * @CreateTime 2021/5/10 10:03
 */
public class WebUtils {
    /**
     * @param map  请求
     * @param bean 传入的数据库对象
     */
    public static < T > T copyParamToBean(Map map, T bean) {
        try {
            BeanUtils.populate(bean, map);
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        }
        return bean;
    }

    public static int parseInt(String strInt, int defaultValue) {
       try {
           if (strInt == null || strInt==""){
               return defaultValue;
           }else {
               return Integer.parseInt(strInt);
           }
       }catch (NumberFormatException e){
           e.printStackTrace();
       }
       return defaultValue;
    }
}

Servlet工具类根据反射调用方法——BaseServlet

package com.xzh.web;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.lang.reflect.Method;

/**
 * 这个类继承了HttpServlet并且重写了doPost 和 get方法
 * 主要通过反射来调用servlet的方法
 *
 * @Author Lin
 * @CreateTime 2021/5/10 9:45
 */
public abstract class BaseServlet extends HttpServlet {

    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //解决请求和转发乱码
        request.setCharacterEncoding("UTF-8");
        response.setCharacterEncoding("UTF-8");

        //得到网页的信息 判断该页面是什么页面
        //<%--action后面的值表示要调用的servlet的方法--%>
        String action = request.getParameter("action");
        try {
            //通过反射调用方法  getDeclaredMethod(方法名,参数类型 ...args){}
            Method method = this.getClass().getDeclaredMethod(action, HttpServletRequest.class, HttpServletResponse.class);
            //调用方法 并传入参数
            method.invoke(this, request, response);
        } catch (Exception e) {
            e.printStackTrace();
            throw new RuntimeException(e);
        }
    }

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        doPost(req, resp);
    }
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
这个网上书城系统使用Eclipse开发的,代码完整,jar包齐全,sql脚本包含在里面,将下载下来的项目导入到Eclipse中即可运行,本项目做了很多校验,对可能出现的bug做了考虑,属于比较完善的系统。本系统包含九个模块,前台模块分为:用户模快,分类模块,图书模块,购物车模块,订单模块;后台模块分为:管理员模块,分类管理模快,图书管理模快,订单管理模块。书城界面简洁,易于操作,简单易懂,代码均有注释,各模块功能完善。各大模块的功能描述:前台: 1). 用户模块功能有: * 用户注册: > 表单页面是jQuery做校验(包含了ajax异步请求) # 在输入框失去焦点时进行校验; # 在提交时对所有输入框进行校验; # 在输入框得到焦点时,隐藏错误信息。 > 表单页面使用一次性图形验证码; > 在servlet中再次做了表单校验。 * 用户登录: > 表单校验与注册功能相同; > 登录成功时会把当前用户名保存到cookie中,为了在登录页面的输入框中显示! * 用户退出:销毁session 2). 分类模块 * 查询所有分类: > 有1级和2级分类 > 在页面中使用手风琴式菜单(Javascript组件)显示分类。 3). 图书模块: * 按分类查询 * 按作者查询 * 按出版社查询 * 按书名模糊查询 * 多条件组合查询 * 按id查询 除按id查询外,其他都是分页查询。 技术难点: > 组合查询:根据多个条件拼凑sql语句。 > 带条件分页查询:条件可能会丢失。使用自定义的PageBean来传递分页数据! > 页面上的分页导航:页码列表的显示不好计算! 4). 购物车模块: * 添加条目 * 修改条目数量 * 删除条目 * 批量删除条目 * 我的购物车 * 查询被勾选条目 购物车没有使用sesson或cookie,而是存储到数据库中。 技术难点: > 添加条目时,如果两次添加针对同一本书的条目,不是添加,而是合并; > 修改数量时使用ajax时请求服务器端,服务器端返回json。 > 大量js代码 5). 订单模块: * 生成订单 * 我的订单 * 查看订单详细 * 订单支付 * 订单确认收货 * 取消订单 后台 1). 管理员 * 管理员登录 2). 分类管理 * 添加1级分类 * 添加2级分类: 需要为2级分类指定所属1级分类 * 编辑1级分类 * 编辑2级分类: 可以修改所属1级分类 * 删除1级分类: 存在子分类时,不能删除 * 删除2级分类: 当前2级分类下存在图书时不能删除 * 查看所有分类 3). 图书管理 * 各种查询:与前台相同 * 添加图书: > 上传图片 > 页面中使用动态下拉列表显示2级分类,当指定1级分类后,2级分类下拉列表中动态显示该1级分类下所有2级分类名称 * 修改图书: 与添加图书相似,也使用动态下拉列表 * 删除图书: 需要删除图书对应图片,再删除图书 4). 订单管理 * 各种查询 * 订单发货 * 订单取消
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值