使用Dapper.Contrib 開發.net core程式,相容多種資料庫

来源:https://www.cnblogs.com/wuhuacong/archive/2018/11/13/9952900.html
-Advertisement-
Play Games

關於Dapper的介紹,我想很多人都對它有一定的瞭解,這個類似一個輕型的ORM框架是目前應用非常火的一個東西,據說各方面的性能都不錯,而且可以支持多種資料庫,在開始介紹這個文章之前,我花了不少功夫來學習了Dapper 的相關使用。Dapper.Contrib是對Dapper的進一步封裝,使對象的基本... ...


關於Dapper的介紹,我想很多人都對它有一定的瞭解,這個類似一個輕型的ORM框架是目前應用非常火的一個東西,據說各方面的性能都不錯,而且可以支持多種資料庫,在開始介紹這個文章之前,我花了不少功夫來學習了Dapper 的相關使用。Dapper.Contrib是對Dapper的進一步封裝,使對象的基本增刪改查等操作進一步簡化,我做了一個案例使用Dapper.Contrib 開發.net core程式,測試它對多種資料庫的處理。

1、Dapper.Contrib的使用

前面介紹過,Dapper.Contrib是對Dapper的進一步封裝,使對象的基本增刪改查等操作進一步簡化。

它主要是通過特性映射的方式實現自定義類和資料庫之間的關係處理,如下是實體類的定義信息。

    [Table("T_Customer")]
    public class CustomerInfo
    {
        [ExplicitKey]//非自增長的用此標識
        public virtual string ID { get; set; }

        public virtual string Name { get; set; }

        public virtual int Age { get; set; }

        public virtual string Creator { get; set; }

        public virtual DateTime CreateTime { get; set; }

    }

Dapper.Contrib的所有實體配置選項

  • Table:指定實體對應地資料庫表名,如果類名和資料庫表名不同,需要設置(如案例所示)
  • Key:指定此列為自動增長主鍵
  • ExplicitKey:指定此列為非自動增長主鍵(例如guid,字元串列)
  • Computed:計算屬性,此列不作為更新
  • Write:指定列是否可寫

通過定義好實體類和資料庫表的映射關係,就可以通過強類型處理相關的介面了,如下所示。

T Get<T>(id);
IEnumerable<T> GetAll<T>();
int Insert<T>(T obj);
int Insert<T>(Enumerable<T> list);
bool Update<T>(T obj);
bool Update<T>(Enumerable<T> list);
bool Delete<T>(T obj);
bool Delete<T>(Enumerable<T> list);
bool DeleteAll<T>();

這樣通過映射指定表名或者欄位信息後,就可以知道類和表之間的關係,可以封裝對應的強類型處理介面了。

2、Dapper.Contrib 開發.net core程式

我們創建一個空白的.net core程式框架後,就在它的基礎上做一些Dapper的資料庫測試。

首先為了考慮多資料庫的處理,我們需要創建一個配置文件,並可以動態配置不同的資料庫,配置文件appSettings.json如下所示。

上面我配置了多種資料庫的連接字元串,並且通過動態指定節點名稱和資料庫類型,來實現對項目指向不同資料庫的訪問。

例如我們準備需要讓Dapper支持我們常見的資料庫類型,如下定義資料庫類型。

    /// <summary>
    /// 資料庫類型定義
    /// </summary>
    public enum DatabaseType
    {
        SqlServer,  //SQLServer資料庫
        MySql,      //Mysql資料庫
        Npgsql,     //PostgreSQL資料庫
        Oracle,     //Oracle資料庫
        Sqlite,     //SQLite資料庫
        DB2         //IBM DB2資料庫
    }

對於不同的資料庫信息,我們需要根據不同的配置連接字元串,並創建對應的資料庫連接對象供Dapper使用,如對於SQLServer的資料庫,那麼創建的是SqlConnection對象,對於Mysql,創建的是MySqlConnection連接對象,對於PostgreSQL對應的是NpgsqlConnection,以此類推。而Dapper則通過對連接對象的擴展實現了多種數據請求。

