首页 后端开发 Golang Golang中使用缓存加速数据库访问效率的实践。

Golang中使用缓存加速数据库访问效率的实践。

Jun 20, 2023 am 10:12 AM
缓存 golang 数据库访问

Golang中使用缓存加速数据库访问效率的实践

随着Web应用越来越复杂,对数据库的访问也变得越来越频繁。而访问数据库通常是非常耗时的操作,特别是在数据量较大的情况下。为了提高访问数据库的效率,可以采用诸如缓存等策略来优化数据库访问。

本文将介绍Golang中如何使用缓存加速数据库访问的实践。我们将使用Golang作为开发语言,Redis作为缓存服务器,MySQL作为数据库服务器进行实验。

1.搭建环境

在开始之前,我们需要搭建好环境。首先安装Golang以及MySQL和Redis服务器,这里不再赘述。

然后在Golang中安装本地Redis和MySQL的Go驱动程序:

go get github.com/go-redis/redis/v8
go get github.com/go-sql-driver/mysql

2.编写代码

接下来,我们编写代码以实现缓存加速数据库访问。

首先是数据库访问的代码。我们定义了一个叫做DB的全局变量,用于连接MySQL。然后,我们定义了一个函数getUserByID,用于从MySQL中查询一个用户的信息:

package main
 
import (
   "database/sql"
   "fmt"
   "log"
 
   _ "github.com/go-sql-driver/mysql"
)
 
var DB *sql.DB
 
type User struct {
   ID       int
   Username string
   Password string
   Age      int
}
 
func init() {
   db, err := sql.Open("mysql", "root:password@tcp(127.0.0.1:3306)/test?charset=utf8")
   if err != nil {
      log.Fatal("Open mysql failed,err:", err)
      return
   }
   DB = db
   fmt.Println("Connect to mysql success")
}
 
func getUserByID(id int) (*User, error) {
   var user User
   query := "SELECT id, username, password, age FROM users WHERE id=?"
   err := DB.QueryRow(query, id).Scan(&user.ID, &user.Username, &user.Password, &user.Age)
   if err != nil {
      log.Println(err)
      return nil, err
   }
   return &user, nil
}

然后,我们在这个getUserByID函数中添加了缓存逻辑。具体而言,我们首先通过getUserByID函数尝试从Redis缓存中读取请求的用户信息。如果Redis中没有该用户的信息记录,则从MySQL中读取用户信息,并将其存入Redis中以供下次访问。如果用户信息在Redis中有记录,则直接从Redis中返回用户信息:

package main
 
import (
   "database/sql"
   "encoding/json"
   "fmt"
   "log"
   "strconv"
 
   "github.com/go-redis/redis/v8"
   _ "github.com/go-sql-driver/mysql"
)
 
var DB *sql.DB
var RedisClient *redis.Client
 
type User struct {
   ID       int
   Username string
   Password string
   Age      int
}
 
func init() {
   db, err := sql.Open("mysql", "root:password@tcp(127.0.0.1:3306)/test?charset=utf8")
   if err != nil {
      log.Fatal("Open mysql failed,err:", err)
      return
   }
   DB = db
   fmt.Println("Connect to mysql success")
 
   RedisClient = redis.NewClient(&redis.Options{
      Addr: "127.0.0.1:6379",
   })
   pong, err := RedisClient.Ping(RedisClient.Context()).Result()
   if err != nil {
      panic(err)
      return
   }
   fmt.Println("Connect to redis success: ", pong)
}
 
func getUserByID(id int) (*User, error) {
   var user User
   key := "user-" + strconv.Itoa(id)
 
   // 1.尝试从Redis中读取用户信息
   val, err := RedisClient.Get(RedisClient.Context(), key).Result()
   if err == redis.Nil {
      fmt.Println("Cache miss")
   } else if err != nil {
      log.Println("Get from Redis fail:", err)
   } else {
      fmt.Println("Get from Redis:", val)
      if err := json.Unmarshal([]byte(val), &user); err != nil { // 将json字符串转换为结构体
         log.Panicln("Unmarshal to user fail:", err)
      }
      return &user, nil
   }
 
   // 2.如果Redis中没有,从MySQL中查询
   query := "SELECT id, username, password, age FROM users WHERE id=?"
   err = DB.QueryRow(query, id).Scan(&user.ID, &user.Username, &user.Password, &user.Age)
   if err != nil {
      log.Println(err)
      return nil, err
   }
 
   // 3.然后更新Redis缓存
   val, err = json.Marshal(user) // 将结构体转换为json字符串
   if err != nil {
      log.Panicln("Marshal user fail:", err)
   }
   err = RedisClient.Set(RedisClient.Context(), key, val, 0).Err()
   if err != nil {
      log.Panicln("Cache to Redis fail:", err)
   }
   return &user, nil
}

3.测试

我们完成了缓存逻辑的编写。现在我们来测试一下这份代码,看看运行效果如何。

首先,我们测试程序第一次查询用户信息需要从MySQL中读取:

func main() {
   id := 1
   user, err := getUserByID(id)
   if err != nil {
      log.Fatal(err)
      return
   }
   fmt.Printf("User info: id=%d, username=%s, password=%s, age=%d
",
      user.ID, user.Username, user.Password, user.Age)
}

运行程序,输出如下:

Cache miss
User info: id=1, username=kirito, password=123456, age=18

可以看到程序从MySQL中取出了用户信息,并将其存入了Redis缓存中。

第二次查询同样的用户后,程序将从Redis中读取用户信息而不是访问MySQL数据库:

