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

注意下文里面的 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

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
前端 WebSocket 线连是一种处理 WebSocket 连接异常开的方法,以确保 WebSocket 连接的可靠性。以下是一个基本的线连示例: ```javascript let websocket = null; let lockReconnect = false; // 避免复连接 function createWebSocket() { try { websocket = new WebSocket('ws://localhost:8080'); websocket.onopen = function () { console.log('WebSocket 连接功'); }; websocket.onclose = function () { console.log('WebSocket 连接关闭'); reconnect(); }; websocket.onerror = function () { console.log('WebSocket 连接出错'); reconnect(); }; websocket.onmessage = function (event) { console.log('收到消息:', event.data); }; } catch (e) { console.log('WebSocket 连接异常:', e.message); reconnect(); } } function reconnect() { if (lockReconnect) { return; } lockReconnect = true; // 连间隔递增 setTimeout(function () { createWebSocket(); lockReconnect = false; }, 5000); } createWebSocket(); ``` 以上代码中,`createWebSocket` 函数用于创建 WebSocket 连接,捕获连接功、关闭、出错和收到消息的事件。如果连接关闭或出错,则调用 `reconnect` 函数进行连。`reconnect` 函数会检查是否已经在连中,如果是则不进行连,否则等待一段时间后新创建 WebSocket 连接。 需要注意的是,WebSocket 线连的实现也需要考虑到服务器的实现,例如服务器可能会限制客户端的连接频率或连接数量。因此,更复杂的实现方式可能需要考虑这些因素。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值