對於多資料庫的支持,我們需要統一解析配置內容appSetting.json的內容,並返回不同資料庫的連接對象,如下是連接工廠的統一處理方式,通過 CreateConnection() 返回配置的連接對象。

    /// <summary>
    /// 資料庫連接輔助類
    /// </summary>
    public class ConnectionFactory
    {
        /// <summary>
        /// 轉換資料庫類型
        /// </summary>
        /// <param name="databaseType">資料庫類型</param>
        /// <returns></returns>
        private static DatabaseType GetDataBaseType(string databaseType)
        {
            DatabaseType returnValue = DatabaseType.SqlServer;
            foreach (DatabaseType dbType in Enum.GetValues(typeof(DatabaseType)))
            {
                if (dbType.ToString().Equals(databaseType, StringComparison.OrdinalIgnoreCase))
                {
                    returnValue = dbType;
                    break;
                }
            }
            return returnValue;
        }

        /// <summary>
        /// 獲取資料庫連接
        /// </summary>
        /// <returns></returns>
        public static IDbConnection CreateConnection()
        {
            IDbConnection connection = null;

            //獲取配置進行轉換
            var type = AppConfig.GetConfig("ComponentDbType");
            var dbType = GetDataBaseType(type);

            //DefaultDatabase 根據這個配置項獲取對應連接字元串
            var database = AppConfig.GetConfig("DefaultDatabase");
            if (string.IsNullOrEmpty(database))
            {
                database = "sqlserver";//預設配置
            }
            var strConn = AppConfig.Configuration.GetConnectionString(database);

            switch (dbType)
            {
                case DatabaseType.SqlServer:
                    connection = new System.Data.SqlClient.SqlConnection(strConn);
                    break;
                case DatabaseType.MySql:
                    connection = new MySql.Data.MySqlClient.MySqlConnection(strConn);
                    break;
                case DatabaseType.Npgsql:
                    connection = new Npgsql.NpgsqlConnection(strConn);
                    break;
                case DatabaseType.Sqlite:
                    connection = new SQLiteConnection(strConn);
                    break;
                case DatabaseType.Oracle:
                    connection = new Oracle.ManagedDataAccess.Client.OracleConnection(strConn);
                    //connection = new System.Data.OracleClient.OracleConnection(strConn);
                    break;
                case DatabaseType.DB2:
                    //connection = new System.Data.OleDb.OleDbConnection(strConn);
                    break;
            }

            return connection;
        }
    }

有了資料庫對象工廠,我們的配置就可以動態化了。

下麵我們來看看,獲得這些連接對象後,如何通過Dapper.Contrib來獲取對應的對象了,下麵的類是常規的對資料庫信息的處理,包括常規的增刪改查等基礎介面。

    /// <summary>
    /// 常規的數據訪問層
    /// </summary>
    public class Customer
    {
        public IDbConnection Connection
        {
            get
            {
                var connection = ConnectionFactory.CreateConnection();
                connection.Open();
                return connection;
            }
        }

        public IEnumerable<CustomerInfo> GetAll()
        {
            using (IDbConnection dbConnection = Connection)
            {
                return dbConnection.GetAll<CustomerInfo>();
                //return dbConnection.Query<CustomerInfo>("SELECT * FROM T_Customer");
            }
        }


        public CustomerInfo FindByID(string id)
        {
            using (IDbConnection dbConnection = Connection)
            {
                return dbConnection.Get<CustomerInfo>(id);
                //string query = "SELECT * FROM T_Customer WHERE ID = @Id";
                //return dbConnection.Query<CustomerInfo>(query, new { Id = id }).FirstOrDefault();
            }
        }

        public bool Insert(CustomerInfo info)
        {
            bool result = false;
            using (IDbConnection dbConnection = Connection)
            {
                result = dbConnection.Insert(info) > 0;
                result = true;
                //string query = "INSERT INTO T_Customer (ID, Name, Age, Creator, CreateTime)"
                //                + " VALUES(@ID, @Name, @Age, @Creator, @CreateTime)";
                //result = dbConnection.Execute(query, info) > 0;
            }
            return result;
        }
        
        public bool Update(CustomerInfo prod)
        {
            bool result = false;
            using (IDbConnection dbConnection = Connection)
            {
                result = dbConnection.Update(prod);
                //string query = "UPDATE T_Customer SET Name = @Name,"
                //               + " Age = @Age, Creator= @Creator, CreateTime=@CreateTime"
                //               + " WHERE ID = @ID";
                //result = dbConnection.Execute(query, prod) > 0;
            }
            return result;
        }
        public bool Delete(string id)
        {
            bool result = false;
            using (IDbConnection dbConnection = Connection)
            {
                result = dbConnection.Delete(new CustomerInfo { ID = id });
                //string query = "DELETE FROM T_Customer WHERE ID = @Id";
                //result = dbConnection.Execute(query, new { ID = id }) > 0;
            }
            return result;
        }
        public bool DeleteAll()
        {
            bool result = false;
            using (IDbConnection dbConnection = Connection)
            {
                result = dbConnection.DeleteAll<CustomerInfo>();
                //string query = "DELETE FROM T_Customer WHERE ID = @Id";
                //result = dbConnection.Execute(query, new { ID = id }) > 0;
            }
            return result;
        }
    }

