项目链接以及目录结构

liuzhixin405/efcore-template (github.com)

这是一个纯webapi的开发框架。

1、支持的orm有efcore6、dapper,可以灵活切换数据库。

using Microsoft.CodeAnalysis.CSharp.Syntax;
using Microsoft.CodeAnalysis.Elfie.Model;
using Microsoft.EntityFrameworkCore;
using project.Context;
using project.Repositories;
using project.Services;
using RepositoryComponent.DbFactories; namespace project.Extensions
{
public static partial class TheExtensions
{
public static void AddDatabase(this WebApplicationBuilder builder)
{
///sqlserver
if (builder.Configuration["DbType"]?.ToLower() == "sqlserver")
{
builder.Services.AddDbContext<ReadProductDbContext>(options => options.UseSqlServer(builder.Configuration["ConnectionStrings:SqlServer:ReadConnection"]), ServiceLifetime.Scoped);
builder.Services.AddDbContext<WriteProductDbContext>(options => options.UseSqlServer(builder.Configuration["ConnectionStrings:SqlServer:WriteConnection"]), ServiceLifetime.Scoped); }
///mysql
else if (builder.Configuration["DbType"]?.ToLower() == "mysql")
{
builder.Services.AddDbContext<ReadProductDbContext>(options => options.UseMySQL(builder.Configuration["ConnectionStrings:MySql:ReadConnection"]), ServiceLifetime.Scoped);
builder.Services.AddDbContext<WriteProductDbContext>(options => options.UseMySQL(builder.Configuration["ConnectionStrings:MySql:WriteConnection"]), ServiceLifetime.Scoped); }
else
{
//throw new ArgumentNullException("δ����ȷ��ע�����ݿ�");
builder.Services.AddDbContext<ReadProductDbContext>(options => options.UseInMemoryDatabase("test_inmemory_db"), ServiceLifetime.Scoped);
builder.Services.AddDbContext<WriteProductDbContext>(options => options.UseInMemoryDatabase("test_inmemory_db"), ServiceLifetime.Scoped); } builder.Services.AddScoped<Func<ReadProductDbContext>>(provider => () => provider.GetService<ReadProductDbContext>() ?? throw new ArgumentNullException("ReadProductDbContext is not inject to program"));
builder.Services.AddScoped<Func<WriteProductDbContext>>(provider => () => provider.GetService<WriteProductDbContext>() ?? throw new ArgumentNullException("WriteProductDbContext is not inject to program")); builder.Services.AddScoped<DbFactory<WriteProductDbContext>>();
builder.Services.AddScoped<DbFactory<ReadProductDbContext>>(); builder.Services.AddTransient<IReadProductRepository, ProductReadRepository>();
builder.Services.AddTransient<IWriteProductRepository, ProductWriteRepository>();
builder.Services.AddTransient<IProductService, ProductService>(); builder.Services.AddTransient<ICustomerService, CustomerService>();
}
}
}

2、至于消息中间件有rabbitmq、kafka,也是通过配置文件来指定哪一个实现。

using MessageMiddleware.Factory;
using MessageMiddleware.RabbitMQ; namespace project.Extensions
{
public static partial class TheExtensions
{
public static void AddMq(this WebApplicationBuilder builder)
{
var rabbitMqSetting = new RabbitMQSetting
{
ConnectionString = builder.Configuration["MqSetting:RabbitMq:ConnectionString"].Split(';'),
Password = builder.Configuration["MqSetting:RabbitMq:PassWord"],
Port = int.Parse(builder.Configuration["MqSetting:RabbitMq:Port"]),
SslEnabled = bool.Parse(builder.Configuration["MqSetting:RabbitMq:SslEnabled"]),
UserName = builder.Configuration["MqSetting:RabbitMq:UserName"],
};
var kafkaSetting = new MessageMiddleware.Kafka.Producers.ProducerOptions
{
BootstrapServers = builder.Configuration["MqSetting:Kafka:BootstrapServers"],
SaslUsername = builder.Configuration["MqSetting:Kafka:SaslUserName"],
SaslPassword = builder.Configuration["MqSetting:Kafka:SaslPassWord"],
Key = builder.Configuration["MqSetting:Kafka:Key"]
};
var mqConfig = new MQConfig
{
ConsumerLog = bool.Parse(builder.Configuration["MqSetting:ConsumerLog"]),
PublishLog = bool.Parse(builder.Configuration["MqSetting:PublishLog"]),
Rabbit = rabbitMqSetting,
Use = int.Parse(builder.Configuration["MqSetting:Use"]),
Kafka = kafkaSetting
};
builder.Services.AddSingleton<MQConfig>(sp => mqConfig);
builder.Services.AddMQ(mqConfig);
}
}
}

