基於XMPP協議的Android IM研究一

之前在做基於XMPP協議的Android IM項目的過程中遇到了不少問題,由於國內這方面的資料相對比較少,而且不夠全面,經過不斷的學習和請教,項目有了一點進展,下面分享一下在項目的過程中遇到的問題和解決辦法。

首先,聲明XMPP連接:

 

  1. Java代碼  
  2.       
  3. public static final ConnectionConfiguration connConfig = new ConnectionConfiguration("169.254.50.19"5222"http://06peng.com");        
  4. public static XMPPConnection connection;    

ConnectionConfiguration類的三個參數分別爲ip地址,端口號,域名。

建立XMPP連接

 

  1. Java代碼  
  2.       
  3. static {                  
  4.     connConfig.setSASLAuthenticationEnabled(true);//不使用SASL驗證,設置爲false          
  5.     connection = new XMPPConnection(connConfig);          
  6.     connection.connect();//連接            
  7.     connection.login("06peng""111111");//登陸            
  8. }      

這樣就能夠和服務器連接了,運行代碼,結果悲劇。報下面的錯誤:

 

  1. Java代碼  
  2.       
  3. java.security.KeyStoreException: KeyStore jks implementation not found  

還有:

 

  1. Java代碼  
  2.       
  3. SASL authentication failed using mechanism PLAIN:     

驗證不通過,可是我已經設置了不使用SASL驗證了,居然還出現這樣的問題,於是看官方網站的例子,發現前面加了下面的配置:

 

  1. Java代碼  
  2.       
  3. ConnectionConfiguration connectionConfig = new ConnectionConfiguration(host, 5222"");           
  4. connectionConfig .setTruststorePath("/system/etc/security/cacerts.bks");           
  5. connectionConfig.setTruststorePassword("changeit");           
  6. connectionConfig.setTruststoreType("bks");     

好吧,加上這樣的驗證代碼應該是沒問題了,哪知道一運行,還是悲劇,繼續報錯:

 

  1. Java代碼  
  2.       
  3. not-authorized(401)         
  4. at org.jivesoftware.smack.NonSASLAuthentication.authenticate(NonSASLAuthentication.java:109)        
  5. at org.jivesoftware.smack.XMPPConnection.login(XMPPConnection.java:239)         
  6. at org.jivesoftware.smack.Connection.login(Connection.java:353)     

鑑權失敗,不過至少不報KeyStore jks implementation not found的錯誤了。於是,繼續看官方例子,尋找相關資料,結果修改代碼如下:

 

  1. Java代碼  
  2.       
  3. connConfig.setReconnectionAllowed(true);        
  4. connConfig.setSecurityMode(ConnectionConfiguration.SecurityMode.enabled);         
  5. connConfig.setSASLAuthenticationEnabled(true);        
  6. connConfig .setTruststorePath("/system/etc/security/cacerts.bks");         
  7. connConfig.setTruststorePassword("changeit");         
  8. connConfig.setTruststoreType("bks");         
  9. connection = new XMPPConnection(connConfig);    

一運行,通過了,心裏那個激動啊。。。

感謝博客園一位叫“沒有代碼”的哥們,給了我一些建議,受到了幫助和啓發。

由於代碼寫的比較複雜,主要就是引用了比較多的類和接口,所以不貼出來了,因爲很麻煩,這裏介紹一些比較簡單的例子:

 

  1. Java代碼  
  2.       
  3. connection.getAccountManager().createAccount(username, password);  //創建一個用戶    

 

  1. Java代碼  
  2.       
  3. roster.removeEntry(roster.getEntry(friendName)); //刪除某個好友    

 

  1. Java代碼  
  2.       
  3. roster.setSubscriptionMode(Roster.SubscriptionMode.accept_all);          
  4. roster.createEntry(user, nickname, friends);//添加一個好友到朋友組上    

 

  1. Java代碼  
  2.       
  3. Collection<RosterEntry> entries = roster.getEntries();          
  4. for(Iterator<RosterEntry> entry  = entries .iterator();entry .hasNext();){          
  5.     RosterEntry re = entry .next();  //獲取所有好友                 
  6. }      

相關屬性的介紹:

 

1、ConnectionConfiguration
作爲用於與XMPP服務建立連接的配置。它能配置;連接是否使用TLS,SASL加密。
包含內嵌類:ConnectionConfiguration.SecurityMode

2、XMPPConnection.
XMPPConnection這個類用來連接XMPP服務.
可以使用connect()方法建立與服務器的連接。disconnect()方法斷開與服務器的連接.
在創建連接前可以使用XMPPConnection.DEBUG_ENABLED = true; 使開發過程中可以彈出一個GUI窗口,用於顯示我們的連接與發送Packet的信息。

3、ChatManager
用於監控當前所有chat。可以使用createChat(String userJID, MessageListener listener)創建一個聊天。
4、Chat

Chat用於監控兩個用戶間的一系列message。使用addMessageListener(MessageListener listener)當有任何消息到達時將會觸發listener的processMessage(Chat chat, Message message)

 

5、Message
Message用於表示一個消息包(可以用調試工具看到發送包和接收包的具體內容)。它有以下多種類型。
Message.Type.NORMAL -- (默認)文本消息(比如郵件)
Message.Type.CHAT -- 典型的短消息,如QQ聊天的一行一行顯示的消息
Message.Type.GROUP_CHAT -- 羣聊消息
Message.Type.HEADLINE -- 滾動顯示的消息
Message.TYPE.ERROR -- 錯誤的消息
Message有兩個內部類:
Message.Body -- 表示消息體
Message.Type -- 表示消息類型

6、Roster
表示存儲了很多RosterEntry的一個花名冊.爲了易於管理,花名冊的項被分貝到了各個group中.
當建立與XMPP服務的連接後可以使用connection.getRoster()獲取Roster對象。
別的用戶可以使用一個訂閱請求(相當於QQ加好友)嘗試訂閱目的用戶。可以使用枚舉類型Roster.SubscriptionMode的值處理這些請求:
accept_all: 接收所有訂閱請求
reject_all:拒絕所有訂閱請求
manual:  手工處理訂閱請求
創建組:RosterGroup group = roster.createGroup("大學");
向組中添加RosterEntry對象: group.addEntry(entry);

7、RosterEntry
表示Roster(花名冊)中的每條記錄.它包含了用戶的JID,用戶名,或用戶分配的暱稱.

 

8、RosterGroup

表示RosterEntry的組。可以使用addEntry(RosterEntry entry)添加。contains(String user) 判斷某用戶是否在組中.當然removeEntry(RosterEntry entry)就是從組中移除了。getEntries()獲取所有RosterEntry.

9、Presence

表示XMPP狀態的packet。每個presence packet都有一個狀態。

用枚舉類型Presence.Type的值表示:

available -- (默認)用戶空閒狀態

unavailable -- 用戶沒空看消息 

subscribe -- 請求訂閱別人,即請求加對方爲好友 

subscribed -- 統一被別人訂閱,也就是確認被對方加爲好友 

unsubscribe -- 他取消訂閱別人,請求刪除某好友

unsubscribed -- 拒絕被別人訂閱,即拒絕對放的添加請求

error -- 當前狀態packet有錯誤

內嵌兩個枚舉類型:Presence.Mode和Presence.Type.

 

具體使用方法可參考前一篇文章中的Smack中文文檔。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章