[项目](美多商城)(数据库课设/软件工程大作业/软件工程课设)项目的创建、配置、说明文档与源码

news2025/1/21 8:47:44

文章目录

  • 💘 相关说明
  • 💘 美多商城前台前端项目创建与配置
    • 💖 项目的创建
    • 💖 配置UI组件库
      • 💝 安装
      • 💝 全局引入与全局注册
      • 💝 局部引入与局部注册
      • 💝 定制主题
    • 💖 配置axios
      • 💝 axios的配置
      • 💝 配置代理
    • 💖 常量声明挂载文件
      • 💝 用户相关
    • 💖 工具文件
  • 💘 美多商城前台前端路由说明
  • 💘 美多商城前台前端Vue挂载自定义变量说明
  • 💘 美多商城前台后端项目创建与配置
    • 💖 项目的创建
    • 💖 去除不需要使用的项目默认子应用与中间件
    • 💖 配置MySQL数据库
    • 💖 配置Redis数据库
    • 💖 配置跨域
    • 💖 子应用的创建
      • 💝 user
    • 💖 常量声明文件
    • 💖 工具文件
      • 💝 发送短信验证码的工具文件
      • 💝 操作数据库的工具文件
      • 💝 与请求处理相关的工具文件
      • 💝 与数据类型转换相关的工具文件
      • 💝 与用户账号相关的工具文件
      • 💝 与日期时间相关的工具文件
      • 💝 与jwt相关的工具文件
  • 💘 美多商城前台后端路由说明(请求路由)
  • 💘 美多商城前台变量与常量说明
    • 💖 与用户相关
      • 💝 变量
      • 💝 常量
    • 💖 与请求和响应相关
      • 💝 常量
      • 💝 响应数据对象的键
    • 💖 用于美多商城服务端功能控制的常量
  • 💘 美多商城数据库表的说明
  • 💘 美多商城数据库SQL语句
    • 💖 数据库的创建
    • 💖 表的创建
    • 💖 表的修改
    • 💖 触发器的创建
    • 💖 视图的创建
  • 💘 美多商城后台前端项目的创建与配置
  • 💘 美多商城后台后端
  • 手机号重复注册
  • 💘💖💝💗💓💞💕💟💔❤️‍🔥🧡💛💚💙


💘 相关说明

本项目采用的为前后端分离,前端使用的框架为Vue2,使用的组件库为Vant2和Element,前端使用的语言为HTML5+CSS3+JavaScript,后端使用的语言为Python,后端使用的框架为Django,使用的数据库为MySQL,使用的缓存数据库为Redis。

项目的前后端源码和数据库SQL脚本源码等我码完再加上。

本项目会使用到的配置基本就是本文写的了,剩下的就是功能的实现和前后端数据交互。这篇文章没写完,应该也不会继续写了。(好吧,其实就是不想写了)

注意点:
在这里插入图片描述
在这里插入图片描述


💘 美多商城前台前端项目创建与配置

💖 项目的创建

美多商城前端使用的框架为Vue2.x,前端搭建的为移动端页面。

创建前端项目的脚手架为vue-cli。

vue create meiduo_move

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

vscode打开项目,删除不需要的文件和代码,并运行测试:

在这里插入图片描述

main.js

import Vue from 'vue'
import App from './App.vue'
import router from './router'

Vue.config.productionTip = false

new Vue({
  router,
  render: h => h(App)
}).$mount('#app')

App.vue

<template>
  <div id="app">
    <h1>MeiDuo Pro</h1>
  </div>
</template>

<script>
export default {
  name: 'App'
}
</script>

<style lang="less">
</style>

router/index.js

import Vue from 'vue'
import VueRouter from 'vue-router'

Vue.use(VueRouter)

const routes = [
]

const router = new VueRouter({
  routes
})

export default router

运行项目测试:

npm run serve

在这里插入图片描述

💖 配置UI组件库

前台前端页面使用UI组件库为Vant UI,Vant的版本为Vant2。

Vant2:https://vant-contrib.gitee.io/vant/v2/#/zh-CN/quickstart

💝 安装

项目中安装Vant2组件库:

npm i vant@latest-v2 -S

安装插件,Vant2自动按需引入:

npm i babel-plugin-import -D

在 babel.config.js 文件中增加配置项 plugins:

  plugins: [
    ['import', {
      libraryName: 'vant',
      libraryDirectory: 'es',
      style: true
    }, 'vant']
  ]
module.exports = {
  presets: [
    '@vue/cli-plugin-babel/preset'
  ],
  plugins: [
    ['import', {
      libraryName: 'vant',
      libraryDirectory: 'es',
      style: true
    }, 'vant']
  ]
}

停止项目重新运行:

npm run serve

💝 全局引入与全局注册

在 src 目录下新建 lib 文件夹用于存放组件库引入的代码文件:

在这里插入图片描述

在lib文件夹中新建vantUI.js文件用于vantUI组件的全局引入与全局注册:

在这里插入图片描述

在vant_ui.js文件中全局按需引入组件,并注册为全局组件:

import Vue from 'vue'

// Vant组件的引入
import { Button } from 'vant'

// Vant组件注册为全局组件
Vue.use(Button)

在main.js文件中导入vant_ui.js文件:

import Vue from 'vue'
import App from './App.vue'
import router from './router'
import '@/lib/vantUI.js' // 引入全局导入与全局注册vant组件的文件

Vue.config.productionTip = false

new Vue({
  router,
  render: h => h(App)
}).$mount('#app')

在App.vue文件中测试:

<template>
  <div id="app">
    <h1>MeiDuo Pro</h1>
    <van-button type="primary">vant按钮</van-button>
  </div>
</template>

<script>
export default {
  name: 'App'
}
</script>

<style lang="less">
</style>

在这里插入图片描述

💝 局部引入与局部注册

局部按需引入组件:

<template>
  <div id="app">
    <h1>MeiDuo Pro</h1>
    <van-button type="primary">vant按钮</van-button>
    <van-rate />
  </div>
</template>

<script>
import {Rate} from 'vant'

export default {
  name: 'App',
  components: {
    [Rate.name]: Rate, // 注册vant组件
  }
}
</script>

<style lang="less">
</style>

在这里插入图片描述

💝 定制主题

配置按需引入vant组件库的样式,在原先的babel.config.js配置文件中的plugins配置项中修改style配置:

// 指定样式路径
style: (name) => `${name}/style/less`,
module.exports = {
  presets: [
    '@vue/cli-plugin-babel/preset'
  ],
  plugins: [
    ['import', {
      libraryName: 'vant',
      libraryDirectory: 'es',
      // 指定样式路径
      style: (name) => `${name}/style/less`,
    }, 'vant']
  ]
}

在src目录下新建less文件夹,用于存放自定义样式的代码文件:

在这里插入图片描述

在less文件夹中,新建vant_theme.less文件,用于自定义组件样式:

在这里插入图片描述

vue-cli 搭建的项目,在 vue.config.js 中进行如下配置:

const { defineConfig } = require('@vue/cli-service')
module.exports = defineConfig({
  transpileDependencies: true,
  css: {
    loaderOptions: {
      less: {
        // 若 less-loader 版本小于 6.0,请移除 lessOptions 这一级,直接配置选项。
        lessOptions: {
          modifyVars: {
            // 通过 less 文件覆盖(文件路径为绝对路径)
            hack: `true; @import "@/less/vant_theme.less";`,
          },
        },
      },
    },
  },
})

重新启动运行项目,接下来即可在vantTheme.less文件中自定义组件的样式。

💖 配置axios

💝 axios的配置

安装axios

npm i axios

在lib文件夹下新建axios.js文件用于配置和挂载axios

在这里插入图片描述

在axios.js文件夹中进行如下配置:

import Vue from "vue"
import axios from "axios"

const req = axios.create({
  baseURL: 'http://127.0.0.1:8000/', // 配置baseURL
  withCredentials: true, // axios 发送请求时允许携带cookie
})

Vue.prototype.req = req // 将req挂载到Vue的原型对象上

在main.js文件中引入axios.js

import Vue from 'vue'
import App from './App.vue'
import router from './router'
import '@/lib/vant_ui.js' // 引入全局导入与全局注册vant组件的文件
import '@/constants/user_constants.js' // 引入与用户相关的常量并挂载到Vue原型上
import '@/lib/axios.js' // 引入axios配置

💝 配置代理

配置代理,实现可以获取到服务器响应回来的set-cookie.

在vue.config.js文件的配置对象中新增devServer配置项:

const { defineConfig } = require('@vue/cli-service')
module.exports = defineConfig({
  transpileDependencies: true,
  css: {
    loaderOptions: {
      less: {
        // 若 less-loader 版本小于 6.0,请移除 lessOptions 这一级,直接配置选项。
        lessOptions: {
          modifyVars: {
            // 通过 less 文件覆盖(文件路径为绝对路径)
            hack: `true; @import "@/less/vant_theme.less";`,
          },
        },
      },
    },
  },
  devServer: {  // 配置代理服务器
    proxy: {
      '/api': {  // '/api' 防止与本地资源冲突
        target: 'http://localhost:8000/',  // 代理服务器向http://127.0.0.1:8000/请求资源
        pathRewrite: { '/api': '' }, // 重写请求路径
        ws: true, // 支持websocket
        changeOrigin: true, // 服务器收到的请求头中的host与服务器一样
      } 
    }
  }
})

修改axios请求的根地址为与页面运行的地址一样:

import Vue from "vue"
import axios from "axios"

const req = axios.create({
  // 记得 api/
  baseURL: 'http://localhost:8080/api/', // 配置baseURL,向代理服务器请求
  withCredentials: true, // axios 发送请求时允许携带cookie
})

Vue.prototype.req = req // 将req挂载到Vue的原型对象上

重启运行项目。

💖 常量声明挂载文件

在src目录下新建constants文件夹用于存放常量声明挂载的文件。

在这里插入图片描述

💝 用户相关

在constants文件夹下新建user_constants.js文件夹用于与用户相关的常量的声明和挂载。

在这里插入图片描述

import Vue from "vue"

Vue.prototype.userConstants = {
  
}

在main.js文件中导入该文件,执行并向Vue原型上挂载。

import Vue from 'vue'
import App from './App.vue'
import router from './router'
import '@/lib/vant_ui.js' // 引入全局导入与全局注册vant组件的文件
import '@/constants/user_constants.js' // 引入与用户相关的常量并挂载到Vue原型上

Vue.config.productionTip = false

new Vue({
  router,
  render: h => h(App)
}).$mount('#app')

💖 工具文件

在src目录下新建utils文件夹用于存放工具文件,在utils中新建utils.js文件用于工具方法的声明。


💘 美多商城前台前端路由说明

  • /container:
    • 用于放置主页、分类界面、购物车界面、用户界面组件的容器的路由
    • 路由命名:container
    • 对应组件:Container.vue
  • /home:
    • /container 下的子路由
    • 完整路由:/container/home
    • 主页界面路由
    • 路由命名:home
    • 对应组件:Home.vue
  • /category:
    • /container 下的子路由
    • 完整路由:/container/category
    • 分类界面路由
    • 路由命名:category
    • 对应组件:Category.vue
  • /cart:
    • /container 下的子路由
    • 完整路由:/container/cart
    • 购物车界面路由
    • 路由命名:cart
    • 对应组件:Cart
  • /user:
    • /container 下的子路由
    • 完整路由:/container/user
    • 用户界面路由
    • 路由命名:user
    • 对应组件:User.vue
  • user/receive/address:
    • 用户收货地址界面路由
    • 路由命名:receiveAddress
    • 对应组件:UserReceiveAddress
  • user/userdetailinfo:
    • 用户详细信息界面路由
    • 路由命名:userdetailinfo
    • 对应组件:UserDetailInfo
  • /user/setting:
    • 用户设置界面路由
    • 路由命名:setting
    • 对应组件:Setting
  • /user/changepassword:
    • 修改界面路由
    • 路由命名: changepassword
    • 对应组件:ChangePassword
  • /login:
    • 登录界面路由
    • 路由命名:login
    • 对应组件:Login.vue
  • /register:
    • 注册界面路由
    • 路由命名:register
    • 对应组件:Register.vue

💘 美多商城前台前端Vue挂载自定义变量说明

  • userConstants
    • 对象类型
    • 该对象上存放与用户相关的常量
    • 声明文件:src/constants/user_constants.js

💘 美多商城前台后端项目创建与配置

美多商城前台后端使用的语言为Python,使用的框架为Django。由于采用的为前后端分离的方式,所以这里不使用Django的模板;同时由于本次主要用户数据库课设,而Django对于数据库的一些功能,是在代码逻辑层面进行实现的,即对于数据库中数据表的一些设置不会在对应的数据库中有很好的体现,如设置外键时表的级联删除,Django实现该功能就是在代码逻辑层面进行实现的,在数据库中数据表的设置中显示的为no action。

使用Python中的Django是由于Python的Web框架目前只学了Django。(交作业要紧 )

💖 项目的创建

使用的虚拟环境工具为virtualenvwrapper。

创建用于本次项目的虚拟环境:

mkvirtualenv -p D:\python3.9.7\python.exe meiduo_environment

在这里插入图片描述

选择本次项目后端存放的文件夹路径,进入新创建的虚拟环境:

workon meiduo_environment

在新创建的虚拟环境中安装Django:

pip install django -i https://pypi.tuna.tsinghua.edu.cn/simple

在这里插入图片描述

进行本次美多商城前台后端项目的创建:

django-admin startproject meiduo_move_server

在这里插入图片描述

在pycharm中打开项目,试运行:

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

至此项目创建成功。

💖 去除不需要使用的项目默认子应用与中间件

由于本项目采用前后端分离的方式,同时不使用Django的模型与模板,所以将默认子应用admin(内置的后台管理系统)、auth(内置的用户认证系统)、contenttype(用于记录项目中model元数据)进行注释。

