本文介绍: 作为一名软件开发工程师,我们需要了解在软件开发过程中的开发流程, 以及软件开发过程中涉及到的岗位角色角色的分工、职责, 并了解软件开发中涉及到的三种软件环境。那么我们先从 软件开发流程角色分工、软件环境 三个方面整体介绍一下软件开发。在开发苍穹外卖这个项目之前,我们需要全方位的来介绍一下当前我们学习这个项目接下来,我们将从项目简介产品原型技术选型三个方面来介绍苍穹外卖这个项目。本项目苍穹外卖)是专门为餐饮企业(餐厅、饭店)定制一款软件产品,包括 系统管理后台小程序应用 两部分。

1. 苍穹外卖项目介绍

1.1 项目介绍

苍穹外卖是专门为餐饮企业(餐厅、饭店)定制的一款软件产品,包括 系统管理后台小程序应用 两部分。其中系统管理后台主要提供给餐饮企业内部员工使用可以对餐厅的分类菜品、套餐、订单员工等进行管理维护,对餐厅的各类数据进行统计,同时也可进行来单语音播报功能小程序端主要提供给消费者使用可以在线浏览菜品添加购物车下单支付催单等。

在这里插入图片描述

项目分为管理用户端

在这里插入图片描述

1). 管理

餐饮企业内部员工使用。 主要功能有:

模块 描述
登录/退出 内部员工必须登录后,才可以访问系统管理后台
员工管理 管理员可以在系统后台员工信息进行管理,包含查询新增编辑禁用功能
分类管理 主要对当前餐厅经营的 菜品分类 或 套餐分类 进行管理维护包含查询新增修改删除功能
菜品管理 主要维护各个分类下的菜品信息,包含查询新增修改删除、启售、停售等功能
套餐管理 主要维护当前餐厅中的套餐信息,包含查询新增修改删除、启售、停售等功能
订单管理 主要维护用户移动端下的订单信息,包含查询、取消、派送、完成,以及订单报表下载功能
数据统计 主要完成对餐厅的各类数据统计,如营业额用户数量、订单等

2). 用户端

移动应用主要提供给消费者使用。主要功能有:

模块 描述
登录/退出 用户需要通过微信授权登录使用小程序进行点餐
点餐菜单 在点餐界面需要展示菜品分类/套餐分类, 并根据当前选择的分类加载其中的菜品信息, 供用户查询选择
点餐-购物车 用户选中的菜品就会加入用户购物车, 主要包含 查询购物车加入购物车删除购物车清空购物车等功能
订单支付 用户选完菜品/套餐后, 可以对购物车菜品进行结算支付, 这时就需要进行订单的支付
个人信息 个人中心页面中会展示当前用户基本信息, 用户可以管理收货地址, 也可以查询历史订单数据

1.2 技术选型

在这里插入图片描述
1). 用户层

项目中在构建系统管理后台前端页面,我们会用到H5、Vue.js、ElementUI、apache echarts(展示图表)等技术。而在构建移动应用时,我们会使用微信小程序。

2). 网关

Nginx一个服务器,主要用来作为Http服务器部署静态资源访问性能高。在Nginx中还有两个比较重要的作用反向代理负载均衡, 在进行项目部署时,要实现Tomcat负载均衡,就可以通过Nginx实现

3). 应用层

SpringBoot快速构建Spring项目, 采用 “约定优于配置” 的思想, 简化Spring项目的配置开发

SpringMVC:SpringMVC是spring框架一个模块springmvcspring无需通过中间整合层进行整合,可以无缝集成

Spring Task: 由Spring提供的定时任务框架

httpclient: 主要实现了对http请求发送

Spring Cache: 由Spring提供的数据缓存框架

JWT: 用于应用程序上的用户进行身份验证标记

阿里云OSS: 对象存储服务,在项目中主要存储文件,如图片等。

Swagger: 可以自动的帮助开发人员生成接口文档,并对接口进行测试

POI: 封装了对Excel表格常用操作

WebSocket: 一种通信网络协议,使客户端服务器之间数据交换更加简单用于项目的来单、催单功能实现

4). 数据

MySQL: 关系数据库, 本项目的核心业务数据都会采用MySQL进行存储。

Redis基于keyvalue格式存储的内存数据库, 访问速度快, 经常使用它做缓存

Mybatis: 本项目持久层将会使用Mybatis开发

pagehelper: 分页插件

spring data redis: 简化java代码操作Redis的API。

5). 工具

git: 版本控制工具, 在团队协作中, 使用该工具对项目中的代码进行管理。

maven: 项目构建工具。

