Go-Gin-Example 第八部分 优化配置接口+图片上传功能

文章目录

    • 前情提要
    • 本节目标
  • 优化配置结构
    • 讲解
    • 落实
      • 修改配置文件
      • 优化配置读取及设置初始化顺序
        • 第一步
      • 验证
    • 抽离file
  • 实现上传图片接口
    • 图片名加密
    • 封装image的处理逻辑
    • 编写上传图片的业务逻辑
    • 增加图片上传的路由
  • 验证
  • 实现前端访问 http.FileServer
    • r.StaticFS
    • 修改文章接口
      • 新增、更新文章接口

前情提要

学习项目github地址

上一部分学习笔记

本节目标

  • 优化配置结构(因为配置项越来越多)
  • 抽离 原 loggingFile 便于公用(logging、upload 各保有一份并不合适)
  • 实现上传图片接口(需限制文件格式、大小)
  • 修改文章接口(需支持封面地址参数)
  • 增加 blog_article (文章)的数据库字段
  • 实现 http.FileServer

优化配置结构

讲解

在先前章节中,我们通过读取KEY的方式读取配置项(建立setting模块)
本次需求中,需要增加图片的配置项,总体就有些冗余了

我们采用以下解决方法:

  • 映射结构体:使用 MapTo 来设置配置参数
  • 配置统管:所有的配置项统管到 setting

落实

修改配置文件

修改 conf/app.ini

增加了 5 个配置项用于上传图片的功能,4文件日志方面的配置项

[app]
PageSize = 10
JwtSecret = 233RuntimeRootPath = runtime/ImagePrefixUrl = http://127.0.0.1:8000
ImageSavePath = upload/images/
# MB
ImageMaxSize = 5
ImageAllowExts = .jpg,.jpeg,.pngLogSavePath = logs/
LogSaveName = log
LogFileExt = log
TimeFormat = 20060102[server]
#debug or release
RunMode = debug
HttpPort = 8000
ReadTimeout = 60
WriteTimeout = 60[database]
Type = mysql
User = root
Password = rootroot
Host = 127.0.0.1:3306
Name = blog
TablePrefix = blog_

优化配置读取及设置初始化顺序

第一步

将散落在其他文件里的配置都删掉,统一在 setting 中处理以及修改 init 函数为 Setup 方法

  1. 打开 pkg/setting/setting.go 文件,修改如下:
package modelsimport ("fmt""log""time""github.com/jinzhu/gorm"_ "github.com/jinzhu/gorm/dialects/mysql""github.com/kingsill/gin-example/pkg/setting"
)// 定义一个全局的数据库连接变量
var db *gorm.DB// Model 设定常用结构体,可以作为匿名结构体嵌入到别的表格对应的结构体
type Model struct {ID         int `gorm:"primary_key" json:"id"`CreatedOn  int `json:"created_on"`ModifiedOn int `json:"modified_on"`DeletedOn  int `json:"deleted_on"`
}func Setup() {//配置文件加载Cfg, err := ini.Load("conf/app.ini")if err != nil {log.Fatalf("Fail to parse 'conf/app.ini': %v", err)}//将app section 部分映射到AppSetting结构体上err = Cfg.Section("app").MapTo(AppSetting)if err != nil {log.Fatalf("Cfg.MapTo AppSetting err: %v", err)}//将图片最大大小设置从5字节Byte转换为5兆字节MBAppSetting.ImageMaxSize = AppSetting.ImageMaxSize * 1024 * 1024err = Cfg.Section("server").MapTo(ServerSetting)if err != nil {log.Fatalf("Cfg.MapTo ServerSetting err: %v", err)}//将读取时自动转换的类型转换为时间间隔了,只不过是最小单位纳秒ServerSetting.ReadTimeout = ServerSetting.ReadTimeout * time.SecondServerSetting.WriteTimeout = ServerSetting.WriteTimeout * time.Seconderr = Cfg.Section("database").MapTo(DatabaseSetting)if err != nil {log.Fatalf("Cfg.MapTo DatabaseSetting err: %v", err)}
}

