Go应该如何实现二级缓存
作者:ReganYue 发布时间:2024-02-19 17:07:53
目录
一、需求
二、实现连接Mysql并执行查询语句
三、写一个错误处理函数
四、设置二级缓存
一、需求
实现二级缓存
程序运行起来后提示:“请输入命令:”,如果输入getall,查询并显示所有人员的信息
第一次时查询mysql并将结果缓存在redis,设置60秒的过期时间
以后的每次查询,如果redis有数据就从redis加载,没有则重复上一步的操作
二、实现连接Mysql并执行查询语句
先实现需求二,当输入命令getall时,查询并显示所有人员的信息。
package main
import (
"fmt"
_"github.com/go-sql-driver/mysql"
"github.com/jmoiron/sqlx"
)
type Human struct {
Name string `db:"name"`
Age int `db:"age"`
}
func main() {
var cmd string
for{
fmt.Println("请输入命令:")
fmt.Scan(&cmd)
switch cmd{
case "getall":
//显示所有人的信息
GetAllPeople()
case "exit":
//退出程序
goto GAMEOVER
default:
fmt.Println("输入的命令有误,请重新输入!")
}
}
GAMEOVER:
fmt.Println("GAME OVER")
}
func GetAllPeople() {
fmt.Println("allPeople")
//先尝试拿缓存
GetPeopleFromRedis()
db, _ := sqlx.Connect("mysql", "root:123456@tcp(localhost:3306)/mydb")
defer db.Close()
var people []Human
err := db.Select(&people, "select name,age from person")
if err!=nil{
fmt.Println("查询失败!err=",err)
}
fmt.Println(people)
CachePeople2Redis(people)
}
第一步还是导包,需要在mysql驱动包前面加上下划线_,因为它只是一个驱动文件,并不需要在代码中调用它的有关API接口.
接下来的这个结构体中后面的db:"name" db:"age"一定要加反单引号,否则运行时会报错。(傻傻的编者刚开始这里就出现问题啦~)
type Human struct {
Name string `db:"name"`
Age int `db:"age"`
}
然后main函数里面都是一些基本语法知识,用了switch和goto这两个内容。
接下来就是连接数据库了,这里要用到数据库扩展包Sqlx,Sqlx包其实最大最大的优点是在查询方面,也就是使用select时优化得比较好。比原来的使用查询方便了不止一点。
db, _ := sqlx.Connect("mysql", "root:123456@tcp(localhost:3306)/mydb")
driverName:mysql,表示驱动器的名称是mysql也就上面"github.com/go-sql-driver/mysql"导入的驱动器。
dataSourceName是root:123456@tcp(localhost:3306)/mydb 它的含义是 账户名:密码@tcp(ip:端口)/数据库名称。
将缓存查询结果到Redis,就是通过这个函数CachePeople2Redis(people)。
三、写一个错误处理函数
func HandleError(err error,why string) {
if err != nil{
fmt.Println(err,why)
os.Exit(1)
}
}
因为后面需要处理很多错误,而错误处理也是GO的一个特性,所以我们这先写一个错误处理函数。
四、设置二级缓存
func CachePeople2Redis(people []Human) {
conn, _ := redis.Dial("tcp", "localhost:6379")
defer conn.Close()
for _,human := range people{
humanStr := fmt.Sprint(human)
_, err := conn.Do("rpush", "people", humanStr)
if err != nil{
fmt.Println("缓存失败(rpush people),err=",err)
return
}
}
_, err := conn.Do("expire", "people", 66)
if err!=nil{
HandleError(err,"@expire people 60")
}
fmt.Println("缓存成功!")
}
redis.Dial()这个函数是用来连接redis的,需要给定网络协议和IP地址及端口号,redis的端口号默认为6379.
defer conn.Close()表示延时结束与redis的连接,为了节省系统的io资源,需要及时关闭连接!刚入门时我们很容易忘记这个,需要我们养成习惯!
conn.Do()是用来执行数据库命令的,第一个参数是命令名,后面的参数是数据库命令的参数。它返回的结果中reply是字节数组[]byte类型,需要根据具体的业务类型进行数据类型转换。
这段代码先将people数组中的每一个human放入到redis的people列表中。然后再执行expire命令,将列表设置过期时间。
执行成功!下面是运行结果:
请输入命令:
getall
allPeople
[{大扬 21} {小飞 21} {大红袍 1} {小芳 18}]
缓存成功!
请输入命令:
然后去看看数据库里面存进去没有。
127.0.0.1:6379> lrange people 0 -1
1) "{\xe5\xa4\xa7\xe6\x89\xac 21}"
2) "{\xe5\xb0\x8f\xe9\xa3\x9e 21}"
3) "{\xe5\xa4\xa7\xe7\xba\xa2\xe8\xa2\x8d 1}"
4) "{\xe5\xb0\x8f\xe8\x8a\xb3 18}"
过了一分钟之后,再查看redis数据库内的数据。
127.0.0.1:6379> lrange people 0 -1
(empty list or set)
已经消失了。
再写一个函数:
func GetPeopleFromRedis() (peopleStrs []string) {
//连数据库
conn, _ := redis.Dial("tcp", "localhost:6379")
//延迟关闭
defer conn.Close()
//执行命令
reply, err := conn.Do("lrange", "people", 0, -1)
//处理错误
HandleError(err,"@lrange people 0 -1")
//类型转换
peopleStrs, err = redis.Strings(reply, err)
//打印结果
fmt.Println("缓存拿取结果:",peopleStrs,err)
return
}
如果redis里面有就不需要从mysql里面取数据了。直接从redis里面利用lrange命令来获取people的所有值。
来源:https://juejin.cn/post/6994600836101832741