junit单元测试工具,开发人员功能实现完毕后,需要通过junit对功能进行单元测试

postman: 接口测工具,模拟用户发起的各类HTTP请求获取对应响应结果

2. 开发环境搭建

在这里插入图片描述

2.1 前端环境

1). 前端工程基于 nginx

双击 nginx.exe 即可启动 nginx 服务访问端口号为 80

http://localhost:80

在这里插入图片描述
(如果启动不了,可能端口号冲突,在nginx配置文件修改端口号即可)。

2.2 后端环境搭建

后端工程基于 maven 进行项目构建,并且进行分模块开发。

1). 用 IDEA 打开初始工程,了解项目的整体结构

在这里插入图片描述
2). 数据库共11张表,每张表的说明

序号 表名 中文
1 employee 员工
2 category 分类表
3 dish 菜品表
4 dish_flavor 菜品口味表
5 setmeal 套餐表
6 setmeal_dish 套餐菜品关系
7 user 用户表
8 address_book 地址
9 shopping_cart 购物车表
10 orders 单表
11 order_detail 订单明细

3). 导入接口文档
项目是基于后端分离开发的·,所有需要根据接口文档来开发。
在这里插入图片描述

2.3 Git版本控制

使用Git进行项目代码版本控制,具体操作:

1). 创建Git本地仓库

在这里插入图片描述
在这里插入图片描述

在这里插入图片描述

当Idea中出现

在这里插入图片描述
说明本地仓库创建成功

2). 创建Git远程仓库

访问https://gitee.com/新建仓库
在这里插入图片描述

点击 创建
在这里插入图片描述
3). 将本地文件推送到Git远程仓库

  1. 提交文件至本地仓库

    忽略以下类型文件

在这里插入图片描述

开始提交

在这里插入图片描述

中间出现:点击commit

在这里插入图片描述

  1. 添加Git远程仓库地址

    复制远程地址

在这里插入图片描述

添加地址

在这里插入图片描述
在这里插入图片描述

  1. 推送

在这里插入图片描述

成功推送至远程仓库

在这里插入图片描述

2.4 nginx反向代理负载均衡

我们先思考一个问题前端发送请求,是如何请求后端服务的?

前端请求地址http://localhost/api/employee/login
在这里插入图片描述
后端接口地址http://localhost:8080/admin/employee/login
在这里插入图片描述

很明显,两个地址不一致,那是如何请求后端服务的呢?

在这里插入图片描述

1). nginx反向代理

nginx 反向代理就是将前端发送动态请求nginx 转发后端服务器

在这里插入图片描述

为什么直接通过浏览器直接请求后台服务端,需要通过nginx反向代理呢?

nginx 反向代理的好处:

在这里插入图片描述

nginx 反向代理的配置方式

server{
    listen 80;
    server_name localhost;
    
    location /api/{
        proxy_pass http://localhost:8080/admin/; #反向代理
    }
}

proxy_pass:该指令用来设置代理服务器的地址,可以是主机名称,IP地址加端口号等形式。

如上代码的含义是:监听80端口号然后当我们访问 http://localhost:80/api/…/…这样的接口的时候,它会通过 location /api/ {} 这样的反向代理到 http://localhost:8080/admin/上来。

接下来,进到nginx-1.20.2conf打开nginx配置

# 反向代理,处理管理端发送的请求
location /api/ {
	proxy_pass   http://localhost:8080/admin/;
    #proxy_pass   http://webservers/admin/;
}

当在访问http://localhost/api/employee/login,nginx接收到请求后转到http://localhost:8080/admin/,故最终的请求地址为http://localhost:8080/admin/employee/login,和后台服务的访问地址一致。

2). nginx 负载均衡

当如果服务以集群方式进行部署时,那nginx在转发请求到服务器时就需要做相应的负载均衡。其实负载均衡从本质上来说也是基于反向代理来实现的,最终都是转发请求。

nginx 负载均衡的配置方式

upstream webservers{
    server 192.168.100.128:8080;
    server 192.168.100.129:8080;
}
server{
    listen 80;
    server_name localhost;
    
    location /api/{
        proxy_pass http://webservers/admin;#负载均衡
    }
}

upstream:如果代理服务器是一组服务器的话,我们可以使用upstream指令配置后端服务器组。

如上代码的含义是:监听80端口号然后当我们访问 http://localhost:80/api/…/…这样的接口的时候,它会通过 location /api/ {} 这样的反向代理到 http://webservers/admin,根据webservers名称找到一组服务器,根据设置的负载均衡策略(默认轮询)转发到具体的服务器。