其中的備註部分的代碼是等同於上面的執行代碼的,是Dapper 的SQL版本的一種處理方式。

我們看到,對於Customer表來說,使用對象的介面處理,我們已經隔離了很多硬編碼的SQL處理,不過我們還可以對它進行進一步的優化處理。

我們定義一個通用的BaseDAL來剝離常規的增刪改查處理,並且把同步和非同步的操作分來兩個文件來管理,同步處理的基類如下代碼所示。

    /// <summary>
    /// 資料庫訪問基類
    /// </summary>
    /// <typeparam name="T">實體類類型</typeparam>
    public partial class BaseDAL<T> where T : class
    {
        /// <summary>
        /// 對象的表名
        /// </summary>
        public string TableName { get; set; }

        /// <summary>
        /// 主鍵屬性對象
        /// </summary>
        public PropertyInfo PrimaryKey { get; set; }

        public BaseDAL()
        {
            this.TableName = EntityHelper.GetTableName(typeof(T));
            this.PrimaryKey = EntityHelper.GetSingleKey<T>();
        }

        /// <summary>
        /// 資料庫連接
        /// </summary>
        protected IDbConnection Connection
        {
            get
            {
                var connection = ConnectionFactory.CreateConnection();
                connection.Open();
                return connection;
            }
        }

        /// <summary>
        /// 返回資料庫所有的對象集合
        /// </summary>
        /// <returns></returns>
        public IEnumerable<T> GetAll()
        {
            using (IDbConnection dbConnection = Connection)
            {
                return dbConnection.GetAll<T>();
            }
        }

        /// <summary>
        /// 查詢資料庫,返回指定ID的對象
        /// </summary>
        /// <param name="id">主鍵的值</param>
        /// <returns></returns>
        public T FindByID(object id)
        {
            using (IDbConnection dbConnection = Connection)
            {
                return dbConnection.Get<T>(id);
            }
        }

        /// <summary>
        /// 插入指定對象到資料庫中
        /// </summary>
        /// <param name="info">指定的對象</param>
        /// <returns></returns>
        public bool Insert(T info)
        {
            bool result = false;
            using (IDbConnection dbConnection = Connection)
            {
                dbConnection.Insert(info);
                result = true;
            }
            return result;
        }
        /// <summary>
        /// 插入指定對象集合到資料庫中
        /// </summary>
        /// <param name="list">指定的對象集合</param>
        /// <returns></returns>
        public bool Insert(IEnumerable<T> list)
        {
            bool result = false;
            using (IDbConnection dbConnection = Connection)
            {
                result = dbConnection.Insert(list) > 0;
            }
            return result;
        }

        /// <summary>
        /// 更新對象屬性到資料庫中
        /// </summary>
        /// <param name="info">指定的對象</param>
        /// <returns></returns>
        public bool Update(T info)
        {
            bool result = false;
            using (IDbConnection dbConnection = Connection)
            {
                result = dbConnection.Update(info);
            }
            return result;
        }
        /// <summary>
        /// 更新指定對象集合到資料庫中
        /// </summary>
        /// <param name="list">指定的對象集合</param>
        /// <returns></returns>
        public bool Update(IEnumerable<T> list)
        {
            bool result = false;
            using (IDbConnection dbConnection = Connection)
            {
                result = dbConnection.Update(list);
            }
            return result;
        }
        /// <summary>
        /// 從資料庫中刪除指定對象
        /// </summary>
        /// <param name="info">指定的對象</param>
        /// <returns></returns>
        public bool Delete(T info)
        {
            bool result = false;
            using (IDbConnection dbConnection = Connection)
            {
                result = dbConnection.Delete(info);
            }
            return result;
        }
        /// <summary>
        /// 從資料庫中刪除指定對象集合
        /// </summary>
        /// <param name="list">指定的對象集合</param>
        /// <returns></returns>
        public bool Delete(IEnumerable<T> list)
        {
            bool result = false;
            using (IDbConnection dbConnection = Connection)
            {
                result = dbConnection.Delete(list);
            }
            return result;
        }
        /// <summary>
        /// 根據指定對象的ID,從資料庫中刪除指定對象
        /// </summary>
        /// <param name="id">對象的ID</param>
        /// <returns></returns>
        public bool Delete(object id)
        {
            bool result = false;
            using (IDbConnection dbConnection = Connection)
            {
                string query = string.Format("DELETE FROM {0} WHERE {1} = @id", TableName, PrimaryKey.Name);
                var parameters = new DynamicParameters();
                parameters.Add("@id", id);

                result = dbConnection.Execute(query, parameters) > 0;
            }
            return result;
        }
        /// <summary>
        /// 從資料庫中刪除所有對象
        /// </summary>
        /// <returns></returns>
        public bool DeleteAll()
        {
            bool result = false;
            using (IDbConnection dbConnection = Connection)
            {
                result = dbConnection.DeleteAll<T>();
            }
            return result;
        }

    }