func main() {
   id := 1
   user, err := getUserByID(id)
   if err != nil {
      log.Fatal(err)
      return
   }
   fmt.Printf("User info: id=%d, username=%s, password=%s, age=%d
",
      user.ID, user.Username, user.Password, user.Age)
}

运行程序,输出如下:

Get from Redis: {"ID":1,"Username":"kirito","Password":"123456","Age":18}
User info: id=1, username=kirito, password=123456, age=18

可以看到程序直接从Redis中读取了用户信息,而没有访问MySQL数据库,这证明了缓存的实用性。

总结

在本文中,我们介绍了Golang中使用Redis缓存机制来优化数据库访问效率的实践。通过编写getUserByID函数,我们在查询用户信息时首先尝试从Redis中读取信息以加速查询速度,如果Redis中没有该用户信息则从MySQL中读取并将其存入Redis中。将数据进行缓存后,程序在多次访问相同数据时,可以直接从缓存中读取,而不需要每次都去访问数据库。

需要注意的是,缓存的使用需要避免脏数据产生。对于数据的CRUD操作,缓存也需要跟随数据进行操作,以确保数据在缓存和数据库中的一致性。

总的来说,使用缓存机制可以大幅提高程序的性能,并且在处理大量数据的时候尤为有效。这种策略在高并发系统中也非常实用,建议开发者进行尝试。

以上是Golang中使用缓存加速数据库访问效率的实践。的详细内容。更多信息请关注PHP中文网其他相关文章!

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Stock Market GPT

Stock Market GPT

人工智能驱动投资研究,做出更明智的决策

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

Golang的标准库记录的替代方案是什么? Golang的标准库记录的替代方案是什么? Aug 05, 2025 pm 08:36 PM

forNewgo1.21项目,使用logforofficial loggingsupport; 2. forhigh-performanceProductionservices,selectzaporzerologduetototheirspeedandlowallowallowallowallocations; 3.ForeaseofusofusofuseanDrichEandrichIntRichIntrationsLikEsentryHooksEntryHooksEntryHooksEntryHooksEntryHooksEntryhooksEnderGrusIsIdeAdeSiteSiteSiteSitePitElowerPertermesterpersemperance; 4

Golang的基准是什么? Golang的基准是什么? Aug 13, 2025 am 12:14 AM

Gobenchmarkingmeasurescodeperformancebytimingfunctionexecutionandmemoryusage,usingbuilt-intestingtools;benchmarksarewrittenin_test.gofileswithnamesstartingwithBenchmark,takeatesting.Bparameter,andruntargetcodeinaloopcontrolledbyb.N,whichGoautomatical

您如何在Golang中实现观察者模式? 您如何在Golang中实现观察者模式? Aug 14, 2025 pm 12:04 PM

在Go中可以通过接口和通道实现观察者模式,定义Observer接口包含Update方法,Subject结构体维护观察者列表和消息通道,通过Attach添加观察者,Notify发送消息,listengoroutine异步广播更新,具体观察者如EmailService和LogService实现Update方法处理通知,主程序注册观察者并触发事件,实现松耦合的事件通知机制,适用于事件驱动系统、日志记录和消息通知等场景。

如何管理Golang的内存分配 如何管理Golang的内存分配 Aug 11, 2025 pm 12:23 PM

UnderstandGo’smemoryallocationmodelbyusingescapeanalysistominimizeheapallocations;2.Reduceheapallocationswithvaluetypes,pre-allocatedslices,andsync.Poolforbufferreuse;3.Optimizestringandbytehandlingusingstrings.Builderandreusablebyteslicestoavoidunne

如何使用GRPC在Golang中的微服务之间进行通信 如何使用GRPC在Golang中的微服务之间进行通信 Aug 12, 2025 am 03:49 AM

使用gRPC在Go微服务间通信的步骤为:1.使用ProtocolBuffers定义服务接口和消息类型,编写.proto文件;2.安装protoc编译器及Go插件,生成greeter.pb.go和greeter_grpc.pb.go代码文件;3.实现gRPC服务器,注册服务并监听指定端口;4.创建gRPC客户端,建立连接并调用远程方法;5.分别运行服务器和客户端验证通信;6.遵循最佳实践,包括启用TLS、使用拦截器、错误处理和版本控制;7.采用清晰的项目结构,便于维护和更新。通过这些步骤可实现高效

如何清除Laravel中的缓存 如何清除Laravel中的缓存 Aug 14, 2025 pm 03:10 PM

Clearapplicationcachewithphpartisancache:cleartoremovecacheddatastoredviacachedriver.2.Clearconfigurationcacheusingphpartisanconfig:clearafterchanging.envorconfigfiles,thenoptionallyre-cachewithphpartisanconfig:cacheinproductiononly.3.Clearroutecache

如何将静态资产嵌入golang二进制 如何将静态资产嵌入golang二进制 Aug 30, 2025 am 04:50 AM

使用Go的embed包可以将静态资源直接嵌入二进制文件中。从Go1.16开始,通过在变量前使用//go:embed指令,可将单个文件、多个文件或整个目录嵌入,支持string、[]byte或embed.FS类型,嵌入内容在编译时固化到二进制中,路径需存在且区分大小写,推荐使用embed而非第三方工具如go-bindata,该方法简洁高效并已成为标准做法。

如何在Golang中构建一个无框架的静态API 如何在Golang中构建一个无框架的静态API Aug 20, 2025 am 01:47 AM

是的,使用Go标准库可以构建RESTfulAPI,通过net/http处理请求、encoding/json处理JSON数据、context管理上下文,结合http.ServeMux路由、手动路径解析、中间件封装和适当的错误处理,即可实现一个轻量、可控且无需外部框架的RESTful服务,最终以模块化结构提升可维护性,并完全掌握底层HTTP机制。

See all articles