SignalR注册成Windows后台服务,并实现web前端断线重连

news2024/12/23 5:03:58

注意下文里面的 SignalR 不是 Core 版本,而是 Framework 下的

本文使用的方式是把 SignalR 写在控制台项目里,再用 Topshelf 注册成 Windows 服务

这样做有两点好处

  1. 传统 Window 服务项目调试时需要“附加到进程”,开发体验比较差,影响效率
  2. 使用控制台不仅可以随时打断点调试,还可以随时打印调试信息,非常方便

Topshelf 的使用方法这里不再阐述,在控制台里使用 Topshelf 三个步骤 :

  1. 定义一个 Owin 自托管作为 SignalR的宿主,里面设置允许跨域,起名为 Startup 
    using Microsoft.AspNet.SignalR;
    using Microsoft.Owin.Cors;
    using Owin;
    using System;
    using System.Diagnostics;
    
    namespace HenryMes.SignalR.Hosting
    {
        /// <summary>
        /// 配置跨域请求、SignalR Server
        /// </summary>
        class Startup
        {
            public void Configuration(IAppBuilder app)
            {
                //app.UseErrorPage();
                app.UseCors(CorsOptions.AllowAll);
    
                // 有关如何配置应用程序的详细信息,请访问 http://go.microsoft.com/fwlink/?LinkID=316888
    
                //Hub Mode
                app.MapSignalR("/lcc", new HubConfiguration());
    
                app.Map("/signalr", map =>
                {
                    var config = new HubConfiguration
                    {
                        // You can enable JSONP by uncommenting this line
                        // JSONP requests are insecure but some older browsers (and some
                        // versions of IE) require JSONP to work cross domain
                        EnableJSONP = true
    
                    };
                    //config.EnableCrossDomain = true;
                    // Turns cors support on allowing everything
                    // In real applications, the origins should be locked down
                    map.UseCors(CorsOptions.AllowAll)
                       .RunSignalR(config);
    
    
                });
    
    
                 Make long polling connections wait a maximum of 110 seconds for a
                 response. When that time expires, trigger a timeout command and
                 make the client reconnect.
                //GlobalHost.Configuration.ConnectionTimeout = TimeSpan.FromSeconds(110);
    
                 Wait a maximum of 30 seconds after a transport connection is lost
                 before raising the Disconnected event to terminate the SignalR connection.
                //GlobalHost.Configuration.DisconnectTimeout = TimeSpan.FromSeconds(30);
    
                 For transports other than long polling, send a keepalive packet every
                 10 seconds. 
                 This value must be no more than 1/3 of the DisconnectTimeout value.
                //GlobalHost.Configuration.KeepAlive = TimeSpan.FromSeconds(10);
    
                // Turn tracing on programmatically
                GlobalHost.TraceManager.Switch.Level = SourceLevels.Information;
            }
        }
    }
    

  2. 建立可在服务里运行的服务类,使用了上面的Startup配置实例化宿主对象,里面定义了服务的启动,暂停,关闭等触发时的一些动作,本文就建立一个 JobManager 类来完成这些工作 
    using HenryMes.Utils;
    using Microsoft.Owin.Hosting;
    using System;
    using System.Threading.Tasks;
    
    namespace HenryMes.SignalR.Hosting
    {
        public class JobManager
        {
            private const string displayName = "SignalR 状态监控";
    
            IDisposable SignalR { get; set; }
    
            public bool Start()
            {
                try
                {
                    //signalr server地址,端口可以更换,确保不被占用,否则服务启动不了
    #if DEBUG
                    var url = $"http://{JsonConfig.Instance.Root()?.Debug?.Ip}:{JsonConfig.Instance.Root()?.Debug?.Port}";
                    var Port = $"{JsonConfig.Instance.Root()?.Debug?.Port}";
    #else
                    var url = $"http://{JsonConfig.Instance.Root()?.Release?.Ip}:{JsonConfig.Instance.Root()?.Release?.Port}";
                    var Port = $"{JsonConfig.Instance.Root()?.Release?.Port}";
    #endif
                    StartOptions options = new StartOptions();
                    options.Urls.Add(url);
                    options.Urls.Add($"http://+:{Port}");
                    //此处需要用一个全局变量来保存WebApp,否则在发布为后台服务的时候生命周期会提前结束,被系统回收掉
                    SignalR = WebApp.Start<Startup>(options);
                    Task.Delay(TimeSpan.FromSeconds(1)).Wait();
                    Console.WriteLine("Server running on {0}", url);
                    Console.WriteLine($"{displayName}服务开始");
                    Console.ReadLine();
                    LogHelper.GetInstance().Information($"{displayName}服务开始,地址 {url}");
                    return true;
                }
                catch (Exception ex)
                {
                    LogHelper.GetInstance().Error(ex);
                }
                return false;
            }
    
            public bool Stop()
            {
                SignalR.Dispose();
                LogHelper.GetInstance().Information($"{displayName}服务停止");
                System.Threading.Thread.Sleep(1500);
                return true;
            }
    
            public bool Shutdown()
            {
                SignalR.Dispose();
                LogHelper.GetInstance().Information($"{displayName}服务停止");
                System.Threading.Thread.Sleep(1500);
                return true;
            }
        }
    }
    
  3. 在 Program.cs 文件,也就是入口函数 main 调用 Topshelf 对服务进行配置