upstream后面的名称自定义,但要上下保持一致。

nginx 负载均衡策略

名称 说明
轮询 默认方式
weight 权重方式,默认为1,权重越高,被分配客户端请求就越多
ip_hash 依据ip分配方式,这样每个访客可以固定访问一个后端服务
least_conn 依据最少连接方式,把请求优先分配连接数少的后端服务
url_hash 依据url分配方式,这样相同url会被分配同一个后端服务
fair 依据响应时间方式,响应时间短的服务将会被优先分配

具体配置方式:

轮询

upstream webservers{
    server 192.168.100.128:8080;
    server 192.168.100.129:8080;
}

weight:

upstream webservers{
    server 192.168.100.128:8080 weight=90;
    server 192.168.100.129:8080 weight=10;
}

ip_hash:

upstream webservers{
    ip_hash;
    server 192.168.100.128:8080;
    server 192.168.100.129:8080;
}

least_conn:

upstream webservers{
    least_conn;
    server 192.168.100.128:8080;
    server 192.168.100.129:8080;
}

url_hash:

upstream webservers{
    hash &request_uri;
    server 192.168.100.128:8080;
    server 192.168.100.129:8080;
}

fair:

upstream webservers{
    server 192.168.100.128:8080;
    server 192.168.100.129:8080;
    fair;
}

3.登录功能

问题员工表中密码是明文存储,安全性太低。

在这里插入图片描述

解决思路

  1. 密码加密后存储,提高安全性

在这里插入图片描述

  1. 使用MD5加密方式对明文密码加密

在这里插入图片描述

实现步骤

  1. 修改数据库中明文密码改为MD5加密后的密文
    在这里插入图片描述1.Controller层

sky-server模块中,com.sky.controller.admin.EmployeeController

/**
     * 登录
     *
     * @param employeeLoginDTO
     * @return
     */
    @PostMapping("/login")
    public Result<EmployeeLoginVO> login(@RequestBody EmployeeLoginDTO employeeLoginDTO) {
        log.info("员工登录:{}", employeeLoginDTO);
		//调用service方法查询数据库
        Employee employee = employeeService.login(employeeLoginDTO);

        //登录成功后,生成jwt令牌
        Map<String, Object> claims = new HashMap<>();
        claims.put(JwtClaimsConstant.EMP_ID, employee.getId());
        String token = JwtUtil.createJWT(
                jwtProperties.getAdminSecretKey(),
                jwtProperties.getAdminTtl(),
                claims);

        EmployeeLoginVO employeeLoginVO = EmployeeLoginVO.builder()
                .id(employee.getId())
                .userName(employee.getUsername())
                .name(employee.getName())
                .token(token)
                .build();

        return Result.success(employeeLoginVO);
    }

2.Service层

sky-server模块中,com.sky.service.impl.EmployeeServiceImpl

 /**
        * 员工登录
        *
        * @param employeeLoginDTO
        * @return
        */
      public Employee login(EmployeeLoginDTO employeeLoginDTO) {
        String username = employeeLoginDTO.getUsername();
        String password = employeeLoginDTO.getPassword();

        //1、根据用户名查询数据库中的数据
        Employee employee = employeeMapper.getByUsername(username);

        //2、处理各种异常情况(用户名存在密码不对、账号锁定
        if (employee ==  null) {
            //账号不存在
            throw new AccountNotFoundException(MessageConstant.ACCOUNT_NOT_FOUND);
        }

        //密码比对
        // 进行md5加密,然后再进行比对
        password = DigestUtils.md5DigestAsHex(password.getBytes());


        if (!password.equals(employee.getPassword())) {
            //密码错误
            throw new PasswordErrorException(MessageConstant.PASSWORD_ERROR);
        }

        if (employee.getStatus()  == StatusConstant.DISABLE) {
            //账号被锁定
            throw new AccountLockedException(MessageConstant.ACCOUNT_LOCKED);
        }

        //3、返回实体对象
        return employee;
    }

3.Mapper

sky-server模块中,com.sky.mapper.EmployeeMapper

package com.sky.mapper;

import com.sky.entity.Employee;
import org.apache.ibatis.annotations.Mapper;
import org.apache.ibatis.annotations.Select;

@Mapper
public interface EmployeeMapper {

    /**
     * 根据用户名查询员工
     * @param username
     * @return
     */
    @Select("select * from employee where username = #{username}")
    Employee getByUsername(String username);

}

4. Swagger

4.1 介绍

