中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

EntityFramework.Extended如何用EF實現指定字段的更新

發布時間:2021-11-15 14:20:11 來源:億速云 閱讀:375 作者:柒染 欄目:大數據

EntityFramework.Extended如何用EF實現指定字段的更新,相信很多沒有經驗的人對此束手無策,為此本文總結了問題出現的原因和解決方法,通過這篇文章希望你能解決這個問題。

今天在將一個項目中使用存儲過程的遺留代碼遷移至新的架構時,遇到了一個問題——如何用EF實現數據庫中指定字段的更新(根據UserId更新Users表中的FaceUrl與AvatarUrl字段)?

原先調用存儲過程的代碼:

public bool UpdateAvatar(Guid userId, string faceUrl, string avatarUrl)
{
DbCommand command = _db.GetStoredProcCommand("User_UpdateFaceAvatar");
_db.AddInParameter(command, "@FaceUrl", DbType.String, faceUrl);
_db.AddInParameter(command, "@AvatarUrl", DbType.String, avatarUrl);
_db.AddInParameter(command, "@UserId", userId);
return _db.ExecuteNonQuery(command) > 0;
}

存儲過程中所使用的SQL語句:

UPDATE Users 
SET FaceUrl=@FaceUrl,AvatarUrl=@AvatarUrlWHERE [UserId]=@UserId

在新的架構中,數據庫訪問用的是Entity Framework,并且用IUnitOfWork接口進行了封裝,Application層對數據庫的操作是通過IUnitOfWork接口完成的。

IUnitOfWork接口是這么定義的:

public interface IUnitOfWork : IDisposable
{
IQueryable<T> Set<T>() where T : class;

T Add<T>(T entity) where T : class;

IEnumerable<T> AddRange<T>(IEnumerable<T> entities) where T : class;

T Attach<T>(T entity) where T : class;

T Remove<T>(T entity) where T : class;

IEnumerable<T> RemoveRange<T>(IEnumerable<T> entities) where T : class;

bool Commit();

Task<bool> CommitAsync();
}

如果不給IUnitOfWork添加新的接口方法,可以使用EF的change tracking完成指定字段的更新操作。

Application.Services中的實現代碼:

public async Task<bool> UpdateFaceAvatar(Guid userId, string faceUrl, string avatarUrl)
{
var user = await _userRepository.GetByUserId(userId);
user.FaceUrl = faceUrl;
user.AvatarUrl = avatarUrl;
return await _unitOfWork.CommitAsync();
}

使用ef change tracking的優點是如果屬性的值沒有被改變,就不會觸發數據庫更新操作,缺點是每次更新前都要進行1次查詢操作。

而對于這里的更新FaceUrl與AvatarUrl的應用場景,更新前就明確知道數據已經改變,直接更新數據庫即可。ef change tracking的更新前查詢不僅沒有必要,而且增加了額外的開銷。

于是,嘗試尋找新的解決方法。

開始嘗試的是EF的DbEntityEntry,未抽象的實現代碼:

public class EfUnitOfWork : DbContext, IUnitOfWork
{
public async Task<bool> UpdateAsync(User user)
{
base.Set<User>().Attach(user);
base.Entry<User>(user).Property(x => x.FaceUrl).IsModified = true;
base.Entry<User>(user).Property(x => x.AvatarUrl).IsModified = true;
return (await base.SaveChangesAsync()) > 0;
}
}

用代碼:

await UpdateAsync(new User { UserId = userId, FaceUrl = faceUrl, AvatarUrl = avatarUrl });

但是基于這個實現,很難抽象出一個好用的接口方法。

后來突然想到前一段時間在一個項目中用到的EntityFramework.Extended。針對Update操作,它實現了一個優雅的EF擴展,為何不直接用它呢?

//example of using an IQueryable as the filter for the updatevar users = context.Users.Where(u => u.FirstName == "firstname");
context.Users.Update(users, u => new User {FirstName = "newfirstname"});

