一 项目架构
1.1 今日指数技术选型
【1】前端技术
data:image/s3,"s3://crabby-images/6fad3/6fad36e459bed024549eb0f5ecf9c6f06e9f6423" alt=""
【2】后端技术栈
data:image/s3,"s3://crabby-images/ad913/ad91306f2f8b6f3d50dce6d72ae7a137b96bf1ee" alt=""
【3】整体概览
data:image/s3,"s3://crabby-images/8b81c/8b81cd9444537aeaad690e168c6d3a85e251c4fa" alt=""
1.2 核心业务介绍
【1】业务结构预览
data:image/s3,"s3://crabby-images/248fb/248fbcfc29232a7e04ebad279dd3abc7c26509fa" alt=""
【2】业务结构预览
1.定时任务调度服务
XXL-JOB通过RestTemplate+多线程动态拉去股票接口数据,刷入数据库;
2.国内指数服务
3.板块指数服务
4.涨幅榜展示功能
5.涨停跌停数展示功能
6.成交量对比展示功能
7.个股涨停服务展示功能
8.个股详情展示功能
包含分时行情、日k线、周K线图等
9.个股描述服务;
10.报表导出服务
二 后端开发环境搭建
开发工具版本要求:
data:image/s3,"s3://crabby-images/c3706/c37064a7fd9854b91fb5c0c76cfa36ee473b8b5a" alt=""
2.1 数据库环境搭建
【1】表结构介绍
data:image/s3,"s3://crabby-images/7f2b7/7f2b75a5d868e405b231c692e9e5524fceb1a6ea" alt=""
data:image/s3,"s3://crabby-images/4c61e/4c61ebd0d23619a2f48611190bb9c997f74e5d10" alt=""
data:image/s3,"s3://crabby-images/e32d8/e32d8018d19ee4d4d0e611e4957244382cde3659" alt=""
注意事项:后期股票相关的数据量非常庞大,表与表之间尽量不要构建外键约束(提升数据库性能),同时也为后期分库分表准备!
【2】数据导入
因为我边的CentOS7连接不上 所以在这里面 用本地(windoms)运行
数据库可视化选用 SQLyog - 64 bit mysql用的是5.7.24
data:image/s3,"s3://crabby-images/9e7dd/9e7dd47f91fc1b744678889e1ae639fa35f2cba5" alt=""
2.2后端工程搭建
【1】构建stock_parent父工程
创建stock_parent maven工程:
data:image/s3,"s3://crabby-images/9a511/9a51143920c6021c8d63ddb477a3ca00ab7fc6e2" alt=""
data:image/s3,"s3://crabby-images/84000/84000e6d2857b533d649c912e41be621313b3a63" alt=""
接下来 复制以下maven到pom.xml
<packaging>pom</packaging>
<properties>
<!--工程构建的代码格式为UTF-8-->
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<!--编译-->
<maven.compiler.encoding>UTF-8</maven.compiler.encoding>
<!--编译原文件-->
<maven.compiler.source>8</maven.compiler.source>
<!--生成的编译目录-->
<maven.compiler.target>8</maven.compiler.target>
<!--打包时跳过测试-->
<maven.test.skip>true</maven.test.skip>
<!--mybatis整合spring-boot场景依赖-->
<mybatis-spring-boot-starter.version>2.1.4</mybatis-spring-boot-starter.version>
<!--pagehelper版本-->
<pagehelper-spring-boot-starter.version>1.2.12</pagehelper-spring-boot-starter.version>
<!--mysql驱动包-->
<mysql-driver.version>5.1.49</mysql-driver.version>
<!--fastjson工具-->
<fastjson.version>1.2.71</fastjson.version>
<!--依赖的版本-->
<springfox-swagger2.version>2.9.2</springfox-swagger2.version>
<!--druid的场景依赖-->
<druid-spring-boot-starter.version>1.1.22</druid-spring-boot-starter.version>
<!--druid的核心依赖-->
<druid-core-version>1.2.8</druid-core-version>
<!--分库分表对应的版本-->
<sharding-jdbc.version>4.0.0-RC1</sharding-jdbc.version>
<!--jwt-->
<jjwt.version>0.9.1</jjwt.version>
<!--easyExcel 报表导入导出-->
<easyExcel.version>3.0.4</easyExcel.version>
<!--xxl-job-->
<xxl-job-core.version>2.3.0</xxl-job-core.version>
<!--spring-boot版本-->
<spring-boot.version>2.5.3</spring-boot.version>
<!--日期小插件-->
<joda-time.version>2.10.5</joda-time.version>
<!--google.guava-->
<google.guava.version>30.0-jre</google.guava.version>
</properties>
<!--定义依赖版本锁定-->
<dependencyManagement>
<dependencies>
<!--引入springboot依赖-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>${spring-boot.version}</version>
<type>pom</type>
<scope>import</scope>
</dependency>
<!--引入mybatis场景依赖-->
<dependency>
<groupId>org.mybatis.spring.boot</groupId>
<artifactId>mybatis-spring-boot-starter</artifactId>
<version>${mybatis-spring-boot-starter.version}</version>
</dependency>
<!--pageHelper场景依赖-->
<dependency>
<groupId>com.github.pagehelper</groupId>
<artifactId>pagehelper-spring-boot-starter</artifactId>
<version>${pagehelper-spring-boot-starter.version}</version>
</dependency>
<!--mysql驱动包-->
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>${mysql-driver.version}</version>
</dependency>
<!--shardingjdbc分库分表-->
<dependency>
<groupId>org.apache.shardingsphere</groupId>
<artifactId>sharding-jdbc-spring-boot-starter</artifactId>
<version>${sharding-jdbc.version}</version>
</dependency>
<!--json工具包-->
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>fastjson</artifactId>
<version>${fastjson.version}</version>
</dependency>
<!--druid-boot依赖-->
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>druid-spring-boot-starter</artifactId>
<version>${druid-spring-boot-starter.version}</version>
</dependency>
<!--druid core-->
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>druid</artifactId>
<version>${druid-core-version}</version>
</dependency>
<!--swagger文档依赖-->
<dependency>
<groupId>io.springfox</groupId>
<artifactId>springfox-swagger2</artifactId>
<version>${springfox-swagger2.version}</version>
</dependency>
<dependency>
<groupId>io.springfox</groupId>
<artifactId>springfox-swagger-ui</artifactId>
<version>${springfox-swagger2.version}</version>
</dependency>
<!--引入jwt依赖-->
<dependency>
<groupId>io.jsonwebtoken</groupId>
<artifactId>jjwt</artifactId>
<version>${jjwt.version}</version>
</dependency>
<!-- 导出 excel -->
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>easyexcel</artifactId>
<version>${easyExcel.version}</version>
</dependency>
<!--xxl-job定义任务框架支持-->
<dependency>
<groupId>com.xuxueli</groupId>
<artifactId>xxl-job-core</artifactId>
<version>${xxl-job-core.version}</version>
</dependency>
<!--时间小工具-->
<dependency>
<groupId>joda-time</groupId>
<artifactId>joda-time</artifactId>
<version>${joda-time.version}</version>
</dependency>
<!--引入google的工具集-->
<dependency>
<groupId>com.google.guava</groupId>
<artifactId>guava</artifactId>
<version>${google.guava.version}</version>
</dependency>
</dependencies>
</dependencyManagement>
<build>
<pluginManagement>
<plugins>
<!--Springboot核心插件-->
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<version>${spring-boot.version}</version>
<configuration>
<excludes>
<!--插件运行时排除依赖-->
<exclude>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-configuration-processor</artifactId>
</exclude>
</excludes>
</configuration>
</plugin>
<!--打包跳过test -->
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-surefire-plugin</artifactId>
<configuration>
<skip>${maven.test.skip}</skip>
</configuration>
</plugin>
</plugins>
</pluginManagement>
</build>
注意事项:
1.打包方式:pom
2.通过dependencyManagement、pluginManagement锁定开发中的依赖和插件的版本;
接下来 删除父工程的src
data:image/s3,"s3://crabby-images/5994e/5994ef6f55cfcd8cb44fbfc7810540e32996353c" alt=""
【2】构建stock_backend基础工程
创建stock_parent的maven子工程stock_backend:
data:image/s3,"s3://crabby-images/c6831/c6831b5315b7c90c531117169247ae95cdc56e68" alt=""
data:image/s3,"s3://crabby-images/6ed47/6ed47a5b8ec921164fa16f769a4ac76961058777" alt=""
data:image/s3,"s3://crabby-images/237fa/237fa411e6586057ce3087f98e91f55f8f4e07b6" alt=""
data:image/s3,"s3://crabby-images/045a4/045a42829d422a24e06399fa1a142857d0df2e35" alt=""
【2.1】引入依赖
stock_backend工程被stock_parent父工程聚合,pom配置如下:
<packaging>jar</packaging>
<artifactId>stock_backend</artifactId>
<properties>
<maven.compiler.source>8</maven.compiler.source>
<maven.compiler.target>8</maven.compiler.target>
</properties>
<dependencies>
<!-- 基本依赖 web的场景依赖-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<!--日志-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
<!--mysql-->
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
</dependency>
<!--单元测试-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
<!--lombok-->
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<optional>true</optional>
</dependency>
<!--mybatis整合spring-boot-->
<dependency>
<groupId>org.mybatis.spring.boot</groupId>
<artifactId>mybatis-spring-boot-starter</artifactId>
</dependency>
<!--分页-->
<dependency>
<groupId>com.github.pagehelper</groupId>
<artifactId>pagehelper-spring-boot-starter</artifactId>
</dependency>
<!--druid-->
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>druid-spring-boot-starter</artifactId>
</dependency>
<!--配置提示-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-configuration-processor</artifactId>
<optional>true</optional>
</dependency>
<!--时间小工具-->
<dependency>
<groupId>joda-time</groupId>
<artifactId>joda-time</artifactId>
</dependency>
</dependencies>
<build>
<!--打包名称-->
<finalName>${project.artifactId}</finalName>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
<!-- 打包跳过test -->
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-surefire-plugin</artifactId>
</plugin>
</plugins>
</build>
【2.2】创建公共包结构
找到资料中的公共包结构
data:image/s3,"s3://crabby-images/82c67/82c67ff05cc7e2c19585a80989931066401660aa" alt=""
复制com
data:image/s3,"s3://crabby-images/8a23a/8a23ab6b6a4cd3a1cda6094447695185c015e7f3" alt=""
进入到项目的物理路径(IntelliJ IDEA 2020.1.3 x64)
data:image/s3,"s3://crabby-images/c446b/c446b839717db29e06ad742816f7df0cb049b8b6" alt=""
不同的idea进行项目物理环境的选项不同
data:image/s3,"s3://crabby-images/3fb7b/3fb7b2b635725be102977a43a2f17de2afcb6a6b" alt=""
data:image/s3,"s3://crabby-images/36403/364034d9f0b9a7d29b4dd1949a52c818f3b158da" alt=""
把刚才复制过来的com复制到main/java下面
data:image/s3,"s3://crabby-images/f929a/f929a6a7fd62f35e5513b6aa53b02af2bba9b63f" alt=""
这样就能得到目录结构(里面为空目录) 这样做的目的是不能一次次创建目录结构
data:image/s3,"s3://crabby-images/d43c5/d43c5d8c6fc9752df8cb8f7654e8a95c53d1faf0" alt=""
【2.3】快速生成yml和main启动类的插件JBL SpringbootAppGen
安装
data:image/s3,"s3://crabby-images/0da48/0da488c67ffdf5eb2e86af3301abe4d42fd73c3b" alt=""
data:image/s3,"s3://crabby-images/2857c/2857c5a45abfd92c74a181ae287da7a4ae6a90ac" alt=""
使用
data:image/s3,"s3://crabby-images/a6df5/a6df50479291d7c727861f7939d5b82bccdedbb3" alt=""
data:image/s3,"s3://crabby-images/95ba6/95ba6ad343c19a06957bc764e8d52b27e28da327" alt=""
生成效果
data:image/s3,"s3://crabby-images/2a35b/2a35b35bfe1ebfd0e555ae58633d0f45c99a6a35" alt=""
data:image/s3,"s3://crabby-images/59e65/59e6586fc4cab47b8cce4ba76fe548cac8b4f1e0" alt=""
【2.4】配置yml
# web定义
server:
port: 8081 #指定当前端口号
spring:
# 配置mysql数据源
datasource:
druid:
username: root
password: root
url: jdbc:mysql://127.0.0.1:3306/stock_db?useUnicode=true&characterEncoding=UTF-8&allowMultiQueries=true&useSSL=false&serverTimezone=Asia/Shanghai
driver-class-name: com.mysql.jdbc.Driver
# 初始化时建立物理连接的个数。初始化发生在显示调用 init 方法,或者第一次 getConnection 时
initialSize: 6
# 最小连接池数量
minIdle: 2
# 最大连接池数量
maxActive: 20
# 获取连接时最大等待时间,单位毫秒。配置了 maxWait 之后,缺省启用公平锁,
# 并发效率会有所下降,如果需要可以通过配置 useUnfairLock 属性为 true 使用非公平锁。
maxWait: 60000
# 配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒
timeBetweenEvictionRunsMillis: 60000
# 配置一个连接在池中最小生存的时间,单位是毫秒
minEvictableIdleTimeMillis: 300000
# 用来检测连接是否有效的 sql 因数据库方言而差, 例如 oracle 应该写成 SELECT 1 FROM DUAL
validationQuery: SELECT 1 FROM DUAL
# 建议配置为 true,不影响性能,并且保证安全性。申请连接的时候检测,
# 如果空闲时间大于 timeBetweenEvictionRunsMillis,执行 validationQuery 检测连接是否有效。
testWhileIdle: true
# 申请连接时执行 validationQuery 检测连接是否有效,做了这个配置会降低性能。
testOnBorrow: false
# 归还连接时执行 validationQuery 检测连接是否有效,做了这个配置会降低性能。
testOnReturn: false
# 是否自动回收超时连接
poolPreparedStatements: true
maxPoolPreparedStatementPerConnectionSize: 20
# 配置监控统计拦截的filters,去掉后监控界面sql无法统计,'wall'用于防火墙
filters: stat,wall
# 通过connectProperties属性来打开mergeSql功能;慢SQL记录
connectionProperties: druid.stat.mergeSql=true;druid.stat.slowSqlMillis=5000
# 配置mybatis
mybatis:
type-aliases-package: com.itheima.stock.pojo #指定的包下类名取别名
mapper-locations: classpath:mapper/*.xml
configuration:
map-underscore-to-camel-case: true #开启驼峰映射
cache-enabled: false #使全局的映射器启用或禁用缓存。
lazy-loading-enabled: false #全局启用或禁用延迟加载。当禁用时,所有关联对象都会即时加载。
aggressive-lazy-loading: true #当启用时,有延迟加载属性的对象在被调用时将会完全加载任意属性。否则,每种属性将会按需要加载。
# pagehelper配置
pagehelper:
helper-dialect: mysql #指定分页数据库类型(方言)
reasonable: true #合理查询超过最大也,则查询最后一页
support-methods-arguments: true # 支持通过Mapper接口参数来传递分页参数,默认false
params: pacount=countSql # POJO或者Map中发现了countSql属性,就会作为count参数使用
returnPageInfo: check # always总是返回PageInfo类型,check检查返回类型是否为PageInfo,none返回Page
因为上面配置了少了一个mapper 所以我们现 在创建这个包
data:image/s3,"s3://crabby-images/4e13a/4e13ac23909c06c92b57449914dd991e083655aa" alt=""
data:image/s3,"s3://crabby-images/88972/88972afdc50d24bbf39afdc57f785d1c7fcc1c45" alt=""
data:image/s3,"s3://crabby-images/fc435/fc435150b09503c2a8450791e1853663a276392e" alt=""
注意:
第一点
url: jdbc:mysql://127.0.0.1:3306/stock_db?useUnicode=true&characterEncoding=UTF-8&allowMultiQueries=true&useSSL=false&serverTimezone=Asia/Shanghai
这里面的url 如果是连接的是linux的 刚要写成Linux上的Ip 如果连接的是本地(win)则url 刚要写成
jdbc:mysql:///数据库名称或jdbc:mysql://localhost:3306/数据库名称jdbc:mysql://127.0.0.1:3306/数据库名称
第二点 本项目用到的mysql为5.7 不是8.0 所以把
driver-class-name: com.mysql.cj.jdbc.Driver 把cj删除
补充知识
driver-class-name: com.mysql.cj.jdbc.Driver mysql8.0
driver-class-name: com.mysql.jdbc.Driver mysql5.7
第三点 接口扫描
IntelliJ IDEA 2020.1.3 x64版本复制路径写法
data:image/s3,"s3://crabby-images/18cc8/18cc8b0f36a0929a2c6c7a331c6d6049d0935cd5" alt=""
视频中的idea版本复制路径写法
data:image/s3,"s3://crabby-images/9eca4/9eca4e85d01f2016a5f1022c2b1521ffcabfbc14" alt=""
data:image/s3,"s3://crabby-images/6c49d/6c49d2cc599138ec516a47189f5309372dcc5b43" alt=""
然后把复制过来的路径给放在下图所示
data:image/s3,"s3://crabby-images/d3c3f/d3c3f06ebcc3f96ffecc572d03aaef37f75dfa25" alt=""
【2.5】定义main启动类
package com.itheima.stock;
import org.mybatis.spring.annotation.MapperScan;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
@SpringBootApplication
@MapperScan("com.itheima.stock.mapper")
public class StockApp {
public static void main(String[] args) {
SpringApplication.run(StockApp.class, args);
}
}
【2.6】定义web测试接口
data:image/s3,"s3://crabby-images/82704/82704499b94531ffeb45a692fc77dc1338b93a49" alt=""
package com.itheima.stock.controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
@RequestMapping("/api")
public class UserController {
@GetMapping("/test")
public String getName(){
return "itheima";
}
}
启动:
data:image/s3,"s3://crabby-images/c7580/c7580db47ebb28c5fe903d24965c46d47f6a6b65" alt=""
注意:这里面启动时 要注意你连接的mysql是linux的还是本地的 要注意他们的账户和密码否正确
启动成功
data:image/s3,"s3://crabby-images/417b1/417b1b8a304faca95635df5a44534f61d379c646" alt=""
在浏览器上测试能够获取到
data:image/s3,"s3://crabby-images/0351b/0351bcc0a8d9eb507d1b55825e4335e7f050e3e3" alt=""
在做接口调试的时候 在这里面建议用postman 因为用浏览器只能用到git
演示使用课程资料中的------股票API接口测试.json
data:image/s3,"s3://crabby-images/ee8f0/ee8f0c8e47924667cb9b22ee241f8d8b7f740a7a" alt=""
data:image/s3,"s3://crabby-images/77e56/77e56cd048a753dfd3b1af9401afa56f48201fb8" alt=""
data:image/s3,"s3://crabby-images/9b798/9b798bebde0321399e8ceb1692163c2b6d56705a" alt=""
data:image/s3,"s3://crabby-images/10790/107902a185655cfbbc96834c651bf8faf8931fc2" alt=""
data:image/s3,"s3://crabby-images/ffc4d/ffc4d3232dfe203c129c5b2fdd7fd9ba5c3142f8" alt=""
至此,工程基础环境搭建完毕!
2.3 快速构建mybatis业务开发环境
【1】安装插件mybatisX工具
data:image/s3,"s3://crabby-images/53eda/53eda34aa501d6ebee129b82107f5d530e99e080" alt=""
我们可借助mybatisX工具生成基础代码,步骤如下:
第一步:通过idea自带的database组件连接数据库:
data:image/s3,"s3://crabby-images/3423f/3423fb02c9ed08dddffc6bb9bc9ca1aa24039baa" alt=""
data:image/s3,"s3://crabby-images/d470f/d470f4cacd0e0fd0d7b74d04d4c0ecf33a4b8f30" alt=""
data:image/s3,"s3://crabby-images/d14d0/d14d0549939e4d499ec1a06a957f02ced0591532" alt=""
全选11张表 点击第一张表 按住Shist 在点击最后一张表
data:image/s3,"s3://crabby-images/92acc/92acce351a2129361070b68b579c8cb02f7bcc67" alt=""
右击
data:image/s3,"s3://crabby-images/79d47/79d4726033c9f13a3256ecf829547214d6522845" alt=""
第二步:配置pojo实体类选项
data:image/s3,"s3://crabby-images/3ede4/3ede4fbcbcd85a0d04cfedfc2c3c3a593439b8fa" alt=""
data:image/s3,"s3://crabby-images/d9fe8/d9fe8ee4f7e843315590d50d43b3fba532de8154" alt=""
data:image/s3,"s3://crabby-images/1dbb1/1dbb14447651533dce5fd153c036f1d9a24be7f1" alt=""
data:image/s3,"s3://crabby-images/f5ac5/f5ac58009f5e45c2c6e0a8c4e9db4b0870988e82" alt=""
【2】集成mybatis的工程结构
data:image/s3,"s3://crabby-images/7964e/7964e3c4738a56092b93bceb9a2c660aebf10aaf" alt=""
【3】环境整体测试
目的:我们通过一个简单的web接口访问数据库,验证工程搭建情况
接口功能说明:查询所有上市公司主营业务数据
接口url:/api/quot/stock/business/all
【3.1】定义mapper接口方法
StockBusinessMapper接口和xml定义查询所有股票业务信息的接口方法:
List<StockBusiness> findall();
data:image/s3,"s3://crabby-images/5127a/5127a9c4c54f59bce8344f51ccb5bc6c83151c3b" alt=""
<select id="findall" resultMap="BaseResultMap">
select
<include refid="Base_Column_List" />
from stock_business
</select>
data:image/s3,"s3://crabby-images/a57b3/a57b3da976b67774fec55a1a7d05c1afba82b12d" alt=""
【3.2】定义服务接口及实现
定义服务接口:
data:image/s3,"s3://crabby-images/05e41/05e4199548d45562e407d7334f43da44fa7090e6" alt=""
data:image/s3,"s3://crabby-images/b58db/b58db51b86066bc5dc9b55bbe08ece956991d520" alt=""
package com.itheima.stock.service;
import com.itheima.stock.pojo.StockBusiness;
import java.util.List;
/*定义股票服务接口*/
public interface StockService {
/*查询所有主营业务信息*/
List<StockBusiness> findAll();
}
data:image/s3,"s3://crabby-images/05e66/05e667050c0503be66c268775b9cdb6c8c3e232a" alt=""
定义服务接口实现:
Alt+Enter
data:image/s3,"s3://crabby-images/79a54/79a54e1bbf31cd5df78a5d936b6679bb3a460e3a" alt=""
data:image/s3,"s3://crabby-images/9db7b/9db7b7d92879c2b237c03dd23815444ceda8c6a5" alt=""
data:image/s3,"s3://crabby-images/9392f/9392f4b2c2856f21eb1da703b9e3432ffb4f569a" alt=""
data:image/s3,"s3://crabby-images/b3c9c/b3c9c0cf5c7e80783579da12f9fdddd3bc4b6067" alt=""
package com.itheima.stock.service.impl;
import com.itheima.stock.mapper.StockBusinessMapper;
import com.itheima.stock.pojo.StockBusiness;
import com.itheima.stock.service.StockService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import java.util.List;
@Service("stockService")
public class StockServiceImpl implements StockService {
@Autowired
private StockBusinessMapper stockBusinessMapper;
@Override
public List<StockBusiness> findAll() {
return stockBusinessMapper.findAll();
}
}
【3.3】定义web访问接口
data:image/s3,"s3://crabby-images/48e25/48e25327897230b7a0b56a1d94882187696d0b9a" alt=""
package com.itheima.stock.controller;
import com.itheima.stock.pojo.StockBusiness;
import com.itheima.stock.service.StockService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import java.util.List;
@RestController
@RequestMapping("/api/quot")
public class StockController {
@Autowired
private StockService stockService;
@GetMapping("/stock/business/all")
public List<StockBusiness> findAllBusiness(){
return stockService.findAll();
}
}
data:image/s3,"s3://crabby-images/dc510/dc510f3f8fbfce34c42192161f6cdc3c13241221" alt=""
【3.4】启动项目测试
data:image/s3,"s3://crabby-images/342b5/342b5251715d9e97abbc144546dae3b46bbeef78" alt=""
成功启动
data:image/s3,"s3://crabby-images/02e32/02e323534106067e6ae9ebec7232cf3fc00871b7" alt=""
data:image/s3,"s3://crabby-images/16f97/16f971aa4dd9277823b6b53ccb39407bb2716d23" alt=""
data:image/s3,"s3://crabby-images/34fd6/34fd669d7bf1f57181edea7e39933f9b1144bb4c" alt=""
data:image/s3,"s3://crabby-images/91456/914565c8eff3b5623b470b2aa7b6bc3129fc71a5" alt=""
data:image/s3,"s3://crabby-images/be698/be69878f649626e05a99009ed76df2784891292f" alt=""
至此,后台基本开发环境构建完毕!
3.前端开发环境搭建
3.1 前端环境准备
【1】node安装
前端node版本:
data:image/s3,"s3://crabby-images/78ecb/78ecbda0fb327c024700f28515c0a705f13e0d82" alt=""
详见资料:
data:image/s3,"s3://crabby-images/427a8/427a81604d3537c66d325eed8ef550d3e777a378" alt=""
【2】vs导入前端代码
资料:day01\资料\前端资料\stock_front_admin
使用vscode打开工程:
data:image/s3,"s3://crabby-images/796b1/796b140bcd746109fc5db9eb4776a0d71a003796" alt=""
data:image/s3,"s3://crabby-images/deaad/deaadd731ba0240a3e182727421c5f3f4f37fed1" alt=""
效果如下:
data:image/s3,"s3://crabby-images/1bc90/1bc90e000fed3bf0e495d0193900973fae6a770c" alt=""
【3】前端工程启动
data:image/s3,"s3://crabby-images/ff60c/ff60c463bf3ed5171aea0c32a898663c296302f4" alt=""
data:image/s3,"s3://crabby-images/dd9be/dd9be5935f4f8a872ff30a03fa1c6ef2621bba21" alt=""
data:image/s3,"s3://crabby-images/bb8cc/bb8cc00f46ec68263e4d3b05598b64293beacbdb" alt=""
npm run dev
data:image/s3,"s3://crabby-images/230c0/230c0954059b0ea5aebd8fef6d3b5ea6465169ba" alt=""
这里面是8080 但是我们后端是8081 所以这里面要注意一下
注意事项:如果启动报错,重新npm install 或者cnpm install
【4】页面效果
data:image/s3,"s3://crabby-images/2a562/2a562f3b04cbad8f0c49cda424739c809c0c417b" alt=""
3.2 前后端分离跨域问题
在前面的知识中,我们已经了解到项目进行前后端分离后,存在跨域问题,只需在前端进行简单配置,即可解决该问题;
【1】前后端跨域配置
在stock_front_admin\src\settings.js文件下配置跨域:
data:image/s3,"s3://crabby-images/e2a56/e2a56eb23d6d5c6be25902d39fa35ebc509e0621" alt=""
devServer: {
port: 8080,
host: '127.0.0.1',// 开发host
open:true,// 是否启动时打开浏览器
disableHostCheck: true, // 映射外网时,需要设置为true
/**
* 域名,他将会基于 window.location来链接服务器,需要使用public配置
* dev-server被代理到nginx中配置的 itheima.com
*/
public: "127.0.0.1:8080",//itheima.com
publicPath:'/',
compress:true,
overlay: {// 是否在浏览器全屏显示错误与警告
warnings: false,
errors:true
},
proxy: {// 跨域请求配置
"/api": {
secure: false,// 关闭安全检测,默认请求 https
//target: "http://192.168.188.131:8081",
target: "http://localhost:8081",
changeOrigin: true,
// pathRewrite: {"^/api" : ""},
}
},
},
【2】前后端交互数据格式
前端与后端使用json格式进行交互。
4.登录功能实现
4.1 需求分析
1)页面原型效果
data:image/s3,"s3://crabby-images/f2180/f2180e3ebe21ce46b43bd4bf9cd6b20532e2613d" alt=""
2)相关的表结构
sys_user表如下:
data:image/s3,"s3://crabby-images/12176/12176b989e8b0a7f77984818977ff5c79ae51cba" alt=""
3)访问接口定义
请求接口:/api/login
请求方式:POST
请求数据示例:
{
username:'zhangsan',//用户名
password:'666',//密码
code:'1234' //校验码
}
响应数据:
{
"code": 1,//成功1 失败0
"data": {
"id":"1237365636208922624",
"username":"zhangsan",
"nickName":"xiaozhang",
"phone":"1886702304"
}
}
4)封装请求和响应vo
请求vo封装:
data:image/s3,"s3://crabby-images/0ad5b/0ad5b4ffa25b0ab53bdd082fe5557bf82c505886" alt=""
data:image/s3,"s3://crabby-images/eb19c/eb19cddeb865da0ad3389a5359988650f62374e9" alt=""
data:image/s3,"s3://crabby-images/9d91a/9d91a59a18b84746e1bd7858763fe06b60a07354" alt=""
package com.itheima.stock.vo.req;
import lombok.Data;
/*用户登录请求vo*/
@Data
public class LoginReqVo {
/**
* 用户名
*/
private String username;
/**
* 密码
*/
private String password;
/**
* 前端发送的验证码
*/
private String code;
}
}
响应vo:
data:image/s3,"s3://crabby-images/09371/093710f35ab199abab09809a9a1e208b780096cc" alt=""
把资料中的LoginRespVo复制到idea中
data:image/s3,"s3://crabby-images/d26eb/d26eb09e2bda856e4e6f4bae441ff5da85f759aa" alt=""
data:image/s3,"s3://crabby-images/6eebf/6eebf7c61cd35b1a11d5b9f9ae54f37bc60a1803" alt=""
package com.itheima.stock.vo.resp;
import lombok.AllArgsConstructor;
import lombok.Builder;
import lombok.Data;
import lombok.NoArgsConstructor;
/**
* @author by itheima
* @Date 2021/12/24
* @Description 登录后响应前端的vo
*/
@Data
@NoArgsConstructor
@AllArgsConstructor
@Builder
public class LoginRespVo {
/**
* 用户ID
*/
private String id;
/**
* 电话
*/
private String phone;
/**
* 用户名
*/
private String username;
/**
* 昵称
*/
private String nickName;
}
定义公共响应vo:
把资料中的R复制过来
data:image/s3,"s3://crabby-images/4896c/4896ce7ed5ddcd6d2c4e89b8764d0b8134e8956c" alt=""
data:image/s3,"s3://crabby-images/f9012/f9012e34c0553acbcaf3617d7c0873c0f9c154d4" alt=""
package com.itheima.stock.vo.resp;
import com.fasterxml.jackson.annotation.JsonInclude;
import java.io.Serializable;
/**
* 返回数据类
* @param <T>
*/
//保证序列化json的时候,如果是null的对象,key也会消失
@JsonInclude(JsonInclude.Include.NON_NULL)
public class R<T> implements Serializable {
private static final long serialVersionUID = 7735505903525411467L;
// 成功值
private static final int SUCCESS_CODE = 1;
// 失败值
private static final int ERROR_CODE = 0;
//状态码
private int code;
//消息
private String msg;
//返回数据
private T data;
private R(int code){
this.code = code;
}
private R(int code, T data){
this.code = code;
this.data = data;
}
private R(int code, String msg){
this.code = code;
this.msg = msg;
}
private R(int code, String msg, T data){
this.code = code;
this.msg = msg;
this.data = data;
}
public static <T> R<T> ok(){
return new R<T>(SUCCESS_CODE,"success");
}
public static <T> R<T> ok(String msg){
return new R<T>(SUCCESS_CODE,msg);
}
public static <T> R<T> ok(T data){
return new R<T>(SUCCESS_CODE,data);
}
public static <T> R<T> ok(String msg, T data){
return new R<T>(SUCCESS_CODE,msg,data);
}
public static <T> R<T> error(){
return new R<T>(ERROR_CODE,"error");
}
public static <T> R<T> error(String msg){
return new R<T>(ERROR_CODE,msg);
}
public static <T> R<T> error(int code, String msg){
return new R<T>(code,msg);
}
public int getCode(){
return code;
}
public String getMsg(){
return msg;
}
public T getData(){
return data;
}
}
定义响应状态枚举:
把资料中的ResponseCode复制到idea中
data:image/s3,"s3://crabby-images/9ef62/9ef62700f8701c950d06b5f69b1511e3ac2aa50a" alt=""
data:image/s3,"s3://crabby-images/62d41/62d411841f665798ef223738d9b41ab931c98070" alt=""
package com.itheima.stock.common.enums;
/**
* @author by itheima
* @Date 2021/12/21
* @Description
*/
public enum ResponseCode{
ERROR(0,"操作失败"),
SUCCESS(1,"操作成功"),
DATA_ERROR(0,"参数异常"),
NO_RESPONSE_DATA(0,"无响应数据"),
SYSTEM_VERIFY_CODE_NOT_EMPTY(0,"验证码不能为空"),
SYSTEM_VERIFY_CODE_ERROR(0,"验证码错误"),
SYSTEM_USERNAME_NOT_EMPTY(0,"账号不能为空"),
SYSTEM_USERNAME_NOT_EXISTS(0,"账号不存在"),
SYSTEM_USERNAME_EXPIRED(0,"账户过期"),
SYSTEM_USERNAME_LOCKED(0,"账户被锁"),
SYSTEM_USERNAME_DISABLED(0,"账户被禁用"),
SYSTEM_PASSWORD_ERROR(0,"账号或密码错误"),
SYSTEM_PASSWORD_EXPIRED(0,"密码过期"),
SYSTEM_USERNAME_OFFLINE(0,"已下线,请重新登录"),
SYSTEM_ERROR(0,"系统异常请稍后再试"),
ACCOUNT_EXISTS_ERROR(0,"该账号已存在"),
TOKEN_ERROR(2,"用户未登录,请先登录"),
NOT_PERMISSION(3,"没有权限访问该资源"),
TOKEN_NOT_NULL(-1,"token 不能为空"),
TOKEN_NO_AVAIL(-1,"token无效或过期"),
TOKEN_PAST_DUE(-1,"登录失效,请重新登录"),
TOKEN_EXISTS(-1,"账号异地登录,你已被迫退出"),
OPERATION_MENU_PERMISSION_CATALOG_ERROR(0,"操作后的菜单类型是目录,所属菜单必须为默认顶级菜单或者目录"),
OPERATION_MENU_PERMISSION_MENU_ERROR(0,"操作后的菜单类型是菜单,所属菜单必须为目录类型"),
OPERATION_MENU_PERMISSION_BTN_ERROR(0,"操作后的菜单类型是按钮,所属菜单必须为菜单类型"),
OPERATION_MENU_PERMISSION_URL_NOT_NULL(0,"菜单权限的url不能为空"),
OPERATION_MENU_PERMISSION_URL_PERMS_NULL(0,"菜单权限的标识符不能为空"),
OPERATION_MENU_PERMISSION_URL_METHOD_NULL(0,"菜单权限的请求方式不能为空"),
OPERATION_MENU_PERMISSION_URL_CODE_NULL(0,"菜单权限的按钮标识不能为空"),
OPERATION_MENU_PERMISSION_UPDATE(0,"操作的菜单权限存在子集关联不允许变更"),
ROLE_PERMISSION_RELATION(0, "该菜单权限存在子集关联,不允许删除"),
OLD_PASSWORD_ERROR(0,"旧密码不匹配");
private int code;
private String message;
ResponseCode(int code, String message) {
this.code = code;
this.message = message;
}
public int getCode() {
return 0;
}
public String getMessage() {
return null;
}
}
说明:上述的vo对象直接在:今日指数\day01\资料\vo导入即可;
4.2 登录功能开发实现
stock_backend导入依赖资源
<!--apache工具包-->
<dependency>
<groupId>org.apache.commons</groupId>
<artifactId>commons-lang3</artifactId>
</dependency>
<!--密码加密和校验工具包-->
<dependency>
<groupId>org.springframework.security</groupId>
<artifactId>spring-security-crypto</artifactId>
</dependency>
<!--工具包-->
<dependency>
<groupId>com.google.guava</groupId>
<artifactId>guava</artifactId>
</dependency>
data:image/s3,"s3://crabby-images/f21ff/f21ff6eba8f5bceb818cecc940771d215a1dfc7e" alt=""
配置密码加密服务
package com.itheima.stock.config;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder;
import org.springframework.security.crypto.password.PasswordEncoder;
/*定义公共配置类*/
@Configuration
public class CommonConfig {
/**
* 密码加密器 定义密码加密器和解密器 bean
* BCryptPasswordEncoder方法采用SHA-256对密码进行加密
* @return
*/
@Bean
public PasswordEncoder passwordEncoder(){
return new BCryptPasswordEncoder();
}
}
data:image/s3,"s3://crabby-images/2324c/2324c32ebdf64cbad4929908a528da958950f0ca" alt=""
密码加密测试:
登录接口方法定义
package com.itheima.stock.controller;
import com.itheima.stock.vo.req.LoginReqVo;
import com.itheima.stock.vo.resp.LoginRespVo;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;
@RestController
@RequestMapping("/api")
public class UserController {
/* @GetMapping("/test")
public String getName(){
return "itheima";
}*/
@Autowired
private UserService userService;
/**
* 用户登录功能实现
* @param vo
* @return
*/
@PostMapping("/login")
public R<LoginRespVo> login(@RequestBody LoginReqVo vo){
return userService.login(vo);
}
}
data:image/s3,"s3://crabby-images/710d5/710d552ab9274907e69a08083c422b6890ce6909" alt=""
4)定义登录服务接口和实现
服务接口:
Alt+Enter 选择以下图上所示
data:image/s3,"s3://crabby-images/ea95d/ea95dcaa1d4b2a8d970a5a22a08470163680909d" alt=""
data:image/s3,"s3://crabby-images/eb2e9/eb2e9ebd816ab2aff37fabfb1e66b34cf54b610b" alt=""
data:image/s3,"s3://crabby-images/20f7f/20f7f1282c53eb864986564007e60d1d2b9c8074" alt=""
data:image/s3,"s3://crabby-images/640af/640afa04224ac155ce8509c48cbe96d035d894b7" alt=""
data:image/s3,"s3://crabby-images/ca79a/ca79ad5081f4de7af0481227b23dbb7bfe33ec85" alt=""
接口服务实现:
data:image/s3,"s3://crabby-images/8393c/8393c70a26e080ced80f16d5697281b9a3aad2b5" alt=""
Alt+Enter
data:image/s3,"s3://crabby-images/0baaf/0baaf94644c63deab6ea0724009f5c1063e6173d" alt=""
data:image/s3,"s3://crabby-images/237d0/237d04f66f4d46a35c26e6d50196c61715d26d0a" alt=""
data:image/s3,"s3://crabby-images/0fdab/0fdab680c19fb12f2e3cc0c569bb71cf6810b955" alt=""
data:image/s3,"s3://crabby-images/4825d/4825d59d4e0608297535714c7388de89b565e7b7" alt=""
package com.itheima.stock.service.impl;
import com.google.common.base.Strings;
import com.itheima.stock.mapper.SysUserMapper;
import com.itheima.stock.pojo.SysUser;
import com.itheima.stock.service.UserService;
import com.itheima.stock.vo.req.LoginReqVo;
import com.itheima.stock.vo.resp.LoginRespVo;
import com.itheima.stock.vo.resp.R;
import com.itheima.stock.vo.resp.ResponseCode;
import org.springframework.beans.BeanUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.security.crypto.password.PasswordEncoder;
import org.springframework.stereotype.Service;
@Service("userService")
public class UserServiceImpl implements UserService {
@Autowired
private SysUserMapper sysUserMapper;
@Autowired
private PasswordEncoder passwordEncoder;
@Override
public R<LoginRespVo> login(LoginReqVo vo) {
//1.判断vo是否存在 或者用户名是否存在或都密码是否存在
if (vo == null || Strings.isNullOrEmpty(vo.getUsername()) || Strings.isNullOrEmpty(vo.getPassword())) {
return R.error(ResponseCode.DATA_ERROR.getMessage());
}
//2. 根据用户名判断用户是否存在
SysUser userInfo = sysUserMapper.findUserInfoByUserName(vo.getUsername());
if (userInfo == null) {
return R.error(ResponseCode.DATA_ERROR.getMessage());
}
//3.判断密码,不匹配
if (!passwordEncoder.matches(vo.getPassword(), userInfo.getPassword())) {
return R.error(ResponseCode.SYSTEM_PASSWORD_ERROR.getMessage());
}
//4.属性赋值 两个类之间属性名称一致
LoginRespVo respVo = new LoginRespVo();
BeanUtils.copyProperties(userInfo,respVo);
return R.ok(respVo);
}
}
测试
package com.itheima.stock;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.security.crypto.password.PasswordEncoder;
@SpringBootTest
public class TestPasswordEncoder {
@Autowired
private PasswordEncoder passwordEncoder;
@Test
public void test1(){
String pwd ="123456";
//加密
String enPwd1 = passwordEncoder.encode(pwd);
String enPwd2 = passwordEncoder.encode(pwd);
System.out.println(enPwd1);
System.out.println(enPwd2);
//验证密码是否正确
boolean r1 = passwordEncoder.matches(pwd, enPwd1);
System.out.println(r1);
boolean r2 = passwordEncoder.matches(pwd, enPwd2);
System.out.println(r2);
}
}
data:image/s3,"s3://crabby-images/47b9f/47b9f65c6e68db2820f8f31c1edd33e5b0620122" alt=""
data:image/s3,"s3://crabby-images/1d564/1d564966f5ed6926daa03f25f22174bde71d511b" alt=""
5)Postman测试
data:image/s3,"s3://crabby-images/86b3c/86b3cd63a8c59a0779dcefbd1efb3020d229ffa6" alt=""
data:image/s3,"s3://crabby-images/8dda2/8dda2abad87d26140100eea2645aa95be49e2413" alt=""
data:image/s3,"s3://crabby-images/798f5/798f5ff487c29812886ee65bdfc3f5cda23e36f1" alt=""
data:image/s3,"s3://crabby-images/1d321/1d32104287bce4ee5fca7bb3b20741e58bb214c4" alt=""
data:image/s3,"s3://crabby-images/fcb1e/fcb1eefe1fcd468fc3571ba3d698682fcb31e017" alt=""
data:image/s3,"s3://crabby-images/69d2c/69d2cd69cd6f8ab144b43776c7d3c6356f229d06" alt=""
data:image/s3,"s3://crabby-images/b3ed5/b3ed54884fc3ec8dac202740bba9953f9fdc6b40" alt=""