3、该项目还集成了mongodb和elasticsearch,在project项目中没有写实现案例,实现起来也很简单。

4、下面是分布式雪花id的实现,先注入代码,使用的时候直接使用distributedid即可。

 builder.Services.AddDistributedLock(x =>
{
x.LockType = LockType.InMemory;
x.RedisEndPoints = new string[] { builder.Configuration["DistributedRedis:ConnectionString"] ?? throw new Exception("$未能获取distributedredis连接字符串")};
}).AddCache(new CacheOptions
{
CacheType = CacheTypes.Redis,
RedisConnectionString = builder.Configuration["DistributedRedis:ConnectionString"] ?? throw new Exception("$未能获取distributedredis连接字符串")
}).AddDistributedId(new DistributedIdOptions
{
Distributed = true
});
 newProduct.Id = _distributedId.NewLongId().ToString();

5、缓存使用的是分布式缓存和内存缓存,其中分布式缓存有一般实现和指定序列化格式的实现。

using System.Text;
using System.Text.Json.Serialization;
using MessagePack;
using StackExchange.Redis.Extensions.Core;
using StackExchange.Redis.Extensions.Core.Abstractions;
using StackExchange.Redis.Extensions.Core.Configuration;
using StackExchange.Redis.Extensions.Core.Implementations; namespace project.Utility.Helper
{
public class CacheHelper
{
private static IRedisClientFactory _factory_with_msgpack;
private static IRedisDatabase _redis_with_msgpack => _factory_with_msgpack.GetDefaultRedisDatabase(); private static IRedisClientFactory _factory;
private static IRedisDatabase _redis => _factory.GetDefaultRedisDatabase();
public static void Init(IConfiguration configuration)
{
var config = configuration.GetSection("Redis").Get<RedisConfiguration>();
_factory = new RedisClientFactory(new[] { config }, null, new RedisSerializer());
_factory_with_msgpack = new RedisClientFactory(new[] { config }, null, new RedisMessagepackSerializer());
}
static CacheHelper() { } public static T Get<T>(string key)
{
return _redis.GetAsync<T>(key).GetAwaiter().GetResult();
}
public static async Task<T> GetAsync<T>(string key)
{
return await _redis.GetAsync<T>(key);
}
public static async Task<T> GetAsync_With_Msgpack<T>(string key)
{
return await _redis_with_msgpack.GetAsync<T>(key);
} public static string Get(string key)
{
return _redis.GetAsync<string>(key).GetAwaiter().GetResult();
} public static bool Set(string key, object value, TimeSpan expiresIn)
{
return _redis.AddAsync(key, value, expiresIn).GetAwaiter().GetResult();
}
public static async Task<bool> SetAsync(string key, object value, TimeSpan expiresIn)
{
return await _redis.AddAsync(key, value, expiresIn);
} public static async Task<bool> SetAsync_With_Msgpack(string key, object value, TimeSpan expiresIn)
{
return await _redis_with_msgpack.AddAsync(key, value, expiresIn);
} /// <summary>
/// 以秒为单位,返回给定 key 的剩余生存时间
/// </summary> public static long GetExpirin(string key)
{
var timespan = _redis.Database.KeyTimeToLive(key);
if (timespan == null) { return 0; }
return (long)timespan.Value.TotalSeconds;
}
public static bool KeyExpire(string key, TimeSpan expiresIn)
{
return _redis.Database.KeyExpire(key, expiresIn);
}
public static async Task<bool> RemoveKeyAsync(string key)
{
return await _redis.Database.KeyDeleteAsync(key);
}
public static long RemoveKey(string key)
{
var result = _redis.Database.KeyDelete(key);
return result ? 1 : 0;
}
} public class RedisSerializer : ISerializer
{
public T? Deserialize<T>(byte[] serializedObject)
{
var data = Encoding.UTF8.GetString(serializedObject);
return System.Text.Json.JsonSerializer.Deserialize<T>(data);
} public byte[] Serialize<T>(T? item)
{
var data = System.Text.Json.JsonSerializer.Serialize(item);
return Encoding.UTF8.GetBytes(data);
}
} public class RedisMessagepackSerializer : ISerializer
{
private MessagePackSerializerOptions _options;
public RedisMessagepackSerializer()
{
_options = MessagePackSerializerOptions.Standard.WithCompression(MessagePackCompression.Lz4BlockArray);
}
public T? Deserialize<T>(byte[] serializedObject)
{
return MessagePackSerializer.Deserialize<T>(serializedObject, _options);
} public byte[] Serialize<T>(T? item)
{
return MessagePackSerializer.Serialize(item, _options);
}
}
}