using Topshelf;

namespace HenryMes.SignalR.Hosting
{
    internal class Program
    {
        private const string displayName = "HenryMes.SignalR.Hosting";
        static void Main(string[] args)
        {
            HostFactory.Run(x => {
                x.Service<JobManager>(s =>
                {
                    s.ConstructUsing(name => new JobManager());
                    s.WhenStarted(tc => tc.Start());
                    s.WhenShutdown(tc => tc.Shutdown());
                    s.WhenStopped(tc => tc.Stop());
                });
                x.RunAsLocalSystem();
                x.StartAutomatically();
                x.SetDescription(displayName);
                x.SetDisplayName(displayName);
                x.SetServiceName(displayName);
            });
        }
    }
}

下面定义一个 SignalR 的 Hub 基类,里面管理了SignalR 的连接和断开,一个线程管理一个连接,连接断开,线程自动取消,建立一个抽象类 BaseHub

using HenryMes.Utils;
using Microsoft.AspNet.SignalR;
using Microsoft.AspNet.SignalR.Hubs;
using Newtonsoft.Json;
using System;
using System.Collections.Concurrent;
using System.Threading;
using System.Threading.Tasks;

namespace HenryMes.SignalR.Hosting
{
    /// <summary>
    /// 
    /// </summary>
    [HubName(nameof(T))]
    public abstract class BaseHub<T> : Hub where T : IHub
    {
        /// <summary>
        /// 线程安全版本的字典,管理SignalR的连接
        /// </summary>
        protected static readonly ConcurrentDictionary<string, CancellationTokenSource> Connections =
            new ConcurrentDictionary<string, CancellationTokenSource>();

        /// <summary>
        /// 异步锁
        /// </summary>
        public static readonly object AsyncObj = new object();

        /// <summary>
        /// 设置超时时间
        /// </summary>
        abstract protected int millisecondsTimeout { get; }

        /// <summary>
        /// 设置线程轮询时间
        /// </summary>
        abstract protected int intervalTime { get; }

        /// <summary>
        /// 跑单个Task
        /// </summary>
        abstract protected Func<object> RunMethod { get; }

        /// <summary>
        /// 跑多个Task, 返回是否超时
        /// </summary>
        abstract protected Func<CancellationTokenSource, (bool, object)> RunMultiTaskMethod { get; }

        /// <summary>
        /// 是否跑多任务
        /// </summary>
        abstract protected bool runMultiTask { get; }

        //当客户端与服务器建立连接后执行的方法
        public override Task OnConnected()
        {
            //获取客户端ID
            Console.WriteLine("{0}已连接", Context.ConnectionId);
            LogHelper.GetInstance().Information($"服务端与客户端:【{typeof(T).Name}】{Context.ConnectionId} 成功建立连接!");
            return base.OnConnected();
        }