非同步類的代碼如下所示。

    /// <summary>
    /// 資料庫訪問基類
    /// </summary>
    /// <typeparam name="T">實體類類型</typeparam>
    public partial class BaseDAL<T> where T : class
    {
        /// <summary>
        /// 返回資料庫所有的對象集合
        /// </summary>
        /// <returns></returns>
        public virtual async Task<IEnumerable<T>> GetAllAsync()
        {
            using (IDbConnection dbConnection = Connection)
            {
                return await dbConnection.GetAllAsync<T>();
            }
        }

        /// <summary>
        /// 查詢資料庫,返回指定ID的對象
        /// </summary>
        /// <param name="id">主鍵的值</param>
        /// <returns></returns>
        public virtual async Task<T> FindByIDAsync(object id)
        {
            using (IDbConnection dbConnection = Connection)
            {
                return await dbConnection.GetAsync<T>(id);
            }
        }
        /// <summary>
        /// 插入指定對象到資料庫中
        /// </summary>
        /// <param name="info">指定的對象</param>
        /// <returns></returns>
        public virtual async Task<bool> InsertAsync(T info)
        {
            bool result = false;
            using (IDbConnection dbConnection = Connection)
            {
                await dbConnection.InsertAsync(info);
                result = true;
            }
            return await Task<bool>.FromResult(result);
        }

        /// <summary>
        /// 插入指定對象集合到資料庫中
        /// </summary>
        /// <param name="list">指定的對象集合</param>
        /// <returns></returns>
        public virtual async Task<bool> InsertAsync(IEnumerable<T> list)
        {
            using (IDbConnection dbConnection = Connection)
            {
                return await dbConnection.InsertAsync(list) > 0;
            }
        }
        /// <summary>
        /// 更新對象屬性到資料庫中
        /// </summary>
        /// <param name="info">指定的對象</param>
        /// <returns></returns>
        public virtual async Task<bool> UpdateAsync(T info)
        {
            using (IDbConnection dbConnection = Connection)
            {
                return await dbConnection.UpdateAsync(info);
            }
        }

        /// <summary>
        /// 更新指定對象集合到資料庫中
        /// </summary>
        /// <param name="list">指定的對象集合</param>
        /// <returns></returns>
        public virtual async Task<bool> UpdateAsync(IEnumerable<T> list)
        {
            using (IDbConnection dbConnection = Connection)
            {
                return await dbConnection.UpdateAsync(list);
            }
        }

        /// <summary>
        /// 從資料庫中刪除指定對象
        /// </summary>
        /// <param name="info">指定的對象</param>
        /// <returns></returns>
        public virtual async Task<bool> DeleteAsync(T info)
        {
            using (IDbConnection dbConnection = Connection)
            {
                return await dbConnection.DeleteAsync(info);
            }
        }

        /// <summary>
        /// 從資料庫中刪除指定對象集合
        /// </summary>
        /// <param name="list">指定的對象集合</param>
        /// <returns></returns>
        public virtual async Task<bool> DeleteAsync(IEnumerable<T> list)
        {
            using (IDbConnection dbConnection = Connection)
            {
                return await dbConnection.DeleteAsync(list);
            }
        }

        /// <summary>
        /// 根據指定對象的ID,從資料庫中刪除指定對象
        /// </summary>
        /// <param name="id">對象的ID</param>
        /// <returns></returns>
        public virtual async Task<bool> DeleteAsync(object id)
        {
            using (IDbConnection dbConnection = Connection)
            {
                string query = string.Format("DELETE FROM {0} WHERE {1} = @id", TableName, PrimaryKey.Name);
                var parameters = new DynamicParameters();
                parameters.Add("@id", id);

                return await dbConnection.ExecuteAsync(query, parameters) > 0;
            }
        }

        /// <summary>
        /// 從資料庫中刪除所有對象
        /// </summary>
        /// <returns></returns>
        public virtual async Task<bool> DeleteAllAsync()
        {
            using (IDbConnection dbConnection = Connection)
            {
                return await dbConnection.DeleteAllAsync<T>();
            }
        }
    }