6、单元测试、集成测试没有写。

更细节的需要自己看代码,这应该是一个基本的开发具备的功能。

该项目下载下来可以直接运行。

webapi开发框架实践的更多相关文章

  1. Restful WebApi开发实践

    随笔分类 - Restful WebApi开发实践   C#对WebApi数据操作 摘要: ## 目标简化并统一程序获取WebApi对应实体数据的过程,方便对实体进行扩充.原理就是数据服务使用反射发现 ...

  2. 一次asp.net core3.1打造webapi开发框架的实践

    aaarticlea/png;base64,iVBORw0KGgoAAAANSUhEUgAAAigAAAAbCAYAAABWfHSvAAAH30lEQVR4nO1dy5GsMAx80RIESRAEST ...

  3. 快读《ASP.NET Core技术内幕与项目实战》WebApi3.1:WebApi最佳实践

    本节内容,涉及到6.1-6.6(P155-182),以WebApi说明为主.主要NuGet包:无 一.创建WebApi的最佳实践,综合了RPC和Restful两种风格的特点 1 //定义Person类 ...

  4. 42岁大龄程序员的迷茫,看我最新尝鲜.net 5+Dapper搭建的WebAPI框架

    42岁大龄程序员的迷茫 我真傻,真的.我单知道雪天是野兽在深山里没有食吃,会到村里来;我不知道春天也会有-- 我真傻,真的.我单知道程序员要活到老学到老,年龄大了要失业;我不知道码农(新型农民工)也会 ...

  5. WebAPI

    WebAPI的Host OWIN IIS WebAPI 的handler和Filter有啥区别? WebAPI  常用 Filters Exception Filter Timer Filter Lo ...

  6. C#对WebApi数据操作

    目标 简化并统一程序获取WebApi对应实体数据的过程,方便对实体进行扩充.原理就是数据服务使用反射发现数据提供者,处理好泛型就行. 相关传送门:Restful WebApi开发实践 先来看下最后的请 ...

  7. 适合WebApi的简单的C#状态机实现

    目标 采用了Restful WebApi的架构,再把业务逻辑状态转移放到后端就有点违背初衷了.实际上只要后端Api的权限设置的好,把状态转移放到前端也未尝不可.我考虑的结果是,一般如果变更这个状态本身 ...

  8. 大道至简、大智若愚—GO语言最佳详解实践

      导读:2007年,受够了C++煎熬的Google首席软件工程师Rob Pike纠集Robert Griesemer和Ken Thompson两位牛人,决定创造一种新语言来取代C++, 这就是Gol ...

  9. 你应该知道Go语言的几个优势

    要说起GO语言的优势,我们就得从GO语言的历史讲起了-- 本文由腾讯技术工程官方号发表在腾讯云+社区 2007年,受够了C++煎熬的Google首席软件工程师Rob Pike纠集Robert Grie ...

  10. 基于.NetCore开发博客项目 StarBlog - (3) 模型设计

    系列文章 基于.NetCore开发博客项目 StarBlog - (1) 为什么需要自己写一个博客? 基于.NetCore开发博客项目 StarBlog - (2) 环境准备和创建项目 基于.NetC ...