        public override Task OnReconnected()
        {
            Console.WriteLine("{0}已重连", Context.ConnectionId);
            LogHelper.GetInstance().Information($"服务端与客户端:【{typeof(T).Name}】{Context.ConnectionId} 已重连!");
            Send(Context.ConnectionId);
            return base.OnReconnected();
        }

        /// <summary>
        /// 所有任务执行完是否超时
        /// </summary>
        /// <param name="tokenSource"></param>
        /// <param name="allTasks"></param>
        /// <returns></returns>
        public bool IsCompletedAllTasks(CancellationTokenSource tokenSource, Task[] allTasks)
        {
            try
            {
                return Task.WaitAll(allTasks, millisecondsTimeout, tokenSource.Token);
            }
            catch (AggregateException ex)
            {
                LogHelper.GetInstance().Error($"系统错误:{this.GetType().Name},{ex.Flatten().InnerException.Message}");
                tokenSource.Cancel();
            }
            return false;
        }

        /// <summary>
        /// 向客户端发送消息
        /// </summary>
        /// <param name="connectId"></param>
        public void Send(string connectId)
        {
            lock (AsyncObj)
            {
                var tokenSource = new CancellationTokenSource();
                Connections.TryAdd(connectId, tokenSource);

                Task.Run(() =>
                {
                    while (!tokenSource.Token.IsCancellationRequested)
                    {
                        try
                        {
                            // 是否是多任务
                            if (runMultiTask == false)
                            {
                                var result = RunMethod();
                                var message = $"【{typeof(T).Name}】 {connectId} 正在回传数据!";
                                LogHelper.GetInstance().Information(message);
                                // 把组装好的数据推送到前端
                                BaseNotifer<T>.Refresh(connectId, JsonConvert.SerializeObject(result));
                                tokenSource.Token.WaitHandle.WaitOne(intervalTime);
                            }
                            else
                            {
                                // 是否超时
                                var (isCompleted, result) = RunMultiTaskMethod(tokenSource);
                                if (isCompleted)
                                {
                                    var message = $"【{typeof(T).Name}】 {connectId} 正在回传数据!";
                                    LogHelper.GetInstance().Information(message);
                                    // 把组装好的数据推送到前端
                                    BaseNotifer<T>.Refresh(connectId, JsonConvert.SerializeObject(result));
                                    // 下一次推送等待N秒后进行
                                    tokenSource.Token.WaitHandle.WaitOne(intervalTime);
                                }
                                else
                                {
                                    // 等待超时
                                    tokenSource.Cancel();
                                    // 打印超时错误日志
                                    LogHelper.GetInstance().Error($@"{this.GetType().Name} 推送超时! 当前超时时间设置为{millisecondsTimeout}毫秒!");
                                    // 重新执行
                                    Connections.TryRemove(connectId, out tokenSource);
                                    Send(connectId);
                                }
                            }
                        }
                        catch(AggregateException ex)
                        {
                            LogHelper.GetInstance().Error($"系统错误:{this.GetType().Name},{ex.Flatten().InnerException.Message}");
                            tokenSource.Token.WaitHandle.WaitOne(intervalTime);
                        }
                    }
                }, tokenSource.Token);
            }
        }

        /// <summary>
        /// 连接断开事件
        /// </summary>
        /// <param name="stopCalled"></param>
        /// <returns></returns>
        public override Task OnDisconnected(bool stopCalled)
        {
            lock (AsyncObj)
            {
                try
                {
                    var tokenSource = Connections[Context.ConnectionId];
                    Connections.TryRemove(Context.ConnectionId, out tokenSource);
                    tokenSource.Cancel();
                    LogHelper.GetInstance().Information($"服务端与客户端:【{typeof(T).Name}】{Context.ConnectionId} 连接已断开!");
                }
                catch (Exception ex)
                {
                    if (Connections.ContainsKey(Context.ConnectionId))
                    {
                        var tokenSource = Connections[Context.ConnectionId];
                        Connections.TryRemove(Context.ConnectionId, out tokenSource);
                    }
                    // 打印错误日志
                    LogHelper.GetInstance().Error($@"{this.GetType().Name} 已断开! {ex.Message}!");
                }
            }
            return base.OnDisconnected(stopCalled);
        }
    }
}

