【小5聊】C# SignalR之主动推送信息的简单解决方案

本文详细介绍了使用SignalR实现客户端与服务端之间的主动推送消息功能。通过实例展示了如何在Form窗体下应用SignalR,包括服务端代码设置、客户端连接与监听、定时器消息推送及服务端启动配置。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

注:Core下的SignalR实现原理一样,只是写法不一样,在后续文章中再写一遍来做对比,敬请期待下下!

主动推送消息的场景使用:页面登录后台管理系统,最新订单信息的通知等

1、描述

Form窗体下使用SignalR,通过页面客户端触发服务端,在服务端触发一次后,开启一个定时器,让定时器给加入到连接列表的客户端主动推送消息

2、主动推送,客户端和服务端相互判定

1)服务端关键代码

using Microsoft.AspNet.SignalR;
using System;
using System.Collections.Generic;
using System.IO;
using System.Threading.Tasks;
using System.Timers;

namespace SignalRForm
{
    public class TestHub:Hub
    {
        #region 初始化 - 构造函数/以及服务
        public static bool isInite = false;
        public static List<string> list = null;
        Timer timer = null;

        static TestHub()
        {
            list = new List<string>();

            if (isInite)
            {
                Console.WriteLine("再次运行!");
            }
            
            //客户端第一个调用服务端方法时,执行一次构造函数
            Console.WriteLine("TestHub_Init");
        }

        public void MyTimer(object sender, ElapsedEventArgs e)
        {
            timer.Stop();

            Clients.Clients(list).getNowTime(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"));
            //Clients.Client("cbffa36b-24a5-41e1-a5d9-1be4127c5cb4").getNowTime(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"));

            timer.Start();
        }

        public void initServer(string userGuid)
        {
            if (!isInite)
            {
                //读取txt文档内容
                string str;
                StreamReader sr = new StreamReader("d://template.txt", false);
                str = sr.ReadLine().ToString();
                sr.Close();

                if (!string.IsNullOrEmpty(str))
                {
                    string[] arr = str.Split(',');
                    foreach (string item in arr)
                    {
                        if (!string.IsNullOrEmpty(item))
                        {
                            list.Add(item);
                        }
                    }
                }

                isInite = true;
                timer = new Timer(1000); //每隔500毫米执行一次方法
                timer.Enabled = true;
                timer.Elapsed += MyTimer;
                if (list.Count > 0)
                    Clients.Clients(list).getNowTime(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"));

                //Task t = new Task(() =>
                //{
                //    //Console.WriteLine("任务开始工作……");
                //    //模拟工作过程
                //    //System.Threading.Thread.Sleep(1000);
                //});
                //t.Start();
                //t.ContinueWith((task) =>
                //{
                //    while (true)
                //    {
                //        System.Threading.Thread.Sleep(100);
                //        if (list.Count > 0)
                //            Clients.Clients(list).getNowTime(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"));
                //    }
                //    //Console.WriteLine("任务完成,完成时候的状态为:");
                //    //Console.WriteLine("IsCanceled={0}\tIsCompleted={1}\tIsFaulted={2}", task.IsCanceled, task.IsCompleted, task.IsFaulted);
                //});
            }
        }
        #endregion

        #region 客户端连接 - 服务器端监听
        public override Task OnConnected()
        {
            //将内容写入到txt文档
            StreamWriter sw = new StreamWriter("d://template.txt", true);
            sw.WriteLine(Context.ConnectionId + ",");
            sw.Close();

            list.Add(Context.ConnectionId);

            return base.OnConnected();
        }
        #endregion

        #region 客户端断开连接
        public override Task OnDisconnected(bool stopCalled)
        {
            //将内容写入到txt文档
            StreamWriter sw = new StreamWriter("d://templateout.txt", true);
            sw.WriteLine(Context.ConnectionId + ",");
            sw.Close();

            return base.OnDisconnected(stopCalled);
        }
        #endregion

        #region 客户端重新连接
        public override Task OnReconnected()
        {
            //将内容写入到txt文档
            StreamWriter sw = new StreamWriter("d://template.txt", true);
            sw.WriteLine("再次连接!" + isInite);
            sw.Close();

            return base.OnConnected();
        }
        #endregion
    }
}

2)客户端关键代码

<!DOCTYPE html>
<html style="height: 100%">
<head>
    <meta charset="utf-8">
    <meta name="viewport" content="width=device-width, initial-scale=1">
    <title>测试SignalR主动推送</title>
    <script type="text/javascript" src="~/jQuery/jquery-2.0.0.min.js"></script>
    <script type="text/javascript" src="~/jQuery/jquery.signalR-2.2.0.min.js"></script>
    @*<script src="http://localhost:8686/signalr/hubs"></script>*@
    <script src="http://192.168.1.25:8686/signalr/hubs"></script>

