IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 大数据 -> 带你了解Go怎样实现二级缓存 -> 正文阅读

[大数据]带你了解Go怎样实现二级缓存

带你了解Go怎样实现二级缓存

一、需求

  • 实现二级缓存
  • 程序运行起来后提示:”请输入命令:“,如果输入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函数里面都是一些基本语法知识,用了switchgoto这两个内容。

接下来就是连接数据库了,这里要用到数据库扩展包SqlxSqlx包其实最大最大的优点是在查询方面,也就是使用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的所有值。

  大数据 最新文章
实现Kafka至少消费一次
亚马逊云科技:还在苦于ETL?Zero ETL的时代
初探MapReduce
【SpringBoot框架篇】32.基于注解+redis实现
Elasticsearch:如何减少 Elasticsearch 集
Go redis操作
Redis面试题
专题五 Redis高并发场景
基于GBase8s和Calcite的多数据源查询
Redis——底层数据结构原理
上一篇文章      下一篇文章      查看所有文章
加:2022-04-24 09:31:36  更:2022-04-24 09:33:05 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/16 12:49:44-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码