在这里,我们做了如下几件事:

  • 编写与配置项保持一致的结构体(App、Server、Database
  • 使用 MapTo 将配置项映射到结构体上
  • 对一些需特殊设置的配置项进行再赋值
  1. 修改models.go
    init函数改为Setup方法,将独立读取的DB配置项删除,改为统一读取setting
package modelsimport (
...
)// 定义一个全局的数据库连接变量
var db *gorm.DB// Model 设定常用结构体,可以作为匿名结构体嵌入到别的表格对应的结构体
type Model struct {ID         int `gorm:"primary_key" json:"id"`CreatedOn  int `json:"created_on"`ModifiedOn int `json:"modified_on"`DeletedOn  int `json:"deleted_on"`
}func Setup() {var err error//使用gorm框架初始化数据库连接db, err = gorm.Open(setting.DatabaseSetting.Type, fmt.Sprintf("%s:%s@tcp(%s)/%s?charset=utf8&parseTime=True&loc=Local",setting.DatabaseSetting.User,setting.DatabaseSetting.Password,setting.DatabaseSetting.Host,setting.DatabaseSetting.Name))if err != nil {log.Println(err)}//自定义默认表的表名,使用匿名函数,在原默认表名的前面加上配置文件中定义的前缀gorm.DefaultTableNameHandler = func(db *gorm.DB, defaultTableName string) string {return setting.DatabaseSetting.TablePrefix + defaultTableName}//gorm默认使用复数映射,当前设置后即进行严格匹配db.SingularTable(true)//log记录打开db.LogMode(true)//进行连接池设置db.DB().SetMaxIdleConns(10)db.DB().SetMaxOpenConns(100)//替换Create和Update回调函数db.Callback().Create().Replace("gorm:update_time_stamp", updateTimeStampForCreateCallback)db.Callback().Update().Replace("gorm:update_time_stamp", updateTimeStampForUpdateCallback)//添加删除的回调CallBacksdb.Callback().Delete().Replace("gorm:delete", deleteCallback)
}// CloseDB 与数据库断开连接函数
func CloseDB() {defer db.Close()
}// updateTimeStampForCreateCallback 在创建记录时设置 `CreatedOn`, `ModifiedOn`
func updateTimeStampForCreateCallback(scope *gorm.Scope) {...
}// updateTimeStampForUpdateCallback 在更新记录时设置 `ModifyOn`
func updateTimeStampForUpdateCallback(scope *gorm.Scope) {
...
}// 设定delete操作的callback逻辑
func deleteCallback(scope *gorm.Scope) {...
}// 判断是否为空来进行空格插入,防止sql注入,保证安全性
func addExtraSpaceIfExist(str string) string {...
}
  1. 修改log.go
    init函数改为Setup方法
func Setup() {//获取log文件目录filePath := getLogFileFullPath()//得到log文件句柄F = openLogFile(filePath)//创建一个新的日志记录器logger = log.New(F, DefaultPrefix, log.LstdFlags)
}
  1. 修改pkg/logging/file.go

    独立的 LOG 配置项删除,改为统一读取 setting,修改这两个函数即可

// 返回log文件的前缀路径,算是一个具有仪式感的函数
func getLogFilePath() string {return fmt.Sprintf("%s", setting.AppSetting.LogSavePath)
}// 获得log文件的整体路径,以当前日期作为.log文件的名字
func getLogFileFullPath() string {prefixPath := getLogFilePath()suffixPath := fmt.Sprintf("%s%s.%s", setting.AppSetting.LogSaveName, time.Now().Format(setting.AppSetting.TimeFormat), setting.AppSetting.LogFileExt)return fmt.Sprintf("%s%s", prefixPath, suffixPath)
}
  1. 其他漏下的未改为统一读取setting的根据报错进行修改即可

验证

在这里为止,针对本需求的配置优化就完毕了,你需要执行 go run main.go 验证一下你的功能是否正常哦

抽离file

  1. pkg目录下新建file/file.go
package fileimport ("io""mime/multipart""os""path"
)// GetSize multipart.file用于处理HTTP请求中文件上传到类型   os.file则主要是本地文件的操作
func GetSize(f multipart.File) (int, error) {content, err := io.ReadAll(f)return len(content), err
}// GetExt 获取文件扩展名
func GetExt(filename string) string {return path.Ext(filename)
}// CheckExist 检查文件是否存在
func CheckExist(src string) bool {//os.stat用于获取文件的相关信息_, err := os.Stat(src)return os.IsNotExist(err)
}// CheckPermission 检查访问文件的权限
func CheckPermission(src string) bool {_, err := os.Stat(src)//检查是否有访问文件的权限return os.IsPermission(err)
}// IsNotExistMkDir 检查是否存在目录,不存在则创建目录
func IsNotExistMkDir(src string) error {if notExist := CheckExist(src); notExist == true {if err := MkDir(src); err != nil {return err}}return nil
}// MkDir 创建目录
func MkDir(src string) error {err := os.MkdirAll(src, os.ModePerm) //权限0777,权限拉满if err != nil {return err}return nil
}// Open 算是简单包装os.openfile
func Open(name string, flag int, perm os.FileMode) (*os.File, error) {f, err := os.OpenFile(name, flag, perm)if err != nil {return nil, err}return f, nil
}

在这里我们用到了 mime/multipart 包,它主要实现了 MIMEmultipart解析,主要适用于 HTTP 和常见浏览器生成的 multipart 主体

  1. 修改原logging包的方法
  • 修改pkg/logging/file.go
package loggingimport ("fmt""github.com/kingsill/gin-example/pkg/file""github.com/kingsill/gin-example/pkg/setting""os""time"
)// 返回log文件的前缀路径,算是一个具有仪式感的函数
func getLogFilePath() string {return fmt.Sprintf("%s", setting.AppSetting.LogSavePath)
}// 获得log文件的整体路径,以当前日期作为.log文件的名字 runtime/log20010212.log
func getLogFileFullPath() string {prefixPath := getLogFilePath()suffixPath := fmt.Sprintf("%s%s.%s",setting.AppSetting.LogSaveName,time.Now().Format(setting.AppSetting.TimeFormat),setting.AppSetting.LogFileExt,)return fmt.Sprintf("%s%s", prefixPath, suffixPath)
}// 打开日志文件,返回写入的句柄handle
func openLogFile() (*os.File, error) {//获取文件整体路径fileName := getLogFileFullPath()//创建目录mkDir()//如果.log文件不存在,这里会创建一个handle, err := file.Open(fileName, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644)if err != nil {return nil, fmt.Errorf("fail to open:%s\n", fileName)}return handle, nil
}// 创建log目录
func mkDir() {//获得当前目录 dir: /home/wang2/gin-exampledir, _ := os.Getwd()//检查目录访问权限perm := file.CheckPermission(getLogFilePath())if perm == true {panic("Permission denied")}//如果目录不存在,创建目录err := file.IsNotExistMkDir(dir + "/" + getLogFilePath())if err != nil {panic(err)}
}
  • 修改pkg/logging/log.go
    由于原方法传参有变,这里也进行相关调整
...// Setup 自定义logger的初始化
func Setup() {var err error//得到log文件句柄F, err = openLogFile()if err != nil {log.Fatalln(err)}//创建一个新的日志记录器logger = log.New(F, DefaultPrefix, log.LstdFlags)
}
...

实现上传图片接口

首先需要在 blog_article 中增加字段 cover_image_url,格式为 varchar(255) DEFAULT '' COMMENT '封面图片地址'

alter table blog_article add cover_image_url varchar(255) DEFAULT '' COMMENT '封面图片地址';

图片名加密

我们通过 MD5 对图片进行加密,防止图片名暴露
util目录下新建md5.go,写入文件内容

package utilimport ("crypto/md5""encoding/hex"
)// EncodeMD5 计算给定字符的MD5哈希值,返回其十六进制表示
func EncodeMD5(value string) string {//创建一个新的MD5计算器实例m := md5.New()//将value写入到MD5计算器中m.Write([]byte(value))//nil表示计算完哈希值后不添加后缀return hex.EncodeToString(m.Sum(nil))
}

封装image的处理逻辑

pkg 目录下新建upload/image.go文件,写入文件内容

这里基本是对底层代码的二次封装,为了更灵活的处理一些图片特有的逻辑,并且方便修改,不直接对外暴露下层

package uploadimport (
...
)func GetImageFullUrl(name string) string {return setting.AppSetting.ImagePrefixUrl + "/" + GetImagePath() + name
}// GetImageName 计算MD5加密之后的图片名
func GetImageName(name string) string {//将图片的名字剥离扩展名ext := path.Ext(name)fileName := strings.TrimSuffix(name, ext)//对单纯的图片名进行MD5加密fileName = util.EncodeMD5(fileName)//将MD5加密后的图片名和后缀返回return fileName + ext
}// GetImagePath 包装文件路径 upload/images/
func GetImagePath() string {return setting.AppSetting.ImageSavePath
}// GetImageFullPath 拼凑完整路径 runtime/+upload/images/
func GetImageFullPath() string {return setting.AppSetting.RuntimeRootPath + GetImagePath()
}// CheckImageExt 检查图片格式是否正确
func CheckImageExt(fileName string) bool {ext := file.GetExt(fileName)for _, allowExt := range setting.AppSetting.ImageAllowExts {//都大写进行对比if strings.ToUpper(allowExt) == strings.ToUpper(ext) {return true}}return false
}// CheckImageSize 检查图片的大小是否小于规定的最大值 5M
func CheckImageSize(f multipart.File) bool {size, err := file.GetSize(f)if err != nil {log.Println(err)logging.Warn(err)return false}return size <= setting.AppSetting.ImageMaxSize
}func CheckImage(src string) error {dir, err := os.Getwd()if err != nil {return fmt.Errorf("os.Getwd err: %v", err)}//检查图片目录err = file.IsNotExistMkDir(dir + "/" + src)if err != nil {return fmt.Errorf("file.IsNotExistMkDir err: %v", err)}//检查访问权限perm := file.CheckPermission(src)if perm == true {return fmt.Errorf("file.CheckPermission Permission denied src: %s", src)}return nil
}

编写上传图片的业务逻辑

routers/api 目录下新建 upload.go 文件,写入内容

package apiimport (
...
)func UploadImage(c *gin.Context) {code := e.SUCCESSdata := make(map[string]string)file, image, err := c.Request.FormFile("image")if err != nil {logging.Warn(err)code = e.ERRORc.JSON(http.StatusOK, gin.H{"code": code,"msg":  e.GetMsg(code),"data": data,})}if image == nil {code = e.INVALID_PARAMS} else {imageName := upload.GetImageName(image.Filename) //获取图片名fullPath := upload.GetImageFullPath()            //图片完整路径savePath := upload.GetImagePath()                //仓库内保存路径//图片路径+名字src := fullPath + imageName//检查图片格式和大小if !upload.CheckImageExt(imageName) || !upload.CheckImageSize(file) {code = e.ERROR_UPLOAD_CHECK_IMAGE_FORMAT} else {//检查图片目录、访问权限err := upload.CheckImage(fullPath)if err != nil {logging.Warn(err)code = e.ERROR_UPLOAD_CHECK_IMAGE_FAIL} else if err := c.SaveUploadedFile(image, src); err != nil { //图片保存到指定位置logging.Warn(err)code = e.ERROR_UPLOAD_SAVE_IMAGE_FAIL} else {//data["image_url"] = upload.GetImageFullUrl(imageName)data["image_save_url"] = savePath + imageName}}}c.JSON(http.StatusOK, gin.H{"code": code,"msg":  e.GetMsg(code),"data": data,})
}

在这一大段的业务逻辑中,我们做了如下事情:

  • c.Request.FormFile:获取上传的图片(返回提供的表单键的第一个文件)
  • CheckImageExt、CheckImageSize 检查图片大小,检查图片后缀
  • CheckImage:检查上传图片所需(权限、文件夹)
  • SaveUploadedFile:保存图片
    总的来说,就是 入参 -> 检查 -》 保存 的应用流程

增加图片上传的路由

打开 routers/router.go 文件,增加路由 r.POST("/upload", api.UploadImage)

func InitRouter() *gin.Engine {r := gin.New()...r.GET("/auth", api.GetAuth)r.GET("/swagger/*any", ginSwagger.WrapHandler(swaggerFiles.Handler))r.POST("/upload", api.UploadImage)apiv1 := r.Group("/api/v1")apiv1.Use(jwt.JWT()){...}return r
}

验证

使用 postman,测试图片上传功能
外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传
看到runtime/upload/images下存在我们上传的文件
外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

实现前端访问 http.FileServer

在完成了上一小节后,我们还需要让前端能够访问到图片,一般是如下:

  • CDN
  • http.FileSystem

在公司的话,CDN 或自建分布式文件系统居多,也不需要过多关注。而在实践里的话肯定是本地搭建了,Go 本身对此就有很好的支持,而 Gin 更是再封装了一层,只需要在路由增加一行代码即可

r.StaticFS

打开 routers/router.go 文件,增加路由 r.StaticFS("/upload/images", http.Dir(upload.GetImageFullPath()))

func InitRouter() *gin.Engine {...//网页 请求我们指定目录内的内容r.StaticFS("/upload/images", http.Dir(upload.GetImageFullPath()))r.GET("/auth", api.GetAuth)r.GET("/swagger/*any", ginSwagger.WrapHandler(swaggerFiles.Handler))r.POST("/upload", api.UploadImage)...
}

http.dir 创建了文件系统,将 /upload/image 路径映射到我们指定的文件目录中,这里为 runtime/upload/images/ ,即我们放置图片的文件夹下

更多内容可以查看源码进行学习,到这里可以自行进行验证,访问 127.0.0.1:8000/upload/images/图片名

修改文章接口

新增、更新文章接口

支持入参 cover_image_url、增加对cover_image_url的非空、最长长度的检验

  1. 修改 models/article.go
...// Article 建立对应article表的struct结构体,方便进行信息读写
type Article struct {
...CoverImageUrl string `json:"cover_image_url"`
}// AddArticle 添加文章
func AddArticle(data map[string]interface{}) bool {db.Create(&Article{
...CoverImageUrl: data["cover_image_url"].(string),})return true
}
...
  1. 修改 routers/api/v1/article.go
    AddArticleEditArticle 方法在原来的基础上进行修改,首先将之前为了方便验证写的使用 查询参数 ,改为 表单参数, 更安全
// @Summary	新增文章
// @Produce	json
// @Param		tagId		body		int		true	"tagId"
// @Param		title		body        string	true	"title"
// @Param		desc		body		string	true	"desc"
// @Param		content		body		string	true	"content"
// @Param		createdBy	body		string	true	"createdBy"
// @Param		state		body		int		true	"state"
// @Success	200			{string}	json	"{"code":200,"data":{},"msg":"ok"}"
// @Router		/api/v1/tags [post]
func AddArticle(c *gin.Context) {tagId := com.StrTo(c.PostForm("tag_id")).MustInt()title := c.PostForm("title")desc := c.PostForm("desc")content := c.PostForm("content")createdBy := c.PostForm("created_by")coverImageUrl := c.PostForm("cover_image_url")//**********state := com.StrTo(c.DefaultQuery("state", "0")).MustInt()valid := validation.Validation{}valid.Min(tagId, 1, "tag_id").Message("标签ID必须大于0")valid.Required(title, "title").Message("标题不能为空")valid.Required(desc, "desc").Message("简述不能为空")valid.Required(content, "content").Message("内容不能为空")valid.Required(createdBy, "created_by").Message("创建人不能为空")valid.Range(state, 0, 1, "state").Message("状态只允许0或1")valid.Required(coverImageUrl, "cover_image_url").Message("封面地址不能为空")//***********code := e.INVALID_PARAMSif !valid.HasErrors() {if models.ExistTagByID(tagId) {data := make(map[string]interface{})data["tag_id"] = tagIddata["title"] = titledata["desc"] = descdata["content"] = contentdata["created_by"] = createdBydata["state"] = statedata["cover_image_url"] = coverImageUrl//****************models.AddArticle(data)code = e.SUCCESS} else {code = e.ERROR_NOT_EXIST_TAG}} else {for _, err := range valid.Errors {logging.Info("err.key: %s, err.message: %s", err.Key, err.Message)}}c.JSON(http.StatusOK, gin.H{"code": code,"msg":  e.GetMsg(code),"data": make(map[string]interface{}),})
}// @Summary	修改文章
// @Produce	json
// @Param		id			path		int		true	"id"
// @Param		tagId		body		int		true	"tagId"
// @Param		title		body		string	true	"title"
// @Param		desc		body		string	true	"desc"
// @Param		content		body		string	true	"content"
// @Param		modifiedBy	body		string	true	"modifiedBy"
// @Param		state		body		int		false	"state"
// @Success	200			{string}	json	"{"code":200,"data":{},"msg":"ok"}"
// @Router		/api/v1/tags [post]
func EditArticle(c *gin.Context) {valid := validation.Validation{}id := com.StrTo(c.Param("id")).MustInt()tagId := com.StrTo(c.PostForm("tag_id")).MustInt()title := c.PostForm("title")desc := c.PostForm("desc")content := c.PostForm("content")coverImageUrl := c.PostForm("cover_image_url")//******modifiedBy := c.PostForm("modified_by")var state int = -1if arg := c.Query("state"); arg != "" {state = com.StrTo(arg).MustInt()valid.Range(state, 0, 1, "state").Message("状态只允许0或1")}valid.Min(id, 1, "id").Message("ID必须大于0")valid.Min(tagId, 1, "tag_id").Message("标签ID必须大于0")valid.MaxSize(title, 100, "title").Message("标题最长为100字符")valid.Required(title, "title").Message("标题不能为空")valid.MaxSize(desc, 255, "desc").Message("简述最长为255字符")valid.Required(desc, "desc").Message("简述不能为空")valid.MaxSize(content, 65535, "content").Message("内容最长为65535字符")valid.Required(modifiedBy, "modified_by").Message("修改人不能为空")valid.Required(coverImageUrl, "cover_image_url").Message("封面地址不能为空")//****************valid.MaxSize(coverImageUrl, 255, "cover_image_url").Message("封面地址最长为255字符")//*************valid.MaxSize(modifiedBy, 100, "modified_by").Message("修改人最长为100字符")code := e.INVALID_PARAMSif !valid.HasErrors() {if models.ExistArticleByID(id) {if models.ExistTagByID(tagId) {data := make(map[string]interface{})data["tag_id"] = tagIddata["title"] = titledata["desc"] = descdata["content"] = contentdata["modified_by"] = modifiedBymodels.EditArticle(id, data)code = e.SUCCESS} else {code = e.ERROR_NOT_EXIST_TAG}} else {code = e.ERROR_NOT_EXIST_ARTICLE}} else {for _, err := range valid.Errors {logging.Info("err.key: %s, err.message: %s", err.Key, err.Message)}}c.JSON(http.StatusOK, gin.H{"code": code,"msg":  e.GetMsg(code),"data": make(map[string]string),})
}

接下来进行验证即可

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

以太网/USB 数据采集卡 24位16通道 labview 256K同步采样

XM7016以太网SUB数据采集卡 XM7016是一款以太网/USB高速数据采集卡&#xff0c;具有16通道真差分输入&#xff0c;24位分辨率&#xff0c;单通道最高采样率256ksps. 16通道同步共计4.096Msps、精密前置增益放大、集成IEPE/ICP硬件支持的特点。本产品采用了多个高精度24位ADC单元…

学习JavaEE的日子 Day32 线程池

Day32 线程池 1.引入 一个线程完成一项任务所需时间为&#xff1a; 创建线程时间 - Time1线程中执行任务的时间 - Time2销毁线程时间 - Time3 2.为什么需要线程池(重要) 线程池技术正是关注如何缩短或调整Time1和Time3的时间&#xff0c;从而提高程序的性能。项目中可以把Time…

【 MyBatis 】| 关于多表联查返回 List 集合只查到一条的 BUG

目录 一. &#x1f981; 写在前面二. &#x1f981; 探索过程2.1 开端 —— 开始写 bug2.2 发展 —— bug 完成2.3 高潮 —— bug探究2.4 结局 —— 效果展示 三. &#x1f981; 写在最后 一. &#x1f981; 写在前面 今天又是 BUG 气满满的一天&#xff0c;一个 xxxMapper.xm…

跑spark的yarn模式时RM连不上的情况

在linux控制台跑spark on yarn一个测试案例&#xff0c;日志中总显示RM连yarn服务的时候是&#xff1a;0.0.0.0:8032 具体情况如下图&#xff1a; 我问题出现的原因&#xff0c;总结如下&#xff1a; 1.防火墙没关闭&#xff0c;关闭 2.spark-env.sh这个文件的YARN_CONF_DIR…

MyBatis基础使用

MyBatis首页https://mybatis.net.cn/ MyBatis细节注意&#xff0c;让你更加熟悉MyBatishttps://blog.csdn.net/m0_61160520/article/details/137173558?spm1001.2014.3001.5501 1.项目目录 2.数据库 CREATE DATABASE mybatis-example;USE mybatis-example;CREATE TABLE t_e…

Linux文件与进程交互的窥探者lsof

lsof 是一个 Linux 和 UNIX 系统中的实用工具,用于列出系统中打开文件的所有信息。这个名字代表 “List Open Files”,但它也可以显示进程相关的其他信息,如: 打开的文件描述符列表 打开网络连接的列表 被进程使用的信号和内核对象等 在Linux系统中,有一个经典的概念: …

vue3+threejs新手从零开发卡牌游戏(二十):添加卡牌被破坏进入墓地逻辑

在game目录下新建graveyard文件夹存放墓地相关代码&#xff1a; game/graveyard/p1.vue&#xff0c;这里主要设置了墓地group的位置&#xff1a; <template><div></div> </template><script setup lang"ts"> import { reactive, ref,…

【刷题】 二分查找入门

送给大家一句话: 总有一天&#xff0c;你会站在最亮的地方&#xff0c;活成自己曾经渴望的模样—— 苑子文 & 苑子豪《我们都一样 年轻又彷徨》 二分查找入门 1 前言2 Leetcode 704. 二分查找2.1 题目描述2.2 算法思路 3 Leetcode 34. 在排序数组中查找元素的第一个和最后…

求组合背包II(acwing)

题目描述&#xff1a; 给定n组循问&#xff0c;每组询问给定两个整数a&#xff0c;b&#xff0c;请你输出Ca^b mod (1e9 7)的值&#xff0c;。 输入格式&#xff1a; 第一行包含整数n。 接下来2行&#xff0c;每行包含一组a和b。 输出格式&#xff1a; …

Leetcode 4.1

LeetCode 热题 100 贪心算法1.买卖股票的最佳时机2.跳跃游戏3.跳跃游戏 II4.划分字母区间 区间合并1.合并区间 贪心算法 1.买卖股票的最佳时机 买卖股票的最佳时机 买的那天一定是卖的那天之前的最小值。 每到一天&#xff0c;维护那天之前的最小值即可。 在题目中&#xff0…

面试题:MySQL 优化篇

定位慢查询 &#x1f496; 开源工具 调试工具&#xff1a;Arthas&#xff08;阿尔萨斯&#xff09;运维工具&#xff1a;Prometheus&#xff08;普罗米修斯&#xff09;、Skywalking &#x1f496; MySQL 慢查询日志 # 开启 MySQL 慢查询日志开关 slow_query_log1 # 设置慢…

微信小程序wx.navigateTo无法跳转到Component组件问题解决。(共享元素动画必备)

关于Component构造器官方是有文档说明的&#xff0c;然后官方文档内部也给出了组件是可以通过像pages一样跳转的。但是官方文档缺少了必要的说明&#xff0c;会引起wx.navigateTo无法跳转到组件问题&#xff01; 扫码查看小程序&#xff0c;可以体验效果&#xff1a; 以下是官方…

ElementUI表格table组件实现单选及禁用默认选中效果

在使用ElementUI&#xff0c;需要ElementUI表格table组件实现单选及禁用默认选中效果, 先看下效果图&#xff1a; 代码如下&#xff1a; <template><el-tableref"multipleTable":data"tableData"tooltip-effect"dark"style"widt…

C语言之位段

1.位段的声明 位段的声明和结构是类似的&#xff0c;有两个不同&#xff1a; 1.位段的成员必须是 int、unsigned int 或signed int 。 2.位段的成员名后边有一个冒号和一个数字。 比如&#xff1a; struct A {int _a:2;int _b:5;int _c:10;int _d:30; }; A 就是一个位段类型…

vue3 渲染一个后端返回的图片字段渲染、table表格内放置图片

一、后端直接返回图片url 当图片字段接口直接返回的是图片url&#xff0c;可以直接放到img标签上 <img v-if"thumbLoader" class"r-image-loader-thumb" :src"resUrl" /> 二、当图片字段接口直接返回的是图片Id 那么就需要去拼一下图片…

商品服务 - 三级分类

1.递归查询树形结构 Overridepublic List<CategoryEntity> listWithTree() {//1.查出所有分类List<CategoryEntity> all this.list();//2.组装成父子的属性结构List<CategoryEntity> level1Menus all.stream().filter(c -> c.getParentCid().equals(0L)…

LeetCode-560. 和为 K 的子数组【数组 哈希表 前缀和】

LeetCode-560. 和为 K 的子数组【数组 哈希表 前缀和】 题目描述&#xff1a;解题思路一&#xff1a;一边算前缀和一边统计。这里用哈希表统计前缀和出现的次数&#xff0c;那么和为k的子数组的个数就是当前前缀和-k的个数&#xff0c;即preSums[presum - k]。画个图表述就是&a…

FPGA之状态机学习

作为一名逻辑工程师&#xff0c;掌握和应用状态机设计是必不可少的。能够灵活的应用状态机是对逻辑工程师最基本的要求&#xff0c;状态机设计的好坏能够直接影响到设计系统的稳定性&#xff0c;所以学会状态机是非常的重要。 1.状态机的概念 状态机通过不同的状态迁移来完成特…

Java封装最佳实践:打造高内聚、低耦合的优雅代码~

​ 个人主页&#xff1a;秋风起&#xff0c;再归来~ 文章专栏&#xff1a;javaSE的修炼之路 个人格言&#xff1a;悟已往之不谏&#xff0c;知来者犹可追 克心守己&#xff0c;律己则安&#xff01; 1、封装 1.1 封装的概念 面向对象程序三大…

从0到1利用express搭建后端服务

目录 1 架构的选择2 环境搭建3 安装express4 创建启动文件5 express的核心功能6 加入日志记录功能7 日志记录的好处本节代码总结 不知不觉学习低代码已经进入第四个年头了&#xff0c;既然低代码很好&#xff0c;为什么突然又自己架构起后端了呢&#xff1f;我有一句话叫低代码…