Swagger 是一个规范完整框架用于生成、描述、调用可视化 RESTful 风格的 Web 服务(https://swagger.io/)。 它的主要作用是:

  1. 使得前后分离开发更加方便,有利于团队协作

  2. 接口的文档在线自动生成,降低后端开发人员编写接口文档的负担

  3. 功能测试

    Spring已经将Swagger纳入自身的标准,建立了Spring-swagger项目,现在叫Springfox。通过在项目中引入Springfox ,即可非常简单快捷的使用Swagger。

knife4j是为Java MVC框架集成Swagger生成Api文档的增强解决方案,前身是swaggerbootstrapui,取名kni4j是希望它能像一把匕首一样小巧,轻量,并且功能强悍!

目前,一般都使用knife4j框架

4.2 使用步骤

  1. 导入 knife4j 的maven坐标

    pom.xml添加依赖

    <dependency>
       <groupId>com.github.xiaoymin</groupId>
       <artifactId>knife4j-spring-boot-starter</artifactId>
    </dependency>
    
  2. 在配置类中加入 knife4j 相关配置

    WebMvcConfiguration.java

    /**
         * 通过knife4j生成接口文档
         * @return
    */
        @Bean
        public Docket docket() {
            ApiInfo apiInfo = new ApiInfoBuilder()
                    .title("苍穹外卖项目接口文档")
                    .version("2.0")
                    .description("苍穹外卖项目接口文档")
                    .build();
            Docket docket = new Docket(DocumentationType.SWAGGER_2)
                    .apiInfo(apiInfo)
                    .select()
                    .apis(RequestHandlerSelectors.basePackage("com.sky.controller"))
                    .paths(PathSelectors.any())
                    .build();
            return docket;
        }
    
  3. 设置静态资源映射,否则接口文档页面无法访问

    WebMvcConfiguration.java

    /**
         * 设置静态资源映射
         * @param registry
    */
    protected void addResourceHandlers(ResourceHandlerRegistry registry) {
            registry.addResourceHandler("/doc.html").addResourceLocations("classpath:/META-INF/resources/");
            registry.addResourceHandler("/webjars/**").addResourceLocations("classpath:/META-INF/resources/webjars/");
    }
    
  4. 访问测试

    接口文档访问路径http://ip:port/doc.html —> http://localhost:8080/doc.html

在这里插入图片描述

4.3 常用注解

通过注解可以控制生成的接口文档,使接口文档拥有更好的可读性,常用注解如下

注解 说明
@Api 用在类上,例如Controller,表示对类的说明
@ApiModel 用在类上,例如entity、DTO、VO
@ApiModelProperty 用在属性上,描述属性信息
@ApiOperation 用在方法上,例如Controller的方法,说明方法的用途作用

接下来,使用上述注解生成可读性更好的接口文档

在sky-pojo模块中

EmployeeLoginDTO.java

@Data
@ApiModel(description = "员工登录时传递数据模型")
public class EmployeeLoginDTO implements Serializable {

    @ApiModelProperty("用户名")
    private String username;

    @ApiModelProperty("密码")
    private String password;

}

EmployeeLoginVo.java

@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
@ApiModel(description = "员工登录返回数据格式")
public class EmployeeLoginVO implements Serializable {

    @ApiModelProperty("主键值")
    private Long id;

    @ApiModelProperty("用户名")
    private String userName;

    @ApiModelProperty("姓名")
    private String name;

    @ApiModelProperty("jwt令牌")
    private String token;

}

在sky-server模块中

EmployeeController.java

/**
 * 员工管理
 */
@RestController
@RequestMapping("/admin/employee")
@Slf4j
@Api(tags = "员工相关接口")
public class EmployeeController {
    /**
     * 登录
     */
    @PostMapping("/login")
    @ApiOperation(value = "员工登录")
    public Result<EmployeeLoginVO> login(@RequestBody EmployeeLoginDTO employeeLoginDTO) 	{
        //.............
    }

    /**
     * 退出
     */
    @PostMapping("/logout")
    @ApiOperation("员工退出")
    public Result<String> logout() {
        return Result.success();
    }

}

启动服务:访问http://localhost:8080/doc.html

在这里插入图片描述

后记
👉👉💕💕美好的一天,到此结束,下次继续努力!欲知后续,请看下回分解写作不易,感谢大家支持!! 🌹🌹🌹

原文地址:https://blog.csdn.net/m0_59230408/article/details/132768205

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任

如若转载,请注明出处:http://www.7code.cn/show_49475.html

如若内容造成侵权/违法违规/事实不符,请联系代码007邮箱suwngjj01@126.com进行投诉反馈,一经查实,立即删除

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注