    <style type="text/css">

        body, html {
            margin: 0;
            padding: 0;
        }

        /*-----clear-----*/
        .clear {
            *zoom: 1;
        }

            .clear:after {
                content: '';
                display: table;
                clear: both;
            }
    </style>
</head>
<body>

    <div id="nowTime"></div>
    <input id="runTime" type="hidden" />
</body>
</html>

<script type="text/javascript">

    var runTime = "";
    test();
    function test() {

        $(function () {

            try {
                //$.connection.hub.url = "http://localhost:8686/signalr"; //连接到SignalR服务端
                $.connection.hub.url = "http://192.168.1.25:8686/signalr"; //连接到SignalR服务端

                $.connection.hub.start().done(function (data) {

                    //服务端方法的回调函数:OnConnected、OnDisconnected、OnReconnected?
                    //在此方法初始化
                    chat.server.initServer("companyid");
                }).fail(function () {
                    console.log("实时消息服务连接失败!");
                });

                var chat = $.connection.testHub; //继承了Hub的类,用于客户端调用服务端类的方法

                //监听服务端方法判断是否初始化服务成功
                //chat.server.getNowTime(); //Connection must be started before data can be sent. Call .start() before .send()
                chat.client.getNowTime = function (data) {
                    
                    $("#runTime").val(data);

                    $("#nowTime").html(data);
                }
            }
            catch (ex) {
                console.log(ex);
            }
        });
    }

    setInterval(function () {
        
        if (runTime == $("#runTime").val()) {
            test();
        }
        else {
            runTime = $("#runTime").val();
        }

    }, 10 * 1000);
</script>

3)补上服务端启动代码

using System;
using Microsoft.AspNet.SignalR;
using Owin;
using Microsoft.Owin.Cors;
using Microsoft.Owin.Hosting;

namespace SignalRForm
{
    /*
     Microsoft.Owin.Cors -version=4.2.2
     Microsoft.Owin.Host.HttpListener -version=4.2.2
     */

    class Program
    {
        static void Main(string[] args)
        {
            string url = "http://192.168.10.62:8686"; //"http://*:8899";//IP和端口
            using (WebApp.Start(url)) //netsh http add urlacl url=http://www.test.com user=Everyone
            {
                Console.Read();
            }
        }
    }

    public class Startup
    {
        public void Configuration(IAppBuilder app)
        {
            var config = new HubConfiguration
            {
                EnableJSONP = true,
                EnableDetailedErrors = true,
                EnableJavaScriptProxies = true
            };
            app.UseCors(CorsOptions.AllowAll);
            app.MapSignalR(config);
        }
    }
}

4)服务端启动常见情况

找不到给定输入的服务器工厂:Microsoft.Owin.Host.HttpListener“

原因:没有添加到上面提示的dll

解决方法:直接Nuget搜索添加-Microsoft.Owin.Host.HttpListener

 如果启动成功,那么就会直接到using里面

 5)可能还需要如下操作

1、可以先删除再添加【注意:最后一定要加斜杠】
netsh http delete urlacl url=http://localhost:8686/ 
2、添加网址【注意:最后一定要加斜杠】
netsh http add urlacl url=http://localhost:8686/ user=Everyone

3、查看所有使用到的端口

netstat -ano

6)添加URL保留项失败,错误:87

7)确保有Startup类

这个类可以创建单独的类文件,也可以放到Program.cs同一个命名空间下,不能放到类里面

否则会提示如下错误

8)特别注意 - 拒绝访问

最后面需要加一个斜杠

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

全栈小5

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值