INSTALLED_APPS = [
    # 'django.contrib.admin',
    # 'django.contrib.auth',
    # 'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
]

由于不需要使用项目内置的用户认证系统,所以将AuthenticationMiddleware中间件进行注释。

MIDDLEWARE = [
    'django.middleware.security.SecurityMiddleware',
    'django.contrib.sessions.middleware.SessionMiddleware',
    'django.middleware.common.CommonMiddleware',
    # 'django.middleware.csrf.CsrfViewMiddleware',  # 注释避免出现403 (Forbidden)
    # 'django.contrib.auth.middleware.AuthenticationMiddleware',
    'django.contrib.messages.middleware.MessageMiddleware',
    'django.middleware.clickjacking.XFrameOptionsMiddleware',
]

💖 配置MySQL数据库

安装PyMySQL:

pip install pymysql -i https://pypi.tuna.tsinghua.edu.cn/simple

由于使用的是原生的SQL语句脚本,所以不进行后序的配置。

python使用原生的SQL操作数据库的代码写在子应用的models.py文件中。

💖 配置Redis数据库

安装 django-redis:

pip install django-redis -i https://pypi.tuna.tsinghua.edu.cn/simple

在Django项目的配置文件中,新增CACHES配置项,用于Redis数据库的配置:

与DATABASES配置项平级

DATABASES = {
	......
}

# 配置redis数据库
CACHES = {
    "default": {  # 默认使用的redis的0号数据库
        "BACKEND": "django_redis.cache.RedisCache",  # 使用redis作为缓存
        "LOCATION": "redis://127.0.0.1:6379/0",  # redis 数据库所在的主机地址 端口号 选择第几个数据库
        "OPTIONS": {
            "CLIENT_CLASS": "django_redis.client.DefaultClient",
            "PASSWORD": "123123",  # 访问redis数据库的密码
        }
    },
    "user": {  # 与用户相关的数据存放在Redis的1号数据库
        "BACKEND": "django_redis.cache.RedisCache",  
        "LOCATION": "redis://127.0.0.1:6379/1",
        "OPTIONS": {
            "CLIENT_CLASS": "django_redis.client.DefaultClient",
            "PASSWORD": "123123",
        }
    },
}

💖 配置跨域

安装django-cors-headers:

pip install django-cors-headers

在项目的配置文件中注册跨域子应用corsheaders

INSTALLED_APPS = [
    # 'django.contrib.admin',
    # 'django.contrib.auth',
    # 'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
    'corsheaders',  # 处理跨域子应用
    'user',
]

在项目的配置文件中新增处理跨域的中间件:

MIDDLEWARE = [
    'django.middleware.security.SecurityMiddleware',
    'django.contrib.sessions.middleware.SessionMiddleware',
    'corsheaders.middleware.CorsMiddleware',  # 处理跨域的中间件
    'django.middleware.common.CommonMiddleware',
    'django.middleware.csrf.CsrfViewMiddleware',
    # 'django.contrib.auth.middleware.AuthenticationMiddleware',
    'django.contrib.messages.middleware.MessageMiddleware',
    'django.middleware.clickjacking.XFrameOptionsMiddleware',
]

在配置文件中新增如下配置项:

CORS_ALLOW_CREDENTIALS = True  # 设置请求是否允许携带Cookie信息 默认为False
CORS_ORIGIN_ALLOW_ALL = True  # 允许所有的域名发送请求
CORS_ORIGIN_WHITELIST = ()   # 设置可以发送请求的域名
CORS_ALLOW_METHODS = (  # 允许的请求方式
    'DELETE',
    'GET',
    'OPTIONS',
    'PATCH',
    'POST',
    'PUT',
    'VIEW',
)
CORS_ALLOW_HEADERS = (  # 允许的请求头
    'accept',
    'accept-encoding',
    'authorization',
    'content-type',
    'dnt',
    'origin',
    'user-agent',
    'x-csrftoken',
    'x-requested-with',
)

💖 子应用的创建

💝 user

user子应用用于处理和用户相关的请求并向浏览器返回相应的数据响应。

在项目的根目录下创建user子应用:

python manage.py startapp user

在这里插入图片描述

在项目配置文件的INSTALLED_APPS配置项中注册user子应用:

INSTALLED_APPS = [
    # 'django.contrib.admin',
    # 'django.contrib.auth',
    # 'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
    'user'
]

💖 常量声明文件

在项目配置文件所在的文件夹中新建project_constants.py文件用于与本项目相关的常量的声明。

在这里插入图片描述

在user子应用的目录下新建user_constants.py文件用于与用户相关的常量的声明。

在这里插入图片描述

💖 工具文件

在子应用user目录下新建utils包用于存放和用户相关的工具文件.

在这里插入图片描述

💝 发送短信验证码的工具文件

在utils文件夹中新建sms_utils.py文件,用于声明和短信相关的工具类或方法。

在这里插入图片描述

安装容联云短信API平台的SDK

pip install ronglian-sms-sdk -i https://pypi.tuna.tsinghua.edu.cn/simple

在sms_utils.py文件中,使用如下代码:

记得修改accId accToken appId

from ronglian_sms_sdk import SmsSDK
import json


# accId = '容联云通讯分配的主账号ID'
accId = '...'
# accToken = '容联云通讯分配的主账号TOKEN'
accToken = '...'
# appId = '容联云通讯分配的应用ID'
appId = '...'


class SendSmsVerificationCode:
    """发送短信验证码的单例类"""

    def __new__(cls, *args, **kwargs):
        """
        发送短信验证码单例类的初始化方法
        :return: 返回一个发送短信验证码的对象
        """
        # 判断类中发送短信验证码的对象 _instance 是否已经存在
        # 如果不存在, 创建一个发送短信验证码的对象, 并将其作为类属性
        if not hasattr(cls, '_instance'):
            # 创建发送短信验证码的对象
            cls._instance = super(SendSmsVerificationCode, cls).__new__(cls, *args, **kwargs)
            # 创建SmsSDK对象作为 _instance 的对象属性
            cls._instance.sdk = SmsSDK(accId, accToken, appId)
        # 如果存在, 返回发送短信验证码的对象
        return cls._instance

    def send_message(self, mobile, datas, tid='1'):
        """
        发送短信的方法
        @params mobile 字符串类型  mobile = '手机号1,手机号2'
        @params tid tid = '容联云通讯平台创建的模板' 默认模板的编号为1
        @params datas 元组类型  第一个参数为验证码 第二个参数为验证码的有效时间(对于短信模板1)
        :return: 返回发送短信后的响应参数
        """
        # 发送短信
        resp = self.sdk.sendMessage(tid, mobile, datas)
        print(json.loads(resp), type(json.loads(resp)))
        return resp


