本文章是根据 微软MVP solenovex(杨旭)老师的视频教程编写而来,再加上本身的一些理解。
视频教程地址:https://www.bilibili.com/video/BV1xa4y1v7rR
GitHub源码:https://github.com/hllive/LearnEFCore3.1git
如何在EFCore中使用原生SQL语句或执行存储过程以及视图
首先执行两个SQL脚本,一个视图和一个存储过程github
CREATE VIEW ViewPlayerClub AS SELECT p.Id[PlayerId],p.Name[PlayerName],c.Name[ClubName],c.City[ClubCity] FROM Players p INNER JOIN Clubs c ON p.ClubId=c.Id GO
CREATE PROCEDURE RemoveGamePlayer @PlayerId UNIQUEIDENTIFIER AS BEGIN SET NOCOUNT ON; DELETE FROM GamePlayer WHERE PlayerId=@PlayerId END GO
根据视图ViewPlayerClub建立一个类sql
public class PlayerClub { public Guid PlayerId { get; set; } public string PlayerName { get; set; } public string ClubName { get; set; } public string ClubCity { get; set; } }
再把这个类添加到DBContext中ide
public DbSet<PlayerClub> PlayerClubs { get; set; }
DbSet使用这个没有主键的类是不行的,怎么才能使用无主键的类呢?
能够在OnModelCreating()
方法里设置一下ui
modelBuilder.Entity<PlayerClub>() .HasNoKey()//设置没有主键 .ToView("ViewPlayerClub");//若是不写这句,当迁移的时候还会建立一个PlayerClub的Table,应该把这个类映射到一个视图上
针对这种没有主键的model查询出来都是没法追踪的code
例子视频
[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()里的参数是主键get
执行原生SQL查询有两种方法源码
.FromSqlRaw("SELECT *...")
直接写SQL语句.FromSqlRawAsync("SELECT *...")
.FromSqlInterpolated("$SELECT *...WHERE x={var}")
SQL语句须要使用参数.FromSqlInterpolatedAsync("$SELECT *...WHERE x={var}")
FromSqlRaw
和FromSqlInterpolated
是DbSet()的方法,因此只能针对DbSet方法执行,须要在DbSet()后使用FromSqlRaw
[HttpGet("SqlTest")] public IActionResult GetSqlTest1() { var leagues = _dbContext.Leagues.FromSqlRaw("SELECT * FROM Leagues").ToList(); return Ok(leagues); }
生成的SQL语句就是FromSqlRaw()方法里的语句
[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'足球'
SELECT *
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语句两个都同样
博客文章能够转载,但不能够声明为原创