【AI图像生成网站Golang】项目架构

news/2024/12/23 3:06:28/

AI图像生成网站

目录

一、项目介绍

二、雪花算法

三、JWT认证与令牌桶算法

四、项目架构

五、图床上传与图像生成API搭建

六、项目测试与优化


四、项目架构

本项目的后端基于Golang和Gin框架开发,主要包括的模块有:

backend/
├── controller   // 路由控制器
├── dao          // 数据访问模块
│   ├── mysql    // MySQL 数据库操作
│   └── api      // 调用第三方 API 的模块
├── logic        // 核心业务逻辑
├── models       // 数据结构定义
├── pkg          // 工具包(JWT、Snowflake 等)
└── router       // 路由定义
  • controller: 负责处理 HTTP 请求并将请求数据传递给逻辑层,控制器层作为入口,将不同的请求指向相应的业务逻辑。
  • dao/mysql: 负责数据库操作,封装了数据查询和持久化逻辑,便于管理数据库交互。
  • logic: 业务逻辑层,实现具体的业务功能,比如用户的登录注册和相关的账号管理。
  • model: 用于定义数据模型与表结构,是项目中数据对象的核心描述。

以用户登录功能为例,从前端请求到后端处理的完整调用流程如下:

在这里插入图片描述

1. 前端发送请求

Vue.js 前端会通过表单收集用户输入的 usernamepassword,并调用后端登录 API(如 POST /api/login)。请求中会包含 JSON 格式的用户凭证。

2. 路由层解析请求

后端的路由通过 controller 文件夹中定义的控制器来接收前端的 HTTP 请求。例如:

v1.POST("/login", controller.LoginHandler)

3.控制器层处理请求

控制器负责将请求转发到逻辑层,并对输入参数进行基本的校验:


// LoginHandler 登录业务
func LoginHandler(c *gin.Context) {// 1、获取请求参数及参数校验var u *models.LoginFormif err := c.ShouldBindJSON(&u); err != nil {// 请求参数有误,直接返回响应zap.L().Error("Login with invalid param", zap.Error(err))// 判断err是不是 validator.ValidationErrors类型的errorserrs, ok := err.(validator.ValidationErrors)if !ok {// 非validator.ValidationErrors类型错误直接返回ResponseError(c, CodeInvalidParams) // 请求参数错误return}// validator.ValidationErrors类型错误则进行翻译ResponseErrorWithMsg(c, CodeInvalidParams, removeTopStruct(errs.Translate(trans)))return}// 2、业务逻辑处理——登录user, err := logic.Login(u)if err != nil {zap.L().Error("logic.Login failed", zap.String("username", u.UserName), zap.Error(err))if err.Error() == mysql.ErrorUserNotExit {ResponseError(c, CodeUserNotExist)return}ResponseError(c, CodeInvalidParams)return}// 3、返回响应ResponseSuccess(c, gin.H{"user_id":        fmt.Sprintf("%d", user.UserID), //js识别的最大值:id值大于1<<53-1  int64: i<<63-1"user_name":      user.UserName,"access_token":   user.AccessToken,"refresh_token":  user.RefreshToken,"avatarImageUrl": user.Avatar,"gender":         user.Gender,})
}

这里使用了 models.LoginForm 来解析和校验前端传递的 JSON 数据。

// LoginForm 登录请求参数
type LoginForm struct {UserName string `json:"username" binding:"required"`Password string `json:"password" binding:"required"`
}

4. 逻辑层处理核心业务逻辑

logic 文件夹中封装了核心的业务逻辑,例如验证用户凭证、生成令牌等:

func Login(username, password string) (string, error) {user = &models.User{UserName: p.UserName,Password: p.Password,}if err := mysql.Login(user); err != nil {return nil, err}// 生成JWTaccessToken, refreshToken, err := jwt.GenToken(user.UserID, user.UserName)if err != nil {return}user.AccessToken = accessTokenuser.RefreshToken = refreshTokenreturn
}

5. 数据访问层与数据库交互

dao/mysql 中的方法负责与数据库交互,如查询用户信息:

// Login 登录业务
func Login(user *models.User) (err error) {originPassword := user.Password // 记录一下原始密码(用户登录的密码)sqlStr := "select user_id, username, password,avatar,gender from user where username = ?"err = db.Get(user, sqlStr, user.UserName)// 查询数据库出错if err != nil && err != sql.ErrNoRows {return err}// 用户不存在if err == sql.ErrNoRows {return errors.New(ErrorUserNotExit)}// 生成加密密码与查询到的密码比较password := encryptPassword([]byte(originPassword))if user.Password != password {return errors.New(ErrorPasswordWrong)}return nil
}

6. 返回响应

数据流逐步返回到控制器,最终通过 JSON 格式的响应返回给前端。

// 3、返回响应ResponseSuccess(c, gin.H{"user_id":        fmt.Sprintf("%d", user.UserID), //js识别的最大值:id值大于1<<53-1  int64: i<<63-1"user_name":      user.UserName,"access_token":   user.AccessToken,"refresh_token":  user.RefreshToken,"avatarImageUrl": user.Avatar,"gender":         user.Gender,})

一些项目中的model建立技巧

1. 字段标签

(1)json 标签
指定 JSON 数据的序列化和反序列化的字段名,如:

	UserID uint64 `json:"user_id,string"`

