[C#] 折騰海康威視的人體測溫 模組

来源:https://www.cnblogs.com/catzhou/archive/2020/05/21/12928921.html
-Advertisement-
Play Games

單位的項目需要測溫,同事買了個海康威視的人體測溫機芯,型號位:TB 4117 3/S,給了一份pdf的說明書。 按說明書把設備連接設置好,從官網下載了sdk,我的個乖乖,壓縮包就有70多M,把他家的所有東西都給了我,有各種Demo,就是沒有測溫的,暈死,差點想打退堂鼓不玩了。 最後,最後得到如下成果 ...


單位的項目需要測溫,同事買了個海康威視的人體測溫機芯,型號位:TB-4117-3/S,給了一份pdf的說明書。

按說明書把設備連接設置好,從官網下載了sdk,我的個乖乖,壓縮包就有70多M,把他家的所有東西都給了我,有各種Demo,就是沒有測溫的,暈死,差點想打退堂鼓不玩了。

最後,最後得到如下成果:

一、所需的DLL

  1. HCCore.dll
  2. HCCoreDevCfg.dll 這玩意必須在運行目錄
  3. HCGeneralCfgMgr.dll 這鬼也必須在運行目錄
  4. HCNetSDK.dll
  5. libeay32.dll
  6. ssleay32.dll
    以上6個文件必須的,幹啥用俺也不知道,反正缺一不可。

二、委托、dll封裝、數據結果


        public delegate void LOGINRESULTCALLBACK(int lUserID, int dwResult, IntPtr lpDeviceInfo, IntPtr pUser);
        public delegate void UpdateTextStatusCallback(string strLogStatus, IntPtr lpDeviceInfo);
        public delegate void RemoteConfigCallback(uint dwType, IntPtr lpBuffer, uint dwBufLen, IntPtr pUserData);


        #region DllWrapper
        const string DllFileName = "HCNetSDK.dll";
        [DllImport(DllFileName)]
        public static extern bool NET_DVR_Init();
        [DllImport(DllFileName)]
        public static extern int NET_DVR_Login_V40(ref NET_DVR_USER_LOGIN_INFO pLoginInfo, ref NET_DVR_DEVICEINFO_V40 lpDeviceInfo);
        [DllImportAttribute(DllFileName)]
        public static extern int NET_DVR_StartRemoteConfig(int lUserID, int dwCommand, IntPtr lpInBuffer, Int32 dwInBufferLen, RemoteConfigCallback cbStateCallback, IntPtr pUserData);
        [DllImport(DllFileName)]
        public static extern bool NET_DVR_Logout(int iUserID);
        [DllImport(DllFileName)]
        public static extern bool NET_DVR_Cleanup();
        [DllImport(DllFileName)]
        public static extern uint NET_DVR_GetLastError();
        #endregion

        #region Data

        [StructLayout(LayoutKind.Sequential)]
        public struct NET_DVR_THERMOMETRY_UPLOAD
        {
            public uint dwSize;
            public uint dwRelativeTime;     // 相對時標
            public uint dwAbsTime;            // 絕對時標
            [MarshalAsAttribute(UnmanagedType.ByValArray, SizeConst = 32, ArraySubType = UnmanagedType.I1)]
            public byte[] szRuleName;//規則名稱
            public byte byRuleID;//規則ID號
            public byte byRuleCalibType;//規則標定類型 0-點,1-框,2-線
            public ushort wPresetNo; //預置點號
            [MarshalAs(UnmanagedType.Struct)]
            public NET_DVR_POINT_THERM_CFG struPointThermCfg;
            [MarshalAs(UnmanagedType.Struct)]

            public NET_DVR_LINEPOLYGON_THERM_CFG struLinePolygonThermCfg;
            public byte byThermometryUnit;//測溫單位: 0-攝氏度(℃),1-華氏度(℉),2-開爾文(K)
            public byte byDataType;//數據狀態類型:0-檢測中,1-開始,2-結束
            public byte byRes1;
            /*
            bit0-中心點測溫:0-不支持,1-支持;
            bit1-最高點測溫:0-不支持,1-支持;
            bit2-最低點測溫:0-不支持,1-支持;
            */
            public byte bySpecialPointThermType;// 是否支持特殊點測溫
            public float fCenterPointTemperature;//中心點溫度,精確到小數點後一位(-40-1500),(浮點數+100)*10 (由bySpecialPointThermType判斷是否支持中心點)
            public float fHighestPointTemperature;//最高點溫度,精確到小數點後一位(-40-1500),(浮點數+100)*10(由bySpecialPointThermType判斷是否支持最高點)
            public float fLowestPointTemperature;//最低點溫度,精確到小數點後一位(-40-1500),(浮點數+100)*10(由bySpecialPointThermType判斷是否支持最低點)
            [MarshalAs(UnmanagedType.Struct)]
            public NET_VCA_POINT struHighestPoint;//線、框測溫最高溫度位置坐標(當規則標定類型為線、框的時候生效)
            [MarshalAs(UnmanagedType.Struct)]
            public NET_VCA_POINT struLowestPoint;//線、框測溫最低溫度位置坐標(當規則標定類型為線、框的時候生效)
            public byte byIsFreezedata;//是否數據凍結 0-否 1-是
            [MarshalAsAttribute(UnmanagedType.ByValArray, SizeConst = 3, ArraySubType = UnmanagedType.I1)]
            public byte[] byRes2;
            public uint dwChan; //通道號,查詢條件中通道號為0xffffffff時該欄位生效
            [MarshalAsAttribute(UnmanagedType.ByValArray, SizeConst = 88, ArraySubType = UnmanagedType.I1)]
            public byte[] byRes;
        }

        [StructLayout(LayoutKind.Sequential)]
        public struct NET_DVR_POINT_THERM_CFG
        {
            public float fTemperature;
            public NET_VCA_POINT struPoint;
            [MarshalAsAttribute(UnmanagedType.ByValArray, SizeConst = 120, ArraySubType = UnmanagedType.I1)]
            public byte[] byRes;

        }

        [StructLayout(LayoutKind.Sequential)]
        public struct NET_DVR_LINEPOLYGON_THERM_CFG
        {
            public float fMaxTemperature;
            public float fMinTemperature;
            public float fAverageTemperature;
            public float fTemperatureDiff;
            public NET_VCA_POLYGON struRegion;
            [MarshalAsAttribute(UnmanagedType.ByValArray, SizeConst = 32, ArraySubType = UnmanagedType.I1)]
            public byte[] byRes;
        }
        public const int VCA_MAX_POLYGON_POINT_NUM = 10;//檢測區域最多支持10個點的多邊形

        [StructLayoutAttribute(LayoutKind.Sequential)]
        public struct NET_VCA_POLYGON
        {
            /// DWORD->unsigned int
            public uint dwPointNum;
            [MarshalAsAttribute(UnmanagedType.ByValArray, SizeConst = VCA_MAX_POLYGON_POINT_NUM, ArraySubType = UnmanagedType.Struct)]
            public NET_VCA_POINT[] struPos;
        }
        [StructLayoutAttribute(LayoutKind.Sequential)]
        public struct NET_VCA_POINT
        {
            public float fX;// X軸坐標, 0.001~1
            public float fY;//Y軸坐標, 0.001~1
        }



        [StructLayout(LayoutKind.Sequential)]
        public struct NET_DVR_REALTIME_THERMOMETRY_COND
        {
            public uint dwSize;
            public uint dwChan;//通道號,從1開始,0xffffffff代表獲取全部通道
            public byte byRuleID; //規則ID 0-代表獲取全部規則,具體規則ID從1開始 
            /*
            1-定時模式:設備每隔一秒上傳各個規則測溫數據的最高溫、最低溫和平均溫度值、溫差
            2-溫差模式:若上一秒與下一秒的最高溫或者最低溫或者平均溫或者溫差值的溫差大於等於2攝氏度,則上傳最高溫、最低溫和平均溫度值。若大於等於一個小時溫差值均小於2攝氏度,則上傳最高溫、最低溫、平均溫和溫差值
            */
            public byte byMode; //長連接模式, 0-保留(為相容老設備),1-定時模式,2-溫差模式
            public ushort wInterval; //上傳間隔,僅溫差模式支持,1~3600S,填0則預設3600S上傳一次
            [MarshalAsAttribute(UnmanagedType.ByValArray, SizeConst = 60, ArraySubType = UnmanagedType.I1)]
            public byte[] byRes; //保留
        }
        public const int NET_DVR_DEV_ADDRESS_MAX_LEN = 129;
        public const int NET_DVR_LOGIN_USERNAME_MAX_LEN = 64;
        public const int NET_DVR_LOGIN_PASSWD_MAX_LEN = 64;

        [StructLayout(LayoutKind.Sequential)]
        public struct NET_DVR_USER_LOGIN_INFO
        {
            [MarshalAsAttribute(UnmanagedType.ByValArray, SizeConst = NET_DVR_DEV_ADDRESS_MAX_LEN, ArraySubType = UnmanagedType.I1)]
            public byte[] sDeviceAddress;
            public byte byUseTransport;
            public ushort wPort;
            [MarshalAsAttribute(UnmanagedType.ByValArray, SizeConst = NET_DVR_LOGIN_USERNAME_MAX_LEN, ArraySubType = UnmanagedType.I1)]
            public byte[] sUserName;
            [MarshalAsAttribute(UnmanagedType.ByValArray, SizeConst = NET_DVR_LOGIN_PASSWD_MAX_LEN, ArraySubType = UnmanagedType.I1)]
            public byte[] sPassword;
            public LOGINRESULTCALLBACK cbLoginResult;
            public IntPtr pUser;
            public bool bUseAsynLogin;
            public byte byProxyType; //0:不使用代理,1:使用標準代理,2:使用EHome代理
            public byte byUseUTCTime;    //0-不進行轉換,預設,1-介面上輸入輸出全部使用UTC時間,SDK完成UTC時間與設備時區的轉換,2-介面上輸入輸出全部使用平臺本地時間,SDK完成平臺本地時間與設備時區的轉換
            public byte byLoginMode; //0-Private, 1-ISAPI, 2-自適應
            public byte byHttps;    //0-不適用tls,1-使用tls 2-自適應
            public int iProxyID;    //代理伺服器序號,添加代理伺服器信息時,相對應的伺服器數組下表值
            public byte byVerifyMode;  //認證方式,0-不認證,1-雙向認證,2-單向認證;認證僅在使用TLS的時候生效;    
            [MarshalAsAttribute(UnmanagedType.ByValArray, SizeConst = 119, ArraySubType = UnmanagedType.I1)]
            public byte[] byRes3;
        }

        [StructLayout(LayoutKind.Sequential)]
        public struct NET_DVR_DEVICEINFO_V40
        {
            public NET_DVR_DEVICEINFO_V30 struDeviceV30;
            public byte bySupportLock;        //設備支持鎖定功能,該欄位由SDK根據設備返回值來賦值的。bySupportLock為1時,dwSurplusLockTime和byRetryLoginTime有效
            public byte byRetryLoginTime;	    //剩餘可嘗試登陸的次數,用戶名,密碼錯誤時,此參數有效
            public byte byPasswordLevel;      //admin密碼安全等級0-無效,1-預設密碼,2-有效密碼,3-風險較高的密碼。當用戶的密碼為出廠預設密碼(12345)或者風險較高的密碼時,上層客戶端需要提示用戶更改密碼。      
            public byte byProxyType;//代理類型,0-不使用代理, 1-使用socks5代理, 2-使用EHome代理
            public uint dwSurplusLockTime;	//剩餘時間,單位秒,用戶鎖定時,此參數有效
            public byte byCharEncodeType;     //字元編碼類型
            public byte bySupportDev5;//支持v50版本的設備參數獲取,設備名稱和設備類型名稱長度擴展為64位元組
            public byte bySupport;  //能力集擴展,位與結果:0- 不支持,1- 支持
            // bySupport & 0x1:  保留
            // bySupport & 0x2:  0-不支持變化上報 1-支持變化上報
            public byte byLoginMode; //登錄模式 0-Private登錄 1-ISAPI登錄
            public int dwOEMCode;
            public int iResidualValidity;   //該用戶密碼剩餘有效天數,單位:天,返回負值,表示密碼已經超期使用,例如“-3表示密碼已經超期使用3天”
            public byte byResidualValidity; // iResidualValidity欄位是否有效,0-無效,1-有效
            [MarshalAsAttribute(UnmanagedType.ByValArray, SizeConst = 243, ArraySubType = UnmanagedType.I1)]
            public byte[] byRes2;
        }

        public const int SERIALNO_LEN = 48;//序列號長度

        //NET_DVR_Login_V30()參數結構
        [StructLayoutAttribute(LayoutKind.Sequential)]
        public struct NET_DVR_DEVICEINFO_V30
        {
            [MarshalAsAttribute(UnmanagedType.ByValArray, SizeConst = SERIALNO_LEN, ArraySubType = UnmanagedType.I1)]
            public byte[] sSerialNumber;  //序列號
            public byte byAlarmInPortNum;		        //報警輸入個數
            public byte byAlarmOutPortNum;		        //報警輸出個數
            public byte byDiskNum;				    //硬碟個數
            public byte byDVRType;				    //設備類型, 1:DVR 2:ATM DVR 3:DVS ......
            public byte byChanNum;				    //模擬通道個數
            public byte byStartChan;			        //起始通道號,例如DVS-1,DVR - 1
            public byte byAudioChanNum;                //語音通道數
            public byte byIPChanNum;					//最大數字通道個數,低位  
            public byte byZeroChanNum;			//零通道編碼個數 //2010-01-16
            public byte byMainProto;			//主碼流傳輸協議類型 0-private, 1-rtsp,2-同時支持private和rtsp
            public byte bySubProto;				//子碼流傳輸協議類型0-private, 1-rtsp,2-同時支持private和rtsp
            public byte bySupport;        //能力,位與結果為0表示不支持,1表示支持,
                                          //bySupport & 0x1, 表示是否支持智能搜索
                                          //bySupport & 0x2, 表示是否支持備份
                                          //bySupport & 0x4, 表示是否支持壓縮參數能力獲取
                                          //bySupport & 0x8, 表示是否支持多網卡
                                          //bySupport & 0x10, 表示支持遠程SADP
                                          //bySupport & 0x20, 表示支持Raid卡功能
                                          //bySupport & 0x40, 表示支持IPSAN 目錄查找
                                          //bySupport & 0x80, 表示支持rtp over rtsp
            public byte bySupport1;        // 能力集擴充,位與結果為0表示不支持,1表示支持
                                           //bySupport1 & 0x1, 表示是否支持snmp v30
                                           //bySupport1 & 0x2, 支持區分回放和下載
                                           //bySupport1 & 0x4, 是否支持佈防優先順序	
                                           //bySupport1 & 0x8, 智能設備是否支持佈防時間段擴展
                                           //bySupport1 & 0x10, 表示是否支持多磁碟數(超過33個)
                                           //bySupport1 & 0x20, 表示是否支持rtsp over http	
                                           //bySupport1 & 0x80, 表示是否支持車牌新報警信息2012-9-28, 且還表示是否支持NET_DVR_IPPARACFG_V40結構體
            public byte bySupport2; /*能力,位與結果為0表示不支持,非0表示支持							
							bySupport2 & 0x1, 表示解碼器是否支持通過URL取流解碼
							bySupport2 & 0x2,  表示支持FTPV40
							bySupport2 & 0x4,  表示支持ANR
							bySupport2 & 0x8,  表示支持CCD的通道參數配置
							bySupport2 & 0x10,  表示支持佈防報警回傳信息(僅支持抓拍機報警 新老報警結構)
							bySupport2 & 0x20,  表示是否支持單獨獲取設備狀態子項
							bySupport2 & 0x40,  表示是否是碼流加密設備*/
            public ushort wDevType;              //設備型號
            public byte bySupport3; //能力集擴展,位與結果為0表示不支持,1表示支持
                                    //bySupport3 & 0x1, 表示是否多碼流
                                    // bySupport3 & 0x4 表示支持按組配置, 具體包含 通道圖像參數、報警輸入參數、IP報警輸入、輸出接入參數、
                                    // 用戶參數、設備工作狀態、JPEG抓圖、定時和時間抓圖、硬碟盤組管理 
                                    //bySupport3 & 0x8為1 表示支持使用TCP預覽、UDP預覽、多播預覽中的"延時預覽"欄位來請求延時預覽(後續都將使用這種方式請求延時預覽)。而當bySupport3 & 0x8為0時,將使用 "私有延時預覽"協議。
                                    //bySupport3 & 0x10 表示支持"獲取報警主機主要狀態(V40)"。
                                    //bySupport3 & 0x20 表示是否支持通過DDNS功能變數名稱解析取流

            public byte byMultiStreamProto;//是否支持多碼流,按位表示,0-不支持,1-支持,bit1-碼流3,bit2-碼流4,bit7-主碼流,bit-8子碼流
            public byte byStartDChan;		//起始數字通道號,0表示無效
            public byte byStartDTalkChan;	//起始數字對講通道號,區別於模擬對講通道號,0表示無效
            public byte byHighDChanNum;		//數字通道個數,高位
            public byte bySupport4;
            public byte byLanguageType;// 支持語種能力,按位表示,每一位0-不支持,1-支持  
                                       //  byLanguageType 等於0 表示 老設備
                                       //  byLanguageType & 0x1表示支持中文
                                       //  byLanguageType & 0x2表示支持英文
            [MarshalAsAttribute(UnmanagedType.ByValArray, SizeConst = 9, ArraySubType = UnmanagedType.I1)]
            public byte[] byRes2;		//保留
        }

        #endregion

三、測溫

 const string IPAddress = "192.168.1.64";
        const string UserName = "admin";
        const string Password = "13245678";
        const ushort PortNo = 8000;
        static void Main(string[] args)
        {
            //初始化
            if (!NET_DVR_Init())
            {
                Console.WriteLine("NET_DVR_Init error!");
                return;
            }
            //登陸
            var struLogInfo = new NET_DVR_USER_LOGIN_INFO();

            //設備IP地址或者功能變數名稱
            byte[] byIP = System.Text.Encoding.Default.GetBytes(IPAddress);
            struLogInfo.sDeviceAddress = new byte[129];
            byIP.CopyTo(struLogInfo.sDeviceAddress, 0);

            //設備用戶名
            byte[] byUserName = System.Text.Encoding.Default.GetBytes(UserName);
            struLogInfo.sUserName = new byte[64];
            byUserName.CopyTo(struLogInfo.sUserName, 0);

            //設備密碼
            byte[] byPassword = System.Text.Encoding.Default.GetBytes(Password);
            struLogInfo.sPassword = new byte[64];
            byPassword.CopyTo(struLogInfo.sPassword, 0);

            struLogInfo.wPort = PortNo;//設備服務埠號

            struLogInfo.bUseAsynLogin = false; //是否非同步登錄:0- 否,1- 是 

            var DeviceInfo = new NET_DVR_DEVICEINFO_V40();

            //登錄設備 Login the device
            var m_lUserID = NET_DVR_Login_V40(ref struLogInfo, ref DeviceInfo);
            if (m_lUserID < 0)
            {
                Console.WriteLine("登陸失敗,錯誤代碼:" + NET_DVR_GetLastError()); //登錄失敗,輸出錯誤號
                return;
            }


            //配置測溫
            var size = Marshal.SizeOf(typeof(NET_DVR_REALTIME_THERMOMETRY_COND));

            NET_DVR_REALTIME_THERMOMETRY_COND struThermCond = new NET_DVR_REALTIME_THERMOMETRY_COND();
            struThermCond.dwSize = (uint)size;
            struThermCond.byRuleID = 0;       //規則ID,0代表獲取全部規則,具體規則ID從1開始
            struThermCond.dwChan = 1;// dwChannel; //從1開始,0xffffffff代表獲取全部通道

            IntPtr pCond = Marshal.AllocCoTaskMem(size);
            Marshal.StructureToPtr(struThermCond, pCond, false);
            var ret = NET_DVR_StartRemoteConfig(m_lUserID, 3629, pCond, size, Callback, IntPtr.Zero);
            if (ret < 0)
            {
                Console.WriteLine("配置測溫失敗, 錯誤代碼:" + NET_DVR_GetLastError()); //登錄失敗,輸出錯誤號
                return;
            }

            Console.WriteLine("開始測溫...");
            Console.ReadKey();

            //退出登錄
            NET_DVR_Logout(m_lUserID);

            NET_DVR_Cleanup();


        }
        /// <summary>
        /// 只顯示最高溫度
        /// </summary>
        /// <param name="dwType"></param>
        /// <param name="lpBuffer"></param>
        /// <param name="dwBufLen"></param>
        /// <param name="pUserData"></param>
        static void Callback(uint dwType, IntPtr lpBuffer, uint dwBufLen, IntPtr pUserData)
        {
            var ret = Marshal.PtrToStructure<NET_DVR_THERMOMETRY_UPLOAD>(lpBuffer);
            Console.WriteLine(ret.struLinePolygonThermCfg.fMaxTemperature);
        }

哎,就這麼點事,弄得很複雜。


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

-Advertisement-
Play Games
更多相關文章
  • 近年來,企業管理軟體開發領域掀起了一陣快速開發平臺的風氣,很多人覺得這是開發界有人帶的節奏,很快就會消失了,軟體開發應該回歸最真實的代碼之中。 不過隨著時間的推移,越來越多的企業認識到快速開發平臺的重要性,相比於傳統的代碼式開發風格,快速開發平臺的優勢體現的淋漓盡致。 傳統開發平臺最突出的特點就是開 ...
  • 背景:我們的應用程式通常都是由多個程式集組成,例如一個 exe 程式依賴於多個 dll 程式集。在某些情況下,我們希望程式的分發能夠簡單,單獨一個 exe 就能正常運行。這種情況下,就需要將 dll 依賴項合併到 exe 主程式中。 本文章給大家講下非常好用的NuGet 包,Costura.Fody ...
  • 一、概述 在Window伺服器部署程式後,可能因為代碼的不合理或者其他各種各樣的問題,會導致CPU暴增,甚至達到100%等情況,嚴重危及到伺服器的穩定以及系統穩定,但是一般來說對於已發佈的程式,沒法即時看到出問題的代碼,而微軟提供了一個很好的工具“WinDbg”,使得我們能夠回溯問題。下麵講一下操作 ...
  • 前言, Blazor Assembly 需要最少 1.9M 的下載量. ( Blazor WebAssembly 船新項目下載量測試 , 僅供參考. ) 隨著程式越來越複雜, 引用的東西越來越多, 需要更多的下載量 , 有一些網站的網路可能較差, 載入這些文件需要一定的時間. 對於一些網站而言, 它 ...
  • 背景介紹:基於netcore2.2開發api介面程式,自定義了一個異常捕獲中間件,用於捕獲未經處理的異常以及狀態碼404、500等訪問(設計的出發點就是,出現了非200的響應,我這邊全部會進行處理成200,並返回固定格式的JSON格式數據),併進行統一的信息返回。 返回的JSON實體定義如下: 中間 ...
  • 前言: 昨天 Blazor WebAssembly 3.2 正式發佈了. 更新 VS2019後就能直接使用. 新建了兩個PWA項目, 一個不用asp.net core (靜態部署), 一個使用asp.net core (項目模板與伺服器交互) 其中下載量主要是預壓縮有沒有被使用 , 分別為 3.2M ...
  • 上一篇文章(https://www.cnblogs.com/meowv/p/12916613.html)使用自定義倉儲完成了簡單的增刪改查案例,有心的同學可以看出,我們的返回參數一塌糊塗,顯得很不友好。 在實際開發過程中,每個公司可能不盡相同,但都大同小異,我們的返回數據都是包裹在一個公共的模型下麵 ...
  • 現象: 用Microsoft.Office.Interop.Outlook取得日曆項,然後根據業務要求篩選。 items.Restrict方法中的篩選器,使用like進行模糊查詢時,會出COMException異常。 代碼: 1 //folder取得前略 2 3 Outlook.Items item ...
一周排行
    -Advertisement-
    Play Games
  • 1. 說明 /* Performs operations on System.String instances that contain file or directory path information. These operations are performed in a cross-pla ...
  • 視頻地址:【WebApi+Vue3從0到1搭建《許可權管理系統》系列視頻:搭建JWT系統鑒權-嗶哩嗶哩】 https://b23.tv/R6cOcDO qq群:801913255 一、在appsettings.json中設置鑒權屬性 /*jwt鑒權*/ "JwtSetting": { "Issuer" ...
  • 引言 集成測試可在包含應用支持基礎結構(如資料庫、文件系統和網路)的級別上確保應用組件功能正常。 ASP.NET Core 通過將單元測試框架與測試 Web 主機和記憶體中測試伺服器結合使用來支持集成測試。 簡介 集成測試與單元測試相比,能夠在更廣泛的級別上評估應用的組件,確認多個組件一起工作以生成預 ...
  • 在.NET Emit編程中,我們探討了運算操作指令的重要性和應用。這些指令包括各種數學運算、位操作和比較操作,能夠在動態生成的代碼中實現對數據的處理和操作。通過這些指令,開發人員可以靈活地進行算術運算、邏輯運算和比較操作,從而實現各種複雜的演算法和邏輯......本篇之後,將進入第七部分:實戰項目 ...
  • 前言 多表頭表格是一個常見的業務需求,然而WPF中卻沒有預設實現這個功能,得益於WPF強大的控制項模板設計,我們可以通過修改控制項模板的方式自己實現它。 一、需求分析 下圖為一個典型的統計表格,統計1-12月的數據。 此時我們有一個需求,需要將月份按季度劃分,以便能夠直觀地看到季度統計數據,以下為該需求 ...
  • 如何將 ASP.NET Core MVC 項目的視圖分離到另一個項目 在當下這個年代 SPA 已是主流,人們早已忘記了 MVC 以及 Razor 的故事。但是在某些場景下 SSR 還是有意想不到效果。比如某些靜態頁面,比如追求首屏載入速度的時候。最近在項目中回歸傳統效果還是不錯。 有的時候我們希望將 ...
  • System.AggregateException: 發生一個或多個錯誤。 > Microsoft.WebTools.Shared.Exceptions.WebToolsException: 生成失敗。檢查輸出視窗瞭解更多詳細信息。 內部異常堆棧跟蹤的結尾 > (內部異常 #0) Microsoft ...
  • 引言 在上一章節我們實戰了在Asp.Net Core中的項目實戰,這一章節講解一下如何測試Asp.Net Core的中間件。 TestServer 還記得我們在集成測試中提供的TestServer嗎? TestServer 是由 Microsoft.AspNetCore.TestHost 包提供的。 ...
  • 在發現結果為真的WHEN子句時,CASE表達式的真假值判斷會終止,剩餘的WHEN子句會被忽略: CASE WHEN col_1 IN ('a', 'b') THEN '第一' WHEN col_1 IN ('a') THEN '第二' ELSE '其他' END 註意: 統一各分支返回的數據類型. ...
  • 在C#編程世界中,語法的精妙之處往往體現在那些看似微小卻極具影響力的符號與結構之中。其中,“_ =” 這一組合突然出現還真不知道什麼意思。本文將深入剖析“_ =” 的含義、工作原理及其在實際編程中的廣泛應用,揭示其作為C#語法奇兵的重要角色。 一、下劃線 _:神秘的棄元符號 下劃線 _ 在C#中並非 ...