随机推荐

  1. Kubernetes 证书详解

    K8S 证书介绍 在 Kube-apiserver 中提供了很多认证方式,其中最常用的就是 TLS 认证,当然也有 BootstrapToken,BasicAuth 认证等,只要有一个认证通过,那么 ...

  2. docker部署gitlab CI/CD (一)第一篇:部署gitlab及汉化

    网上很多类似教程,但多少有点夹带私货,有的竟然拉取的第三方镜像,而且很多都要修改配置文件,完全不知道是为什么,于是结合其他人的博客和官方文档,知其然也要知其所以然,于2023年4月17日写下这篇. 官 ...

  3. 聊聊CSS 缓动函数的新成员linear()

    CSS 缓动函数是一种用于控制 CSS 动画过渡效果的函数,可以让动画变得更加自然.这篇文章将介绍一种新的 CSS easing function,即 linear(),它可以模拟出更复杂的缓动效果, ...

  4. VuePress2.0构建项目文档系统

    VuePress2.0构建项目文档系统 参考TerraMours 官网.https://terramours.site/ 文件结构参考: 1.修改首页README.md 修改项目下的README.md ...

  5. R 包 optparse 之命令行参数传递

    对于经常与 Linux 打交道的生物信息分析童鞋们,我们今天分享一下怎么在命令行下通过传递参数(类似perl.python) 的方式执行 R 脚本. 一般来说,命令行下使用 Rscript 执行 R ...

  6. 三分钟免费将 Claude API 接入个人服务

    首先我们介绍一下今天的主角 Claude Claude 是最近新开放的一款 AI 聊天机器人,是世界上最大的语言模型之一,比之前的一些模型如 GPT-3 要强大得多,因此 Claude 被认为是 Ch ...

  7. 案例实践 | 某能源企业API安全实践

    随着智能电网.全球能源互联网."互联网+电力".新电改的全面实施,分布式能源.新能源.电力交易.智能用电等新型业务不断涌现,运营模式.用户群体都将发生较大变化,电力市场由相对专业向 ...

  8. 五年磨一剑——Sealos 云操作系统正式发布!

    这是个宏伟的计划 这是一个宏伟的计划,漫长且有趣. 2018 年的某个夜晚,夜深人静,我挥舞键盘,敲下了 Sealos 的第一行代码.当时仓库命名为 "kubeinit",后来觉得 ...

  9. 2023-06-16:给你一份工作时间表 hours,上面记录着某一位员工每天的工作小时数。 我们认为当员工一天中的工作小时数大于 8 小时的时候,那么这一天就是「劳累的一天」。 所谓「表现良好的时间

    2023-06-16:给你一份工作时间表 hours,上面记录着某一位员工每天的工作小时数. 我们认为当员工一天中的工作小时数大于 8 小时的时候,那么这一天就是「劳累的一天」. 所谓「表现良好的时间 ...

  10. Subset Sum 问题单个物品重量限制前提下的更优算法

    前言 看了 ShanLunjiaJian 关于这个问题的文章,是完全没看懂,沙东队爷的中枢神经内核配置把我偏序了.叉姐在下面提了个论文,论文找不到资源,谁搞到了可以 Q 我一份之类的拜谢了.然后找到了 ...