springboot搭建web应用

2022-11-02  本文已影响0人  愤怒的阿昆达

一、学习情景:

作为Java程序员,现在领导要求你负责新闻信息系统后端中用户模块,要求你实现该模块的用户注册和登录功能。
要求技术栈:idea、springboot、maven、mybatis、swagger、lombok

二、前置环境:

三、创建项目

  1. 打开idea


    image.png
  2. 新建项目


    image.png

    设置项目的基本信息,其中注意jdk版本要与Java版本匹配,这里使用jdk1.8和java8


    image.png
    下一步选择springboot版本和项目依赖
    image.png

    选择新窗口编辑器


    image.png
  3. 目录结构(初始)


    image.png

四、Mysql数据库

数据库下载安装和配置

参见:windows下安装mysql - 简书 (jianshu.com)

脚本建库、建表导入

脚本:

-- 零、删库:
-- DROP DATABASE IF EXISTS `db_news`;

-- 一、建库
CREATE DATABASE IF NOT EXISTS `db_news` CHARACTER SET 'utf8mb4' COLLATE 'utf8mb4_general_ci';

-- 二、建表、索引等
-- -------------------------------------------------------------------------------------------------------
use db_news;

DROP TABLE IF EXISTS `dt_user`;

-- 用户表:
CREATE TABLE IF NOT EXISTS `dt_user`(
    `id` INT AUTO_INCREMENT PRIMARY KEY COMMENT '用户编号',
    `username` VARCHAR(32) NOT NULL UNIQUE COMMENT '用户名',
    `password` VARCHAR(32) NOT NULL COMMENT '密码',
    `mobile` CHAR(11) NOT NULL COMMENT '手机号',
    `age` TINYINT UNSIGNED NOT NULL DEFAULT 18 COMMENT '年龄'
) ENGINE=InnoDB COMMENT = '用户表' DEFAULT CHARSET=utf8mb4 COLLATE = utf8mb4_general_ci ROW_FORMAT = Dynamic;

使用ideaDatabase插件连接Mysql数据库

image.png
image.png
image.png

至此可以看到用户表:dt_user

五、项目架构

image.png

六、根据架构调整项目目录

pom.xml配置

pom.xml

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.7.5</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>

    <groupId>com.sunning</groupId>
    <artifactId>news</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>news</name>
    <description>news</description>

    <properties>
        <java.version>1.8</java.version>
    </properties>

    <dependencies>
        <!--  springboot 整合web组件-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>

        <!-- MySql 8.0.27 Connector -->
        <!-- 使用sql查询`select version() from dual;`查看mysql版本号 -->
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <scope>runtime</scope>
        </dependency>

        <!-- druid 数据连接池-->
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>druid</artifactId>
            <version>1.1.19</version>
        </dependency>

        <!--mybatis 依赖-->
        <dependency>
            <groupId>org.mybatis.spring.boot</groupId>
            <artifactId>mybatis-spring-boot-starter</artifactId>
            <version>2.1.0</version>
        </dependency>

        <!--lombok 依赖-->
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <optional>true</optional>
        </dependency>

        <!-- swagger2 依赖 -->
        <dependency>
            <groupId>io.springfox</groupId>
            <artifactId>springfox-swagger2</artifactId>
            <version>2.9.2</version>
        </dependency>
        <dependency>
            <groupId>io.springfox</groupId>
            <artifactId>springfox-swagger-ui</artifactId>
            <version>2.9.2</version>
        </dependency>


    </dependencies>

    <build>
        <!--  如果你想定义jar包的名称可直接在这里配置否则可以注释或删掉-->
        <finalName>AppNews</finalName>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <!--    maven build插件-->
                <artifactId>spring-boot-maven-plugin</artifactId>
                <!--    spring-boot-maven-plugin报红添加相匹配的springboot版本号-->
                <version>2.7.5</version>
            </plugin>
        </plugins>
    </build>

</project>


application.yml配置

application.yml

server:
  host: localhost
  port: 8081