# 测试
if __name__ == '__main__':
    sendSmsVerificationCode1 = SendSmsVerificationCode()
    sendSmsVerificationCode2 = SendSmsVerificationCode()
    sendSmsVerificationCode3 = SendSmsVerificationCode()
    print(sendSmsVerificationCode1)
    print(sendSmsVerificationCode2)
    print(sendSmsVerificationCode3)
    res = sendSmsVerificationCode1.send_message('...', ('123456', 5), '1')
    print(res)

💝 操作数据库的工具文件

在项目配置文件所在的文件目录下新建utils包用于存放和本项目相关的工具文件。

在这里插入图片描述

在utils文件夹下新建mysql_utils.py文件用于声明和操作mysql数据库相关的工具类或方法。

在这里插入图片描述

在mysql_utils.py文件中写如下用于操作mysql数据库的工具类:

import pymysql


class MySQLUtils():
    """
    操作MySQL数据库的工具类
    """

    @staticmethod
    def getConnection():
        """
        获取MySQL数据库的连接对象

        @return: MySQL数据库的连接对象
        """
        conn = None
        try:
            # 获取MySQL数据库连接对象
            conn = pymysql.connect(
                host='localhost',
                port=3306,
                user='root',
                password='123456',
                database='meiduo',
                charset='utf8'
            )
        except Exception as e:
            print('获取MySQL数据库连接对象失败')
            print(e)
        # 返回数据库的连接对象
        return conn

    @staticmethod
    def update(sql, params=None):
        """
        更新MySQL数据库中数据的通用方法,包括对数据库中数据的增删改

        @param - sql: 需要执行的sql语句
        @param - params: 向sql语句中填充的参数,元组类型,默认值为None

        @return: 返回本次操作对数据库中数据的影响行数
        """
        conn = MySQLUtils.getConnection()  # 获取MySQL的连接对象
        cur = None
        row_count = None
        try:
            cur = conn.cursor()  # 获取执行sql的游标对象
            if params is None:
                row_count = cur.execute(sql)
            else:
                row_count = cur.execute(sql, params)
            conn.commit()  # 提交本次操作
        except Exception as e:
            conn.rollback()  # 如果执行SQL语句时出现异常则回滚本次操作
            print('执行SQL语句时出现异常')
            print(e)
        finally:
            MySQLUtils.closeResource(conn=conn, cur=cur)  # 关闭连接数据库的资源
        return row_count

    @staticmethod
    def select(sql, params=None):
        """
        查询MySQL数据库中数据的通用方法

        @param - sql: 需要执行的sql语句
        @param - params: 向sql语句中填充的参数,元组类型,默认值为None

        @return: 返回的为字典类型数据,字典中row_count为本次查询结果的行数,data为本次查询结果集(元组类型)
        """
        conn = MySQLUtils.getConnection()  # 获取MySQL数据库的连接对象
        cur = None
        row_count = None
        data = None
        try:
            cur = conn.cursor()  # 获取执行SQL的游标对象
            if params is None:
                row_count = cur.execute(sql, params)  # 执行SQL并获取本次查询结果集的行数
            else:
                row_count = cur.execute(sql, params)  # 执行SQL并获取本次查询结果集的行数
            data = cur.fetchall()  # 获取本次查询结果集
        except Exception as e:
            print('执行SQL语句时出现异常')
            print(e)
        finally:
            MySQLUtils.closeResource(conn=conn, cur=cur)
        # 返回查询结果集
        return {
            'row_count': row_count,
            'data': data
        }

    @staticmethod
    def closeResource(conn=None, cur=None):
        """
        关闭与操作MySQL数据库相关的资源对象

        @param - conn: MySQL数据库的连接对象
        @param - cur: 执行sql语句的游标对象

        @return: 无返回值
        """
        try:
            if cur is not None:
                cur.close()
            if conn is not None:
                conn.close()
        except Exception as e:
            print('关闭MySQL资源对象出错')
            print(e)


if __name__ == '__main__':
    # sql = "INSERT INTO user_account_info(user_account, user_telephone, user_password, username) VALUES (%s, %s, %s, %s);"
    # print(MySQLUtils.update(sql, ('123', '123', '123', '123')))
    # sql = 'select * from user_account_info'
    # print(MySQLUtils.select(sql))
    pass

💝 与请求处理相关的工具文件

在项目配置文件所在目录下的utils包中新建request_utils.py文件用于声明与请求处理相关的工具类和方法。

在这里插入图片描述

💝 与数据类型转换相关的工具文件

在项目配置文件所在目录下的utils包中新建data_type_change_utils.py文件用于声明与数据类型转换相关的工具类和方法。

在这里插入图片描述

💝 与用户账号相关的工具文件

在子应用user的utils包中新建user_account_utils.py文件用于声明与用户账号相关的工具类和方法。

在这里插入图片描述

💝 与日期时间相关的工具文件

在这里插入图片描述

💝 与jwt相关的工具文件

安装pyjwt:

pip install pyjwt

在user/utils包中新建文件:

在这里插入图片描述

"""
声明和jwt相关的工具类或方法
"""
import jwt
import datetime
from jwt import exceptions
from meiduo_move_server.settings import SECRET_KEY  # 获取django自动生成的项目密钥
from user.user_constants import TOKEN_VALID_TIME  # token的有效时间


def create_token(userAccount, userPassword):
    """
    根据用户的账号或手机号和密码生成对应的token

    @param - userAccount: 用户账号或手机号
    @param - userPassword: 用户账号密码

    @return: 返回用户账号对应的token
    """
    # 构造header,JWT的元数据
    headers = {
        'typ': 'jwt',  # 令牌类型,JWT
        'alg': 'HS256'  # 生成签名算法
    }
    # 构造payload,存放实际数据
    payload = {
        'userAccount': userAccount,  # token中包含的数据
        'userPassword': userPassword,  # token中包含的数据
        'exp': datetime.datetime.utcnow() + datetime.timedelta(days=TOKEN_VALID_TIME)  # 设置token过期时间
    }
    token = jwt.encode(payload=payload, key=SECRET_KEY, algorithm="HS256", headers=headers)
    return token


def get_payload(token):
    """
    获取token中的payload,判断token是否有效,如果token有效返回payload中的包含数据,否则返回None

    @param - token: token

    @return: 如果token有效返回payload中的包含数据,否则返回None
    """
    try:
        # 从token中获取payload【校验合法性】
        verified_payload = jwt.decode(jwt=token, key=SECRET_KEY, algorithms=['HS256'])
        # print(verified_payload)
        return verified_payload
    except exceptions.ExpiredSignatureError:
        # print('token已失效')
        return None
    except jwt.DecodeError:
        # print('token认证失败')
        return None
    except jwt.InvalidTokenError:
        # print('非法的token')
        return None

if __name__ == '__main__':
    token = create_token('15892014635', '11111111')
    try:
        # 从token中获取payload【不校验合法性】
        verified_payload = jwt.decode(jwt=token, key=SECRET_KEY, algorithms=['HS256'])
        print(verified_payload)
    except exceptions.ExpiredSignatureError:
        print('token已失效')
    except jwt.DecodeError:
        print('token认证失败')
    except jwt.InvalidTokenError:
        print('非法的token')

