產品代碼都給你看了,可別再說不會DDD(六):聚合根與資源庫

来源:https://www.cnblogs.com/davenkin/archive/2023/09/17/ddd-aggregate-and-repository.html
-Advertisement-
Play Games

這是一個講解DDD落地的文章系列,作者是《實現領域驅動設計》的譯者滕雲。本文章系列以一個真實的並已成功上線的軟體項目——碼如雲(https://www.mryqr.com)為例,系統性地講解DDD在落地實施過程中的各種典型實踐,以及在面臨實際業務場景時的諸多取捨。 本系列包含以下文章: DDD入門 ...


這是一個講解DDD落地的文章系列,作者是《實現領域驅動設計》的譯者滕雲。本文章系列以一個真實的並已成功上線的軟體項目——碼如雲https://www.mryqr.com)為例,系統性地講解DDD在落地實施過程中的各種典型實踐,以及在面臨實際業務場景時的諸多取捨。

本系列包含以下文章:

  1. DDD入門
  2. DDD概念大白話
  3. 戰略設計
  4. 代碼工程結構
  5. 請求處理流程
  6. 聚合根與資源庫(本文)
  7. 實體與值對象
  8. 應用服務與領域服務
  9. 領域事件
  10. CQRS

案例項目介紹

既然DDD是“領域”驅動,那麼我們便不能拋開業務而只講技術,為此讓我們先從業務上瞭解一下貫穿本文章系列的案例項目 —— 碼如雲(不是馬雲,也不是碼雲)。如你已經在本系列的其他文章中瞭解過該案例,可跳過。

碼如雲是一個基於二維碼的一物一碼管理平臺,可以為每一件“物品”生成一個二維碼,並以該二維碼為入口展開對“物品”的相關操作,典型的應用場景包括固定資產管理、設備巡檢以及物品標簽等。

在使用碼如雲時,首先需要創建一個應用(App),一個應用包含了多個頁面(Page),也可稱為表單,一個頁面又可以包含多個控制項(Control),比如單選框控制項。應用創建好後,可在應用下創建多個實例(QR)用於表示被管理的對象(比如機器設備)。每個實例均對應一個二維碼,手機掃碼便可對實例進行相應操作,比如查看實例相關信息或者填寫頁面表單等,對錶單的一次填寫稱為提交(Submission);更多概念請參考碼如雲術語

在技術上,碼如雲是一個無代碼平臺,包含了表單引擎、審批流程和數據報表等多個功能模塊。碼如雲全程採用DDD完成開發,其後端技術棧主要有Java、Spring Boot和MongoDB等。

碼如雲的源代碼是開源的,可以通過以下方式訪問:

碼如雲源代碼:https://github.com/mryqr-com/mry-backend

聚合根與資源庫

在上一篇請求處理流程中我們講到,領域模型是DDD的核心,而聚合根又是領域模型的核心。從某種意義上講,DDD中其它組件均可看作是對聚合根的支撐或輔助。在本文中,我們將對聚合根以及與之密切相關的資源庫(Repository)做詳細的講解。

聚合根是什麼

DDD概念大白話一文中,我們講到了“什麼是聚合根”,這裡再重覆一下。聚合根中的“聚合”即“高內聚,低耦合”中的“內聚”之意;而“根”則是“根部”的意思,也即聚合根是一種統領式的存在。事實上,並不存在一個教科書式的對聚合根的理論定義,你可以將聚合根理解為一個系統中最重要最顯著的那些名詞,這些名詞是其所在的軟體系統之所以存在的原因。為了給你一個直觀的理解,以下是幾個聚合根的例子:

  • 在一個電商系統中,一個訂單(Order)對象表示一個聚合根
  • 在一個CRM系統中,一個客戶(Customer)對象表示一個聚合根
  • 在一個銀行系統中,一次交易(Transaction)對象表示一個聚合根

你可能會問,軟體中的概念已經很多了,為什麼還要搞出個聚合根的概念?我們認為這裡至少有2點原因:

  1. 聚合根遵循了軟體中“高內聚,低耦合”的基本原則
  2. 聚合根體現了一種模塊化的原則,模塊化思想是被各個行業所證明的可以降低系統複雜度的一種思想。所謂的DDD是“軟體核心複雜性應對之道”,也即這個意思,它將軟體系統在人腦中所呈現地更加有序和簡單,讓人可以更好地理解和管控軟體系統。

在實際項目中識別聚合根時,我們需要對業務有深入的瞭解,因為只有這樣你才知道到底哪些業務邏輯是內聚在一起的。這也是我們一直建議程式員和架構師們不要一味地埋頭於技術而要多關註業務的原因。

事實上,如果讓一個從來沒有接觸過DDD的人來建模,十有八九也能設計出上面的訂單、客戶和交易對象出來。沒錯,DDD絕非什麼顛覆式的發明,依然只是在前人基礎上的一種進步而已,這種進步更多的體現在一些設計原則上,對此我們將在下文進行詳細闡述。

聚合根基類

在代碼實現層面,一般的實踐是將所有的聚合根都繼承自一個公共基類AggregateRoot

//AggregateRoot

@Getter
public abstract class AggregateRoot implements Identified {
    private String id;//聚合根ID
    private String tenantId;//租戶ID
    private Instant createdAt;//創建時間
    private String createdBy;//創建人的MemberId
    private String creator;//創建人姓名
    private Instant updatedAt;//更新時間
    private String updatedBy;//更新人MemberId
    private String updater;//更新人姓名
    private List<DomainEvent> events;//臨時存放領域事件
    private LinkedList<OpsLog> opsLogs;//操作日誌

    @Version
    @Getter(PRIVATE)
    private Long _version;//版本號,實現樂觀鎖

    //...此處省略了AggregateRoot中行為方法

    @Override
    public String getIdentifier() {
        return id;
    }
}

源碼出處:com/mryqr/core/common/domain/AggregateRoot.java

AggregateRoot中,包含聚合根ID(id)、創建信息(createdAtcreatedBy)和更新信息(updatedAtupdatedBy)等數據。租戶ID(tenantId)用於標定聚合根所在的租戶(碼如雲是一個多租戶系統)。另外,events用於臨時性存放聚合根中所產生的領域事件,我們將在領域事件一文中對此所詳細解釋。

實際的聚合根繼承自AggregateRoot,例如,在碼如雲中,分組(Group)聚合根的實現如下:

@Getter
@Document(GROUP_COLLECTION)
@TypeAlias(GROUP_COLLECTION)
@NoArgsConstructor(access = PRIVATE)
public class Group extends AggregateRoot {
    private String name;//名稱
    private String appId;//所在的app
    private List<String> managers;//管理員
    private List<String> members;//普通成員
    private boolean archived;//是否歸檔
    private String customId;//自定義編號
    private boolean active;//是否啟用
    private String departmentId;//由哪個部門同步而來
    
    //...此處省略了Group的行為方法
}

源碼出處:com/mryqr/core/group/domain/Group.java

聚合根基本原則

從上面的代碼例子可以看出,聚合根只是普通的Java對象而已,真正使之成為聚合根的是一些特定的設計原則。

內聚性原則

這個原則不用我們再細講了吧,估計你在大學里就學過,只舉個例子,對於上面的分組Group對象來說,管理員managers、普通成員members以及啟用標誌active均是Group不可分割的屬性,這些屬性獨立於Group是無法存在的。

對外黑盒原則

對外黑盒原則講的是,聚合根的外部(也即聚合根的調用方或客戶方)不需要關心聚合根內部的實現細節,而只需要通過調用聚合根向外界暴露的共有業務方法即可。具體表現為,外部對聚合根的調用只能通過根對象完成,而不能調用聚合根內部對象上的方法。舉個例子,在碼如雲中,管理員可以向分組(Group)中添加成員,具體的實現代碼如下:

//Group

public void addMembers(List<String> memberIds, User user) {
    if (isSynced()) {
        throw new MryException(GROUP_SYNCED,
                "無法添加成員,已設置從部門同步。",
                "groupId", this.getId());
    }

    this.members = concat(members.stream(), memberIds.stream())
            .distinct()
            .collect(toImmutableList());
    
    addOpsLog("設置成員", user);
}

源碼出處:com/mryqr/core/group/domain/Group.java

這裡,外部在向分組中添加成員時,需要調用Group上的addMembers()方法,該方法知道將memberIds添加到自身的members欄位中,這個過程對外部是不可見的。與之相對的另一種方式是,外部調用法先拿到Membermembers引用,然後由外部自行向members中添加memberIds

//外部調用方

@Transactional
public void addGroupMembers(String groupId, List<String> memberIds, User user) {
    Group group = groupRepository.byIdAndCheckTenantShip(groupId, user);

    if (group.isSynced()) {
        throw new MryException(GROUP_SYNCED,
                "無法添加成員,已設置從部門同步。",
                "groupId", group.getId());
    }

    List<String> members = group.getMembers();
    members.addAll(memberIds);
    groupRepository.save(group);

    log.info("Added members{} to group[{}].", memberIds, groupId);
}

源碼出處:com/mryqr/core/group/command/GroupCommandService.java

這種方式是一種反模式,存在以下缺點:

  • 外部需要瞭解Group的內部結構,背離了對外黑盒原則,本例中,外部通過group.getMembers()獲取到了Group內部的members屬性
  • 聚合根內部的業務邏輯泄漏到了外部,背離了內聚性原則,本例中,對group.isSynced()的調用原本應該放在Group中的,結果卻由外部承擔了該職責

在對外黑盒原則的指導下,聚合根自然形成了一個邊界,它站在這個邊界上向外聲明:“我所包圍著的內部的所有均由我負責,如果誰想訪問我的內部,直接訪問是被禁止的,只能通過我這個“根”來訪問。”

不變條件原則

不變條件(Invariants)表示聚合根需要保證其內部在任何時候均處於一種合法的狀態(也即數據一致性需要得到保證),一個常見的例子是訂單(Order)中有訂單項(OrderItem)和訂單價格(Price),當訂單項發生變化時,其價格應該隨之發生變化,並且這兩種變化應該在訂單的同一個業務方法中完成。這一點是好理解的,既然聚合根對外是一個黑盒,那麼外界便不會負責給你聚合根擦屁股,你聚合根自己需要保證自身的正確性。

在碼如雲中,應用管理員可以向分組(Group)中添加分組管理員。這其中有層隱含意思是,既然分組管理員也是分組成員,那麼在添加分組管理員的同時需要一併將其添加到分組成員中,具體實現代碼如下:

//Group

public void addManager(String memberId, User user) {
    if (!this.members.contains(memberId)) {
        this.members = concat(members.stream(), Stream.of(memberId))
                .distinct()
                .collect(toImmutableList());
    }

    this.managers = concat(this.managers.stream(), Stream.of(memberId))
            .distinct()
            .collect(toImmutableList());
    
    raiseEvent(new GroupManagersChangedEvent(this.getId(), this.getAppId(), user));
    
    addOpsLog("添加管理員", user);
}

源碼出處:com/mryqr/core/group/domain/Group.java

在本例的添加分組管理員addManager()方法中,我們除了向managers中添加成員外,還保證了該成員也出現在members中。這裡的“分組管理員也是分組成員”即是一種不變條件,我們需要在聚合根內部保證不變條件不被破壞,因為不變條件往往意味著核心的業務邏輯。

通過ID引用其他聚合根原則

當一個聚合根需要引用另一個聚合根時,並不需要維持對另一聚合根的整體引用,而是只需通過ID進行引用即可。這個原則的出發點是:聚合根和聚合根之間是一種平級關係,並不是隸屬關係,每個聚合根本身是一個相對獨立的模塊,其與其他聚合根的關係應該通過ID這種松耦合的方式進行引用,如果整體引用則更像是一種包含關係。

在碼如雲中,分組(Group)通過appId引用其所屬的應用(App),通過departmentId引用所同步的部門(Department),而在managersmembers欄位中,則是以memberId引用相應成員(Member):

@Getter
@Document(GROUP_COLLECTION)
@TypeAlias(GROUP_COLLECTION)
@NoArgsConstructor(access = PRIVATE)
public class Group extends AggregateRoot {
    private String name;//名稱
    private String appId;//所在的app
    private List<String> managers;//管理員
    private List<String> members;//普通成員
    private boolean archived;//是否歸檔
    private String customId;//自定義編號
    private boolean active;//是否啟用
    private String departmentId;//由哪個部門同步而來
   
    //...省略其他代碼
}

源碼出處:com/mryqr/core/group/domain/Group.java

與基礎設施無關原則

既然整個領域模型與基礎設施無關,那麼位於領域模型之內的聚合根自然也不能與基礎設施相關,這樣好處是將業務複雜度與技術複雜度解耦開來,讓業務模型可以獨立於技術設施而完成自身的演變。比如,假設一個項目需要從Spring框架遷移到Guice框架,此時如果能夠保證領域模型與基礎設施的無關性,那麼對領域模型的遷移過程講變得非常簡單,基本上無需修改任何代碼直接拷貝到新的項目中即可。

事實上,碼如雲尚未完全做到這一點,從上面的例子中可以看到,AggregateRootGroup對Spring框架中的@Version@Document@TypeAlias3個與持久化相關的註解存在引用。如需解決這個問題,可以考慮在領域模型之外另建專門用於資料庫訪問的持久化對象(Persistence Model)。但是,引入持久化對象是有成本的,比如需要維護領域對象與持久化對象之間的相互轉化等。在碼如雲,我們選擇了妥協,一方面考慮到持久化對象的成本,另一方面我們也預見在將來要遷移出Spring框架的幾率是非常小的。不過,除了前面提到的3個註解之外,碼如雲中的聚合根可以做到對基礎設施沒有任何其他引用。關於持久化對象,在Stackoverflow上有過非常有意義的討論,讀者可自行閱覽。

跨聚合根用例

通常來講,一個業務用例只會操作一個(或一種)聚合根。但有時,一個業務用例可能會導致多個(或多種)聚合根對象的更新,此時可分兩種情況:

  1. 如果聚合根位於不同的進程空間(比如不同的微服務)中,那麼解決方式一是可以使用事件驅動架構(EDA),二是通過全局事務(比如JTA)完成。基於全局事務的性能和效率低下等問題,DDD社區一般建議採用事件驅動架構,即在一個進程空間中只對其包含的聚合根進行操作,然後通過向其他進程空間發送事件通知的方式,使得其他進程空間做相應的聚合根更新。
  2. 如果聚合根位於同一個進程空間,此時依然可以選擇事件驅動架構,但是另一種更簡單實用的方式是直接同時更新多個聚合根,畢竟此時對所有聚合根的更新均處於同一個本地事務中。

碼如雲是一個單體系統,因此屬於以上的第2種情況,我們根據聚合根之間的業務緊密程度的不同,在有些場景下選擇了同時更新多個聚合根,在另一些場景下則選擇通過事件驅動機制解決。比如,在“創建實例”的用例中,除了創建實例(QR)之外,還需要創建該實例對應的碼牌(Plate),由於“有實例就必有碼牌”,因此它們之間是緊密聯繫的,故在碼如雲中我們選擇了在同一個本地事務中同時更新實例和碼牌:

//QrCommandService
    
@Transactional
public CreateQrResponse createQr(CreateQrCommand command, User user) {
    String name = command.getName();
    String groupId = command.getGroupId();

    Group group = groupRepository.cachedByIdAndCheckTenantShip(groupId, user);
    String appId = group.getAppId();
    App app = appRepository.cachedById(appId);

    PlatedQr platedQr = qrFactory.createPlatedQr(name, group, app, user);
    QR qr = platedQr.getQr();
    Plate plate = platedQr.getPlate();

    //同時保存QR和Plate
    qrRepository.save(qr);
    plateRepository.save(plate);

    log.info("Created qr[{}] of group[{}] of app[{}].",
            qr.getId(), groupId, appId);

    return CreateQrResponse.builder()
            .qrId(qr.getId())
            .plateId(plate.getId())
            .groupId(groupId)
            .appId(appId)
            .build();
}

源碼出處:com/mryqr/core/group/command/GroupCommandService.java

可以看到,在用例方法createQr()中,我們先後調用qrRepository.save(qr)plateRepository.save(plate)分別完成了對QRPlate的持久化。

如果你希望瞭解事件驅動架構相關的知識,請參考本系列的領域事件一文。

資源庫

在DDD中,資源庫(Repository)以聚合根為單位完成對資料庫的訪問。這裡的重點是“以聚合根為單位”,也即只有聚合根才配得上擁有資源庫(畢竟在DDD中大家都是圍繞著聚合根轉的嘛),其他對象(比如非聚合根實體)是沒有對應資源庫的,這也是資源庫和DAO最大的區別。在編碼實現時,資源庫方法所接受的參數和返回的數據都應該是聚合根對象,例如,在碼如雲中,成員(Member)聚合根對應的資源庫定義如下:

public interface MemberRepository {
    Member byId(String id); //返回聚合根

    Optional<Member> byIdOptional(String id); //返回聚合根

    Member byIdAndCheckTenantShip(String id, User user); //返回聚合根

    void save(Member member); //聚合根作為參數

    void delete(Member member); //聚合根作為參數
}

源碼出處:com/mryqr/core/member/domain/MemberRepository.java

行業中這麼一個現象,很多程式員在面對一個新的業務需求時,首先想到的是如何設計資料庫的表結構,然後再編寫業務代碼。在DDD中,這是一種反模式,既然是“領域驅動”,那麼我們首先應該關心的是如何業務建模,而不是資料庫建模。事實上,正如Robert C. Martin在《整潔架構》一書中所說,資料庫只是一個實現細節而已,不應該成為軟體建模的主體。

資源庫的作用,在於它在業務複雜度和技術複雜度之間做了一層很好的隔離,讓我們可以獨立地看待軟體的業務模型而不受技術設施的影響。從本質上講,資源庫做的事情只是實現數據在記憶體和磁碟之間相互傳輸而已。在編程實現業務邏輯的時候,我們只需關心記憶體中的那個聚合根對象即可,當聚合根對象的狀態由於業務操作發生了改變之後,再調用資源庫將新的聚合根狀態同步到磁碟中完成持久化,在調用時我們假設並相信資源庫一定可以完成其自身的使命。

@Transactional
public void addGroupManager(String groupId, String memberId, User user) {
    Group group = groupRepository.byIdAndCheckTenantShip(groupId, user);

    group.addManager(memberId, user);
    
    groupRepository.save(group);
    
    log.info("Added manager[{}] to group[{}].", memberId, groupId);
}

源碼出處:com/mryqr/core/group/command/GroupCommandService.java

在上例的“向分組中添加管理員”用例中,首先通過資源庫GroupRepositorybyIdAndCheckTenantShip()方法得到聚合根Group對象,然後再完成後續操作。這裡的addGroupManager()無需知道Group是如何載入的,甚至不用知道後臺使用的是MySQL還是MongoDB或是其他,反正通過調用GroupRepository.byIdAndCheckTenantShip()可以得到一個完整合法的Group對象即可。

在資源庫中,最重要的方法有以下3個:

public interface GroupRepository {
    
    Group byId(String id);
    
    void save(Group group);
    
    void delete(Group group);
}

源碼出處:com/mryqr/core/member/domain/MemberRepository.java

其中,byId()用於根據ID獲取指定聚合根,save()用於保存聚合根,delete()則用於刪除聚合根。除此之外,資源庫中還可以包含更多的查詢方法,比如在GroupRepository中還包含以下方法:

//根據部門ID查找分組
List<Group> byDepartmentId(String departmentId);

//根據ID查找分組,返回Optional
Optional<Group> byIdOptional(String id);

//根據ID查找分組,同時檢查租戶
Group byIdAndCheckTenantShip(String id, User user);

源碼出處:com/mryqr/core/member/domain/MemberRepository.java

需要註意的是,這裡的查詢方法指的是在實現業務邏輯的過程中需要做的查詢操作,並不是為了前端顯示那種純粹的查詢,因為純粹的查詢操作不見得一定要放到資源庫中,而是可以作為一個單獨的關註點通過CQRS解決。

在DDD項目中,通常將資源庫分為介面類和實現類,將介面類放置在領域模型domain包中,而將實現類放置在基礎設施infrastructure包中,這種做法有2點好處:

  1. 通過依賴反轉,使得領域模型不依賴於基礎設施
  2. 實現資源庫的可插拔性,比如未來需要從MongoDB遷移到MySQL,那麼只需創建新的實現類即可

總結

在本文中,我們講到了作為DDD核心的聚合根的設計原則及實現,其中包含內聚原則、對外黑盒原則和不變條件原則等。此外,我們也對與聚合根密切相關的資源庫做了講解。在下一篇實體與值對象中,我們將講到實體和值對象之間的區別,以及各自的典型編碼實踐。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1.date文件的備份 2.mysqldump 備份 說明:mysqldump是MySQL資料庫中的一個實用程式,它主要用於轉儲(備份)資料庫。mysqldump通過生成一個SQL腳本文件,包含從頭開始重新創建資料庫所必需的(如 CREATE TABLE和INSERT等),來實現資料庫的備份和轉儲。 ...
  • 1.分組 group by 詳情見,發佈的第七篇博客文章,7- MySQL函數 2.排序 order by 說明:在MySQL中,ORDER BY是一種用於對查詢結果進行排序的關鍵字。它可以根據一列或多列的值,以升序或降序的方式對查詢結果進行排序,使得查詢者可以更加方便 地查看、分析和處理數據。 使 ...
  • 1.分組group by 在MySQL中,GROUP BY的意思是“分組查詢”,它可以根據一個或多個欄位對查詢結果進行分組。 GROUP BY的作用是通過一定的規則將一個數據集劃分成若幹個小的區域,然後針對若幹個小區域進行數據處理。這可以理解為將數據按照某個欄位或者多個欄位進行分組。 使用GROUP ...
  • MySQL資料庫管理 資料庫-->數據表-->行(記錄):用來描述一個對象的信息 列(欄位):用來描述對象的一個屬性 常用的數據類型: int :整型 無符號[0,2^32-1],有符號[-2^31,2^31-1] float :單精度浮點 4位元組32位 double :雙精度浮點 8位元組64位 c ...
  • 在MySQL中,高級查詢是指使用更複雜的查詢語句和操作符來檢索和操作資料庫中的數據。高級查詢可以幫助您更精確地找到所需的信息,並提高查詢的效率和靈活性。 以下是高級查詢的一些常見應用場景和意義: 連接多個表:使用JOIN操作符將多個表連接起來,以便在一次查詢中獲取相關聯的數據。這對於在多個表之間建立 ...
  • 背景: 隨著項目體量越來越大,用戶群體越來越多,用戶的聲音也越來越明顯;關於應用發版之後用戶無感知,導致用戶用的是仍然還是老版本功能,除非用戶手動刷新,否則體驗不到最新的功能;這樣的體驗非常不好,於是我們團隊針對該問題給出了相應的解決方案來處理;技術棧:vue3+ts+vite+ant-design ...
  • 相比用戶停留時間短、用完即走的 Web 頁面,桌面 QQ 用戶在一次登錄後,可能會掛機一周以上,這段期間,如果沒有嚴格控制好 QQ 記憶體占用,那麼結果可能是用戶交互響應變慢、甚至 Crash。在系統監控工具里,高記憶體占用也會被直觀地反映出來,帶來不好的口碑。MAC QQ 灰度期間,也聽到了一些用戶關... ...
  • 介紹 ESLint 是一個根據方案識別並報告 ECMAScript/JavaScript 代碼問題的工具,其目的是使代碼風格更加一致並避免錯誤。在很多地方它都與 JSLint 和 JSHint 類似,除了: ESLint 使用 Espree 對 JavaScript 進行解析。 ESLint 在代碼 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...