(三)毛子整潔架構(Infrastructure層/DapperHelper/樂觀鎖)


文章目錄

  • 項目地址
  • 一、Infrastructure Layer
    • 1.1 創建Application層需要的服務
      • 1. Clock服務
      • 2. Email 服務
      • 3. 注冊服務
    • 1.2 數據庫服務
      • 1. 表配置Configurations
      • 2. Respository實現
      • 3. 數據庫鏈接Factory實現
      • 4. Dapper的DataOnly服務實現
      • 5. 所有數據庫服務注冊
    • 1.3 基于RowVersion的樂觀鎖
      • 1. 添加RowVersion字段
      • 2. 修改ReserveBookingCommandHandler


項目地址

  • 教程作者:
  • 教程地址:
  • 代碼倉庫地址:
  • 所用到的框架和插件:
dbt 
airflow

一、Infrastructure Layer

在這里插入圖片描述

添加對Application層和Domain層的引用
在這里插入圖片描述

1.1 創建Application層需要的服務

1. Clock服務

  • 當前時間接口的實現
using Bookify.Application.Abstractions.Clock;
namespace Bookify.Infrastructure.Clock;
internal sealed class DateTimeProvider : IDateTimeProvider
{public DateTime UtcNow => DateTime.UtcNow;
}

2. Email 服務

  • Email的服務,實現Appliaction的接口
using Bookify.Application.Abstractions.Email;
namespace Bookify.Infrastructure.Email;
internal sealed class EmailService : IEmailService
{public Task SendAsync(Domain.Users.Email recipient, string subject, string body){return Task.CompletedTask;}
}

3. 注冊服務

  • 注冊上面的兩個服務
public static class DependencyInjection
{public static IServiceCollection AddInfrastructure(this IServiceCollection services,IConfiguration configuration){services.AddTransient<IDateTimeProvider, DateTimeProvider>();services.AddTransient<IEmailService, EmailService>();return services;}
}

1.2 數據庫服務

1. 表配置Configurations

  1. BookingConfiguration.cs
internal sealed class BookingConfiguration : IEntityTypeConfiguration<Booking>
{public void Configure(EntityTypeBuilder<Booking> builder){//設置數據庫表名為 bookingsbuilder.ToTable("bookings");//設置主鍵為 Idbuilder.HasKey(booking => booking.Id);//OwnsOne處理值對象builder.OwnsOne(booking => booking.PriceForPeriod, priceBuilder =>{priceBuilder.Property(money => money.Currency).HasConversion(currency => currency.Code, code => Currency.FromCode(code));});builder.OwnsOne(booking => booking.CleaningFee, priceBuilder =>{priceBuilder.Property(money => money.Currency).HasConversion(currency => currency.Code, code => Currency.FromCode(code));});builder.OwnsOne(booking => booking.AmenitiesUpCharge, priceBuilder =>{priceBuilder.Property(money => money.Currency).HasConversion(currency => currency.Code, code => Currency.FromCode(code));});builder.OwnsOne(booking => booking.TotalPrice, priceBuilder =>{priceBuilder.Property(money => money.Currency).HasConversion(currency => currency.Code, code => Currency.FromCode(code));});builder.OwnsOne(booking => booking.Duration);builder.HasOne<Apartment>().WithMany().HasForeignKey(booking => booking.ApartmentId);builder.HasOne<User>().WithMany().HasForeignKey(booking => booking.UserId);}
}
  1. ApartmentConfiguration.cs
internal sealed class ApartmentConfiguration : IEntityTypeConfiguration<Apartment>
{public void Configure(EntityTypeBuilder<Apartment> builder){builder.ToTable("apartments");builder.HasKey(apartment => apartment.Id);builder.OwnsOne(apartment => apartment.Address);builder.Property(apartment => apartment.Name).HasMaxLength(200).HasConversion(name => name.Value, value => new Name(value));builder.Property(apartment => apartment.Description).HasMaxLength(2000).HasConversion(description => description.Value, value => new Description(value));builder.OwnsOne(apartment => apartment.Price, priceBuilder =>{priceBuilder.Property(money => money.Currency).HasConversion(currency => currency.Code, code => Currency.FromCode(code));});builder.OwnsOne(apartment => apartment.CleaningFee, priceBuilder =>{priceBuilder.Property(money => money.Currency).HasConversion(currency => currency.Code, code => Currency.FromCode(code));});builder.Property<uint>("Version").IsRowVersion();}
}

2. Respository實現