于是,如獲珍寶地基于EntityFramework.Extended進行實現。

首先在IUnitOfWork中添加一個UpdateAsync()的接口方法:

public interface IUnitOfWork : IDisposable
{
    Task<bool> UpdateAsync<T>(IQueryable<T> source, Expression<Func<T, T>> updateExpression) where T : class;
}

然后在IUnitOfWork.UpdateAsync()的實現中,調用EntityFramework.Extended的UpdateAsync擴展方法,完成Update操作:

using EntityFramework.Extensions;
public class EfUnitOfWork : DbContext, IUnitOfWork
{
async Task<bool> IUnitOfWork.UpdateAsync<T>(IQueryable<T> source,
Expression<Func<T, T>> updateExpression)
{
return (await source.UpdateAsync<T>(updateExpression)) > 0;
}
}

隨之,Application.Services中的實現代碼改為這樣:

public async Task<bool> UpdateFaceAvatar(Guid userId, string faceUrl, string avatarUrl)
{
IQueryable<User> userQuery = _userRepository.GetByUserId(userId);
return await _unitOfWork.UpdateAsync<User>(
userQuery,
x => new User { FaceUrl = faceUrl, AvatarUrl = avatarUrl }
);
}

(注:這里的_userRepository.GetByUserId的返回類型需要改為IQueryable,再一次驗證了Repository返回IQueryable的必要性,相關博文:開發筆記:用不用UnitOfWork以及Repository返回什么集合類型)

跑單元測試驗證一下。

單元測試代碼:

[Fact]
public async Task UpdateFaceAvatar()
{
var result = await _userService.UpdateFaceAvatar(userId, faceUrl, avatarUrl);
Assert.True(result);
}

試結果:

1 passed, 0 failed, 0 skipped, took 11.38 seconds (xUnit.net 1.9.1 build 1600).

測試通過!

用SQL Profiler查看一下數據庫中實際執行的SQL語句:

exec sp_executesql N'UPDATE [dbo].[Users] SET
[FaceUrl] = @p__update__0,
[AvatarUrl] = @p__update__1
FROM [dbo].[Users] AS j0 INNER JOIN (
SELECT
AS [C1],
[Extent1].[UserId] AS [UserId]
FROM [dbo].[Users] AS [Extent1]
WHERE [Extent1].[UserId] = @p__linq__0
) AS j1 ON (j0.[UserId] = j1.[UserId])',N'@p__linq__0 uniqueidentifier,@p__update__0 nvarchar(24),@p__update__1 nvarchar(24)',
@p__linq__0='BD42420B-63CF-DD11-9E4D-001CF0CD104B',@p__update__0=N'20150810180742.png',@p__update__1=N'20150810180743.png'

就是我們期望的SQL語句。

最終驗證了,添加IUnitOfWork.UpadteAsync()接口,基于EntityFramework.Extended,用EF實現數據庫中指定字段的更新,這種方法在實際開發中使用完全可行。

于是,又少了一個使用存儲過程的理由。

看完上述內容,你們掌握EntityFramework.Extended如何用EF實現指定字段的更新的方法了嗎?如果還想學到更多技能或想了解更多相關內容,歡迎關注億速云行業資訊頻道,感謝各位的閱讀!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

东兰县| 桂东县| 施甸县| 图片| 三亚市| 水富县| 溧水县| 汉沽区| 高安市| 股票| 济阳县| 昭苏县| 龙岩市| 安顺市| 扎兰屯市| 汕头市| 赫章县| 新源县| 格尔木市| 张家港市| 义马市| 扶沟县| 岱山县| 芦溪县| 抚松县| 江阴市| 永和县| 库车县| 大足县| 嘉定区| 察哈| 南靖县| 淮北市| 石家庄市| 陆良县| 桦南县| 达日县| 凌源市| 鹤庆县| 聂拉木县| 调兵山市|