使用SignalR实现比特币价格实时刷新

ASP.NET SignalR是微软支持的一个运行在 Dot NET 平台上的 HTML Websocket 框架。它出现的主要目的是实现服务器主动推送(Push)消息到客户端页面,这样客户端就不必重新发送请求或使用轮询技术来获取消息。SignalR提供用于连接管理(例如,连接和断开的事件),分组连接和授权、发送、监听等一些简单易用的API,目前API支持JavaScript和C#两个平台,我们可以在NuGet上面轻松获取这些SDK。

image

创建一个Startup.cs类在系统启动时注册SignalR。

using Owin;
using Microsoft.Owin;
[assembly: OwinStartup(typeof(SignalRDemo.Startup))]
namespace SignalRDemo
{
    public partial class Startup
    {
        public void Configuration(IAppBuilder app)
        {
            app.MapSignalR();
        }
    }
}

Hub提供与ASP.NET SignalR通信的连接,使您可以从一台服务器连接客户端,并从客户端到服务器进行远程过程调用(RPC)。在服务器代码中,你定义可以被客户端调用方法,并调用该客户端上运行的方法。在客户端代码中,你定义可以从服务器上调用的方法,并且调用服务器上运行的方法。

实现比特币价格实时刷新,则只需要服务器到客户端的调用,我在下述StockHub中,开了一个Timer,每间隔9s去OKCoin获取交易行情,并将数据返回(refresh)给所有用户(Clients),OKCoin平台的交易行情API返回格式为:{"ticker":{"buy":"4003.95","high":"4046.0","last":"4003.95","low":"3857.0","sell":"4007.06","vol":"50344.8216"}},包括了买卖价格、最高、最低、交易量等数据。

using Microsoft.AspNet.SignalR;
using Microsoft.AspNet.SignalR.Hubs;
using System.IO;
using System.Net;
using System.Threading;

namespace SignalRDemo.Hubs
{
    [HubName("stock")]
    public class StockHub : Hub
    {
        private const string BTC_CNY = "https://www.okcoin.com/api/ticker.do";

        private static Timer _timer;

        static StockHub()
        {
            _timer = new Timer(Refresh, null, 3000, 9000);
        }

        private static void Refresh(object state)
        {
            GlobalHost.ConnectionManager.GetHubContext<StockHub>().Clients.All.refresh(GetData(BTC_CNY).Replace("ticker", "BTC"));
        }

        private static string GetData(string url)
        {
            HttpWebRequest request = (HttpWebRequest)HttpWebRequest.Create(url);
            using (HttpWebResponse response = (HttpWebResponse)request.GetResponse())
            {
                using (StreamReader reader = new StreamReader(response.GetResponseStream()))
                {
                    return reader.ReadToEnd();
                }
            }
        }
    }
}

修改Default.aspx,引入jQuery和SignalR插件,创建一个refresh方法供StockHub回调使用,该方法在接收到到数据后插入表格(为演示价格变化过程,记录每一次更新,而不是修改价格)。

<asp:Content runat="server" ID="BodyContent" ContentPlaceHolderID="MainContent">
    <!--Script references. -->
    <script src="Scripts/jquery-1.8.2.min.js"></script>
    <script src="Scripts/jquery.signalR-2.0.2.min.js"></script>
    <script src="signalr/hubs"></script>
    <script type="text/javascript">
        $(function () {
            $.connection.stock.client.refresh = function (data) {
                var obj = eval('(' + data + ')');
                var newRow = "<tr><td>" + obj.BTC.last + "</td><td>" + obj.BTC.buy + "</td><td>" + obj.BTC.sell + "</td><td>" + obj.BTC.high + "</td><td>" + obj.BTC.low + "</td><td>" + obj.BTC.vol + "</td></tr>";
                //if ($("#stock tr").length >= 1) {
                //    $("#stock tr:gt(0)").remove();
                //}
                $("#stock tr:last").after(newRow);
            };
            $.connection.hub.start().done();
        });
    </script>
    <table id="stock" style="width: 100%">
        <tr>
            <th>最近成交价</th>
            <th>买一价</th>
            <th>卖一价</th>
            <th>最高</th>
            <th>最低</th>
            <th>成交量</th>
        </tr>
    </table>
</asp:Content>

自此,所有代码实现完毕,使用非常简单,CTRL+F5启动项目,最终效果如下:

image

再来开发一个简单的WinForm客户端,实现WinForm客户端需要引入Microsoft ASP.NET SignalR .NET Client库。

image

与JavaScript类似,依据地址声明一个HubConnection,在由其创建一个Client Hub Proxy,由Proxy注册refresh时间,供服务端回调。

namespace SignalR.Client
{
    public partial class FrmMain2 : Form
    {
        private readonly HubConnection _hubConnection = new HubConnection("http://localhost:2041/signalr");
        private readonly IHubProxy _buildingApiHubProxy;

        public FrmMain2()
        {
            InitializeComponent();
            _buildingApiHubProxy = _hubConnection.CreateHubProxy("stock");
            _buildingApiHubProxy.On("refresh",
                data => this.refresh(data)
            );
            _hubConnection.Start().Wait();
        }

        private void refresh(string data)
        {
            this.lblMessage.Text = data;
        }
    }
}

image

posted @ 2014-02-16 13:21  冯翔  阅读(2394)  评论(0编辑  收藏  举报