以一个具体的 Hub 为例,继承上面的 BaseHub, 建立一个 具体实现的 Hub 名为 OperationKanBanHub ,使用 RunMultiTaskMethod 并行执行一些任务,这是项目里的一个真实案例,不必关心细节

using HenryMes.Entitys;
using HenryMes.WebApi.Controllers;
using HenryMes.WebApi.Controllers.Other;
using System;
using System.Collections.Generic;
using System.Threading;
using System.Threading.Tasks;

namespace HenryMes.SignalR.Hosting.Hubs
{
    /// <summary>
    /// 运营看板
    /// </summary>
    public class OperationKanBanHub : BaseHub<OperationKanBanHub>
    {
        /// <summary>
        /// 设置超时时间
        /// </summary>
        protected override int millisecondsTimeout => 10000;

        /// <summary>
        /// 设置线程轮询时间
        /// </summary>
        protected override int intervalTime => 5000;

        /// <summary>
        /// 是否跑多任务
        /// </summary>
        protected override bool runMultiTask => true;

        /// <summary>
        /// 跑单个Task
        /// </summary>
        protected override Func<object> RunMethod => throw new NotImplementedException();
        /// <summary>
        /// 跑多个Task
        /// </summary>
        protected override Func<CancellationTokenSource, (bool, object)> RunMultiTaskMethod => (TokenSource) =>
        {

            #region Task取数
            // 菜籽收购
            var taskSum4Rapeseed = Task.Run(() =>
            {
                KanbanController controller = new KanbanController();
                dynamic data = controller.ControlCenter_Center_Sum4Rapeseed();
                return data.Content;
            });

            // 油品生产,销售
            var taskSum4Oil = Task.Run(() =>
            {
                KanbanController controller = new KanbanController();
                dynamic data = controller.ControlCenter_Center_Sum4Oil();
                return data.Content;
            });

            // 库存量前10的存货
            var taskSum4Top = Task.Run(() =>
            {
                KanbanController controller = new KanbanController();
                dynamic data = controller.ControlCenter_Center_Sum4Top();
                return data.Content;
            });

            // 罐区存油,读取 mongodb
            var taskTankOilQuantity = Task.Run(() =>
            {
                return new List<dynamic>
                {
                    new { tank = "Tank1001",temperature = "21.4℃",pressure = "21PA", quantity = "100" },
                    new { tank = "Tank1002",temperature = "21.4℃",pressure = "21PA", quantity = "100" },
                    new { tank = "Tank1003",temperature = "21.4℃",pressure = "21PA", quantity = "100" },
                    new { tank = "Tank1004",temperature = "21.4℃",pressure = "21PA", quantity = "100" },
                    new { tank = "Tank1005",temperature = "21.4℃",pressure = "21PA", quantity = "100" },
                    new { tank = "Tank1006",temperature = "21.4℃",pressure = "21PA", quantity = "100" },
                    new { tank = "Tank1007",temperature = "21.4℃",pressure = "21PA", quantity = "100" },
                    new { tank = "Tank1008",temperature = "21.4℃",pressure = "21PA", quantity = "100" },
                    new { tank = "Tank1009",temperature = "21.4℃",pressure = "21PA", quantity = "100" },
                    new { tank = "Tank1010",temperature = "21.4℃",pressure = "21PA", quantity = "100" },
                    new { tank = "Tank1011",temperature = "21.4℃",pressure = "21PA", quantity = "100" },
                    new { tank = "Tank1012",temperature = "21.4℃",pressure = "21PA", quantity = "100" },
                    new { tank = "Tank1013",temperature = "21.4℃",pressure = "21PA", quantity = "100" },
                    new { tank = "Tank1014",temperature = "21.4℃",pressure = "21PA", quantity = "100" },
                };
            });

            // 近一年产出销售 1-12月
            var taskSaleDispatch4Month = Task.Run(() =>
            {
                KanbanController controller = new KanbanController();
                dynamic data = controller.ControlCenter_Center_SaleDispatch4Month();
                return data.Content;
            });

            // 最近10条采购信息
            var taskPreInStore4Lately = Task.Run(() =>
            {
                KanbanController controller = new KanbanController();
                dynamic data = controller.ControlCenter_Center_PreInStore4Lately();
                return data.Content;
            });

            // 最近10条生产计划
            var taskProductionTask = Task.Run(() =>
            {
                ProductionTaskController controller = new ProductionTaskController();
                dynamic data = controller.QueryTake(new SqlSugarPageRequest
                {
                    PageIndex = 1,
                    PageSize = 10,
                    Filter = new List<SqlSugar.ConditionalModel>()
                });
                return data.Content;
            });

            #endregion

            #region 同步阻塞等待所有Task执行完
            // 所有线程任务是否完成 默认false
            var isCompleted = IsCompletedAllTasks(TokenSource, new Task[] {
                                    taskSum4Rapeseed,
                                    taskSum4Oil,
                                    taskSum4Top,
                                    taskTankOilQuantity,
                                    taskSaleDispatch4Month,
                                    taskPreInStore4Lately,
                                    taskProductionTask
                                });
            #endregion

            if (isCompleted)
            {
                #region 所有Task已完成
                // 菜籽
                var RapeseedResult = taskSum4Rapeseed.Result;
                var Rapeseed = new
                {
                    GYS = new
                    {
                        PurchaseReceiveQuantity = RapeseedResult?.Data?.Rapeseed_GYS?.PurchaseReceiveQuantity,
                        BalanceQuantity = RapeseedResult?.Data?.Rapeseed_GYS?.BalanceQuantity,
                    },
                    SD = new
                    {
                        PurchaseReceiveQuantity = RapeseedResult?.Data?.Rapeseed_SD?.PurchaseReceiveQuantity,
                        BalanceQuantity = RapeseedResult?.Data?.Rapeseed_SD?.BalanceQuantity,
                    }
                };

                // 油品生产,销售
                dynamic Sum4OilResult = taskSum4Oil.Result;
                var Sum4Oil = new
                {
                    // 产出成品油
                    TankOilQuantity = Sum4OilResult?.Data?.TankOil?.ProductReceiveQuantity,
                    // 产出包装油
                    PackageOilQuantity = Sum4OilResult?.Data?.PackageOil?.ProductReceiveQuantity,
                    // 销售包装油
                    SaleOilQuantity = Sum4OilResult?.Data?.PackageOil?.SaleDispatchQuantity
                };

                // 存货中库存量前10的存货
                var Sum4TopResult = taskSum4Top.Result;
                var Sum4Top = new
                {
                    Sum4TopResult?.Data?.DataSource
                };

                // 罐区存油
                var TankOilQuantity = taskTankOilQuantity.Result;

                // 近一年产出销售 1-12月
                var TaskSaleDispatch4MonthResult = taskSaleDispatch4Month.Result;
                var SaleDispatch4Month = new
                {
                    Sale = TaskSaleDispatch4MonthResult?.Data?.SaleDispatch.Details,
                    Product = TaskSaleDispatch4MonthResult?.Data?.ProductReceive.Details
                };

                // 最近10条采购信息
                var TaskPreInStore4LatelyResult = taskPreInStore4Lately.Result;
                var PreInStore4Lately = TaskPreInStore4LatelyResult?.Data?.DataSource;

                // 最近10条生产计划
                var taskProductionTaskResult = taskProductionTask.Result;
                var ProductionTask = taskProductionTaskResult?.Data;

                return (isCompleted, new
                {
                    Rapeseed,
                    Sum4Oil,
                    Sum4Top,
                    TankOilQuantity,
                    SaleDispatch4Month,
                    PreInStore4Lately,
                    ProductionTask
                });
                #endregion
            }
            return (isCompleted, new { });
        };
    }
}

 此时 SignalR 的后台推送基本就完成了,再来就是web前端的接收推送和断线下的自动重新连接(比如说后台服务程序做了更新,此时需要关闭服务再启动服务,这个时候要求web端不断尝试重新连接,直到后台服务启动并重新连接上为止)