這樣,我們如果需要增加一個如客戶信息表的管理類,就很簡單的繼承基類就可以了,代碼很少,但是增刪改查介面一個也少不了。

    /// <summary>
    /// 繼承基類對象管理
    /// </summary>
    public class CustomerDAL :BaseDAL<CustomerInfo>
    {
    }

為了測試一下數據訪問層的處理介面,我創建了一個.net core的控制台程式進行測試,如下項目視圖所示。

主要目的是確認數據處理的效果。

我們在Program.cs類裡面增加相關的測試代碼,為了簡便和處理效果沒有用UnitTest處理。

            //創建管理對象,並測試介面
            var customer = new CustomerDAL();
            var list = customer.GetAll();
            foreach (var item in list)
            {
                Console.WriteLine(item.ToJson());
                var info = customer.FindByID(item.ID);
                Console.WriteLine(info.ToJson());
                Console.WriteLine();
            }

            //插入記錄
            var insertInfo = new CustomerInfo() { Name = "test", Age = 30, Creator = "test" };
            var insertList = new List<CustomerInfo>() { insertInfo };
            var flag = customer.Insert(insertList);
            Console.WriteLine("插入操作" + (flag ? "成功" : "失敗"));

            Console.WriteLine("插入的新內容");
            insertInfo = customer.FindByID(insertInfo.ID);
            Console.WriteLine(insertInfo.ToJson());

            Console.WriteLine("更新內容");
            insertInfo.Name = "Test" + DateTime.Now.ToShortDateString();
            flag = customer.Update(insertInfo);
            Console.WriteLine("更新操作" + (flag ? "成功" : "失敗"));

            Console.WriteLine("更新的新內容");
            insertInfo = customer.FindByID(insertInfo.ID);
            Console.WriteLine(insertInfo.ToJson());

            Console.WriteLine("刪除內容");
            flag = customer.Delete(insertInfo.ID);
            Console.WriteLine("刪除操作" + (flag ? "成功" : "失敗"));


            Console.WriteLine("所有內容");
            list = customer.GetAll();
            foreach (var item in list)
            {
                Console.WriteLine(item.ToJson());
                Console.WriteLine();
            }

            Console.ReadLine();