💘 美多商城前台后端路由说明(请求路由)

  • user/
    • 处理与用户相关的请求
    • 所有处理与用户相关请求的路由均为该路由的子路由
  • user/register/
    • 处理新用户的注册
    • get:
      • 向该路由发起get请求用于获取用户注册时所需的短信验证码,发送get请求的同时传递注册用户的手机号,会向该手机号发送短信验证码,同时会向浏览器返回相应的响应数据对象
      • 所需参数:
        • userTelephone:用户手机号
      • 响应数据对象中的数据项:
        • mark:本次处理成功或失败的标记,成功为1,失败为0
        • message:本次请求处理相关的信息说明
    • post:
      • 向该路由发起post请求,用于实现新用户的注册,发送post请求的同时需要传递用户手机号、用户账号密码以及短信验证码。
      • 所需参数:
        • userTelephone:用户手机号
        • userPassword:用户账号密码
        • smsVerificationCode:短信验证码
      • 响应数据对象中的数据项:
        • mark:本次请求处理成功或失败的标记,成功为1,失败为0
        • message:本次请求处理相关的信息说明
  • user/login/
    • 处理用户的登录请求
    • get:
      • 向该路由发送get请求,用于判断当前用户token是否有效,发送请求时需要传递token(保存于localStorage中)
      • 所需参数:
        • token
      • 响应数据对象中的数据项:
        • mark:本次请求处理成功或失败的标记,成功为1,失败为0
        • message:本次请求处理相关的信息说明
    • post:
      • 向该路由发送post请求,用于处理用户的登录请求,发送post请求的同时需要传递用户登录的账号或手机号和用户账号密码,收到请求后会对账号进行验证,向浏览器返回相应的响应对象。
      • 所需参数:
        • userAccount / userTelephone:用户账号或用户手机号
        • userPassword:用户账号密码
      • 响应数据对象中的数据项:
        • mark:本次请求处理成功或失败的标记,成功为1,失败为0,失效为2
        • message:本次请求处理相关的信息说明
        • token:登录的凭证,响应对象的data中,浏览器页面会保存于localStorage中
  • user/username/userAccount/
    • 向该路由发起get请求,获取当前登录用户的用户名和用户账户以及用户头像的路径地址,发送get请求时需要传递当前用户的登录令牌token(保存于localStorage中)
    • 所需参数:
      • token:当前用户的登录令牌token
    • 响应数据对象中的数据项:
      • mark:本次请求处理成功或失败的标记,成功为1,失败为0,失效为2
      • message:本次请求处理相关的信息说明
      • username:当前登录用户的用户名,响应数据对象的data中
      • userAccount:当前登录用户的用户账号,响应数据对象的data中
      • userHeadPicture:当前登录用户的用户头像的路径地址,响应数据对象的data中
  • user/userinfo/
    • 向该路由发起get请求,获取当前登录用户的个人信息,发送get请求时需要传递当前用户的登录令牌token(保存于localStorage中),获取的信息用于个人信息页面
    • 所需参数:
      • token:当前用户的登录令牌token
    • 响应数据对象中的数据项:
      • mark:本次请求处理成功或失败的标记,成功为1,失败为0,失效为2
      • message:本次请求处理相关的信息说明
      • userHeadPicture:当前登录用户的用户头像的路径地址,响应数据对象的data中
      • userAccount:当前登录用户的用户账号,响应数据对象的data中
      • username:当前登录用户的用户名,响应数据对象的data中
      • userRealName:用户真实新名,响应数据对象的data中
      • userGender:用户性别,响应数据对象的data中
      • userIdCardNumber:用户身份证号,响应数据对象的data中
      • userTelephone:用户手机号,响应数据对象的data中
      • userEmail:用户邮箱,响应数据对象的data中
  • user/change/userinfo/
    • 向该路由发起post请求,修改当前登录用户的指定用户信息,发送post请求时需要传递当前用户的登录令牌token(保存于localStorage中)和修改的信息项(变量名)与修改后的值
    • 所需参数:
      • token:当前用户的登录令牌
      • field_name:当前要修改的信息项名
      • param:要修改信息项的修改后的值
    • 响应数据对象中的数据项:
      • mark:本次请求处理成功或失败的标记,成功为1,失败为0,失效为2
      • message:本次请求处理相关的信息说明

💘 美多商城前台变量与常量说明

常量均通过对象挂载到Vue原型对象上

💖 与用户相关

💝 变量

  • userAccount
    • 用户账号
    • 字符串类型
    • 用户账号的生成规则:mdyyyymmddxxxx,其中,md为每个账号的固定开头(meiduo拼音的缩写),yyyy为注册时间的年份,mm为注册时间的月份,dd为注册时间的天,xxxx为注册用户手机号的后四位。
    • 规定一个手机号只能注册一个账号。
  • userTelephone
    • 用户手机号
    • 字符串类型
  • userPassword
    • 用户账号密码
    • 字符串类型
  • userHeadPicture
    • 用户头像路径地址
    • 字符串类型
  • username
    • 用户账号的用户名
    • 字符串类型
    • 最多30个字符
    • 规定用户的用户名唯一,即不能出现不同账号具有相同的用户名,用户账号的用户名默认为用户账号
  • userRegisterTime
    • 用户账号注册的时间
    • 日期时间类型
    • 格式:yyyy-mm-dd HH:MM:SS
  • userLoginTime
    • 用户账号最近一次的登录时间
    • 日期时间类型
    • 格式:yyyy-mm-dd HH:MM:SS
  • userRealName
    • 用户的真实姓名
    • 字符串类型
    • 最多30个字符
  • userIdCardNumber
    • 用户的身份证号码
    • 字符串类型
  • userGender
    • 用户性别
    • 字符串类型
    • 取值:‘F’ - 女性;‘M’ - 男性
    • 默认为男性
  • userEmail
    • 用户邮箱
    • 字符串类型
    • 最多30个字符
  • smsVerificationCode
    • 短信验证码
    • 字符串类型

