GO学习记录 —— 创建一个GO项目
文章目录
- 前言
- 一、项目介绍
- 二、目录介绍
- 三、创建过程
- 1.引入Gin框架、创建main
- 2.加载配置文件
- 3.连接MySQL、redis
- 4.创建结构体
- 5.错误处理、返回响应处理
前言
代码地址
下载地址:https://github.com/Lee-ZiMu/Golang-Init.git
一、项目介绍
1、使用Gin框架来创建项目
2、使用viper读取配置文件
3、使用gorm连接MySQL、生成结构体
4、使用redigo连接redis
5、包含异常处理、jwt、日志、ws的初始化或使用
二、目录介绍
其中业务代码部分可以拆分为多个业务模块
其中业务代码部分:
公共组件部分
三、创建过程
1.引入Gin框架、创建main
Gin文档地址:https://www.topgoer.com/gin%E6%A1%86%E6%9E%B6/
package mainimport ("github.com/gin-gonic/gin"viperInit "github.com/spf13/viper""hy_heymate/api/route""hy_heymate/common/cron""hy_heymate/common/logger""hy_heymate/config/viper""hy_heymate/database"
)func main() {r := gin.Default()// 读取配置文件viper.Init("../config/config.yaml")// 初始化日志logger.Init()// 连接mysql数据库database.ConnectToMySQL()// 连接redisdatabase.ConnectToRedis()// 加载路由route.LoadRouters(r)// 定时cron.Cron()r.Run(":" + viperInit.GetString("server.port"))}
2.加载配置文件
package viperimport ("fmt""github.com/fsnotify/fsnotify""github.com/spf13/viper"
)func Init(configPath string) {//制定配置文件的路径viper.SetConfigFile(configPath)// 读取配置信息err := viper.ReadInConfig()if err != nil {// 读取配置信息失败panic(fmt.Errorf("配置文件读取失败,错误信息:%v", err))}//监听修改viper.WatchConfig()//为配置修改增加一个回调函数viper.OnConfigChange(func(in fsnotify.Event) {fmt.Println("配置文件修改了...")})
}
3.连接MySQL、redis
连接mysql
package databaseimport ("fmt"viperInit "github.com/spf13/viper""gorm.io/driver/mysql""gorm.io/gorm""gorm.io/gorm/logger""gorm.io/gorm/schema""hy_heymate/common/file""hy_heymate/gen/query""log""os""time"
)var db = new(gorm.DB)func ConnectToMySQL() {// 创建日志文件夹if err := file.MkdirAll(viperInit.GetString("mysql.logpath")); err != nil {panic(err)}logFile, err := os.Create(viperInit.GetString("mysql.logpath") + "db.log")if err != nil {panic(err)}sqlStr := fmt.Sprintf("%s:%s@tcp(%s:%d)/%s?%s",viperInit.GetString("mysql.username"),viperInit.GetString("mysql.password"),viperInit.GetString("mysql.host"),viperInit.GetInt("mysql.port"),viperInit.GetString("mysql.dbname"),viperInit.GetString("mysql.conf"),)db, err = gorm.Open(mysql.Open(sqlStr), &gorm.Config{Logger: logger.New(log.New(logFile, "\n", log.LstdFlags),logger.Config{SlowThreshold: time.Second,Colorful: true,LogLevel: logger.Info,}),NamingStrategy: schema.NamingStrategy{SingularTable: true},})if err != nil {panic(fmt.Errorf("mysql连接失败,错误信息:%s", err))}sqlDB, _ := db.DB()sqlDB.SetMaxIdleConns(viperInit.GetInt("mysql.maxIdleConns")) // 最大连接数sqlDB.SetMaxOpenConns(viperInit.GetInt("mysql.maxOpenConns")) // 最大打开连接数sqlDB.SetConnMaxLifetime(time.Duration(viperInit.GetInt64("mysql.vonnMaxLifetime")) * time.Second) // 设置可以重用连接的最大时间量query.SetDefault(db)}func Get() *gorm.DB {return db
}
连接redis
package databaseimport ("fmt""github.com/gomodule/redigo/redis"viperInit "github.com/spf13/viper""time"
)var (RedisClient *redis.Pool
)func ConnectToRedis() {host := viperInit.GetString("redis.host")port := viperInit.GetString("redis.port")//username := viperInit.GetString("redis.username")//password := viperInit.GetString("redis.password")MaxIdle := viperInit.GetInt("redis.MaxIdle")MaxActive := viperInit.GetInt("redis.MaxActive")IdleTimeout := viperInit.GetInt("redis.IdleTimeout")pool := &redis.Pool{MaxIdle: MaxIdle, // 最大空闲连接数MaxActive: MaxActive, // 最大活跃连接数IdleTimeout: time.Duration(IdleTimeout) * time.Second, // 空闲连接超时时间Dial: func() (redis.Conn, error) {c, err := redis.Dial("tcp", host+":"+port)if err != nil {panic(fmt.Errorf("redis连接失败,错误信息:%s", err))}return c, err},}RedisClient = pool
}
4.创建结构体
package mainimport ("fmt"viperInit "github.com/spf13/viper""gorm.io/driver/mysql""gorm.io/gen""gorm.io/gorm""hy_heymate/config/viper"
)func main() {// 读取配置文件viper.Init("../config/config.yaml")// 连接mysql数据库sqlStr := fmt.Sprintf("%s:%s@tcp(%s:%d)/%s?%s",viperInit.GetString("mysql.username"),viperInit.GetString("mysql.password"),viperInit.GetString("mysql.host"),viperInit.GetInt("mysql.port"),viperInit.GetString("mysql.dbname"),viperInit.GetString("mysql.conf"),)var err errordb, err := gorm.Open(mysql.Open(sqlStr), &gorm.Config{})if err != nil {panic(fmt.Errorf("mysql连接失败,错误信息:%s", err))}// 生成实例g := gen.NewGenerator(gen.Config{// 相对执行`go run`时的路径, 会自动创建目录OutPath: "./query",// WithDefaultQuery 生成默认查询结构体(作为全局变量使用), 即`Q`结构体和其字段(各表模型)// WithoutContext 生成没有context调用限制的代码供查询// WithQueryInterface 生成interface形式的查询代码(可导出), 如`Where()`方法返回的就是一个可导出的接口类型Mode: gen.WithDefaultQuery | gen.WithQueryInterface,// 表字段可为 null 值时, 对应结体字段使用指针类型FieldNullable: true, // generate pointer when field is nullable// 表字段默认值与模型结构体字段零值不一致的字段, 在插入数据时需要赋值该字段值为零值的, 结构体字段须是指针类型才能成功, 即`FieldCoverable:true`配置下生成的结构体字段.// 因为在插入时遇到字段为零值的会被GORM赋予默认值. 如字段`age`表默认值为10, 即使你显式设置为0最后也会被GORM设为10提交.// 如果该字段没有上面提到的插入时赋零值的特殊需要, 则字段为非指针类型使用起来会比较方便.FieldCoverable: false, // generate pointer when field has default value, to fix problem zero value cannot be assign: https://gorm.io/docs/create.html#Default-Values// 模型结构体字段的数字类型的符号表示是否与表字段的一致, `false`指示都用有符号类型FieldSignable: false, // detect integer field's unsigned type, adjust generated data type// 生成 gorm 标签的字段索引属性FieldWithIndexTag: false, // generate with gorm index tag// 生成 gorm 标签的字段类型属性FieldWithTypeTag: true, // generate with gorm column type tag})// 设置目标 dbg.UseDB(db)// 自定义字段的数据类型// 统一数字类型为int64,兼容protobufdataMap := map[string]func(detailType gorm.ColumnType) (dataType string){"tinyint": func(detailType gorm.ColumnType) (dataType string) { return "int64" },"smallint": func(detailType gorm.ColumnType) (dataType string) { return "int64" },"mediumint": func(detailType gorm.ColumnType) (dataType string) { return "int64" },"bigint": func(detailType gorm.ColumnType) (dataType string) { return "int64" },"int": func(detailType gorm.ColumnType) (dataType string) { return "int64" },}// 要先于`ApplyBasic`执行g.WithDataTypeMap(dataMap)// 自定义模型结体字段的标签// 将特定字段名的 json 标签加上`string`属性,即 MarshalJSON 时该字段由数字类型转成字符串类型//jsonField := gen.FieldJSONTagWithNS(func(columnName string) (tagContent string) {// toStringField := `balance, `// if strings.Contains(toStringField, columnName) {// return columnName + ",string"// }// return columnName//})// 将非默认字段名的字段定义为自动时间戳和软删除字段;// 自动时间戳默认字段名为:`updated_at`、`created_at, 表字段数据类型为: INT 或 DATETIME// 软删除默认字段名为:`deleted_at`, 表字段数据类型为: DATETIME//autoUpdateTimeField := gen.FieldGORMTag("update_time", "column:update_time;type:int unsigned;autoUpdateTime")//autoCreateTimeField := gen.FieldGORMTag("create_time", "column:create_time;type:int unsigned;autoCreateTime")//softDeleteField := gen.FieldType("delete_time", "soft_delete.DeletedAt")// 模型自定义选项组//fieldOpts := []gen.ModelOpt{jsonField, autoCreateTimeField, autoUpdateTimeField, softDeleteField}// 创建模型的结构体,生成文件在 model 目录; 先创建的结果会被后面创建的覆盖// 这里创建个别模型仅仅是为了拿到`*generate.QueryStructMeta`类型对象用于后面的模型关联操作中//User := g.GenerateModel("user")// 创建全部模型文件, 并覆盖前面创建的同名模型allModel := g.GenerateAllTable()// 创建有关联关系的模型文件// 可以用于指定外键//Score := g.GenerateModel("score",// append(// fieldOpts,// // user 一对多 address 关联, 外键`uid`在 address 表中// gen.FieldRelate(field.HasMany, "user", User, &field.RelateConfig{GORMTag: "foreignKey:UID"}),// )...,//)// 创建模型的方法,生成文件在 query 目录; 先创建结果不会被后创建的覆盖//g.ApplyBasic(User)g.ApplyBasic(allModel...)g.Execute()}
5.错误处理、返回响应处理
在service层对错误进行处理
在controller返回给ginresult
在ginResult处理
如果错误是我们定义的错误,那么返回给用户,如果不是则返回通用错误 ServeError
package resultimport ("fmt""github.com/gin-gonic/gin""github.com/pkg/errors""go.uber.org/zap""hy_heymate/common/errType""hy_heymate/common/logger""net/http"
)func GinResult(c *gin.Context, data interface{}, err error) {if err == nil {// 成功返回c.JSON(http.StatusOK, Success(data))return}// 打印错误日志到控制台log, _ := zap.NewDevelopment()log.Error(fmt.Sprintf("操作失败,错误信息:%s", err))// 错误返回errcode := errType.ServeErrorerrmsg := errType.GetErrorMsg(errType.ServeError)causeErr := errors.Cause(err) // err类型e, ok := causeErr.(*errType.CodeError)if ok { // 自定义错误类型// 自定义CodeErrorerrcode = e.GetErrCode()errmsg = e.GetErrMsg()}logger.Errorf("http-error: %v", err)c.JSON(http.StatusOK, Error(errcode, errmsg))}
定义的错误如下,可根据业务需求添加
相关文章:

GO学习记录 —— 创建一个GO项目
文章目录 前言一、项目介绍二、目录介绍三、创建过程1.引入Gin框架、创建main2.加载配置文件3.连接MySQL、redis4.创建结构体5.错误处理、返回响应处理 前言 代码地址 下载地址:https://github.com/Lee-ZiMu/Golang-Init.git 一、项目介绍 1、使用Gin框架来创建项…...
C语言中的goto语句:使用、争议与最佳实践
各位少年: 引言: 在C语言编程中,goto语句是一个历史悠久且颇具争议的控制流结构。作为无条件跳转指令,它允许程序执行从当前点直接跳转到同一函数内的任意位置,由一个标签(label)来指定目标。尽…...
wpf-动态设置组件【按钮为例】样式
文章速览 解决方案具体实现Converter 部分创建样式Binding样式 坚持记录实属不易,希望友善多金的码友能够随手点一个赞。 共同创建氛围更加良好的开发者社区! 谢谢~ 解决方案 创建一个Converter,返回对应的style实现对应的修改 创建多个样式…...

40道MyBatis面试题带答案(很全)
1. 什么是MyBatis (1)Mybatis是一个半ORM(对象关系映射)框架,它内部封装了JDBC,开发时只需要关注SQL语句本身,不需要花费精力去处理加载驱动、创建连接、创建statement等繁杂的过程。程序员直接…...
python:PyCharm更改.PyCharm配置文件夹存储位置
关联账号文章:另外的账号 在启动 PyCharm 后选择 Help -> Edit Custom Properties 的选项,弹出: 选择 Create ,之后在文件中添加配置文件新的存储位置即可,例如: idea.config.pathD:/Program Files/.Py…...

Centos安装Kafka(KRaft模式)
1. KRaft引入 Kafka是一种高吞吐量的分布式发布订阅消息系统,它可以处理消费者在网站中的所有动作流数据。其核心组件包含Producer、Broker、Consumer,以及依赖的Zookeeper集群。其中Zookeeper集群是Kafka用来负责集群元数据的管理、控制器的选举等。 由…...

学习笔记13——Spring整合Mybatis、junit、AOP、事务
学习笔记系列开头惯例发布一些寻亲消息 链接:https://baobeihuijia.com/bbhj/ Mybatis - Spring(使用第三方包new一个对象bean) 原始的Mybatis与数据库交互【通过sqlmapconfig来配置和连接】 初始化SqlSessionFactory获得连接获取数据层接口…...
【12月比赛合集】4场可报名的「创新应用」、「数据分析」和「程序设计」大奖赛,任君挑选!
CompHub[1] 实时聚合多平台的数据类(Kaggle、天池…)和OJ类(Leetcode、牛客…)比赛。本账号会推送最新的比赛消息,欢迎关注! 以下信息仅供参考,以比赛官网为准 目录 数据分析赛(1场比赛)程序设计赛&#…...

Cisco模拟器-企业网络部署
某企业园区网有:2个分厂(分别是:零件分厂、总装分厂)1个总厂网络中心 1个总厂会议室; (1)每个分厂有自己的路由器,均各有:1个楼宇分厂网络中心 每个楼宇均包含&#x…...

WPF+Halcon 培训项目实战(12):WPF导出匹配模板
文章目录 前言相关链接项目专栏运行环境匹配图片WPF导出匹配模板如何了解Halcon和C#代码的对应关系逻辑分析:添加截取ROI功能基类矩形圆形 生成导出模板运行结果:可能的报错你的文件路径不存在你选择的区域的内容有效信息过少 前言 为了更好地去学习WPF…...

uniapp中uview组件库的丰富Upload 上传上午用法
目录 基础用法 #上传视频 #文件预览 #隐藏上传按钮 #限制上传数量 #自定义上传样式 API #Props #Methods #Slot #Events 基础用法 可以通过设置fileList参数(数组,元素为对象),显示预置的图片。其中元素的url属性为图片路径 <template>…...

Unity关于动画混合树(Blend Tree)的使用
在动画与动画的切换过程中,常因为两个动画之间的差距过大,而显得动画的切换很不自然。 这时候就需要动画混合树Blend Tree这个功能。使用混合树可以将多个动画混合在一起,例如在处理角色的移动中,走动画与跑动画切换的时候&#x…...

怎么下载landsat 8影像并在ArcGIS Pro中进行波段组合
Landsat 8(前身为Landsat数据连续性任务,或 LDCM)于2013年2月11日由 Atlas-V火箭从加利福尼亚州范登堡空军基地发射升空,这里为大家介绍一下该数据的下载的方法,希望能对你有所帮助。 注册账号 如果之前已经注册过的…...
编程新手IDE
身为一个前端开发者,我深知一个好的开发环境对于编程体验的重要性。对于新手来说,选择一个合适的IDE(集成开发环境)更是至关重要。一个好的IDE可以提高编程效率,减少错误,让新手更专注于学习编程本身。 今…...
如何将一个JSON字符串解析为JavaScript对象或值
JSON.parse(JSON.stringify(data)) 将后端传入的JSON数据data放入该方法的参数中,返回的结果就是JavaScript对象 比如将后端传入的对象key作为对象,而不是字符串双引号格式 {"path": "/home","name": "home",…...

idea配置docker推送本地镜像到远程私有仓库
目录 1,搭建远程Docker 私有仓库 Docker registry 2,Windows10/11系统上安装Docker Desktop 3,idea 配置远程私有仓库地址 4,idea 配置Docker 5,idea在本地构建镜像 6,推送本地Docker镜像到远程 Dock…...

Spring Boot学习随笔- 集成MyBatis-Plus(二)条件查询QueryWrapper、聚合函数的使用、Lambda条件查询
学习视频:【编程不良人】Mybatis-Plus整合SpringBoot实战教程,提高的你开发效率,后端人员必备! 查询方法详解 普通查询 // 根据主键id去查询单个结果的。 Test public void selectById() {User user userMapper.selectById(1739970502337392641L);System.out.print…...
十二、K8S之污点和容忍
污点和容忍 一、概念 k8s 集群中可能管理着非常庞大的服务器,这些服务器可能是各种各样不同类型的,比如机房、地理位置、配置等,有些是计算型节点,有些是存储型节点,此时我们希望能更好的将 pod 调度到与之需求更匹配…...
llvm后端之指令选择源码分析
llvm后端之指令选择源码分析 引言1 主要流程1.1 参数降级1.2 构建DAG1.3 类型合法化1.4 向量合法化1.5 DAG合法化1.6 DAG合并 2 目标实现2.1 TargetLowering2.2 SelectionDAGISel 引言 llvm后端指令选择主要是class SelectionDAGISel的子类实现。整个过程将llvm IR转为有向无环…...

【消息中间件】Rabbitmq消息可靠性、持久化机制、各种消费
原文作者:我辈李想 版权声明:文章原创,转载时请务必加上原文超链接、作者信息和本声明。 文章目录 前言一、常见用法1.消息可靠性2.持久化机制3.消息积压批量消费:增加 prefetch 的数量,提高单次连接的消息数并发消费:…...

eNSP-Cloud(实现本地电脑与eNSP内设备之间通信)
说明: 想象一下,你正在用eNSP搭建一个虚拟的网络世界,里面有虚拟的路由器、交换机、电脑(PC)等等。这些设备都在你的电脑里面“运行”,它们之间可以互相通信,就像一个封闭的小王国。 但是&#…...

日语AI面试高效通关秘籍:专业解读与青柚面试智能助攻
在如今就业市场竞争日益激烈的背景下,越来越多的求职者将目光投向了日本及中日双语岗位。但是,一场日语面试往往让许多人感到步履维艰。你是否也曾因为面试官抛出的“刁钻问题”而心生畏惧?面对生疏的日语交流环境,即便提前恶补了…...
OpenLayers 可视化之热力图
注:当前使用的是 ol 5.3.0 版本,天地图使用的key请到天地图官网申请,并替换为自己的key 热力图(Heatmap)又叫热点图,是一种通过特殊高亮显示事物密度分布、变化趋势的数据可视化技术。采用颜色的深浅来显示…...

黑马Mybatis
Mybatis 表现层:页面展示 业务层:逻辑处理 持久层:持久数据化保存 在这里插入图片描述 Mybatis快速入门 
中南大学无人机智能体的全面评估!BEDI:用于评估无人机上具身智能体的综合性基准测试
作者:Mingning Guo, Mengwei Wu, Jiarun He, Shaoxian Li, Haifeng Li, Chao Tao单位:中南大学地球科学与信息物理学院论文标题:BEDI: A Comprehensive Benchmark for Evaluating Embodied Agents on UAVs论文链接:https://arxiv.…...
工业自动化时代的精准装配革新:迁移科技3D视觉系统如何重塑机器人定位装配
AI3D视觉的工业赋能者 迁移科技成立于2017年,作为行业领先的3D工业相机及视觉系统供应商,累计完成数亿元融资。其核心技术覆盖硬件设计、算法优化及软件集成,通过稳定、易用、高回报的AI3D视觉系统,为汽车、新能源、金属制造等行…...
Swagger和OpenApi的前世今生
Swagger与OpenAPI的关系演进是API标准化进程中的重要篇章,二者共同塑造了现代RESTful API的开发范式。 本期就扒一扒其技术演进的关键节点与核心逻辑: 🔄 一、起源与初创期:Swagger的诞生(2010-2014) 核心…...
Java 二维码
Java 二维码 **技术:**谷歌 ZXing 实现 首先添加依赖 <!-- 二维码依赖 --><dependency><groupId>com.google.zxing</groupId><artifactId>core</artifactId><version>3.5.1</version></dependency><de…...

听写流程自动化实践,轻量级教育辅助
随着智能教育工具的发展,越来越多的传统学习方式正在被数字化、自动化所优化。听写作为语文、英语等学科中重要的基础训练形式,也迎来了更高效的解决方案。 这是一款轻量但功能强大的听写辅助工具。它是基于本地词库与可选在线语音引擎构建,…...
在QWebEngineView上实现鼠标、触摸等事件捕获的解决方案
这个问题我看其他博主也写了,要么要会员、要么写的乱七八糟。这里我整理一下,把问题说清楚并且给出代码,拿去用就行,照着葫芦画瓢。 问题 在继承QWebEngineView后,重写mousePressEvent或event函数无法捕获鼠标按下事…...