流媒體伺服器、海康威視 大華攝像頭實現視頻監控、直播解決方案

来源:https://www.cnblogs.com/CDLinXi/archive/2020/04/07/12653445.html
-Advertisement-
Play Games

隨著互聯網+物聯網進程的加快,視頻監控應用領域變得越來越廣泛,其中海康威視 大華等品牌的攝像頭頻繁出現在視野中。由於去年也實現過智慧工地項目上的視頻監控方案,加上當今直播趨勢不減。現在總結一下: 緣由:是1對N 點對多的直播方式, 一般都是採用伺服器轉發,所以此處不考慮WebRTC這種端對端的方式, ...


  隨著互聯網+物聯網進程的加快,視頻監控應用領域變得越來越廣泛,其中海康威視 大華等品牌的攝像頭頻繁出現在視野中。由於去年也實現過智慧工地項目上的視頻監控方案,加上當今直播趨勢不減。現在總結一下:

緣由:是1對N 點對多的直播方式, 一般都是採用伺服器轉發,所以此處不考慮WebRTC這種端對端的方式,WebRTC將在下一篇文章中講解下實現思路。

前提:需要海康威視或大華的攝像頭,大華攝像頭清晰度 品質較好,但相對於海康的攝像頭較貴,所以海康威視的攝像頭更受口袋歡迎。

一.自建流媒體伺服器

  第一種方式就是自建流媒體伺服器,然後自己實現採集推流 到伺服器 拉流到客戶端播放。先看一張圖:

  1. 先客戶端軟體或設備採集視頻流和語音流,或者是攝像頭硬體採集的畫面流等(如何採集就屬於硬體相關的問題了,此處不討論)
  2. 然後通過推流的方式推到流媒體伺服器,推流協議可以使用RTMP RMSP,這2種都是基於tcp的 不會丟包。但是很容易造成高延遲(具體的看伺服器 網路 是否做CDN來支撐)。
    1 //可指定h264或h265編碼,可以把h265編碼看成是h264編碼的升級版,在碼率 體積 清晰度 移動補償上更友好些
    2 //大體結構為:rtsp://攝像頭用戶名:密碼@地址:埠 伺服器上地址參數...
    3 rtsp://admin:[email protected]:554/h264/ch1/main/av_stream
    4 rtsp://admin:[email protected]:554/Streaming/Channels/101?transportmode=unicast

    以上方式只是實現了流推送到了伺服器,並沒有指定它播放地址以及播放的轉碼。因此我們可以考慮使用ffmpeg,這是一套可以用來記錄、轉換數字音頻、視頻,並能將其轉化為流的開源電腦程式。也就是使用ffmpeg不光可以本地採集流還可以指定推送到那一臺伺服器上和它的播放地址等等;

    1 //ffmpeg -re -i表示使用的協議和協議的參數,具體的參數意義請百度
    2 //接著是和上面一樣的推流,這裡使用的是rtsp,建議用rtmp,本帥在使用中感覺rtmp相容性更好 web前端使用rtmp更方便。比如前端用Flash插件。或者Video標簽等等。
    3 //然後是基於tcp 轉碼 播放的地址,比如播放地址是:rtsp://117.250.250.250/Cameratest
    4 ffmpeg -re -i rtsp://admin:[email protected]:554/h264/ch1/main/av_stream -rtsp_transport tcp -vcodec h264 -f rtsp rtsp://localhost/test
    5 ffmpeg -i rtsp://admin:[email protected]:554/h264/ch1/main/av_stream -rtsp_transport tcp -vcodec h264 -f rtsp rtsp://117.250.250.250/Cameratest

     註意播放地址前指定播放協議,比如rtsp rtsp://117.250.250.250/Cameratest。如果是rtmp那麼最後就應該是:rtmp rtmp://117........................

  3. 流媒體伺服器做一些編碼轉碼處理等將流分發給各個客戶端,進而進行拉流播放。那麼問題來了  如何實現流媒體伺服器呢?如何架設???
  4. 架設上我們可以使用nginx rtmp-module模塊來架設,架設好後就可以使用rtmp推流給它。還可以用上面第2點中的ffmpeg命令寫一個bat腳本來測試攝像頭和架設的流媒體。
  5. PC端播放使用rtmp Flash來進行播放(H5中的Video標簽瞭解一下),移動端播放使用HLS m3u8 rtmp來進行播放(具體播放方式視項目框架情況而定)。看網上有人還使用flv + http stream 進行播放的。
  6. 後期出現了併發 播放量增多的壓力可以把nginx做分層(接入層+交換層),或者是轉發一下做負載均衡,或者CDN來支撐。前期如果考慮到後期會使用CDN也可以直接跳過nginx 一開始用cdn的直播服務。