spring:
  application:
    name: news
  # 数据源配置
  datasource:
    url: jdbc:mysql://${server.host}:3306/mysql?useUnicode=true&characterEncoding=utf8&autoReconnect=true&failOverReadOnly=false&rewriteBatchedStatements=TRUE
    driver-class-name: com.mysql.cj.jdbc.Driver
    username: root
    password: '123456'
    type: com.alibaba.druid.pool.DruidDataSource
    # 数据源其他配置
    initialSize: 5
    minIdle: 5
    maxActive: 20
    maxWait: 60000
    timeBetweenEvictionRunsMillis: 60000
    minEvictableIdleTimeMillis: 300000
    validationQuery: SELECT 1
    testWhileIdle: true
    testOnBorrow: false
    testOnReturn: false
    poolPreparedStatements: true
    #   配置监控统计拦截的filters,去掉后监控界面sql无法统计,'wall'用于防火墙
    filters: stat,wall,logback
    maxPoolPreparedStatementPerConnectionSize: 20
    useGlobalDataSourceStat: true
    connectionProperties: druid.stat.mergeSql=true;druid.stat.slowSqlMillis=500
  # 路径匹配策略
  mvc:
    pathmatch:
      matching-strategy: ant_path_matcher

# 日志配置
logging:
  level:
    com:
      dxy:
        odr:
          referee:
            dao:
              mapper: debug

mybatis:
  configuration:
    # 字段下划线转驼峰
    map-underscore-to-camel-case: true
    # 打印sql+查询结果
    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl

开发目录结构

image.png

七、用户注册、登录功能实现

dao层之user实体类定义

User.java

package com.sunning.news.dao.pojo;

import lombok.*;

/**
 * 用户实体类
 *
 * @author ckk
 * @create 2022-11-02 14:47
 **/
@Data
@NoArgsConstructor
@AllArgsConstructor
@Getter
@Setter
public class User {
    private Integer id;
    private String username;
    private String password;
    private String mobile;
    private Integer age;

}

dao层之mapper数据访问层

UserMapper.java

package com.sunning.news.dao.mapper;

import com.sunning.news.dao.pojo.User;
import org.apache.ibatis.annotations.Insert;
import org.apache.ibatis.annotations.Options;
import org.apache.ibatis.annotations.Param;
import org.apache.ibatis.annotations.Select;
import org.springframework.stereotype.Repository;

/**
 * 用户管理mapper
 * 
 * @author ckk
 * @create 2022-11-02 14:48
 **/
@Repository
public interface UserMapper {

    /**
     * 新增用户记录,返回自增主键
     * username 不重复才新增
     * @param user
     * @return
     */
    @Insert(" insert into db_news.`dt_user`(username, password, mobile, age)" +
            " select #{username}, #{password}, #{mobile}, #{age}" +
            " where not exists(select username from db_news.`dt_user` where username = #{username})")
    @Options(useGeneratedKeys = true, keyProperty = "id")
    int register(User user);

    /**
     * 用户登录,获取用户信息
     * @param username
     * @param password
     * @return
     */
    @Select(" select *" +
            " from db_news.`dt_user`" +
            " where username = #{username} and password = #{password}")
    User login(@Param("username") String username, @Param("password") String password);

    /**
     * 根据username获取用户信息
     * @param username
     * @return
     */
    @Select(" select *" +
            " from db_news.`dt_user`" +
            " where username = #{username}")
    User findByUsername(@Param("username") String username);

    /**
     * 根据id获取用户信息
     * @param id
     * @return
     */
    @Select(" select *" +
            " from db_news.`dt_user`" +
            " where id = #{id}")
    User findByUid(@Param("id") Integer id);

}

service层之接口定义

UserService.java

package com.sunning.news.service;

import com.sunning.news.dao.pojo.User;

/**
 * 用户业务逻辑层接口
 *
 * @author ckk
 * @create 2022-11-02 14:48
 **/
public interface UserService {

    /**
     * 注册业务逻辑
     * @param user 要注册的User对象,属性中主键id要为空
     * @return
     */
    User registerService(User user);

    /**
     * 登录业务逻辑
     * @param uname 账户名
     * @param password 密码
     * @return
     */
    User loginService(String uname, String password);

}