💝 常量

  • USER_ACCOUNT_LENGTH
    • 用户账号的长度(位数)
    • 整数类型
    • 14位
    • 根据用户账号的格式 mdyyyymmddxxxx,为14位
  • USER_TELEPHONE_LENGTH
    • 用户手机号位数(长度)
    • 整数类型
    • 11位
  • USER_TELEPHONE_REGULAR
    • 用户手机号格式的正则表达式
    • 正则
    • /^1(3[0-9]|4[01456879]|5[0-35-9]|6[2567]|7[0-8]|8[0-9]|9[0-35-9])\d{8}$/
  • USER_PASSWORD_MAX_LENGTH
    • 用户账号密码的最大长度(位数)
    • 整数类型
    • 16位
  • USER_PASSWORD_MIN_LENGTH
    • 用户账号密码的最小长度(位数)
    • 整数类型
    • 8位
  • USER_PASSWORD_SECRET
    • 用于在用户进行登录时,在用户账号密码加上后缀,加强安全性,和用户的密码之间以点进行分隔
    • 字符串类型
    • 取值:123456
  • USER_HEAD_PICTURE
    • 用户默认头像的路径地址
    • 字符串类型
    • 取值:require('@/assets/user-default-head-picture.jpg')
    • 如果用户没有上传头像则默认使用该路径对应的图片作为用户头像
  • USERNAME_MAX_LENGTH
    • 用户账号用户名最大长度
    • 整数类型
    • 30
  • USER_REAL_NAME_MAX_LENGTH
    • 用户姓名的最大长度
    • 整数类型
    • 30
  • USER_ID_CARD_NUMBER_LENGTH
    • 用户身份证号码的位数(长度)
    • 整数类型
    • 18位
  • USER_GENDER_MALE
    • 用户性别,男性
    • 字符串类型
    • 取值:‘M’
  • USER_GENDER_FEMALE
    • 用户性别,女性
    • 字符串类型
    • 取值:‘F’
  • USER_EMAIL_MAX_LENGTH
    • 用户邮箱的最大长度
    • 整数类型
    • 30
  • SMS_VERIFICATION_CODE_LENGTH
    • 短信验证码位数(长度)
    • 整数类型
    • 6位
  • SMS_VERIFICATION_CODE_SEND_AGAIN_WAIT_TIME
    • 短信验证码再次发送需要等待的时间,单位秒
    • 整数类型
    • 60秒
  • SMS_VERIFICATION_CODE_VALID_TIME
    • 短信验证码的有效时间,单位秒
    • 整数类型
    • 300秒
  • SMS_VERIFICATION_FIXED
    • 没有开启短信验证码功能时,使用的固定默认短信验证码
    • 字符串类型
    • 取值:‘111111’
  • TOKEN_VALID_TIME
    • token的有效时间
    • 整数类型
    • 5天,单位天

💖 与请求和响应相关

💝 常量

  • MARK_ERROR
    • 请求处理失败的标记
    • 取值为0
  • MARK_SUCCESS
    • 请求处理成功的标记
    • 取值为1
  • MARK_INVALID
    • 过期的标记
    • 取值为2

💝 响应数据对象的键

  • mark
    • 本次请求处理成功或失败的标记
    • 取值:0 - 失败;1 - 成功
  • message
    • 本次请求响应的信息描述
    • 字符串类型
  • data
    • 请求响应回来的数据
    • 对象/字典类型

💖 用于美多商城服务端功能控制的常量

  • IS_OPEN_SMS_VERIFICATION
    • 是否开启短信验证码的验证功能
    • 布尔类型
    • 如果不开启短信验证码的验证功能,则在用户注册过程中,使用短信验证码111111,即可实现注册;如果开启短信验证码的验证功能,则用户在注册过程中将会收到短信验证码,用户需要根据收到的短信验证码进行注册。

💘 美多商城数据库表的说明

  • user_account_info
    • 用户账号信息表
    • 包含字段:
      • id:用户账号编号,int
      • user_account:用户账号,char(14)
      • user_telephone:用户手机号,char(11)
      • user_password:用户账号密码,varchar(16)
      • username:用户账号的用户名,varchar()
      • user_register_time:用户注册时间,datetime
      • user_login_time:用户最近一次登录时间,datetime
      • user_head_picture:用户账号头像,varchar(255)
      • user_personal_info_id:外键,关联user_personal_info表的id字段,int
  • user_personal_info
    • 用户个人信息表
    • 包含字段:
      • id:用户个人信息编号,int
      • user_real_name:用户真实姓名,varchar(30)
      • user_id_card_number:用户身份证号码,char(30)
      • user_gender:用户性别,char(1),取值:‘F’/‘M’
      • user_telephone:用户手机号,char(11)
      • user_email:用户邮箱,varchar(30)
  • user_receive_address_info
    • 用户收货地址信息表
    • 包含字段:
      • id

其余表的字段的相关说明在表的创建语句中


💘 美多商城数据库SQL语句

💖 数据库的创建

CREATE DATABASE meiduo;

💖 表的创建

