CameraGroup.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315
  1. // CameraGroup.cs 相机组线程管理类,控制相机采集和识别的线程运行,开放有算法的接口
  2. using MvCameraControl;
  3. using MvvmScaffoldFrame48.DLL.CameraTools;
  4. using MvvmScaffoldFrame48.Model.ResultModel;
  5. using MvvmScaffoldFrame48.Model.StorageModel.SystemConfig;
  6. using System;
  7. using System.Collections.Concurrent;
  8. using System.Threading;
  9. using System.Threading.Tasks;
  10. namespace MvvmScaffoldFrame48.DLL.ThreadManager
  11. {
  12. /// <summary>
  13. /// 相机组配置和状态管理类
  14. /// 管理单个相机的采集和处理线程及相关资源
  15. /// </summary>
  16. public class CameraGroup
  17. {
  18. #region 变量与实例
  19. //相机线程的休眠时间,默认为0,即不休眠。在采集节拍没有那么高时,增加此值降低性能消耗
  20. private int CameraSleepTime = 0;
  21. //相机
  22. public HikCamera Camera { get; set; }
  23. //相机ID
  24. public int CameraId { get; set; }
  25. //是否运行中
  26. public bool IsRunning { get; set; }
  27. //相机处理线程
  28. public Task CameraTask { get; set; }
  29. //图像处理线程
  30. public Task ProcessingTask { get; set; }
  31. //图像队列
  32. public ConcurrentQueue<IImage> ImageQueue { get; set; }
  33. //信号量
  34. public SemaphoreSlim QueueSemaphore { get; set; }
  35. //取消令牌
  36. public CancellationTokenSource CancellationTokenSource { get; set; }
  37. // 图像处理算法(接口方式)
  38. public IImageProcessingAlgorithmHikVision ImageProcessor { get; set; }
  39. // 结果发送事件,当图像处理完成时触发
  40. public event EventHandler<CameraProcessEventArgsResultModel> ProcessResultAvailable;
  41. // 相机配置
  42. public CameraProcessConfigModel Configuration { get; set; }
  43. #endregion
  44. #region 公共方法
  45. /// <summary>
  46. /// 启动相机组(包括采集和处理线程)
  47. /// </summary>
  48. public void Start()
  49. {
  50. if (Camera == null)
  51. {
  52. Console.WriteLine($"相机 {CameraId} 未初始化");
  53. return;
  54. }
  55. if (Camera.Device == null)
  56. {
  57. Console.WriteLine($"相机 {CameraId} 未初始化");
  58. }
  59. if (Camera.Device.IsConnected == false)
  60. {
  61. Console.WriteLine($"相机 {CameraId} 未打开");
  62. return;
  63. }
  64. if (IsRunning)
  65. return;
  66. // 根据配置加载算法
  67. LoadAlgorithmFromConfiguration();
  68. Camera.StartReceiveFuntion();
  69. // 重置取消令牌
  70. CancellationTokenSource = new CancellationTokenSource();
  71. var token = CancellationTokenSource.Token;
  72. // 启动相机采集线程
  73. CameraTask = Task.Factory.StartNew(() => CameraCaptureLoop(token),
  74. token, TaskCreationOptions.LongRunning, TaskScheduler.Default);
  75. // 启动图像处理线程
  76. ProcessingTask = Task.Factory.StartNew(() => ImageProcessingLoop(token),
  77. token, TaskCreationOptions.LongRunning, TaskScheduler.Default);
  78. IsRunning = true;
  79. Console.WriteLine($"相机组 {CameraId} 已启动");
  80. }
  81. /// <summary>
  82. /// 停止相机组
  83. /// </summary>
  84. public async void Stop()
  85. {
  86. if (!IsRunning)
  87. return;
  88. // 发送取消信号
  89. CancellationTokenSource.Cancel();
  90. // 等待线程完成
  91. try
  92. {
  93. if (CameraTask != null)
  94. await CameraTask;
  95. if (ProcessingTask != null)
  96. await ProcessingTask;
  97. }
  98. catch (OperationCanceledException)
  99. {
  100. // 正常取消,忽略异常
  101. }
  102. IsRunning = false;
  103. Camera.StopReceiveFuntion();
  104. Console.WriteLine($"相机组 {CameraId} 已停止");
  105. }
  106. /// <summary>
  107. /// 更新算法参数
  108. /// </summary>
  109. /// <param name="parameters">新参数</param>
  110. public void UpdateAlgorithmParameters(object parameters)
  111. {
  112. if (ImageProcessor != null && parameters != null)
  113. {
  114. string parametersjson = ImageProcessor.GetSaveJson();
  115. ImageProcessor.Configure(parametersjson);
  116. // 更新配置
  117. if (Configuration != null)
  118. {
  119. Configuration.AlgorithmParameters = parametersjson;
  120. }
  121. Console.WriteLine($"相机 {CameraId} 算法参数已更新");
  122. }
  123. }
  124. #endregion
  125. #region 私有方法
  126. /// <summary>
  127. /// 设置图像处理算法
  128. /// </summary>
  129. /// <param name="algorithmName">算法名称</param>
  130. private void SetProcessingAlgorithm(string algorithmName, string parameters = null)
  131. {
  132. if (string.IsNullOrEmpty(algorithmName))
  133. {
  134. ImageProcessor = null;
  135. return;
  136. }
  137. try
  138. {
  139. // 根据算法名称创建实例(这里可以使用简单的工厂方法或反射)
  140. ImageProcessor = ImageProcessingAlgorithmHikVisionFactory.CreateAlgorithm(algorithmName);
  141. // 配置参数
  142. if (parameters != null && ImageProcessor != null)
  143. {
  144. ImageProcessor.Configure(parameters);
  145. }
  146. Console.WriteLine($"相机 {CameraId} 成功设置算法: {algorithmName}");
  147. }
  148. catch (Exception ex)
  149. {
  150. Console.WriteLine($"相机 {CameraId} 设置算法 {algorithmName} 失败: {ex.Message}");
  151. }
  152. }
  153. /// <summary>
  154. /// 根据配置动态加载算法
  155. /// </summary>
  156. private void LoadAlgorithmFromConfiguration()
  157. {
  158. if (Configuration != null && !string.IsNullOrEmpty(Configuration.ProcessingAlgorithmName))
  159. {
  160. SetProcessingAlgorithm(Configuration.ProcessingAlgorithmName, Configuration.AlgorithmParameters);
  161. }
  162. else
  163. {
  164. // 使用默认算法(无参数)
  165. SetProcessingAlgorithm("Default");
  166. }
  167. }
  168. /// <summary>
  169. /// 发送处理结果到通信线程的方法
  170. /// 多个线程可以共用此方法
  171. /// </summary>
  172. /// <param name="resultData">处理结果数据</param>
  173. private void SendProcessResult(object resultData)
  174. {
  175. var eventArgs = new CameraProcessEventArgsResultModel
  176. {
  177. CameraId = this.CameraId,
  178. ResultData = resultData,
  179. Timestamp = DateTime.Now
  180. };
  181. // 触发事件,通知订阅者有新的处理结果
  182. ProcessResultAvailable?.Invoke(this, eventArgs);
  183. }
  184. /// <summary>
  185. /// 图像处理实现
  186. /// </summary>
  187. private void ProcessImage(IImage imageData, int cameraId)
  188. {
  189. object resultData = null;
  190. try
  191. {
  192. // 优先使用接口方式的算法
  193. if (ImageProcessor != null)
  194. {
  195. resultData = ImageProcessor.ProcessImage(imageData, cameraId);
  196. }
  197. // 如果都没有设置,则使用默认处理
  198. else
  199. {
  200. Console.WriteLine($"相机 {CameraId} 未加载算法");
  201. // 默认处理逻辑
  202. }
  203. // 发送处理结果
  204. SendProcessResult(resultData);
  205. // 输出处理结果
  206. }
  207. catch (Exception ex)
  208. {
  209. Console.WriteLine($"相机 {CameraId} 图像处理错误: {ex.Message}");
  210. // 发送错误结果
  211. }
  212. }
  213. #endregion
  214. #region 线程主体方法
  215. /// <summary>
  216. /// 相机采集循环
  217. /// </summary>
  218. private async void CameraCaptureLoop(CancellationToken token)
  219. {
  220. //int frameCount = 0;
  221. try
  222. {
  223. while (!token.IsCancellationRequested)
  224. {
  225. // 模拟相机图像采集(无休眠,持续采集)
  226. if (Camera.GetOnceImage(out IFrameOut imageData))
  227. {
  228. // 将图像放入队列
  229. ImageQueue.Enqueue(imageData.Image.Clone() as IImage);
  230. Console.WriteLine($"相机 {CameraId} 采集到图像,帧号{imageData.FrameNum}");
  231. QueueSemaphore.Release(); // 通知处理线程有新数据
  232. imageData.Dispose();
  233. }
  234. // 这里不添加休眠,保持最大帧率采集
  235. await Task.Delay(CameraSleepTime);
  236. }
  237. }
  238. catch (OperationCanceledException)
  239. {
  240. // 线程被取消,正常退出
  241. }
  242. catch (Exception ex)
  243. {
  244. Console.WriteLine($"相机 {CameraId} 采集异常: {ex.Message}");
  245. }
  246. }
  247. /// <summary>
  248. /// 图像处理循环
  249. /// </summary>
  250. private async void ImageProcessingLoop(CancellationToken token)
  251. {
  252. try
  253. {
  254. while (!token.IsCancellationRequested)
  255. {
  256. // 等待图像数据
  257. await QueueSemaphore.WaitAsync(token);
  258. // 从队列获取图像
  259. if (ImageQueue.TryDequeue(out IImage imageData))
  260. {
  261. // 执行图像处理逻辑
  262. ProcessImage(imageData, CameraId);
  263. imageData.Dispose();
  264. // 可以根据队列长度决定是否短暂休眠以避免过度占用CPU
  265. if (ImageQueue.Count == 0)
  266. {
  267. await Task.Delay(1, token); // 短暂让出CPU
  268. }
  269. }
  270. }
  271. }
  272. catch (OperationCanceledException)
  273. {
  274. // 线程被取消,正常退出
  275. }
  276. catch (Exception ex)
  277. {
  278. Console.WriteLine($"相机 {CameraId} 处理异常: {ex.Message}");
  279. }
  280. }
  281. #endregion
  282. }
  283. }