二.接入第三方平臺

  在之前的項目中是購買了海康威視的攝像頭,所以為了方便快捷的開發,是接入了第三方平臺,由第三方平臺進行管理和轉發。大體流程是往第三方平臺註冊攝像頭信息(序列號 驗證碼),然後流直接走第三方平臺,自己服務端只需要獲取三方平臺的API介面便能得知播放地址 直接客戶端播放。使用的是螢石雲

 

 

 

   我們可以在自己的項目中往螢石雲註冊攝像頭信息(也就是調用螢石雲介面 往螢石雲寫一條數據),然後在需要用的地方獲取螢石雲API介面播放地址。完全不用管流的處理(得充值)。

   提供一份對螢石雲請求封裝的類(C#代碼):  

  1 using Newtonsoft.Json;
  2 using System;
  3 using System.Collections.Generic;
  4 using System.Linq;
  5 using System.Net.Http;
  6 using System.Net.Http.Headers;
  7 using System.Text;
  8 using System.Threading.Tasks;
  9 using YJT.Common;
 10 
 11 /*20190819 by suzong */
 12 namespace YJT.Wisdom.Api.lib
 13 {
 14     /// <summary>
 15     /// 螢石雲請求封裝
 16     /// </summary>
 17     public class YsClient
 18     {
 19         private static readonly string requestUrl = "https://open.ys7.com/";
 20         private static readonly string appKey = "";//官網註冊獲得
 21         private static readonly string appSecret = "";//官網註冊獲得
 22 
 23         /// <summary>
 24         /// 獲得token
 25         /// </summary>
 26         /// <returns>{code:200,data:{accessToken:"",expireTime:精確到毫秒}}</returns>
 27         public static async Task<string> GetToken()
 28         {
 29             string key = ConfigHelper.GetSetting("CacheKey:YsToken") ?? "YsAccessToken";
 30             string tokenStr = MemoryCacheHelper.Get(key)?.ToString();
 31             if (string.IsNullOrEmpty(tokenStr))
 32             {
 33                 string str = await HttpHelper.HttpPostAsync($"{requestUrl}api/lapp/token/get?appKey={appKey}&appSecret={appSecret}");
 34                 YsResult result = JsonConvert.DeserializeObject<YsResult>(str);
 35                 //緩存token 緩存時間為5天
 36                 tokenStr = result?.data?.accessToken;
 37                 MemoryCacheHelper.Set(key, tokenStr, TimeSpan.FromDays(5));
 38             }
 39             return tokenStr;
 40         }
 41 
 42         /// <summary>
 43         /// 添加設備
 44         /// </summary>
 45         /// <param name="deviceSerial">設備序列號</param>
 46         /// <param name="validateCode">設備驗證碼</param>
 47         /// <returns></returns>
 48         public static async Task<YsResult> SaveDevice(string deviceSerial, string validateCode)
 49         {
 50             if (string.IsNullOrEmpty(deviceSerial) || string.IsNullOrEmpty(validateCode))
 51                 return new YsResult() { code = "-1", msg = "缺少驗證碼或序列號" };
 52 
 53             string str = await HttpHelper.HttpPostAsync($"{requestUrl}api/lapp/device/add?accessToken={GetToken().Result}&deviceSerial={deviceSerial.ToUpper()}&validateCode={validateCode.ToUpper()}");
 54             return JsonConvert.DeserializeObject<YsResult>(str);
 55         }
 56 
 57         /// <summary>
 58         /// 關閉視頻加密
 59         /// </summary>       
 60         /// <param name="deviceSerial">設備序列號</param>
 61         /// <param name="validateCode">設備驗證碼</param>
 62         /// <returns>{code:200}</returns>
 63         public static async Task<YsResult> OffEncryption(string deviceSerial, string validateCode)
 64         {
 65             if (string.IsNullOrEmpty(deviceSerial) || string.IsNullOrEmpty(validateCode))
 66                 return new YsResult() { code = "-1", msg = "缺少驗證碼或序列號" };
 67             string str = await HttpHelper.HttpPostAsync($"{requestUrl}api/lapp/device/encrypt/off?accessToken={GetToken().Result}&deviceSerial={deviceSerial.ToUpper()}&validateCode={validateCode.ToUpper()}");
 68             return JsonConvert.DeserializeObject<YsResult>(str);
 69         }
 70 
 71         /// <summary>
 72         /// 刪除設備
 73         /// </summary>
 74         /// <param name="token"></param>
 75         /// <param name="deviceSerial">設備序列號</param>
 76         /// <returns>{code:200}</returns>
 77         public static async Task<YsResult> DeleteDevice(string deviceSerial)
 78         {
 79             if (string.IsNullOrEmpty(deviceSerial))
 80                 return new YsResult() { code = "-1", msg = "缺少序列號" };
 81 
 82             string str = await HttpHelper.HttpPostAsync($"{requestUrl}api/lapp/device/delete?accessToken={GetToken().Result}&deviceSerial={deviceSerial.ToUpper()}");
 83             return JsonConvert.DeserializeObject<YsResult>(str);
 84         }
 85 
 86         /// <summary>
 87         /// 獲取直播地址 WSS地址 #get請求 每次獲取 
 88         /// </summary>
 89         /// <param name="token"></param>
 90         /// <param name="deviceSerial">設備序列號</param>
 91         /// <param name="validateCode">設備驗證碼</param>
 92         /// <returns></returns>
 93         public static async Task<string> GetPlayWss(string deviceSerial, string validateCode)
 94         {
 95             if (string.IsNullOrEmpty(deviceSerial) || string.IsNullOrEmpty(validateCode))
 96                 return null;
 97             //{"retcode":0,"msg":"成功","data":{"tokens":["ot.cadfwa3t0dkdn62x5qf257es7dbq1cie-1vwkltfwtz-1w1jc79-9eabx2bbz"],"params":"&auth=1&biz=4&cln=100"}}
 98             string str = await HttpHelper.HttpGetAsync($"{requestUrl}jssdk/ezopen/getStreamToken?accessToken={GetToken().Result}&num=1&type=live");
 99             YsResult result = JsonConvert.DeserializeObject<YsResult>(str);
100             if (result.retcode == 0)
101             {
102                 string tokensStr = result?.data?.tokens[0];
103                 string paramStr = result?.data["params"];
104                 //wss://jsdecoder.ys7.com:20006/live?dev=設備序列號&chn=1&stream=2&ssn=剛纔獲取的tokens[0]+剛纔獲取的params的字元串。作為wssUrl,此地址可以加上checkCode=驗證碼作為視頻加密傳輸。
105                 return $"wss://jsdecoder.ys7.com:20006/live?dev={deviceSerial}&chn=1&stream=2&ssn={tokensStr}{paramStr}&checkCode={validateCode}";
106             }
107             return null;
108         }
109 
110         /// <summary>
111         /// 獲取直播地址 #返回RTMP地址
112         /// </summary>
113         /// <param name="token"></param>
114         /// <param name="deviceSerial">設備序列號</param>
115         /// <returns>返回rtmp</returns>
116         public static async Task<string> GetPlayRtmp(string deviceSerial)
117         {
118             if (string.IsNullOrEmpty(deviceSerial))
119                 return null;
120             string str = await HttpHelper.HttpPostAsync($"{requestUrl}api/lapp/live/address/get?accessToken={GetToken().Result}&source={deviceSerial}:1");
121             YsResult result = JsonConvert.DeserializeObject<YsResult>(str);
122             if (result.code.Equals("200"))
123                 return result?.data[0]?.rtmp;
124             return null;
125         }
126 
127         /// <summary>
128         /// 獲取設備可有的許可權
129         /// </summary>
130         /// <param name="token"></param>
131         /// <param name="deviceSerial">設備序列號</param>
132         /// <returns>data:
133         ///{
134         ///    supprot_encrypt 是否支持視頻圖像加密 0 - 不支持, 1 - 支持
135         ///    support_modify_pwd 是否支持修改設備加密密碼: 0 - 不支持, 1 - 支持
136         ///    ptz_top_bottom 是否支持雲臺上下轉動 0 - 不支持, 1 - 支持
137         ///    ptz_left_right 是否支持雲台左右轉動 0 - 不支持, 1 - 支持
138         ///    ptz_45 是否支持雲台45度方向轉動 0 - 不支持, 1 - 支持
139         ///    ptz_zoom 是否支持雲台縮放控制 0 - 不支持, 1 - 支持
140         ///    ptz_focus 是否支持焦距模式 0 - 不支持, 1 - 支持
141         ///}code: 200
142         /// </returns>
143         public static async Task<YsResult<YsRoles>> GetDeviceRole(string deviceSerial)
144         {
145             if (string.IsNullOrEmpty(deviceSerial))
146                 return new YsResult<YsRoles>() { code = "-1", msg = "缺少序列號" };
147 
148             string str = await HttpHelper.HttpPostAsync($"{requestUrl}api/lapp/device/capacity?accessToken={GetToken().Result}&deviceSerial={deviceSerial.ToUpper()}");
149             return JsonConvert.DeserializeObject<YsResult<YsRoles>>(str);
150         }
151 
152         /// <summary>
153         /// 雲台控制開始
154         /// </summary>
155         /// <param name="token"></param>
156         /// <param name="deviceSerial">設備序列號</param>
157         /// <param name="direction">方向 (操作命令:0 - 上,1 - 下,2 - 左,3 - 右,4 - 左上,5 - 左下,6 - 右上,7 - 右下,8 - 放大,9 - 縮小,10 - 近焦距,11 - 遠焦距)</param>
158         /// <param name="speed">速度 (雲台速度:0 - 慢,1 - 適中,2 - 快)</param>
159         /// <returns>{code:200}</returns>
160         public static async Task<YsResult> CradleControlStarts(string token, string deviceSerial, int direction, int speed)
161         {
162             if (string.IsNullOrEmpty(token) || string.IsNullOrEmpty(deviceSerial))
163                 return null;
164             string str = await HttpHelper.HttpPostAsync($"{requestUrl}api/lapp/device/ptz/start?accessToken={token}&deviceSerial={deviceSerial}&channelNo=1&direction={direction}&speed={speed}");
165             return JsonConvert.DeserializeObject<YsResult>(str);
166         }
167 
168         /// <summary>
169         /// 雲台控制結束
170         /// </summary>
171         /// <param name="token"></param>
172         /// <param name="deviceSerial">設備序列號</param>
173         /// <param name="direction">方向 (操作命令:0-上,1-下,2-左,3-右,4-左上,5-左下,6-右上,7-右下,8-放大,9-縮小,10-近焦距,11-遠焦距)</param>
174         /// <returns>{code:200}</returns>
175         public static async Task<YsResult> CradleControlEnd(string token, string deviceSerial, int direction)
176         {
177             if (string.IsNullOrEmpty(token) || string.IsNullOrEmpty(deviceSerial))
178                 return null;
179             string str = await HttpHelper.HttpPostAsync($"{requestUrl}api/lapp/device/ptz/stop?accessToken={token}&deviceSerial={deviceSerial}&channelNo=1&direction={direction}");
180             return JsonConvert.DeserializeObject<YsResult>(str);
181         }
182 
183         /// <summary>
184         /// 獲取單個設備信息
185         /// </summary>
186         /// <param name="token"></param>
187         /// <param name="deviceSerial">設備序列號</param>
188         /// <returns></returns>
189         public static async Task<YsResult> GetDeviceInfo(string deviceSerial)
190         {
191             if (string.IsNullOrEmpty(deviceSerial))
192                 return null;
193             string str = await HttpHelper.HttpPostAsync($"{requestUrl}api/lapp/device/info?accessToken={GetToken().Result}&deviceSerial={deviceSerial}");
194             YsResult result = JsonConvert.DeserializeObject<YsResult>(str);
195             if (result.code.Equals("200"))
196                 return result;
197             return null;
198         }
199 
200         /// <summary>
201         /// 開通直播功能
202         /// </summary>
203         /// <param name="deviceSerial">設備序列號</param>
204         /// <returns></returns>
205         public static async Task<YsResult> LiveOpen(string deviceSerial)
206         {
207             if (string.IsNullOrEmpty(deviceSerial))
208                 return null;
209             string str = await HttpHelper.HttpPostAsync($"{requestUrl}api/lapp/live/video/open?accessToken={GetToken().Result}&source={deviceSerial}:1");
210             return JsonConvert.DeserializeObject<YsResult>(str);
211         }
212 
213 
214     }
215 
216     /// <summary>
217     /// 螢石雲返回對象
218     /// </summary>
219     public class YsResult<T>
220     {
221         public string code { get; set; }
222         public T data { get; set; }
223         public string msg { get; set; }
224         public int retcode { get; set; }
225     }
226     public class YsResult : YsResult<dynamic>
227     {
228     }
229 
230     /// <summary>
231     /// 螢石雲設備能力集
232     /// </summary>
233     public class YsRoles
234     {
235         /// <summary>
236         /// 是否支持視頻圖像加密 0 - 不支持, 1 - 支持
237         /// </summary>
238         public int supprot_encrypt { get; set; } = 0;
239         /// <summary>
240         /// 是否支持修改設備加密密碼: 0 - 不支持, 1 - 支持
241         /// </summary>
242         public int support_modify_pwd { get; set; } = 0;
243         /// <summary>
244         /// 是否支持雲臺上下轉動 0 - 不支持, 1 - 支持
245         /// </summary>
246         public int ptz_top_bottom { get; set; } = 0;
247         /// <summary>
248         /// 是否支持雲台左右轉動 0 - 不支持, 1 - 支持
249         /// </summary>
250         public int ptz_left_right { get; set; } = 0;
251         /// <summary>
252         /// 是否支持雲台45度方向轉動 0 - 不支持, 1 - 支持
253         /// </summary>
254         public int ptz_45 { get; set; } = 0;
255         /// <summary>
256         /// 是否支持雲台縮放控制 0 - 不支持, 1 - 支持
257         /// </summary>
258         public int ptz_zoom { get; set; } = 0;
259         /// <summary>
260         /// 是否支持焦距模式 0 - 不支持, 1 - 支持
261         /// </summary>
262         public int ptz_focus { get; set; } = 0;
263     }
264 
265 }
螢石雲請求封裝

 

三.使用開源流媒體框架

   開源流媒體框架就很多了,常見的SRS國產的。安裝 推流 拉流。可用於直播/錄播/視頻客服等多種場景,其定位是運營級的互聯網直播伺服器集群。傳送門:http://www.ossrs.net/srs.release/releases/ 喜歡的可以自己去瞭解瞭解。


提醒:你所購買的攝像頭硬體上都會有攝像頭的名稱 序列號 驗證碼信息,攝像頭廠商比如海康會有搜索區域網內攝像頭的一個工具(官網去找)。一個Web界面的後臺用於設置攝像頭通道 配置信息等,在區域網內連接上攝像頭 瀏覽器地址欄輸入對應的地址就可以登錄當前攝像頭後臺。

附贈幾個rtsp rtmp免費測試地址(可以先讓前端用這些地址先實現播放功能):

1 rtsp://184.72.239.149/vod/mp4:BigBuckBunny_175k.mov
2 rtsp://195.200.199.8/mpeg4/media.amp
3 rtmp://media3.sinovision.net:1935/live/livestream

 End...

 


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

-Advertisement-
Play Games
更多相關文章
  • 在C#中,struct和class都是用戶定義的數據類型,struct和class有許多不同之處,但主要的區別是: Class是引用類型,它保存在堆上並且能夠被垃圾回收;然而stuct是值類型,它保存在棧上或者內嵌在它的包含類型之中。因此,從總體上來說struct比class節省記憶體。 下圖是Cla ...
  • Blazor的哪個特點, 可以讓程式員快速完成任務? 這隨筆講解的是使用代碼上下文來節約代碼, 讓驗證碼機制變得更加簡易. ...
  • Consul是HashiCorp公司推出的開源工具,Consul由Go語言開發,部署起來非常容易,只需要極少的可執行程式和配置文件,具有綠色、輕量級的特點。Consul是`分散式`的、`高可用`的、 `可橫向擴展`的用於實現分散式系統的服務發現與配置。 ...
  • CPU使用率 Linux 通過 /proc 虛擬文件系統,向用戶空間提供了系統內部狀態的信息,而 /proc/stat 提供的就是系統的 CPU 和任務統計信息 proc process information pseudo file system 查詢 man proc 關鍵指標常用參數 user ...
  • 更多電腦使用技巧可以訪問 "https://xiaoheidiannao.com" 查看哦! "Vim" 是Linux上的文本編輯工具,對於用習慣了 "Vim" 的人來說,肯定也想在Windows上使用 "Vim" ,至少我是這麼想的。雖然也有Windows版的 "Vim" ,但是在 "Word" ...
  • 本文分成入門篇和基礎篇。基礎篇包括變數、字元串處理、數學運算三部分。基礎篇包括流控制、函數和函數庫三部分。主要是基於例子進行講解,其中有 4 個複雜一點的腳本,看懂了也就入門了。 ...
  • 1. 讓外設工作起來 只要給相應的控制器中的寄存器發一個指令 向設備控制器的寄存器寫不就可以了嗎? 需要查寄存器地址、內容的格式和語義、操作系統需要給用戶提供一個簡單視圖 文件視圖 ,這樣方便 總的來說就是: 1. 形成文件視圖 2. 發出out指令 3. 形成中斷處理 中斷處理:當CPU(中央處理 ...
  • 補充 /etc/hostname :CenOS7主機名配置文件 /etc/sysconfig/network C6主機名配置文件 修改主機名 永久生效 臨時改一下 /etc/sysctl.conf Linux內核參數信息文件※※※※※ 調整Linux系統、優化需要配置這個文件 sysctl p 讓修 ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...