这次我们关注持久化数据而不是 GraphQL 方面。我们将使用 Postgres 数据库作为后端存储,您可能问:为什么要使用 Postgres?因为大家都熟悉 SQL Server,我们尝试一下新东西。

在数据访问层,我们将使用一个数据源类,或者说是一个仓储。基于抽象的最佳实践,我们首先创建数据源的接口,IDataStore

public interface IDataStore
{
IEnumerable<Item> GetItems();
Item GetItemByBarcode(string barcode);
}

我们已经用过 GetItemByBarcode 方法了,GetItems 方法返回库存的所有条目。我们随后将会添加一个 GraphQL 的集合。

实现该接口非常简单。

public class DataStore : IDataStore
{
private ApplicationDbContext _applicationDbContext; public DataStore(ApplicationDbContext applicationDbContext)
{
_applicationDbContext = applicationDbContext;
} public Item GetItemByBarcode(string barcode)
{
return _applicationDbContext.Items.First(i => i.Barcode.Equals(barcode));
} public IEnumerable<Item> GetItems()
{
return _applicationDbContext.Items;
}
}

我们使用 Entity Framework Core,这里我们引入 ApplicationDbContext。该类扩展自 Entity Framework 的 DbContext。现在它包含单个的关于 Item 实体的 DbSet 。这将会创建一个名为 Items 的表。

public class ApplicationDbContext : DbContext
{
public ApplicationDbContext(DbContextOptions<ApplicationDbContext> options) : base(options)
{ }
public DbSet<Item> Items { get; set; }
}

DbContextOptions 是很酷的传递参数的方式,例如数据库连接串 ConnectionString,在 Startup.cs 的 ConfigureServices 方法中配置 ApplicationDbContext。

services.AddEntityFrameworkNpgsql()
.AddDbContext<ApplicationDbContext>(
options => options.UseNpgsql(Configuration["DefaultConnection"]));

AddEntityFrameworkNpgsql() 扩展方法来自独立的 Npgsql.EntityFrameworkCore.PostgreSQL 包。可以通过 Nuget 或者 dotnet CLI 安装:

dotnet add package Npgsql.EntityFrameworkCore.PostgreSQL --version 2.0.2

Configuration 属性来自类型 IConfigurationRoot。我们在 Startup.cs 的构造函数中构建配置对象,并赋予 Configuration 属性。

public IConfigurationRoot Configuration { get; set; }

public Startup(IHostingEnvironment env)
{
var builder = new ConfigurationBuilder()
.SetBasePath(env.ContentRootPath)
.AddJsonFile("appsettings.json", optional: true, reloadOnChange: true)
.AddJsonFile($"appsettings.{env.EnvironmentName}.json", optional: true); if (env.IsDevelopment())
{
builder.AddUserSecrets<Startup>();
} builder.AddEnvironmentVariables();
Configuration = builder.Build();
}

我们可以把数据库连接串保存在 appsettings.json 文件中,对于我来说,由于保密问题,我总是将它们保存在用户密钥文件中。这里的 builder.AddUserSecrets<Startup>()。你可以在用户密钥文件中如下添加数据库连接串。

dotnet user-secrets set DefaultConnection 'your-connection-string'

然后如下执行初始迁移的命令。

dotnet ef migrations add Initial --output-dir Data\Migrations

使用如下的命令应用迁移命令以创建数据库。

dotnet ef database update

AddDbContext<ApplicationDbContext>() 使用受限的服务生命周期注册 DbContext 。Singleton 和 scope 之间的区别是:

  • Singleton 服务实例仅仅创建一次,在应用启动的时候。同样的实例被以后所有的请求所共享。
  • Scopt 服务实例在每次被请求的时候重新创建

到目前为止,我们一直使用单例模式,但是,如果我们这样用于 IDataStore,会导致下面的问题:

  • 如果你小心检查,我们将 ApplicationDbContext 直接注入到 DataStore,简单来说,我们在通过一个单例的服务访问 scoped 的服务。
  • 尽管 scope 的服务实例会在每次请求的时候创建,但我们是从单例对象来访问它的,这导致永远返回第一次创建的实例,结果与单例一样了。

所以,我们必须也将 IDataStore 也注册为 scoped 。