前端使用 Vue 2.0 + jQuery.signalR 2.4.2 , 只列一下关键代码

import $ from "jquery";
import "signalr";
import echarts from "../../pages/kanban/OperationKanBanEcharts.vue";
export default {
  components: { echarts },
  data() {
    return {
      connection: null,
      proxy: null,
      // 是否需要断线重连的标记,当页面关闭时是不需要继续推送的
      tryReconnect : true
    }
  },
  methods: {
    // 从SignalR推送过来的数据,刷新看板
    refreshKanban(message) {
      // 刷新时间
      this.getDateTime()
      let obj = JSON.parse(message)
      // 省略无关代码......
    },
  },
  mounted() {
    this.$nextTick(() => {
      this.connection = $.hubConnection(process.env.SignalR);
      // 定义服务器端SignalR推送过来的消息接收代理
      this.proxy = this.connection.createHubProxy("OperationKanBanHub");
      this.proxy.on("Refresh", (message) => {
        console.log(`接收到来自服务端 ${this.connection.id} 的数据!`)
        this.refreshKanban(message)
      });
      // 创建连接到服务器端SignalR的连接
      this.connection
        .start()
        .done(() => {
          // 客户端发送信息到服务器
          this.proxy.invoke("Send", this.connection.id);
        })
        .fail((err) => {
          console.log(err);
        });

      this.connection.disconnected(() => {
          if(this.tryReconnect) {
            setTimeout(() => {
              console.log('连接已断开,正尝试重新连接!')
                this.connection
                  .start()
                  .done(() => {
                    this.proxy.invoke("Send", this.connection.id); // 客户端发送信息到服务器
                  })
                  .fail((err) => {
                    console.log(err);
                  });
                }, 5000); // Restart connection after 5 seconds.
          }
        });
    });
  },
  deactivated() {
    if (this.connection) {
      // 关闭SignalR连接
      this.tryReconnect = false
      this.connection.stop();
      // 清除缓存
      this.$vnode.parent.componentInstance.cache = {};
      this.$vnode.parent.componentInstance.keys = [];
    }
  },
};

