第 3 章:GO 的接口和抽象 拓展篇 - CRUD 接口实现示例

第 3 章:GO 的接口和抽象 拓展篇 - CRUD 接口实现示例

在前面的第3章中,我们用简单的代码展示了GO的接口和抽象方法,但是代码的示例较少,部分同学可能会觉得理解起来比较抽象。因此在本章中,我们将通过一个具体的例子来演示如何使用 GO 语言的接口来实现抽象化的设计。我们将定义一个 CrudInterface 接口,该接口将提供 CRUD(创建、读取、更新、删除)操作的通用方法。然后,我们将为两种不同的存储系统(MySQL 和 Redis)提供该接口的具体实现。此外,我们还将展示如何根据传入参数中的 URI 协议来动态选择使用哪种存储实现。
Implements Implements CrudInterface +Create(ctx context.Context, key string, value interface) : error +Read(ctx context.Context, key string) (interface, error) +Update(ctx context.Context, key string, value interface) : error +Delete(ctx context.Context, key string) : error MySQLStore +Create(ctx context.Context, key string, value interface) : error +Read(ctx context.Context, key string) (interface, error) +Update(ctx context.Context, key string, value interface) : error +Delete(ctx context.Context, key string) : error RedisStore +Create(ctx context.Context, key string, value interface) : error +Read(ctx context.Context, key string) (interface, error) +Update(ctx context.Context, key string, value interface) : error +Delete(ctx context.Context, key string) : error

3.1 定义 CrudInterface 接口

首先,我们定义 CrudInterface 接口,它包含 CRUD 操作的方法:

go 复制代码
package main

import (
	"context"
	"errors"
)

// CrudInterface 定义了CRUD操作的方法
type CrudInterface interface {
	Create(ctx context.Context, key string, value interface{}) error
	Read(ctx context.Context, key string) (interface{}, error)
	Update(ctx context.Context, key string, value interface{}) error
	Delete(ctx context.Context, key string) error
}
3.2 实现 MySQL 存储

我们将创建一个 MySQLStore 结构体,它实现了 CrudInterface 接口:

go 复制代码
type MySQLStore struct {
	// 此处包含连接数据库所需的字段
}

func (m *MySQLStore) Create(ctx context.Context, key string, value interface{}) error {
	// 实现创建逻辑
	return nil
}

func (m *MySQLStore) Read(ctx context.Context, key string) (interface{}, error) {
	// 实现读取逻辑
	return nil, nil
}

func (m *MySQLStore) Update(ctx context.Context, key string, value interface{}) error {
	// 实现更新逻辑
	return nil
}

func (m *MySQLStore) Delete(ctx context.Context, key string) error {
	// 实现删除逻辑
	return nil
}
3.3 实现 Redis 存储

同样地,我们将创建一个 RedisStore 结构体,它也实现了 CrudInterface 接口:

go 复制代码
type RedisStore struct {
	// 此处包含连接Redis所需的字段
}

func (r *RedisStore) Create(ctx context.Context, key string, value interface{}) error {
	// 实现创建逻辑
	return nil
}

func (r *RedisStore) Read(ctx context.Context, key string) (interface{}, error) {
	// 实现读取逻辑
	return nil, nil
}

func (r *RedisStore) Update(ctx context.Context, key string, value interface{}) error {
	// 实现更新逻辑
	return nil
}

func (r *RedisStore) Delete(ctx context.Context, key string) error {
	// 实现删除逻辑
	return nil
}
3.4 创建存储的工厂函数

我们将编写一个工厂函数 NewCrudStore,它根据 URI 协议来创建和返回相应的 CrudInterface 实现:

go 复制代码
func NewCrudStore(uri string) (CrudInterface, error) {
	switch uri {
	case "mysql://default":
		return &MySQLStore{}, nil
	case "redis://default":
		return &RedisStore{}, nil
	default:
		return nil, errors.New("unsupported URI scheme")
	}
}
3.5 主程序中的动态选择

在主程序中,我们将使用传入的 URI 参数来动态选择和创建存储实现:

go 复制代码
func main() {
	uri := "mysql://default" // 这可以是命令行参数或其他配置来源
	store, err := NewCrudStore(uri)
	if err != nil {
		log.Fatalf("Failed to create CRUD store: %v", err)
	}

	// 现在可以使用store进行CRUD操作
	// 例如: store.Create(ctx, "key", "value")
}
3.6 用时序图来描述一种可能的行为

基于上面的代码,我们可以得出一种可能的行为,当客户端要创建一条记录时,NewCrudStore能过传入的URL参数,得知这次的创建是要写入redis,于是最终接口调用了RedisStore保存了数据。
客户端 main函数 NewCrudStore工厂函数 RedisStore Redis数据库 调用创建记录 创建CrudInterface实例 传入URI参数 "redis://default" 实例化RedisStore 连接到Redis 连接成功 返回RedisStore实例 调用Create方法 执行SET命令 返回执行结果 返回结果给main函数 返回结果给客户端 客户端 main函数 NewCrudStore工厂函数 RedisStore Redis数据库

在这个时序图中:

  1. 客户端调用main函数来创建一条新记录。
  2. main函数请求NewCrudStore工厂函数创建一个CrudInterface类型的实例。
  3. NewCrudStore工厂函数根据传入的URI参数(在这个场景中是"redis://default")实例化一个RedisStore对象。
  4. RedisStore对象尝试连接到Redis数据库。
  5. 连接成功后,RedisStoreCreate方法被main函数调用,以创建新记录。
  6. RedisStore在Redis数据库中执行SET命令来保存新记录。
  7. Redis数据库返回执行结果给RedisStore
  8. RedisStore将结果返回给main函数。
  9. 最终,main函数将结果返回给客户端。

通过这个例子,我们展示了如何利用 GO 语言的接口特性来实现一个灵活的、可插拔的 CRUD 服务。程序可以根据运行时的配置动态选择使用 MySQL 或 Redis 作为后端存储,而不需要修改业务逻辑代码。这种设计提高了系统的可扩展性和可维护性。

相关推荐
我命由我12345几秒前
Kotlin 数据容器 - List(List 概述、创建 List、List 核心特性、List 元素访问、List 遍历)
java·开发语言·jvm·windows·java-ee·kotlin·list
liulilittle1 分钟前
C++ TAP(基于任务的异步编程模式)
服务器·开发语言·网络·c++·分布式·任务·tap
励志要当大牛的小白菜2 小时前
ART配对软件使用
开发语言·c++·qt·算法
武子康2 小时前
Java-80 深入浅出 RPC Dubbo 动态服务降级:从雪崩防护到配置中心秒级生效
java·分布式·后端·spring·微服务·rpc·dubbo
舒一笑3 小时前
我的开源项目-PandaCoder迎来史诗级大更新啦
后端·程序员·intellij idea
@昵称不存在4 小时前
Flask input 和datalist结合
后端·python·flask
爱装代码的小瓶子4 小时前
数据结构之队列(C语言)
c语言·开发语言·数据结构
zhuyasen4 小时前
Go 分布式任务和定时任务太难?sasynq 让异步任务从未如此简单
后端·go
Hello.Reader4 小时前
Go-Elasticsearch v9 安装与版本兼容性
elasticsearch·golang·jenkins
东林牧之4 小时前
Django+celery异步:拿来即用,可移植性高
后端·python·django