json:"user_id,string" 表示 JSON 中的 user_id 字段是字符串格式,但在程序中使用 uint64 类型。适用于后端接收到的数字 ID 被前端以字符串形式传递的场景,避免因类型不匹配导致解析失败。

(2) db 标签
用于指定数据库中表字段的映射,如:

	Avatar string `db:"avatar"`

可以避免数据库字段名与代码字段名不一致引发问题。

(3) 多标签结合

字段可以同时使用 jsondb 标签,分别处理 JSON 和数据库交互需求,如:

	Email string `json:"email" db:"email"`

这样做可以使同一字段同时适配 JSON 序列化和数据库映射,代码更加清晰统一。

2. 自定义 UnmarshalJSON 方法

        在前端提交到后端的字段中,某些字段(usernamepassword)可能是某个业务逻辑的必须项,而Golang 的默认 JSON 反序列化无法验证字段是否缺失和符合项目要求,这种情况下,我们可以通过自定义UnmarshalJSON 方法来解决这个问题。

实现细节
定义一个嵌套的 required 临时结构体,声明必需字段:

required := struct {Avatar   string `json:"avatar" db:"avatar"`UserName string `json:"username" db:"username"`Password string `json:"password" db:"password"`Email    string `json:"email" db:"email"`Gender   int    `json:"gender" db:"gender"`
}{}

然后通过 json.Unmarshal 解析 JSON 数据:

err = json.Unmarshal(data, &required)

验证必填字段是否存在:

if len(required.UserName) == 0 {err = errors.New("缺少必填字段username")
} else if len(required.Password) == 0 {err = errors.New("缺少必填字段password")
}

如果校验通过,将值赋给 User 结构体:

  u.Avatar = required.Avataru.UserName = required.UserName

3. binding 标签

binding 标签是 Gin 框架提供的参数校验机制,使用方法如下:

UserName string `json:"username" binding:"required"`

其中,binding:"required" 表示字段为必填项。 如果前端未提供 username,请求会被 Gin 自动拒绝,返回 400 Bad Request。


http://www.ppmy.cn/news/1557354.html

相关文章

RabbitMQ的核心组件有哪些?

大家好&#xff0c;我是锋哥。今天分享关于【RabbitMQ的核心组件有哪些&#xff1f;】面试题。希望对大家有帮助&#xff1b; RabbitMQ的核心组件有哪些&#xff1f; 1000道 互联网大厂Java工程师 精选面试题-Java资源分享网 RabbitMQ是一个开源的消息代理&#xff08;Messag…

Restaurants WebAPI(三)——Serilog/FluenValidation

文章目录 项目地址一、Serilog使用1.1 安装 Serilog1.2 注册日志服务1.3 设置日志级别和详情1.4 配置到文件里1.5 给不同的环境配置日志1.5.1 配置appsettings.Development.json二、Swagger的使用三、自定义Exception中间件3.1 使用FluentValidation项目地址 教程作者:教程地址…

TinyML在OBD-II边缘设备上燃油类型分类的实现与优化

论文标题&#xff1a;TinyML Implementation and Optimization for Fuel Type Classification on OBD-II Edge Device&#xff08;TinyML在OBD-II边缘设备上燃油类型分类的实现与优化&#xff09; 作者信息&#xff1a;Miguel Amaral, Morsinaldo Medeiros, Matheus Andrade, …

高并发架构设计:从 Java Callable 到 CompletableFuture 的进阶修炼

前言 在现代的分布式系统中&#xff0c;高并发性能是一个决定系统能否成功的关键因素。而 Java 作为主流的后端开发语言&#xff0c;也提供了许多强大的工具来帮助我们处理并发任务。今天&#xff0c;我们将通过两个关键工具来讲解高并发架构设计的技巧&#xff1a;Callable 和…

1688跨境代购代采业务:利用API实现自动化信息化

在全球化贸易日益频繁的今天&#xff0c;跨境电商已成为推动国际贸易的重要力量。作为中国电商的源头货盘&#xff0c;1688平台拥有大量的工厂、品牌商和一级批发商&#xff0c;为外贸人提供了极其丰富的货源。如何利用这一平台优势&#xff0c;开展跨境代购代采业务&#xff0…

基础二层交换组网(静态IP)

拓扑图 实验目的 掌握并了解网络中静态IP默认无路由影响的情况。 实验过程 1.创建拓扑 2.配置PC1、PC2 IP地址 PC1&#xff1a; IP地址&#xff1a;192.168.0.1 子网掩码&#xff1a;255.255.255.0PC2&#xff1a; IP地址&#xff1a;192.168.0.2 子网掩码&#xff1a;255.…

selenium 在已打开浏览器上继续调试

关闭浏览器&#xff0c;终端执行如下指令&#xff0c;--user-data-dir换成自己的User Data路径 chrome.exe --remote-debugging-port9222 --user-data-dir"C:\Users\xxx\AppData\Local\Google\Chrome\User Data" 会打开浏览器&#xff0c;打开百度&#xff0c;如下状…

南海信息学竞赛高频考查点系列-1枚举2下标记数3部分和

这套题包含了历年真题&#xff0c;十分重要&#xff01;&#xff01;&#xff01;&#xff01;要考试的同学可以参考一下&#xff01;&#xff01; 此套题限时3小时。 #A. C05.L05.枚举及优化&#xff08;二&#xff09;.课堂练习4.线段覆盖 题目描述 在一条数轴上&#xf…