service层之实现类

UserServiceImpl.java

package com.sunning.news.service.impl;

import com.sunning.news.dao.mapper.UserMapper;
import com.sunning.news.dao.pojo.User;
import com.sunning.news.service.UserService;
import org.springframework.stereotype.Service;

import javax.annotation.Resource;

/**
 * 用户业务逻辑层实现类
 *
 * @author ckk
 * @create 2022-11-02 14:49
 **/
@Service
public class UserServiceImpl implements UserService {

    @Resource
    private UserMapper userMapper;

    @Override
    public User registerService(User user) {
        // 结果
        User userDTO = null;

        // 当新用户的用户名已存在时
        if(userMapper.findByUsername(user.getUsername()) != null){
            return null;
        }

        // 入库
        int id = userMapper.register(user);
        if(id > 0){
            // 查询
            userDTO = userMapper.findByUid(id);
        }

        // 抹掉密码等重要信息
        if(userDTO != null){
            userDTO.setPassword("");
        }

        // 返回结果
        return userDTO;
    }

    @Override
    public User loginService(String uname, String password) {
        // 登录
        User user = userMapper.login(uname, password);

        // 抹掉密码等重要信息
        if(user != null){
            user.setPassword("");
        }

        return user;
    }

}

controller控制层

UserController.java

package com.sunning.news.controller;

import com.sunning.news.dao.pojo.User;
import com.sunning.news.service.UserService;
import com.sunning.news.utils.Result;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import io.swagger.annotations.ApiResponse;
import io.swagger.annotations.ApiResponses;
import org.springframework.web.bind.annotation.*;

import javax.annotation.Resource;

/**
 * 用户控制层
 *
 * @author ckk
 * @create 2022-11-02 14:42
 **/
@Api(tags = "用户模块")
@RestController
@RequestMapping("/user")
public class UserController {
    @Resource
    private UserService userService;

    @ApiOperation("用户注册")
    @PostMapping("/register")
    public Result<User> registController(@RequestBody User userAO){
        User user = userService.registerService(userAO);
        if(user!=null){
            return Result.success(user,"用户注册成功!");
        }else{
            return Result.error("-1","用户名已存在!");
        }
    }

    @ApiOperation("用户登录")
    @PostMapping("/login")
    public Result<User> loginController(@RequestParam String uname, @RequestParam String password){
        User user = userService.loginService(uname, password);
        if(user!=null){
            return Result.success(user,"用户登录成功!");
        }else{
            return Result.error("-2","账号或密码错误!");
        }
    }


}

其他

除了以上包之外,还有config包、utils包,这两者一个是配置类包,另一个是工具类包。

工具类

Result.java

package com.sunning.news.utils;

/**
 * 响应结果类
 *
 * @author ckk
 * @create 2022-11-02 14:51
 **/
public class Result<T> {
    private String code;
    private String msg;
    private T data;

    public String getCode() {
        return code;
    }

    public void setCode(String code) {
        this.code = code;
    }

    public String getMsg() {
        return msg;
    }

    public void setMsg(String msg) {
        this.msg = msg;
    }

    public T getData() {
        return data;
    }

    public void setData(T data) {
        this.data = data;
    }

    public Result() {
    }

    public Result(T data) {
        this.data = data;
    }

    public static Result success() {
        Result result = new Result<>();
        result.setCode("0");
        result.setMsg("成功");
        return result;
    }

    public static <T> Result<T> success(T data) {
        Result<T> result = new Result<>(data);
        result.setCode("0");
        result.setMsg("成功");
        return result;
    }

    public static <T> Result<T> success(T data,String msg) {
        Result<T> result = new Result<>(data);
        result.setCode("0");
        result.setMsg(msg);
        return result;
    }

    public static Result error(String code, String msg) {
        Result result = new Result();
        result.setCode(code);
        result.setMsg(msg);
        return result;
    }
}

配置类

swagger配置

SwaggerConfig.java

package com.sunning.news.config;