  1. Repository.cs
internal abstract class Repository<T>where T : Entity
{protected readonly ApplicationDbContext DbContext;protected Repository(ApplicationDbContext dbContext){DbContext = dbContext;}public async Task<T?> GetByIdAsync(Guid id,CancellationToken cancellationToken = default){return await DbContext.Set<T>().FirstOrDefaultAsync(user => user.Id == id, cancellationToken);}public void Add(T entity){DbContext.Add(entity);}
}
  1. BookingRepository.cs
namespace Bookify.Infrastructure.Repositories;internal sealed class BookingRepository : Repository<Booking>, IBookingRepository
{private static readonly BookingStatus[] ActiveBookingStatuses ={BookingStatus.Reserved,BookingStatus.Confirmed,BookingStatus.Completed};public BookingRepository(ApplicationDbContext dbContext): base(dbContext){}public async Task<bool> IsOverlappingAsync(Apartment apartment,DateRange duration,CancellationToken cancellationToken = default){return await DbContext.Set<Booking>().AnyAsync(booking =>booking.ApartmentId == apartment.Id &&booking.Duration.Start <= duration.End &&booking.Duration.End >= duration.Start &&ActiveBookingStatuses.Contains(booking.Status),cancellationToken);}
}

3. 數據庫鏈接Factory實現

//創建Dapper鏈接數據庫的工廠
internal sealed class SqlConnectionFactory : ISqlConnectionFactory
{private readonly string _connectionString;public SqlConnectionFactory(string connectionString){_connectionString = connectionString;}public IDbConnection CreateConnection(){var connection = new NpgsqlConnection(_connectionString);connection.Open();return connection;}
}

4. Dapper的DataOnly服務實現

namespace Bookify.Infrastructure.Data;
/// <summary>
/// A Dapper type handler for <see cref="DateOnly"/>.
/// </summary>
internal sealed class DateOnlyTypeHandler : SqlMapper.TypeHandler<DateOnly>
{public override DateOnly Parse(object value) => DateOnly.FromDateTime((DateTime)value);public override void SetValue(IDbDataParameter parameter, DateOnly value){parameter.DbType = DbType.Date;parameter.Value = value;}
}

5. 所有數據庫服務注冊

  • 注冊所有infrastructure的服務

public static class DependencyInjection
{public static IServiceCollection AddInfrastructure(this IServiceCollection services,IConfiguration configuration){services.AddTransient<IDateTimeProvider, DateTimeProvider>();services.AddTransient<IEmailService, EmailService>();string connectionString =configuration.GetConnectionString("Database") ??throw new ArgumentNullException(nameof(configuration));services.AddDbContext<ApplicationDbContext>(options =>{options.UseNpgsql(connectionString).UseSnakeCaseNamingConvention();});services.AddScoped<IUserRepository, UserRepository>();services.AddScoped<IApartmentRepository, ApartmentRepository>();services.AddScoped<IBookingRepository, BookingRepository>();services.AddScoped<IUnitOfWork>(sp => sp.GetRequiredService<ApplicationDbContext>());services.AddSingleton<ISqlConnectionFactory>(_ =>new SqlConnectionFactory(connectionString));SqlMapper.AddTypeHandler(new DateOnlyTypeHandler());return services;}
}

1.3 基于RowVersion的樂觀鎖

1. 添加RowVersion字段

  • 這里我們預定apartment需要樂觀鎖,所以在他的數據庫configuration里添加

在這里插入圖片描述

2. 修改ReserveBookingCommandHandler

