[Asp.net 開發系列之SignalR篇]專題四:使用SignalR實現發送圖片

来源:http://www.cnblogs.com/zhili/archive/2016/04/10/SignalRSendImage.html
-Advertisement-
Play Games

一、引言 在前一篇博文已經介紹瞭如何使用SignalR來實現聊天室的功能,在這篇文章中,將實現如何使用SignalR來實現發送圖片的功能。 二、實現發送圖片的思路 我還是按照之前的方式來講述這篇文章,首先,讓我們來理清下實現發送圖片功能的思路。 圖片的顯示,除了直接指定圖片的路徑外(這種實現方式也稱 ...


一、引言

  在前一篇博文已經介紹瞭如何使用SignalR來實現聊天室的功能,在這篇文章中,將實現如何使用SignalR來實現發送圖片的功能。

二、實現發送圖片的思路

  我還是按照之前的方式來講述這篇文章,首先,讓我們來理清下實現發送圖片功能的思路。

  圖片的顯示,除了直接指定圖片的路徑外(這種實現方式也稱為:http URI schema),還可以通過Data Uri Schema的方式來顯示圖片。這種方式允許在網頁里以字元串形式直接內嵌圖片。形式如下所示:

<img src=" 
7ljmRAAAAGElEQVQIW2P4DwcMDAxAfBvMAhEQMYgcACEHG8ELxtbPAAAAAElFTkSuQmCC" />

  上面代碼的方式就是Data Url Schema方式來顯示圖片。關於Data Uri Schema的優缺點有:

  • 優點:

  可以減少Http請求,因為如果你使用http Uri Schema去指定圖片地址的話,這樣客戶端對每個圖片都需要發出Http請求,通過使用Data Uri的方式可以節省帶寬和Http請求

  缺點:

  1.   IE8以上的版本才支持,且限制大小不可超過32KB。
  2.   另外Base64的內容會將圖片的內容變大33%,但可以通過服務端啟用GZIP壓縮來減少增大內容。儘管這樣,由於發送Http請求會附加很多額外的信息(如Http Header等),這樣累計下來一般內容大小還是大於使用Base64編碼所增加的內容。

  關於更多Data Uri的介紹可以參考百度百科:Data URI

  因為SignalR是基於文本方式的傳輸,所以要實現圖片的發送。

  1. 只能通過發送圖片的Base64編碼的字元串到SignalR伺服器,
  2. 然後伺服器再將該Base64字元串推送到需要接收圖片的客戶端,
  3. 客戶端再使用Data Uri的方式將圖片顯示在頁面上,從而完成圖片的傳輸。

  當然你也可以像Jabbr(一個使用SignalR實現即時聊天的開源項目)那樣將圖片上傳到Azure Bob Table中,然後再將Blob 的Uri 返回所有客戶端來顯示圖片。其實這樣的實現方式和我們這裡實現類似,客戶端可以通過blob的Uri來讀取到圖片來顯示。總之實現思路就是將圖片二進位文件的內容間接轉換成文本的形式傳輸。

三、使用SignalR發送圖片的實現代碼

  在具體實現之前,這裡需要介紹一個文件上傳插件——boostrap-fileinput。該插件用來提供圖片的預覽功能。關於插件的具體使用可以參考github站點或本文章的實現代碼。

  1. 第一步還是實現我們的集線器

 

public class ChatHub : Hub
    {
        /// <summary>
        /// 供客戶端調用的伺服器端代碼
        /// </summary>
        /// <param name="name"></param>
        /// <param name="message"></param>
        public void Send(string name,string message)
        {
            // 調用所有客戶端的sendMessage方法
            Clients.All.sendMessage(name, message);
        }

        // 發送圖片
        public void SendImage(string name,IEnumerable<ImageData> images)
        {
            foreach (var item in images ?? Enumerable.Empty<ImageData>())
            {
                if(String.IsNullOrEmpty(item.Image)) continue;
                Clients.All.receiveImage(name, item.Image); // 調用客戶端receiveImage方法將圖片進行顯示
            }
        }

        /// <summary>
        /// 客戶端連接的時候調用
        /// </summary>
        /// <returns></returns>
        public override Task OnConnected()
        {
            Trace.WriteLine("客戶端連接成功");
            return base.OnConnected();
        }
    }

 

  2. HomeController的實現代碼,主要為每個客戶端生成隨機的用戶名,再將用戶名存入Session中。

 

public class HomeController : Controller
    {
        private static readonly char[] Constant =
        {
            '0', '1', '2', '3', '4', '5', '6', '7', '8', '9',
            'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v',
            'w', 'x', 'y', 'z',
            'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V',
            'W', 'X', 'Y', 'Z'
        };

        // GET: Home
        public ActionResult Index()
        {
            Session["username"] = GenerateRandomName(4);
            return View();
        }

        /// <summary>
        /// 產生隨機用戶名函數
        /// </summary>
        /// <param name="length">用戶名長度</param>
        /// <returns></returns>
        private static string GenerateRandomName(int length)
        {
            var newRandom = new System.Text.StringBuilder(62);
            var rd = new Random(DateTime.Now.Millisecond);
            for (var i = 0; i < length; i++)
            {
                newRandom.Append(Constant[rd.Next(62)]);
            }

            return newRandom.ToString();
        }
    }

  3. 接下來就是實現前端頁面了。

<html>
<head>
    <meta name="viewport" content="width=device-width" />
    <title>使用SignalR實現發送圖片</title>
    <link href="/Content/bootstrap.min.css" rel="stylesheet">
    <link href="/Content/bootstrap-fileinput/css/fileinput.min.css" media="all" rel="stylesheet" type="text/css" />
</head>
<body>
    <div class="container">
        <div>用戶名:<p id="username"></p></div>
        <input type="text" id="message" />
        <br/>
        <br />
        <input id="fileinput" type="file">
        <br />
        <input type="button" id="sendmessage" value="Send" />
        <input type="hidden" id="displayname" />
        <ul id="discussion"></ul>
    </div>
    <script type="text/javascript" src="~/Scripts/jquery-2.2.2.min.js"></script>
    <script src="~/Scripts/jquery.signalR-2.2.0.min.js"></script>
    <script src="~/signalr/hubs"></script>
    <script src="/Scripts/fileinput.js" type="text/javascript"></script>
    <script src="/Scripts/bootstrap.min.js" type="text/javascript"></script>
    <script>
        $(function () {
            var userName = '@Session["username"]';
            $('#username').html(userName);
            // 引用自動生成的集線器代理
            var chat = $.connection.chatHub;
            // 定義伺服器端調用的客戶端sendMessage來顯示新消息

            chat.client.sendMessage = function (name, message) {
                // 向頁面添加消息
                $('#discussion').append('<li><strong>' + htmlEncode(name)
                    + '</strong>: ' + htmlEncode(message) + '</li>');
            };

            chat.client.receiveImage = function (name, base64) {
                // 向頁面添加消息
                $('#discussion').append('<image class = "file-preview-image" style="width:auto;height:100px;" src=' + base64
                    + '/>');
            };

            // 設置焦點到輸入框
            $('#message').focus();
            // 開始連接伺服器
            $.connection.hub.start().done(function () {
                $('#sendmessage').click(function () {
                    // 調用伺服器端集線器的Send方法
                    chat.server.send(userName, $('#message').val());
                    // 清空輸入框信息並獲取焦點
                    $('#message').val('').focus();
                });
            });

            $("#fileinput").fileinput({
                allowedFileExtensions: ["jpg", "png", "gif", "jpeg"],
                maxImageWidth: 700,
                maxImageHeight: 700,
                resizePreference: 'height',
                maxFileCount: 1,
                resizeImage: true
            });

            $("#fileinput").on('fileloaded', function (event, file, previewId, index, reader) {
                var readers = new FileReader();
                readers.onloadend = function () {
                    $(".file-preview-image").attr('src', readers.result);
                };
                readers.readAsDataURL(file);
            });

            $('#sendmessage').click(function() {
                var imagesJson = $('.file-preview-image').map(function() {
                    var $this = $(this);
                    return {
                        image: $this.attr('src'),
                        filename: $this.attr('data-filename')
                    };
                }).toArray();

                chat.server.sendImage(userName, imagesJson);
            });
        });

    // 為顯示的消息進行Html編碼
    function htmlEncode(value) {
        var encodedValue = $('<div />').text(value).html();
        return encodedValue;
    }
    </script>
    
</body>
</html>

四、運行效果

  經過上面的三步,使用SignalR發送圖片的功能就已經可以運作了。接下來讓我們一起看看具體的運行效果到底如何。

五、總結

  到這裡,本文的所有內容的介紹就結束了,接下來的將介紹如何使用Html5 Notification API 來實現有新消息的提醒功能。

  本文所有源碼下載地址:SignalRSendFile

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 介紹 本篇文章主要從查看MySQL的啟動命令的代碼來詳細瞭解MySQL的啟動過程,內容多為概念知識;理解MySQL的啟動原理對熟悉MySQL至關重要,啟動mysql服務有三種方式分別是:mysql.sever,mysqld,mysqld_safe。 my.cnf mysql.server 預設的my ...
  • Knowledge Dependence:閱讀文本前,你需要瞭解基本的關係型資料庫與非關係型(NoSQL)資料庫的概念和區別,以及 MongoDB(Mongoose)的簡單實踐。 ​ 這兩三年來,伴隨著大數據(Big Data)的空前火熱,無論是在工程界還是科研界,非關係型資料庫(NoSQL)都已經 ...
  • 實驗環境:Ubnuntu 64位(推薦使用14.04)+Xshell 阿裡雲現在提供的雲伺服器很好用的,用來編譯內核性能也不錯。本文介紹最基本的內核編譯方法,為了方便,所有操作均在root用戶下進行。 如果不是root用戶可以使用su命令切換到root用戶。 註:使用xshell的時候最好把這一項給 ...
  • In order to provide an environment for network experiments in the future, I use VirutalBox to create some hosts. My real host is Ubuntu 14.04 LTS. And ...
  • 1.安裝MySQL需要的依賴的包和編譯軟體 (1)安裝MySQL需要的依賴包 安裝MySQL之前,最好先安裝MySQL需要的依賴包,不然後面會出現報錯,還得回來安裝MySQL的依賴包。 [root@Mysql-server tools]# yum install -y ncurses-devel l ...
  • 首先從www.zabbix.com下載rpm包: 接下來我要配置一臺zabbix server,自己監控自己即使服務端又是客戶端,zabbix web gui和zabbix資料庫都放在同一臺主機上,除了第4,5個包不裝外,其他的包都給裝上,為瞭解決某些包的依賴關係還需事先安裝yum -y insta ...
  • 2016-04-07 張超《Linux內核分析》MOOC課程http://mooc.study.163.com/course/USTC-1000029000#/info 一、理解編譯鏈接的過程和ELF可執行文件格式 我給出了一個例子: 第一步:先編輯一個hello.c,如下 vi hello.c 1 ...
  • 在System.Net.Http中,提供了使用Http與遠程伺服器通訊的httpClient,但是裡面都是非同步方法,有時候我們並不需要使用非同步操作。這個時候可以使用如下的方式來進行同步調用。 不阻塞主線程的非同步操作,可以參考:HttpClient介紹。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...