-- 表的创建
# 创建用户个人信息表
CREATE TABLE user_personal_info
(
    id                  INT PRIMARY KEY AUTO_INCREMENT,
    user_real_name      VARCHAR(30)     DEFAULT NULL COMMENT '用户真实姓名',
    user_id_card_number CHAR(18) UNIQUE DEFAULT NULL COMMENT '用户身份证号码',
    user_gender         CHAR(1)         DEFAULT 'M' CHECK ( user_gender IN ('F', 'M') ) COMMENT '用户性别',
    user_telephone      CHAR(11) UNIQUE NOT NULL COMMENT '用户手机号',
    user_email          VARCHAR(30)     DEFAULT NULL COMMENT '用户邮箱'
);
# 创建用户账号信息表
CREATE TABLE user_account_info
(
    id                    INT PRIMARY KEY AUTO_INCREMENT,
    user_account          CHAR(14) UNIQUE                  NOT NULL COMMENT '用户账号',
    user_telephone        CHAR(11) UNIQUE                  NOT NULL COMMENT '用户手机号',
    user_password         VARCHAR(16)                      NOT NULL COMMENT '用户账号密码',
    username              VARCHAR(30) UNIQUE DEFAULT NULL COMMENT '用户账号的用户名,默认为用户账号',
    user_register_time    DATETIME           DEFAULT NOW() NOT NULL COMMENT '用户账号的注册时间',
    user_login_time       DATETIME           DEFAULT NOW() NOT NULL COMMENT '用户账号最近一次登录时间',
    user_personal_info_id INT                              NOT NULL COMMENT '外键,关联用户个人信息表的id,级联删除',
    CONSTRAINT fk_user_account_info_user_personal_info_id
        FOREIGN KEY (user_personal_info_id)
            REFERENCES user_personal_info (id)
            ON DELETE CASCADE
);
# 创建用户收货地址信息表
CREATE TABLE user_receive_address_info
(
    id                   INT PRIMARY KEY AUTO_INCREMENT COMMENT '用户收货地址信息编号',
    receive_name         VARCHAR(30)         NOT NULL COMMENT '收货人姓名',
    receive_telephone    CHAR(11)            NOT NULL COMMENT '收货人手机号',
    receive_address      VARCHAR(255)        NOT NULL COMMENT '收货地址',
    is_default           CHAR(1) DEFAULT 'F' NOT NULL COMMENT '是否为默认地址',
    user_account_info_id INT                 NOT NULL COMMENT '外键,关联用户账号信息表的id,级联删除',
    CONSTRAINT fk_user_receive_address_info_user_account_info_id
        FOREIGN KEY (user_account_info_id)
            REFERENCES user_account_info (id)
            ON DELETE CASCADE
);
# 商品信息表
CREATE TABLE goods_info
(
    id                   INT PRIMARY KEY AUTO_INCREMENT COMMENT '商品信息编号',
    goods_name           VARCHAR(255)   NOT NULL COMMENT '商品名',
    goods_price          DECIMAL(10, 2) NOT NULL COMMENT '商品价格',
    goods_category       INT            NOT NULL COMMENT '商品类别',
    goods_sale_number    INT            NOT NULL DEFAULT 0 COMMENT '商品销量',
    goods_comment_number INT            NOT NULL DEFAULT 0 COMMENT '商品评价数',
    goods_freight        DECIMAL(6, 2)  NOT NULL DEFAULT 0 COMMENT '商品运费'
);
# 商品参数信息表
CREATE TABLE goods_param_info
(
    id             INT PRIMARY KEY AUTO_INCREMENT COMMENT '商品参数信息编号',
    param          VARCHAR(255)  NOT NULL COMMENT '商品参数',
    param_category INT DEFAULT 0 NOT NULL COMMENT '商品参数类别',
    goods_info_id  INT COMMENT '外键,关联商品信息表的id,商品信息被删除时级联删除',
    CONSTRAINT fk_goods_param_info_goods_info_id
        FOREIGN KEY (goods_info_id)
            REFERENCES goods_info (id)
            ON DELETE CASCADE
);
# 商品图片信息表
CREATE TABLE goods_picture_info
(
    id                 INT PRIMARY KEY AUTO_INCREMENT COMMENT '商品图片信息编号',
    goods_picture_path VARCHAR(255) NOT NULL COMMENT '商品图片的存放路径',
    goods_info_id      INT COMMENT '外键,关联商品信息表的id,商品信息被删除时级联删除',
    CONSTRAINT fk_goods_picture_info_goods_info_id
        FOREIGN KEY (goods_info_id)
            REFERENCES goods_info (id)
            ON DELETE CASCADE
);
# 订单信息表
CREATE TABLE order_info
(
    id                      INT PRIMARY KEY AUTO_INCREMENT COMMENT '订单信息编号',
    order_status            CHAR(1)        NOT NULL DEFAULT 'F' COMMENT '订单状态',
    goods_number            INT            NOT NULL COMMENT '购买商品数量',
    pay_time                DATETIME       NOT NULL COMMENT '订单支付时间',
    pay_money               DECIMAL(10, 2) NOT NULL COMMENT '订单支付金额',
    create_time             DATETIME       NOT NULL COMMENT '订单创建时间',
    user_account_info_id    INT COMMENT '外键,关联用户账号信息表id,用户被删除时设置为空',
    goods_info_id           INT COMMENT '外键,关联商品信息表的id,商品信息被删除时设置为空',
    user_receive_address_id INT COMMENT '外键,关联用户收货地址信息表的id,用户收货地址信息被删除时设置为空',
    CONSTRAINT fk_order_info_user_account_info_id
        FOREIGN KEY (user_account_info_id)
            REFERENCES user_account_info (id)
            ON DELETE SET NULL,
    CONSTRAINT fk_order_info_goods_info_id
        FOREIGN KEY (goods_info_id)
            REFERENCES goods_info (id)
            ON DELETE SET NULL,
    CONSTRAINT fk_order_info_user_receive_address_id
        FOREIGN KEY (user_receive_address_id)
            REFERENCES user_receive_address_info (id)
            ON DELETE SET NULL
);
# 用户商品评价信息表
CREATE TABLE user_goods_comment_info
(
    id                   INT PRIMARY KEY AUTO_INCREMENT COMMENT '用户商品评价信息编号',
    comment_score        INT     NOT NULL DEFAULT 0 COMMENT '商品评价评分',
    comment_description  VARCHAR(255)     DEFAULT '' COMMENT '商品评价',
    is_anonymous         CHAR(1) NOT NULL DEFAULT 'F' COMMENT '是否为匿名评价',
    user_account_info_id INT COMMENT '外键,关联用户账号信息表的id,用户信息被删除时设置为空',
    goods_info_id        INT COMMENT '外键,关联商品信息表的id,商品信息被删除时设置为空',
    CONSTRAINT fk_user_goods_comment_info_user_account_info_id
        FOREIGN KEY (user_account_info_id)
            REFERENCES user_account_info (id)
            ON DELETE SET NULL,
    CONSTRAINT fk_user_goods_comment_info_goods_info_id
        FOREIGN KEY (goods_info_id)
            REFERENCES goods_info (id)
            ON DELETE SET NULL
);

💖 表的修改

-- 表的修改
# 向用户账号信息表中新增用户头像路径地址列
ALTER TABLE user_account_info
ADD COLUMN user_head_picture VARCHAR(255) DEFAULT '' NOT NULL COMMENT '用户头像路径';

💖 触发器的创建

-- 触发器的创建
# 向用户账号信息表插入数据时,先向用户个人信息表插入该用户账号对应的用户个人信息
CREATE TRIGGER before_insert_user_account_info
    BEFORE INSERT
    ON user_account_info
    FOR EACH ROW
BEGIN
    # 向用户个人信息表中新增数据记录,向该记录中填写用户手机号
    INSERT INTO user_personal_info(user_telephone)
    VALUES (NEW.user_telephone);
    # 将新增的用户个人信息对应的id填写到新增的用户账号信息记录中
    SET NEW.user_personal_info_id = (
        SELECT id
        FROM user_personal_info
        WHERE user_telephone = NEW.user_telephone
    );
    # 默认用户账号的用户名为用户账号
    SET NEW.username = NEW.user_account;
END;

💖 视图的创建

-- 视图的创建
# 创建用于用户登录的视图
CREATE OR REPLACE VIEW v_user_account_info_login
AS
SELECT user_account, user_telephone, user_password
FROM user_account_info;

💘 美多商城后台前端项目的创建与配置

美多商城的后台前端项目的创建与配置和美多商城前台前端项目的创建和配置类似,这里不再讲述。

💘 美多商城后台后端

美多商城的后台后端服务器与美多商城前台后端服务器使用同一个服务器。这里也不在讲述


手机号重复注册


💘💖💝💗💓💞💕💟💔❤️‍🔥🧡💛💚💙

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/40754.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【并发】Java并发线程池底层原理详解与源码分析(上)

【并发】Java并发线程池底层原理详解与源码分析&#xff08;上&#xff09; 线程池与线程对比 使用线程的方式运行 使用线程池的方式运行 分析 单线程、多线程、线程池效率问题 详细代码 结果分析 单线程为什么是最快的&#xff1f; 单线程都这么快了&#xff0c;我…

第四周 牛背山云海——拍拍大场景,自拍延时片段

目录4.1 面对广阔云海拍张全景照片&#xff0c;再来一组延时片段4.2 认识镜头滤镜4.3 补充技术基础知识&#xff1a;白平衡、色温4.4 小博士课堂——延时摄影课件4.1 面对广阔云海拍张全景照片&#xff0c;再来一组延时片段 云海 雪山 雅安境内的牛背山拍摄云海 牛背山可以看到…

java刷题day 04