最后的一个步骤,怎么把后台的控制台SignalR宿主程序安装成 Windows 服务?在项目里建立两个批处理文件,Install.bat 安装服务,UnInstall.bat 卸载服务,点击右键点文件属性,把他们的编码改为 ansi(不要问我为什么......因为不改的话,打开批处理命令窗口的时候中文会显示成乱码)

 Install.bat

@echo on
 
rem 设置DOS窗口的背景颜色及字体颜色
color 2f
 
rem 设置DOS窗口大小 
mode con: cols=80 lines=25
 
@echo off
echo 请按任意键开始安装 HenryMes.SignalR.Hosting 服务

rem 以管理员身份运行
%1 mshta vbscript:CreateObject("Shell.Application").ShellExecute("cmd.exe","/c %~s0 ::","","runas",1)(window.close)&&exit
:Admin
 
rem 输出空行
echo.
pause

cd /d %~dp0
HenryMes.SignalR.Hosting install --autostart start
net start HenryMes.SignalR.Hosting
 
pause

UnInstall.bat

@echo on
 
rem 设置DOS窗口的背景颜色及字体颜色
color 2f
 
rem 设置DOS窗口大小 
mode con: cols=80 lines=25
 
@echo off
echo 请按任意键开始卸载 HenryMes.SignalR.Hosting 服务

rem 以管理员身份运行
%1 mshta vbscript:CreateObject("Shell.Application").ShellExecute("cmd.exe","/c %~s0 ::","","runas",1)(window.close)&&exit
:Admin
 
rem 输出空行
echo.
pause

cd /d %~dp0
net stop HenryMes.SignalR.Hosting
HenryMes.SignalR.Hosting uninstall
 
pause

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/340139.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

RibbitMQ 入门到应用 ( 二 ) 安装