services.AddScoped<IDataStore, DataStore>();

直至现在的 EF Core 2.0,我们仍然没有默认的 Seed 方法。所以,我们使用下面的方式准备一些数据。

public class ApplicationDatabaseInitializer
{
public async Task SeedAsync(IApplicationBuilder app)
{
using (var scope = app.ApplicationServices.GetService<IServiceScopeFactory>().CreateScope())
{
var applicationDbContext = scope.ServiceProvider.GetRequiredService<ApplicationDbContext>(); await applicationDbContext.Database.EnsureDeletedAsync();
await applicationDbContext.Database.MigrateAsync();
await applicationDbContext.Database.EnsureCreatedAsync(); var items = new List<Item>
{
new Item { Barcode= "123", Title="Headphone", SellingPrice=50},
new Item { Barcode= "456", Title="Keyboard", SellingPrice= 40},
new Item { Barcode= "789", Title="Monitor", SellingPrice= 100}
}; await applicationDbContext.Items.AddRangeAsync(items); await applicationDbContext.SaveChangesAsync();
}
}
}

一旦我们应用启动,我们期望初始化数据,所以,在 Configure 方法中添加如下代码行。

new ApplicationDatabaseInitializer().SeedAsync(app).GetAwaiter();

一旦进入产品阶段,我们就不再需要这些种子数据。所以,这里并不介意使用 new 来创建对象实例。

其它的调整包括变更服务的生命周期:

services.AddScoped<HelloWorldQuery>();
services.AddScoped<ISchema, HelloWorldSchema>();

此刻,应用可以运行但是不能注册该架构。要记住 .net core 中间件仅在第一次启动的时候注册。但是在中间件中使用 scoped/transient 服务,我们就需要调用 InvokeAsync() 方法了。

public async Task InvokeAsync(HttpContext httpContext, ISchema schema)
{
....
....
}

最后,我希望添加一个新的集合类型的字段,以用于 Items,该字段的类型应该是 ItemType 的 ListGraphType 。

Field<ListGraphType<ItemType>>(
"items",
resolve: context =>
{
return dataStore.GetItems();
}
);

运行应用,查询 items,你将收到如下的结果。

GraphQL Part VI: 使用 Postgres 和 EF Core 持久化数据的更多相关文章

  1. 【asp.net core 系列】8 实战之 利用 EF Core 完成数据操作层的实现

    0. 前言 通过前两篇,我们创建了一个项目,并规定了一个基本的数据层访问接口.这一篇,我们将以EF Core为例演示一下数据层访问接口如何实现,以及实现中需要注意的地方. 1. 添加EF Core 先 ...

  2. 深入理解 EF Core:EF Core 读取数据时发生了什么?

    阅读本文大概需要 11 分钟. 原文:https://bit.ly/2UMiDLb 作者:Jon P Smith 翻译:王亮 声明:我翻译技术文章不是逐句翻译的,而是根据我自己的理解来表述的.其中可能 ...

  3. 深入理解 EF Core:EF Core 写入数据时发生了什么?

    阅读本文大概需要 14 分钟. 原文:https://bit.ly/2C67m1C 作者:Jon P Smith 翻译:王亮 声明:我翻译技术文章不是逐句翻译的,而是根据我自己的理解来表述的.其中可能 ...

  4. ASP.NET Core MVC+EF Core从开发到部署

    笔记本电脑装了双系统(Windows 10和Ubuntu16.04)快半年了,平时有时间就喜欢切换到Ubuntu系统下耍耍Linux,熟悉熟悉Linux命令.Shell脚本以及Linux下的各种应用的 ...

  5. EF Core 实现读写分离的最佳方案

    前言 公司之前使用Ado.net和Dapper进行数据访问层的操作, 进行读写分离也比较简单, 只要使用对应的数据库连接字符串即可. 而最近要迁移到新系统中,新系统使用.net core和EF Cor ...

  6. C# 数据操作系列 - 8. EF Core的增删改查

    0.前言 到目前为止,我们看了一下如何声明EF Core的初步使用,也整体的看了下EF Core的映射关系配置以及导航属性的配置. 这一篇,我带大家分享一下,我在工作中需要的EF Core的用法. 1 ...

  7. 深入理解 EF Core:使用查询过滤器实现数据软删除

    原文:https://bit.ly/2Cy3J5f 作者:Jon P Smith 翻译:王亮 声明:我翻译技术文章不是逐句翻译的,而是根据我自己的理解来表述的.其中可能会去除一些本人实在不知道如何组织 ...

  8. 万字长文,带你彻底理解EF Core5的运行机制,让你成为团队中的EF Core专家

    在EF Core 5中,有很多方式可以窥察工作流程中发生的事情,并与该信息进行交互.这些功能点包括日志记录,拦截,事件处理程序和一些超酷的最新出现的调试功能.EF团队甚至从Entity Framewo ...

  9. EntityFramework 6.x和EntityFramework Core插入数据探讨

    前言 一直以来对EF和EF Core都是单独分开来写,从未以比较的形式来讲解,如果您既用过EF 6.x也用过用EF Core是否有了解过EF和EF Core在插入数据时和返回主键有何异同呢?本篇博客是 ...

  10. ASP.NET Core中使用GraphQL - 第六章 使用EF Core作为持久化仓储

    ASP.NET Core中使用GraphQL ASP.NET Core中使用GraphQL - 第一章 Hello World ASP.NET Core中使用GraphQL - 第二章 中间件 ASP ...