  • 發成沖突后,返回已經預定的錯誤
    在這里插入圖片描述

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/bicheng/80886.shtml
繁體地址,請注明出處:http://hk.pswp.cn/bicheng/80886.shtml
英文地址,請注明出處:http://en.pswp.cn/bicheng/80886.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

uni-app微信小程序登錄流程詳解

文章目錄 uni-app微信小程序登錄流程實戰詳解微信小程序登錄流程概述1. 獲取登錄憑證&#xff08;code&#xff09;2. 發送登錄請求3. 保存登錄態4. 登錄狀態管理5. 應用登錄狀態請求攔截器中添加 token自動登錄頁面路由守衛 使用 Vuex 集中管理登錄狀態登錄組件示例登錄流程最…

GUC并發編程和SpringCloud,二者之間的關系

一.提問 我認為&#xff0c;Java開發中&#xff0c;如果項目的每一個小模塊需要不同人員并行開發時&#xff0c;就需要使用SpringCloud&#xff1b;如果要解決系統用戶激增&#xff0c;就是用GUC并發編程。 這個說法對么&#xff1f; 二.解答 你的理解部分正確&#xff0c;但不…

在 Vue 3 中使用 canvas-confetti 插件

&#x1f389; 在 Vue 3 中使用 canvas-confetti 插件 canvas-confetti 是一個輕量、無依賴的 JavaScript 動畫庫&#xff0c;用于在網頁上展示彩帶、慶祝動畫。非常適合用于抽獎、支付成功、活動慶祝等場景。 本教程將指導你如何在 Vue 3 項目中集成并使用該插件。 &#x1…

深入解析Spring Boot項目目錄結構:從新手到規范實踐

一、標準項目結構全景圖 典型的Spring Boot項目&#xff08;Maven構建&#xff09;目錄結構如下&#xff1a; my-spring-project/ ├── src/ │ ├── main/ │ │ ├── java/ # 核心代碼 │ │ │ └── com/ │ │ │ └── exa…

【C語言】宏經典練習題,交換奇偶位

交換奇偶位 寫一個宏&#xff0c;可以將一個整數的二進制位的奇數位和偶數位交換。 #define Swap(x) x(((x&0x55555555)<<1)((x&0xaaaaaaaa)>>1)) int main() {int a 10;Swap(a);printf("%d\n", a);return 0; } 寫宏的思路&#xff1a; 假設…

VSCode-插件:codegeex:ai coding assistant / 清華智普 AI 插件

一、官網 https://codegeex.cn/ 二、vscode 安裝插件 點擊安裝即可&#xff0c;無需復雜操作&#xff0c;國內軟件&#xff0c;無需科學上網&#xff0c;非常友好 三、智能注釋 輸入 // 或者 空格---后邊自動出現注釋信息&#xff0c;&#xff0c;按下 Tab 鍵&#xff0c;進…

FFmpeg 與 C++ 構建音視頻處理全鏈路實戰(三)—— FFmpeg 內存模型

經過前面文章的 FFmpeg 編程實踐&#xff0c;相信你已經對AVPacket和AVFrame這兩個核心結構體不再陌生。當我們編寫代碼時&#xff0c;頻繁調用unref系列 API 釋放內存的操作&#xff0c;或許讓你心生疑惑&#xff1a;這些函數究竟是如何實現內存釋放的&#xff1f;又該在何時準…

c 中的哈希表

哈希是一種可以接受各種類型、大小的輸入&#xff0c;輸出一個固定長度整數的過程。你可以將哈希理解成一種特殊的映射&#xff0c;哈希映射&#xff0c;將一個理論無限的集合A映射到有限整數集合B上。 哈希函數&#xff1a;哈希函數是哈希過程的核心&#xff0c;它決定了哈希映…

【一次成功!】Ubuntu22.04安裝cartographer

之前在ubuntu20.04上成功安裝cartographer&#xff0c;但是翻遍全網都沒找到官方的22.04安裝教程&#xff0c;然后找到小魚的&#xff0c;試了一下&#xff0c;一次成功&#xff0c;連接如下&#xff1a; gd2l-ros2/docs/humble/chapt10/get_started/2.Carto介紹及安裝.md at …

【WPF】Opacity 屬性的使用

在WPF&#xff08;Windows Presentation Foundation&#xff09;中&#xff0c;Opacity 屬性是定義一個元素透明度的屬性&#xff0c;其值范圍是從 0.0&#xff08;完全透明&#xff09;到 1.0&#xff08;完全不透明&#xff09;。由于 Opacity 是在 UIElement 類中定義的&…

8天Python從入門到精通【itheima】-6~10

目錄 7節-開發出第一個Python程序: 1.在cmd窗口寫下第一個最簡單的程序:Hello World!!! 9節: 1.如何卸載python: 2.報錯:不是可運行的程序 ?編輯 3.報錯:無法初始化設備PRN: 4.報錯:語法錯誤——非法的字符 10節-python解釋器: 1.python解釋器的原理: 2.解…

Mac 3大好用的復制粘貼管理工具對比

剪貼板管理器是查看復制粘貼歷史記錄的工具&#xff0c;幾乎是每個蘋果電腦用戶必備工具。市面上的工具很多&#xff0c;我結合了功能豐富、設計簡潔、交互便利整理了目前3款頭部剪貼板應用 Paste、PasteNow、PasteMe。 Paste 優勢&#xff1a;老牌剪切板應用&#xff0c;功能…

2025年全國青少年信息素養大賽初賽模擬測試網站崩了的原因及應對比賽流程

2025年全國青少年信息素養大賽初賽模擬測試昨天開始&#xff0c;由于同一時間涌入太多的人&#xff0c;導致網站的服務器奔了&#xff0c;出現了各種狀況&#xff0c;導致很多人沒有模擬上&#xff0c;大家今天可以刷新或者提前打開網頁。 有的是一直“轉圈圈”&#xff0c;有的…

02 | 大模型微調 | 從0學習到實戰微調 | 從數學概率到千億參數大模型

一、導讀 作為非AI專業技術開發者&#xff08;我是小小爬蟲開發工程師&#x1f60b;&#xff09; 本系列文章將圍繞《大模型微調》進行學習&#xff08;也是我個人學習的筆記&#xff0c;所以會持續更新&#xff09;&#xff0c;最后以上手實操模型微調的目的。 &#xff08;…

十四、繼承與組合(Inheritance Composition)

十四、繼承與組合&#xff08;Inheritance & Composition&#xff09; 引言 C最引人注目的特性之一是代碼復用。組合&#xff1a;在新類中創建已有類的對象。繼承&#xff1a;將新類作為已有類的一個類型來創建。 14.1 組合的語法 Useful.h //C14:Useful.h #ifndef US…

2025年5月-信息系統項目管理師高級-軟考高項一般計算題

決策樹和期望貨幣值 加權算法 自制和外購分析 溝通渠道 三點估算PERT 當其他條件一樣時&#xff0c;npv越大越好

OpenJDK 17 中線程啟動的完整流程用C++ 源碼詳解

1. 線程創建入口&#xff08;JNI 層&#xff09; 當 Java 層調用 Thread.start() 時&#xff0c;JVM 通過 JNI 進入 JVM_StartThread 函數&#xff1a; JVM_ENTRY(void, JVM_StartThread(JNIEnv* env, jobject jthread))// 1. 檢查線程狀態&#xff0c;防止重復啟動if (java_…

Spring MVC參數傳遞

本內容采用最新SpringBoot3框架版本,視頻觀看地址:B站視頻播放 1. Postman基礎 Postman是一個接口測試工具,Postman相當于一個客戶端,可以模擬用戶發起的各類HTTP請求,將請求數據發送至服務端,獲取對應的響應結果。 2. Spring MVC相關注解 3. Spring MVC參數傳遞 Spri…

Python面向對象編程(OOP)深度解析:從封裝到繼承的多維度實踐

引言 面向對象編程(Object-Oriented Programming, OOP)是Python開發中的核心范式&#xff0c;其三大特性——??封裝、繼承、多態??——為構建模塊化、可維護的代碼提供了堅實基礎。本文將通過代碼實例與理論結合的方式&#xff0c;系統解析Python OOP的實現機制與高級特性…

0.66kV0.69kV接地電阻柜常規配置單

0.66kV/0.69kV接地電阻柜是變壓器中性點接地電阻柜中的特殊存在&#xff0c;主要應用于低壓柴油發電機組220V、火力發電廠380V、煤炭企業660V/690V等電力系統或電力用戶1000V的低壓系統中。 我們來看看0.66kV0.69kV接地電阻柜配置單&#xff1a; 配置特點如下&#xff1a; 1…