IM系統中聊天記錄模塊的設計與實現

  看到不少開發IM系統的朋友都想實現聊天記錄存儲和查詢這一不可或缺的功能,這裏我就把本身前段時間爲傲瑞通(OrayTalk)開發聊天記錄模塊的經驗分享出來,供須要的朋友參考下。sql

一.整體設計

1.存儲位置  數據庫

  從一開始咱們就打算在服務端和客戶端本地同時存儲聊天記錄,並且,在客戶端查看聊天記錄時,能夠選擇是從本地加載、仍是從服務器加載。這樣作的好處有兩個: 服務器

(1)從本地加載聊天記錄速度很是快。框架

(2)當更換了登陸的機器,在任何地方任什麼時候刻均可以從服務器加載完整的聊天記錄,記錄永遠不會丟失。性能

2.存儲方案spa

(1)在服務端存儲聊天記錄固然使用咱們主流的數據庫SqlServer或Mysql等。設計

(2)在客戶端,咱們開始選擇的是使用序列化技術,可是,考慮到當聊天記錄數據量龐大時,序列化方案就不夠靈活了,並且性能也跟不上。因此,最後決定使用輕量級的數據庫Sqlite。code

3.ORM框架對象

  DataRabbit的最新版本增長了對Sqlite的支持,而且對不一樣數據庫的操做API是徹底一致的,因此咱們使用DataRabbit寫了一個小組件來完成聊天記錄的存儲與查詢等數據庫訪問操做。而不管是客戶端仍是服務端的聊天記錄存儲相關的工做,都交給這個組件來完成。blog

 

二.具體實現

1.ChatMessageRecord類

  一條聊天記錄基本上包含了如下幾個內容:發送人、接收人、內容、時間等。而且,咱們想將兩人聊天及羣聊天抽象成同一個模型,因而,聊天記錄的Entity類ChatMessageRecord設計成以下模樣:

    public class ChatMessageRecord
    {     
        #region AutoID
        private long autoID = 0;
        /// <summary>
        /// 自增ID,編號。
        /// </summary>
        public long AutoID
        {
            get { return autoID; }
            set { autoID = value; }
        }
        #endregion

        #region SpeakerID
        private string speakerID = "";
        /// <summary>
        /// 發言人的ID。
        /// </summary>
        public string SpeakerID
        {
            get { return speakerID; }
            set { speakerID = value; }
        }
        #endregion

        #region AudienceID
        private string audienceID = "";
        /// <summary>
        /// 聽衆ID,能夠爲GroupID。
        /// </summary>
        public string AudienceID
        {
            get { return audienceID; }
            set { audienceID = value; }
        }
        #endregion

        #region OccureTime
        private DateTime occureTime = DateTime.Now;
        /// <summary>
        /// 聊天記錄發生的時間。
        /// </summary>
        public DateTime OccureTime
        {
            get { return occureTime; }
            set { occureTime = value; }
        }
        #endregion

        #region ContentRtf
        private string contentRtf = "";
        /// <summary>
        /// 聊天的內容。
        /// </summary>
        public string ContentRtf
        {
            get { return contentRtf; }
            set { contentRtf = value; }
        }
        #endregion

        #region IsGroupChat
        private bool isGroupChat = false;
        /// <summary>
        /// 是否爲羣聊記錄。
        /// </summary>
        public bool IsGroupChat
        {
            get { return isGroupChat; }
            set { isGroupChat = value; }
        }
        #endregion
    } 

    在ChatMessageRecord的定義中,聊天內容字段被設計爲string類型,這是由於在OrayTalk中,聊天內容是富文本RTF格式的。若是須要,能夠更改成byte[]類型,這樣經過自定義的序列化操做就能夠承載更復雜的聊天格式。

  最後一個字段IsGroupChat代表當前記錄是否爲羣聊記錄,若是是羣聊記錄,那麼,AudienceID就不是好友的ID了,而是目標羣組的ID。

  最後請注意:ChatMessageRecord實體與數據庫中的ChatMessageRecord表是徹底映射的關係,這才使得DataRabbit的ORM數據訪問成爲可能。

2.ChatRecordPage類

  當咱們請求聊天記錄時,因爲記錄數量可能很是龐大,因此,採用分頁是不可避免的。咱們用ChatRecordPage來封裝查詢返回的一頁聊天記錄:

根據ChatRecordPage中的TotalCount字段,查詢者能夠知道符合條件的記錄數是多少,如此,就能夠知道總共有多少頁。

3.IChatRecordPersister接口

  不管是客戶端仍是服務端存儲與查詢聊天記錄,咱們都使用同一個接口IChatRecordPersister來進行抽象:

    public interface IChatRecordPersister
    {
        /// <summary>
        /// 插入一條聊天記錄(包括羣聊天記錄)。
        /// </summary>  
        void InsertChatMessageRecord(ChatMessageRecord record);

        
        /// <summary>
        /// 獲取一頁與好友的聊天記錄。
        /// </summary>
        /// <param name="timeScope">日期範圍</param>
        /// <param name="myID">本身的UserID</param>
        /// <param name="friendID">好友的ID</param>
        /// <param name="pageSize">頁大小</param>
        /// <param name="pageIndex">頁索引</param>      
        /// <returns>聊天記錄頁</returns>
        ChatRecordPage GetChatRecordPage(DateTimeScope timeScope, string myID, string friendID, int pageSize, int pageIndex);

        /// <summary>
        /// 獲取一頁羣聊天記錄。
        /// </summary>
        /// <param name="timeScope">日期範圍</param>
        /// <param name="groupID">羣ID</param>
        /// <param name="pageSize">頁大小</param>
        /// <param name="pageIndex">頁索引</param>     
        /// <returns>聊天記錄頁</returns>
        ChatRecordPage GetGroupChatRecordPage(DateTimeScope timeScope, string groupID, int pageSize, int pageIndex);        
    }

(1)插入遊戲記錄時,與好友聊天記錄以及羣聊天記錄使用同一個InsertChatMessageRecord方法便可,只是在構造ChatMessageRecord對象時,字段的賦值有所區別。

(2)使用DataRabbit實現該接口時(如ChatRecordPersister類),經過屬性DataBaseType來控制訪問的是否爲Sqlite數據庫。而後在服務端使用ChatRecordPersister存取聊天記錄時,就將DataBaseType設置爲SqlServer;客戶端則設置爲Sqlite。

 

三.可能的Remoting的接口

  當咱們從服務器加載聊天記錄時,能夠考慮使用Remoting技術來實現,若是是這樣,只須要在服務端把IChatRecordPersister接口暴露爲Remoting服務,而後客戶端使用這一Remoting服務進行聊天記錄查詢。這樣一來,客戶端在切換從本地加載和從服務器加載時,只須要切換IChatRecordPersister爲本地ChatRecordPersister對象的引用或remoting遠程引用便可。整個的代碼實現將會很是簡潔一致。

  到這裏,關於聊天記錄模塊的設計與實現就介紹得差很少了,依照這樣的思路,你們在本身的IM系統中增長聊天記錄的功能應該是很簡單的了。最後,上一張OrayTalk客戶端查詢聊天記錄界面的截圖:

      

 就到這裏了,還有疑問的朋友,請給我留言,我會及時回覆的。

相關文章
相關標籤/搜索