2020-08-26

10、Entity Framework Core 3.1入门教程

本文章是根据 微软MVP solenovex(杨旭)老师的视频教程编写而来,再加上自己的一些理解。
视频教程地址:https://www.bilibili.com/video/BV1xa4y1v7rR
GitHub源码:https://github.com/hllive/LearnEFCore3.1

无主键的Entity

  • .NetCore3.1允许无主键的Entity或Model
  • 它们不会被追踪,相当于是只读的
  • 映射到没有主键的Table或View

如何在EFCore中使用原生SQL语句或执行存储过程以及视图
首先执行两个SQL脚本,一个视图和一个存储过程

CREATE VIEW ViewPlayerClubAS SELECT p.Id[PlayerId],p.Name[PlayerName],c.Name[ClubName],c.City[ClubCity] FROM Players p INNER JOIN Clubs c ON p.ClubId=c.IdGO
CREATE PROCEDURE RemoveGamePlayer @PlayerId UNIQUEIDENTIFIERASBEGIN SET NOCOUNT ON; DELETE FROM GamePlayer WHERE PlayerId=@PlayerIdENDGO

根据视图ViewPlayerClub创建一个类

public class PlayerClub{ public Guid PlayerId { get; set; } public string PlayerName { get; set; } public string ClubName { get; set; } public string ClubCity { get; set; }}

再把这个类添加到DBContext中

public DbSet<PlayerClub> PlayerClubs { get; set; }

DbSet使用这个没有主键的类是不行的,怎么才能使用无主键的类呢?
可以在OnModelCreating()方法里设置一下

modelBuilder.Entity<PlayerClub>() .HasNoKey()//设置没有主键 .ToView("ViewPlayerClub");//如果不写这句,当迁移的时候还会创建一个PlayerClub的Table,应该把这个类映射到一个视图上

针对这种没有主键的model查询出来都是无法追踪的

例子

[HttpGet("PlayerClub")]public IActionResult GetViewPlayerClub(){ var playerClub = _dbContext.PlayerClubs.Where(px => px.ClubCity.Contains("贵州")).ToList(); return Ok(playerClub);}

生成的SQL语句

SELECT [v].[ClubCity], [v].[ClubName], [v].[PlayerId], [v].[PlayerName]FROM [ViewPlayerClub] AS [v]WHERE CHARINDEX(N'贵州', [v].[ClubCity]) > 0

如果使用_dbContext.PlayerClubs.Find(Guid id)在编译的时候不会出错,但运行肯定会报错,因为Find()里的参数是主键

原生SQL查询

执行原生SQL查询有两种方法

  • .FromSqlRaw("SELECT *...") 直接写SQL语句
  • .FromSqlRawAsync("SELECT *...")
  • .FromSqlInterpolated("$SELECT *...WHERE x={var}") SQL语句需要使用参数
  • .FromSqlInterpolatedAsync("$SELECT *...WHERE x={var}")

FromSqlRawFromSqlInterpolated是DbSet()的方法,所以只能针对DbSet方法执行,需要在DbSet()后使用FromSqlRaw

使用FromSqlRaw()方法

[HttpGet("SqlTest")]public IActionResult GetSqlTest1(){ var leagues = _dbContext.Leagues.FromSqlRaw("SELECT * FROM Leagues").ToList(); return Ok(leagues);}

生成的SQL语句就是FromSqlRaw()方法里的语句

使用FromSqlInterpolated()方法

[HttpGet("SqlTest1")]public IActionResult GetSqlTest1([FromQuery] string name){ //使用带参数的FromSqlInterpolated var leagues = _dbContext.Leagues  .FromSqlInterpolated($"SELECT * FROM Leagues WHERE Name LIKE N'%{name}%'")  .ToList(); return Ok(leagues);}

生成的SQL语句就是FromSqlInterpolated()方法的语句,但是会使用SQL参数作为查询值

exec sp_executesql N'SELECT * FROM Leagues WHERE Name LIKE N''%@p0%''',N'@p0 nvarchar(4000)',@p0=N'足球'

原生SQL查询的要求

  • 必须返回Entity类型的所有属性,一般都是SELECT *
  • 字段名和Entity的属性名必须匹配
  • 无法包含关联的数据,不能写JOIN语句
  • 只能查询已知的Entity,也就是Context中设定好的或间接能追踪到的数据

执行非查询类SQL

  • Context.Database.ExecuteSqlRaw()
  • Context.Database.ExecuteSqlRawAsync()
  • Context.Database.ExecuteSqlInterpolated()
  • Context.Database.ExecuteSqlInterpolatedAsync()
  • 无法用于查询
  • 只能返回影响的行数
[HttpGet("SqlTest2")]public IActionResult GetSqlTest2(){ //使用ExecuteSqlRaw var count = _dbContext.Database.ExecuteSqlRaw("EXEC [dbo].[RemoveGamePlayer] {0}", new Guid("")); //使用ExecuteSqlInterpolated var counts = _dbContext.Database.ExecuteSqlInterpolated($"EXEC [dbo].[RemoveGamePlayer] {new Guid("")}"); return Ok(new { count, counts });}

生成的SQL语句两个都一样

博客文章可以转载,但不可以声明为原创

10、Entity Framework Core 3.1入门教程详细解析后台操作如何在ebay选品与刊登FBA实战Shopee广告三兄弟结义篇:关键词广告、关联广告和商店广告(一)亚马逊更新配送要求!发货更难了吗?武汉病毒所致全所职工:若干谣言伤害极大 问心无愧清远玄真漂流惊险刺激吗?清远黄腾峡漂流,古龙峡哪个危险?清远清泉湾漂流的介绍,到清泉湾漂流怎么走?

No comments:

Post a Comment