import com.google.common.base.Predicates;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.ResourceHandlerRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurationSupport;
import springfox.documentation.builders.ApiInfoBuilder;
import springfox.documentation.builders.PathSelectors;
import springfox.documentation.service.ApiInfo;
import springfox.documentation.spi.DocumentationType;
import springfox.documentation.spring.web.plugins.Docket;
import springfox.documentation.swagger2.annotations.EnableSwagger2;

/**
 * Swagger2配置类
 *
 * @author ckk
 * @create 2022-11-02 16:01
 **/
@Configuration
@EnableSwagger2
public class SwaggerConfig extends WebMvcConfigurationSupport {

    @Bean
    public Docket newsApiConfig(){
        return new Docket(DocumentationType.SWAGGER_2)
                .groupName("newsApi")
                // 调用apiInfo方法,创建一个ApiInfo实例,
                // 里面是展示在文档页面信息内容
                .apiInfo(newsApiInfo())
                .select()
                .paths(Predicates.not(PathSelectors.regex("/admin/.*")))
                .paths(Predicates.not(PathSelectors.regex("/error.*")))
                .build();

    }


    // api文档的详细信息
    private ApiInfo newsApiInfo(){

        return new ApiInfoBuilder()
                .title("新闻系统API文档接口测试")    //标题
                .description("本文档描述接口测试用例")  //描述
                .version("1.0")  //版本
                .build();
    }

    /**
     * 解决swagger-ui.html 404无法访问的问题
     */
    @Override
    protected void addResourceHandlers(ResourceHandlerRegistry registry) {
        // 解决静态资源无法访问
        registry.addResourceHandler("/**")
                .addResourceLocations("classpath:/static/");
        // 解决swagger无法访问
        registry.addResourceHandler("/swagger-ui.html")
                .addResourceLocations("classpath:/META-INF/resources/");
        // 解决swagger的js文件无法访问
        registry.addResourceHandler("/webjars/**")
                .addResourceLocations("classpath:/META-INF/resources/webjars/");
    }
}

跨域配置

GlobalCorsConfig.java

package com.sunning.news.config;

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.CorsRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;

/**
 * 全局跨域配置
 *
 * @author ckk
 * @create 2022-11-02 15:52
 **/
@Configuration
public class GlobalCorsConfig {
    @Bean
    public WebMvcConfigurer corsConfigurer() {
        return new WebMvcConfigurer() {
            @Override
            public void addCorsMappings(CorsRegistry registry) {
                registry.addMapping("/**")    //添加映射路径,“/**”表示对所有的路径实行全局跨域访问权限的设置
                        .allowedOriginPatterns("*")    //开放哪些ip、端口、域名的访问权限,不用allowedOrigins是为了适配swagger2
                        .allowCredentials(true)  //是否允许发送Cookie信息
                        .allowedMethods("GET", "POST", "PUT", "DELETE", "OPTIONS")     //开放哪些Http方法,允许跨域访问
                        .allowedHeaders("*")     //允许HTTP请求中的携带哪些Header信息
                        .exposedHeaders("*");   //暴露哪些头部信息(因为跨域访问默认不能获取全部头部信息)
            }
        };
    }
}

启动类配置

NewsApplication.java是启动类,在这里需要配置mybatis的包扫描路径,使用@MapperScan({"com.sunning.news.dao.mapper"})注解即可,如图:

image.png

八、debug运行,swagger测试

在NewsApplication.java启动类上右键点击Debug模式运行项目,如图:


image.png

看到此日志,启动成功,如图:


image.png

swagger-ui页面进行测试,访问:http://localhost:8081/swagger-ui.html

image.png
image.png
image.png

九、打成jar包运行,swagger测试

maven打包:在idea右侧边的Maven栏目中,找到news-Lifecycle-install功能,双击运行,如图:


image.png

在项目的target路径下找到maven打好的jar包:AppNews.jar文件,如图:


image.png

在该目录下启动cmd,执行命令:java -jar AppNews.jar,启动此web服务,如图:

image.png

使用swagger进行测试,同第八步骤中所示。

上一篇 下一篇

猜你喜欢

热点阅读