猜你喜欢
- 目录为什么需要分区?分区的策略分区隐患为什么需要分区?当面对巨大的数据表的时候,至少有一件事情是确定的,表太大了以至于每次查询的时候我们没法
- 之前就想要把自己的BlogsToWordpress打开成exe了。一直没去弄。又看到有人提到python打开成exe的问题。所以打算现在就去
- 金额大小写转换的asp完全无错版本, 这个版本解决了小数位不能到分的问题,处理方式符合会计方式,值得推荐!<!--#inc
- <script language="javascript"> functio
- <title>动态序列比对</title> <script> function pairaln(seq1
- 安装jieba库教程jieba库是一款优秀的 Python 第三方中文分词库,jieba 支持三种分词模式:精确模式、全模式和搜索引擎模式,
- Python字典中的键是唯一的,但不同的键可以对应同样的值,比如说uid,可以是1001。id同样可以是1001。这样的话通过值来获取指定的
- ASPError Object 这个新增的,内置与ASP 3.0中的对象提供了一个以往版本中没有的专门用来处理错误的对象,这样,我们来操纵错
- 本文实例讲述了php通过获取头信息判断图片类型的方法。分享给大家供大家参考。具体实现方法如下:$filename = '617.gi
- Mint UI 是饿了么开源的,基于 Vue.js 的移动端组件库。关于Mint UI,有文档不够准确详尽,组件略显粗糙,功能不够完善等问题
- 先来看一段代码:# ~*~ Twisted - A Python tale ~*~from time import sleep# Hello
- 叨逼叨首先,介绍一下 pdb 调试,pdb 是 python 的一个内置模块,用于命令行来调试 Python 代码。或许你会说,现在用 Py
- 一、说明压缩和解压缩是日常常用的操作,不管是windows上图形界面的操作,还是linux上用命令来进行压缩解压缩,总的而言都还是比较方便的
- CLI工程全局安装vue-clinpm install -g @vue/cli通过cli创建uni-app项目 vue creat
- 曲线一解释这里是使用matplotlib来绘制正态分布的曲线。代码实现import numpy as npimport matplotlib
- 一、前言春节即将来临,大家肯定各种掏腰包花花花,小编相信大家在支付时候,微信、支付宝支付肯定是优先选择。今天小编心血来潮,为大家带来一个很有
- 一、v-if和v-show区别① v-show严格意义来说其实是条件隐藏,直接在页面初始化的时候将DOM(对象模型)元素也初始化,因为它就是
- 在一个大型的项目中,不可避免会出现操作时间的业务,比如时间的格式化,比如时间的加减,我们一般会直接使用moment.js库来做,毕竟稳定可靠
- 由于下学期报了一个Python的入门课程所以寒假一直在自己摸索,毕竟到时候不能挂科,也是水水学分最近心血来潮打算试试爬一下百度翻译肝了一天终
- 为了使一个MySQL系统安全,强烈要求你考虑下列建议……当你连接一个MySQL服务器时,你通常应