測試Mysql、SQLite資料庫同樣沒有問題

Mysql配置信息如下

處理的Mysql記錄信息如下。 

SQLite配置信息如下

處理SQLite數據信息如下

 

 而在處理PostgreSQL的信息(配置節點npgsql裡面)的時候,查詢的主鍵好像和大小寫有關係,導致插入記錄出錯。

而Oracle我採用的是Oracle.ManagedDataAccess.Core進行訪問,由於我本地Oracle資料庫偵聽處理有點問題,因此沒有測試成功,暫不予置評。

而對於資料庫的支持問題,導致我重新審核一下是否採用Dapper.Contrib還是其他Dapper方式來構建資料庫訪問基類的問題,我需要相容多種資料庫的信息,並且能夠儘可能的封裝常規的增刪改查等操作,其中目前的基類還沒有加入更加複雜的查詢操作,分頁操作等功能,在解決這些困惑問題,才會繼續考慮把底層支持的介面全部完善。 


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

-Advertisement-
Play Games
更多相關文章
  • 前言 在前文中我們瞭解了幾種常見的數據結構,這些數據結構有著各自的應用場景,並且被廣泛的應用於編程語言中,其中,Java中的集合類就是基於這些數據結構為基礎。 Java的集合類是一些非常實用的工具類,主要用於存儲和裝載數據 (包括對象),因此,Java的集合類也被成為容器。在Java中,所有的集合類 ...
  • python模塊導入細節 官方手冊:https://docs.python.org/3/tutorial/modules.html 可執行文件和模塊 python源代碼文件按照功能可以分為兩種類型: 1. 用於執行的可執行程式文件 2. 不用與執行,僅用於被其它python源碼文件導入的模塊文件 例 ...
  • 在Python中有以下幾種標準的內置數據類型: 1.None: The Null object--空對象 None是Python的特殊類型,表示一個空對象,值為None2.Numerics(數值): int-整數, long-長整數, float-浮點數, complex-複數, and bool- ...
  • 數組作為一種組合形式的數據類型,必然要求提供一些處理數組的簡便辦法,包括數組比較、數組複製、數組排序等等。為此Java專門設計了Arrays工具,該工具包含了幾個常用方法,方便程式員對數組進行加工操作。Arrays工具的方法說明如下: 下麵分別對以上的四個數組處理方法進行介紹: 1、Arrays.e ...
  • Spring Security Oauth2 示例,基於SpringBoot搭建授權服務和資源服務 ...
  • Java的serialization提供了一種持久化對象實例的機制。當持久化對象時,可能有一個特殊的對象數據成員,我們不想用serialization機制來保存它。 為了在一個特定對象的一個域上關閉serialization,可以在這個域前加上關鍵字transient。 當一個對象被序列化的時候,t ...
  • 首先請using這個類庫。 這個類庫老牛逼了,首先是包含了一個IdentityUser類。我們可以自己寫一個User類繼承IdentityUser,這樣一來,很多屬性我就不用手動建了。如你所見,我的User沒有Emai這個欄位,但是IdentityUser有這個欄位。 如何把用戶發來的數據變成一個U ...
  • /*普通行列轉換 假設有張學生成績表(tb)如下:Name Subject Result張三 語文 74張三 數學 83張三 物理 93李四 語文 74李四 數學 84李四 物理 94*/ /*想變成 姓名 語文 數學 物理 李四 74 84 94張三 74 83 93*/createtable t ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...