3.安装基本操作 3.1.下载安装 3.1.1.官网 下载地址 https://rabbitmq.com/download.html 与Erlang语言对应版本 https://rabbitmq.com/which-erlang.html 3.1.2.安装 Erlang 在确定了RabbitMQ版本号后&#xff0c;先下载安装Erlang环境 Erlang下载链接 https://packa…

vue通用后台管理系统

用到的js库 遇到的问题 vuex和 localStorage区别 vuex在内存中&#xff0c;localStorage存在本地localStorage只能存储字符串类型数据&#xff0c;存储对象需要JSON.stringify() 和 parse()…读取内存比读取硬盘速度要快刷新页面vuex数据丢失&#xff0c;localStorage不会vuex…

图的基本介绍和表示方式

图的基本介绍 为什么要有图这个基本数据结构? 我们还学习过线性表&#xff08;数组、队列、链表和栈&#xff09;和树&#xff0c;但是我们可以发现&#xff0c;线性表局限于一个直接前驱&#xff08;就是只能有唯一一个前面的结点&#xff09;和一个直接后继的&#xff08;…

Python+dilb 简单实现人脸检测(附代码)

如果dilb和face_recognition第三方包安装失败&#xff0c;请移步到Python 解决dilb和face_recognition第三方包安装失败_水w的博客-CSDN博客 目录 一、环境配置 二、前期知识 1 读取图片 2 灰度转换 解决报错cv2.error: OpenCV(4.7.0) D:\a\opencv-python\opencv-python\o…

【前端vue2面试题】2023前端最新版vue模块,高频17问(上)

&#x1f973;博 主&#xff1a;初映CY的前说(前端领域) &#x1f31e;个人信条&#xff1a;想要变成得到&#xff0c;中间还有做到&#xff01; &#x1f918;本文核心&#xff1a;博主收集的关于vue2面试题(上) 目录 vue2面试题 1、$route 和 $router的区别 2、一个…

【Spark分布式内存计算框架——Spark Core】9. Spark 内核调度(上)

第八章 Spark 内核调度 Spark的核心是根据RDD来实现的&#xff0c;Spark Scheduler则为Spark核心实现的重要一环&#xff0c;其作用就是任务调度。Spark的任务调度就是如何组织任务去处理RDD中每个分区的数据&#xff0c;根据RDD的依赖关系构建DAG&#xff0c;基于DAG划分Stag…

PVE虚拟机篇-PVE虚拟机安装

安装包下载 前往Proxmox Virtual Environment下载ISO镜像文件&#xff0c;可以直接选择最新版本。 但是如果没有海外线路一般下载十分缓慢&#xff0c;甚至下不动&#xff0c;所以可以去中科大镜像源&#xff1a;中科大Proxmox Virtual Environment 制作U盘启动盘 可以使用…

SpringMVC:拦截器(12)

拦截器1. 拦截器概念2. 拦截器入门案例2.1 环境准备2.2 拦截器开发步骤1: 创建拦截器类步骤2: 配置拦截器类步骤3: SpringMVC添加SpringMvcSupport包扫描和interceptor包扫描步骤4: 简化SpringMvcSupport的编写5 测试3. 拦截器参数解析&#xff08;了解&#xff09;3.1 前置处理…

字节一面:在浏览器地址栏输入一个 URL 后回车,背后发生了什么?

近段时间&#xff0c;有小伙伴面试字节&#xff0c;说遇到一个面试题&#xff1a; 在浏览器地址栏输入一个 URL 后回车&#xff0c;背后发生了什么&#xff1f; 这里尼恩给大家做一下系统化、体系化的梳理&#xff0c;使得大家可以充分展示一下大家雄厚的 “技术肌肉”&#xf…

c# 自定义隐式转换与运算符重载