一. 单选题&#xff1a; 解析&#xff1a;队列&#xff1a;先到先服务解析&#xff1a;final不能修饰接口&#xff0c;成员变量可以用final修饰 解析&#xff1a; A&#xff1a; 算法是指解题方案的准确而完整的描述&#xff0c;算法不等于程序&#xff0c;也不等于计算方法 BC…

Linux 时间同步 ntpdchrony 内网

Linux 时间同步 ntpd&chrony 在任何服务器集群中&#xff0c;为了更好的协同工作&#xff0c;除了一些互信机制外&#xff0c;需要的就是时间同步功能&#xff0c;如果时间不同步&#xff0c;就好比让在中国的同事与美国的同事进行沟通&#xff0c;会造成各种奇奇怪怪的时…

excel笔记

1.快速填充 当需要提取出电子邮件里的名称&#xff0c;可以在第一行里手动输入Nancy后&#xff0c;直接按快捷键ctrl e进行快速填充&#xff0c;可以直接获得预期数据 如果不用快捷键&#xff0c;也可以点击指定位置的图标&#xff0c;实现这一功能 快速填充不仅可以将进行…

大数据中的R语言——中国大学MOOC课程笔记

第一章 R语言简介 1.1 简介 R的特性&#xff1a; • 免费的 • 一个全面的统计研究平台 • 拥有顶尖水准的制图功能 • 一个可进行交互式数据分析和探索的强大平台 • 轻松地从各种类型的数据源导入数据&#xff0c;包括文本文件、数据库管理系统、统计软件&#xff0c;乃…

rabbitMQ 消息顺序性、消息幂等性、消息不丢失、最终一致性、补偿机制、消息队列设计

一、消息顺序性 消息队列中的若干消息如果是对同一个数据进行操作&#xff0c;这些操作具有前后的关系&#xff0c;必须要按前后的顺序执行&#xff0c;否则就会造成数据异常。 举例&#xff1a;   比如通过mysql binlog进行两个数据库的数据同步&#xff0c;由于对数据库的…

内存和函数

程序的内存布局 Linux默认情况下将高地址的1GB空间分配给内核&#xff0c;用户进程使用剩下2GB或者3GB的内存空间。在用户空间里&#xff0c;也有很多地址区间有特殊的地位&#xff0c;一般来讲&#xff0c;应用程序使用的内存空间里有如下"默认"的区域 1、栈&#…

疫情可视化part2

前言 这是疫情可视化最开始的文章&#xff0c;有需要了解的可前往查看&#xff1a;https://blog.csdn.net/xi1213/article/details/126824752。 本来说有时间就把这个项目完结了的&#xff0c;结果后面一直有事拖着&#xff0c;直到现在十一月份了才搞完。老样子&#xff0c;先…

F1. 生活在树上(easy version)树,dfs

题目链接 F1. 生活在树上&#xff08;easy version&#xff09; 题目背景 本题是 B 组的最后一题&#xff0c;是 F2 题的简单版本&#xff0c;两道题目的解法略有不同。本题和 F2 题在题意上的区别在于本题给定树上的边权&#xff0c;而不是点权。 小智生活在「传智国」&am…

汽车 Automotive > SOME/IP VS DDS调研和未来方向

参考&#xff1a;JASPAR, General incorporated association&#xff1a;What is the conqueror in the SOA platform for the future in-vehicle networks? 目录 SOME/IP介绍参考 DDS介绍 SOME/IP VS DDS 研究方向 SOME/IP介绍参考 汽车Automotive &#xff1e; SOME/…

MAC安全(防MAC泛洪攻击)

一、MAC地址表项分类&#xff1a; 1.1 动态表项&#xff1a;通过对帧内的源MAC进行学习而来&#xff0c;有老化时间 1.2 静态表项&#xff1a;由管理员手工配置&#xff0c;不会老化 1.3 黑洞表项&#xff1a;丢弃特定源MAC或目的MAC,不会老化 静态和黑洞表项不会被动态表项…

类与对象(下篇)

类与对象&#xff08;下&#xff09;再谈构造函数回顾构造函数初始化列表explicit 关键字拷贝构造函数也具有初始化列表友元 friend友元函数输入输出流的重载友元类static 成员内部类再谈构造函数 回顾构造函数 在上一篇博客中提到了构造函数&#xff0c;构造函数其主要目的是…

类与对象(中篇)

类中六个默认成员函数构造函数基本概念构造函数特性析构函数基本概念析构函数特性拷贝构造函数基本概念拷贝构造函数特性赋值运算符重载概念引入运算符重载函数的特性部分运算符的重载函数判等赋值前置 、前置--后置、后置--const 成员函数取地址只要生成一个类 &#xff0c;那…

iOS_Custom Transition Animation 自定义转场动画

文章目录1、push-pop 动画协议2、present-dismiss 动画协议3、实现转场动画协议3.1 动画时长3.2 push or present animation (显示动画)3.3 动画结束3.4 pop or dismiss animation (消失动画)4、UIPresentationController4.1 设置presentVC的frame4.2 present 动画4.3 dismiss …

Docker快速安装Oracle 12c

【Oracle系列3】Docker快速安装Oracle 12c 背景 现在还很多企业用12c&#xff0c;以这个版本为例&#xff0c;介绍docker快速启动Oracle并做实验 步骤 1、docker环境的安装&#xff08;略&#xff09; 2、查询镜像&#xff0c;挑选镜像 docker search oracle结果 StoneM…

阿里P8架构师都在学习参考的SpringCloud微服务实战文档

我一直在使用Spring Boot、Spring Data等框架来进行开发工作。 作为一名Spring系列的忠实粉丝&#xff0c;我自然希望能够有更多的开发者参与进来&#xff0c;于是自己坚持写Spring Cloud相关的文章&#xff0c;并且将文章涉及的代码整理后放在GitHub上分享。 这使我得到了很…

【Hack The Box】Linux练习-- Luanne

HTB 学习笔记 【Hack The Box】Linux练习-- Luanne &#x1f525;系列专栏&#xff1a;Hack The Box &#x1f389;欢迎关注&#x1f50e;点赞&#x1f44d;收藏⭐️留言&#x1f4dd; &#x1f4c6;首发时间&#xff1a;&#x1f334;2022年11月24日&#x1f334; &#x1f3…

零基础搭建基于知识图谱的电影问答系统

零基础搭建基于知识图谱的电影问答系统一、项目准备二、项目数据三、训练问题分类器四、准备问答模板五、搭建webapp六、问题预处理一、项目准备 首先需要一款python编译器&#xff0c;本人选用的是PyCharm&#xff0c;搭建好Python环境&#xff1b;安装第三方依赖库&#xff…

【Hack The Box】linux练习-- Delivery

HTB 学习笔记 【Hack The Box】linux练习-- Delivery &#x1f525;系列专栏&#xff1a;Hack The Box &#x1f389;欢迎关注&#x1f50e;点赞&#x1f44d;收藏⭐️留言&#x1f4dd; &#x1f4c6;首发时间&#xff1a;&#x1f334;2022年11月17日&#x1f334; &#x1…