随机推荐

  1. auto` 作为返回值类型的一些限制

    在 C++ 中,auto 作为返回值类型有一些限制,这与类型推导的方式和时机有关. 虽然在很多场景下 auto 可以简化代码,但它不能直接用于函数返回类型,这是因为在编译时类型推导的机制不同于局部变量 ...

  2. 为 AntdUI 扩展一个 MessageBox 方法

    AntdUI是个很不错的开源 WinFrom 界面组件,使用中感觉消息对话框调用有点麻烦,于是按照 MessageBox.Show 的使用习惯,增加了一个扩展方法来调用,废话不多说,直接上代码. 1 ...

  3. Notification Manager 2.0.0 发布:新增飞书通知、通知路由、通知静默

    Notification Manager 是 KubeSphere 可观测团队开源的一款 Kubernetes 平台上的多租户通知管理系统,其从 Kubernetes 接收告警.事件.审计,根据用户设 ...

  4. C#/.NET/.NET Core技术前沿周刊 | 第 10 期(2024年10.14-10.20)

    前言 C#/.NET/.NET Core技术前沿周刊,你的每周技术指南针!记录.追踪C#/.NET/.NET Core领域.生态的每周最新.最实用.最有价值的技术文章.社区动态.优质项目和学习资源等. ...

  5. 不用PLC和板卡,一台电脑就可以控制伺服

    1.前言 大家好!我是付工. EtherCAT是运动控制领域使用最广泛的总线通信协议之一. 如果我们只有一台电脑,能不能直接控制EtherCAT总线伺服呢? 这个是完全可以的. 我们可以在电脑上安装实 ...

  6. kerberos系列之spark认证配置

    大数据安全系列的其它文章 https://www.cnblogs.com/bainianminguo/p/12548076.html-----------安装kerberos https://www. ...

  7. 5.5 Vim移动光标命令汇总

    Vim 文本编辑器中,最简单的移动光标的方式是使用方向键,但这种方式的效率太低,更高效的方式使用快捷键. Vim 移动光标常用的快捷键及其功能如下面各表所示,需要注意的是,表中所有的快捷键都在命令模式 ...

  8. DearPyGui学习

    1.所有DPG应用程序必须做3件事: 创建和销毁上下文 (create_context) 创建和显示视区 (create_viewport.show_viewport) 设置和启动DearPyGui ...

  9. 根据docker镜像反推dockerfile

    Dockerfile 是一个文本文件,其中包含我们为了构建 Docker 镜像而手动执行的所有命令. Docker 可以从 Dockerfile 中读取指令来自动构建镜像.我们可以使用 docker ...

  10. Ubuntu使用dpkg查看与修改architecture的用法

    dpkg是Debian的包管理器,因为Ubuntu是Debian的变体,在Ubuntu下也有这个工具. 两个常用的命令是: dpkg -i package-file和dpkg -r package 分 ...