用户定义的显式和隐式转换运算符 参考代码 用户定义的显式和隐式转换运算符 - 提供对不同类型的转换 | Microsoft Learn 代码例程 using System;public readonly struct Digit {private readonly byte digit;public Digit(byte digit){if (digit > 9){throw new Argumen…

电子科技大学操作系统期末复习笔记(一):操作系统概述

目录 前言 操作系统概述 操作系统的目标与功能 操作系统的定义 目标 功能 操作系统的历史 单用户系统 简单批处理系统 多道批处理系统 分时系统 个人电脑 → 分布式系统 → 互联网时代 → 移动计算时代 → ...... 实时系统 操作系统的基本特征 并发 共享 虚拟…

第一章 PDF语法

第一章 PDF语法PDF ObjectsNull ObjectsBoolean ObjectsNumeric ObjectsName ObjectsString ObjectsArray ObjectsDictionary ObjectsName treesNumber treesStream ObjectsDirect versus Indirect ObjectsFile StructureWhite-SpaceThe Four Sections of a PDFHeaderTrailerBo…

Sa-Token实现分布式登录鉴权(Redis集成 前后端分离)

文章目录1. Sa-Token 介绍2. 登录认证2.1 登录与注销2.2 会话查询2.3 Token 查询3. 权限认证3.1 获取当前账号权限码集合3.2 权限校验3.3 角色校验4. 前后台分离&#xff08;无Cookie模式&#xff09;5. Sa-Token 集成 Redis6. SpringBoot 集成 Sa-Token6.1 创建项目6.2 添加依…

Python 高级编程之网络编程 Socket(六)

文章目录一、概述二、Python socket 模块1&#xff09;Socket 类型1、创建 TCP Socket2、创建 UDP Socket2&#xff09;Socket 函数1、服务端socket函数2、客户端socket函数3、公共socket函数三、单工&#xff0c;半双工以及全双工通信方式的区别四、单工&#xff0c;半双工以及…

MWORKS--MoHub介绍

MWORKS--MoHub介绍1 介绍1.1 简介1.2 功能特征2 快速上手2.1 进入工作台2.2 新建仓库并进入建模空间2.3 建模进入建模工作空间加载模型库新建模型2.4 仿真2.5 后处理曲线、动画2.6 查看模型信息3 使用手册参考1 介绍 1.1 简介 MWORKS.MoHub 支持工业知识、经验、数据的模型化…

电子技术——BJT差分输入对

电子技术——BJT差分输入对 本节我们来讨论BJT差分输入对。 共模输入 下图是BJT差分输入对的基本原理图&#xff1a; 首先我们考虑两端输入共模信号 VCMV_{CM}VCM​ &#xff1a; 此时 vB1vB2VCMv_{B1} v_{B2} V_{CM}vB1​vB2​VCM​ 因为电路的对称结构&#xff0c;所以 i…

LeetCode 热题 C++ 114. 二叉树展开为链表

给你二叉树的根结点 root &#xff0c;请你将它展开为一个单链表&#xff1a; 展开后的单链表应该同样使用 TreeNode &#xff0c;其中 right 子指针指向链表中下一个结点&#xff0c;而左子指针始终为 null 。展开后的单链表应该与二叉树 先序遍历 顺序相同。 示例 1&#xf…

详细解读503服务不可用的错误以及如何解决503服务不可用

文章目录1. 问题引言2. 什么是503服务不可用错误3 尝试解决问题3.1 重新加载页面3.2 检查该站点是否为其他人关闭3.3 重新启动设备3.3 联系网站4. 其他解决问的方法1. 问题引言 你以前遇到过错误503吗&#xff1f; 例如&#xff0c;您可能会收到消息&#xff0c;如503服务不可…

ccc-Logistic Regression-李宏毅(5)

文章目录Step 1: Function SetStep 2: Goodness of a FunctionStep 3: Find the best functionWhy not Logistic Regression Square ErrorDiscriminative v.s. GenerativeMulti-class Classification(3 Class)Limitation of Logistic RegressionCascading logistic regression…

sklearn降维算法1 - 降维思想与PCA实现

目录1、概述1.1 维度概念2、PCA与SVD2.1 降维实现2.2 重要参数n_components2.2.1 案例&#xff1a;高维数据的可视化2.2.2 最大似然估计自选超参数2.2.3 按信息量占比选超参数1、概述 1.1 维度概念 shape返回的结果&#xff0c;几维几个方括号嵌套 特征矩阵特指二维的 一般来…