图文无关一起娱乐:

这一篇我们开始写Android端的Smack版主类,后面Android的IM功能都是通过这个帮助类实现的

引用类库:

因为我用的是IDE是Android Studio,所以我通过gradle进行jar包管理了,非常方便,jar包如下:

compile 'org.igniterealtime.smack:smack-core:4.1.4'
compile 'org.igniterealtime.smack:smack-tcp:4.1.4'
compile 'org.igniterealtime.smack:smack-extensions:4.1.4'
compile 'org.igniterealtime.smack:smack-android:4.1.4'
compile 'org.igniterealtime.smack:smack-android-extensions:4.1.4'
compile 'org.igniterealtime.smack:smack-experimental:4.1.4'
compile 'org.igniterealtime.smack:smack-bosh:4.1.4'
compile 'org.igniterealtime.smack:smack-resolver-dnsjava:4.1.4'
compile 'org.igniterealtime.smack:smack-legacy:4.1.4'

加上这些jar包后我们就可以使用Smack库了,大家很容易看到,我这个是4.1.4的包。它的最新包是4.2.0不过是alpha版本,所以我就用这个4.1.4最新正式包了。它和它的一些老版本差别还是比较大的。所以如果你是第一次使用还是和我一样,防止出现一些问题。更新gradle后就可以使用了。

写Smack帮助类:

我这里先定义一个ISmack接口,这种命名方式是我从C#那边带过来的,别纠结。本人很久之前是.NET平台的,.NET平台还是有很多东西是非常不错的。这个接口就是一些约束协议。然后再写了一个Smack 类,它继承并且实现了ISmack接口。

ISmack接口代码如下:

/**
*
* @备注:samck操作接口协议
* @作者:高露
* @时间:2015-10-24
* @QQ:408365330
*
*/
public interface ISmack { /**
* 登录
* @param name 账号
* @param pwd 密码
* @return
*/
public boolean login(String name,String pwd); /**
* 消息发送
* @param from 谁发送
* @param to 发送给谁
* @param content 发送内容
*/
public void sendMessage(String from,String to,String content) throws SmackException.NotConnectedException; /**
*添加好友
* @param user 用户jid
* @param name 添加好友备注名称
* @param groups 好友添加到的分组,可以过个组
*/
public void addRosterItem(String user,String name,String[] groups) throws Exception; /**
* 删除好友
* @param user 好友jid
*/
public void removeRoster(String user); }

可以通过这份协议知道我这里定义了 登录,消息发送,添加好友,删除好友 这些方法,后面还是会添加的,每个方法备注还是比较详细的,看看就知道是什么意思了。

然后我们再看看和分析Smack代码

Smack代码分析:

静态构造函数:

首先定义了一个java静态构造函数

static {
if (connection == null) {
XMPPTCPConnectionConfiguration configuration = XMPPTCPConnectionConfiguration.builder()
.setConnectTimeout(Constant.IM_TIMEOUT)
.setSecurityMode(ConnectionConfiguration.SecurityMode.disabled)
.setHost(Constant.IM_SERVER)//ip
.setPort(Constant.IM_SERVER_PORT)//端口号设置一般式5222
.setServiceName(Constant.IM_SERVER_DOMAIN)//服务器名称
.setDebuggerEnabled(true)//设置开启调试
.setSendPresence(true)//设置是否发送Presece信息
.build();
connection = new XMPPTCPConnection(configuration); connection.addConnectionListener(new ConnectionListener() {
@Override
public void connected(XMPPConnection connection) {
LogHelper.i(TAG, "connected");
} @Override
public void authenticated(XMPPConnection connection, boolean resumed) {
LogHelper.i(TAG, "authenticated");
} @Override
public void connectionClosed() {
LogHelper.i(TAG, "connectionClosed");
} @Override
public void connectionClosedOnError(Exception e) {
LogHelper.i(TAG, "connectionClosedOnError");
} @Override
public void reconnectionSuccessful() {
LogHelper.i(TAG, "reconnectionSuccessful");
} @Override
public void reconnectingIn(int seconds) {
LogHelper.i(TAG, "reconnectingIn");
} @Override
public void reconnectionFailed(Exception e) {
LogHelper.i(TAG, "reconnectionFailed");
}
}); ReconnectionManager reconnectionManager = ReconnectionManager.getInstanceFor(connection);
reconnectionManager.setFixedDelay(Constant.IM_RE_CONNET_TIME);//重联间隔
reconnectionManager.enableAutomaticReconnection();//开启重联机制
try {
connection.connect();
} catch (SmackException | IOException | XMPPException ex) {
ex.printStackTrace();
} }
}
XMPPTCPConnectionConfiguration类是连接的配置,连接相关配置都是通过这个类传递给XMPPTCPConnection对象的。配置信息的设置请看上面的代码注释,connection.addConnectionListener是给连接添加监听,这里面我们可以监听连接的各种状态。然后做相应的处理。
ReconnectionManager这个类是重联管理设置连接断开后是否允许重新连接。这样实现自动重联
最后通过connection.connect()连接XMPP服务器
登录:
/**
* 登录
* @param name 账号
* @param pwd 密码
* @return
*/
@Override
public boolean login(String name, String pwd) {
try {
if (connection.isConnected()) {
connection.login(name, pwd); } else {
connection.connect();
connection.login(name, pwd);
} } catch (XMPPException | SmackException | IOException ex) {
ex.printStackTrace();
} /**
* 消息监听
*/
registerMessageListener();
/**
* 通讯录监听
*/
registerRosterListener();
return false;
}

通过用户名和密码进行登录。登录之前判断是否连接了XMPP服务器。登录后添加消息监听处理消息,通讯录监听处理通讯录。下面再看看这两个监听方法

通讯录监听:

/**
* 通讯录监听
*/
private void registerRosterListener() {
Roster roster = Roster.getInstanceFor(connection);
roster.setSubscriptionMode(Roster.SubscriptionMode.manual);//设置添加好友,需要对方确认
roster.addRosterListener(new RosterListener() {
@Override
public void entriesAdded(Collection<String> collection) {
LogHelper.i(TAG, "通讯录用户添加");
} @Override
public void entriesUpdated(Collection<String> collection) {
LogHelper.i(TAG, "通讯录用户变更");
} @Override
public void entriesDeleted(Collection<String> collection) {
LogHelper.i(TAG, "通讯录用户删除");
} @Override
public void presenceChanged(Presence presence) {
LogHelper.i(TAG, "通讯录用户presence变化");
}
});
}
Roster类管理这通讯录。给通讯录添加了监听这样可以处理通讯里,请看上面的代码。这里重点讲解roster.setSubscriptionMode(Roster.SubscriptionMode.manual);这个设置通讯里添加好友的模式。这里设置了添加好友需要确认,而不是直接成为好友,一般都是这样,但是这里也支持直接成为好友。我们通过源码来看看SubscriptionMode这个枚举。
/**
* 好友请求订阅的模式枚举.
*/
public enum SubscriptionMode { /**
* 自动接收所有好友请求. This is
* 这是默认的模式,适合简单的客户端. 更复杂的客户端希望手动处理好友添加请求.
*/
accept_all, /**
* 自动拒绝所有请求
*/
reject_all, /**
* 好友请求被忽略,意味着必须手动注册和处理presence监听,然后处理类型是
* Presence.Type.SUBSCRIBE类型或者是Presence.Type.UNSUBSCRIBE类型的 presence包
*/
manual
}

以上是我通过源码中的英文转译的,鄙人英文就这水平见谅,见谅。这里说的presence包不懂的话看看第一篇XMPP协议简析,这样就懂了。我的原则是任何东西先懂原理(先修炼好《易筋经》然后再学招式,否则累也记不住,不懂硬记很累,有些学编程,语言没学好然后直接上平台性的东西,发现容易出问题,同样一个道理)。

消息监听:

/**
* 各种消息包监听
*/
private void registerMessageListener() {
connection.addSyncStanzaListener(new StanzaListener() {
@Override
public void processPacket(Stanza stanza) throws SmackException.NotConnectedException {
if (stanza instanceof Message) {//表示接收到是消息包
Message message = (Message) stanza;
if (message.getType() == Message.Type.chat) {//表示单聊 }
if (message.getType() == Message.Type.groupchat) {//表示群聊 }
if (message.getType() == Message.Type.error) {//表示错误信息 }
} if (stanza instanceof Presence) {//表示接收到的是Presence包 } if (stanza instanceof IQ) {//表示接收到的是IQ包 }
}
}, new StanzaFilter() {
@Override
public boolean accept(Stanza stanza) {
return true;
}
});

虽然还有其它监听消息的方法,但是我选择这种了,因为这种可以监听所有各种消息包,看上面注释能理解。如果不懂这些包神马意思还是推荐你看前面第一篇XMPP协议简析。那么久知道神马是IQ,神马是Message,什么是Presence包了

new StanzaFilter() {
@Override
public boolean accept(Stanza stanza) {
return true;
}
}

这个是过滤包,我这里返回ture就是不过滤直接接受所有XMPP包。你还可以过滤特定的包比如IQ包  StanzaFilter filter=new StanzaTypeFilter(IQ.class) ,然后把这个作为 addSyncStanzaListener方法的第二个参数这样就只能接受IQ消息了。

消息发送

 /**
* 消息发送
*
* @param from 谁发送
* @param to 发送给谁
* @param content 发送内容
*/
@Override
public void sendMessage(String from, String to, String content) throws SmackException.NotConnectedException { Message msg = new Message(to, content);
msg.setFrom(from);
connection.sendStanza(msg); // ChatManager chatmanager = ChatManager.getInstanceFor(connection);
// Chat newChat = chatmanager.createChat(to, new ChatMessageListener() {
// @Override
// public void processMessage(Chat chat, Message message) {
// LogHelper.i(TAG, "接收到消息:" + message);
//
// }
// });
// newChat.sendMessage(content); }

我这里可以看到有两种方式,一种直接原始的发送Message包,另一种就是被我注释的代码段,通过ChatManager来发送消息

添加好友:

/**
* 添加好友
* @param user 用户jid
* @param name 添加好友备注名称
* @param groups 好友添加到的分组,可以过个组
* @throws Exception
*/
@Override
public void addRosterItem(String user, String name, String[] groups) throws Exception{
Roster roster=Roster.getInstanceFor(connection);
roster.createEntry(user,name,null);
}

这里没神马好讲的,因为上面设置了 roster.setSubscriptionMode(Roster.SubscriptionMode.manual);//设置添加好友,需要对方确认  这种模式,添加好友需要对方确认

最后:

今天就此结束,虽然只有这些方法,如果你结合前面第一篇XMPP协议简介弄懂原理,我们就共同进步了:)。希望我们一起每天进步一点。

合肥程序员群:49313181。    合肥实名程序员群:128131462 (不愿透露姓名和信息者勿加入)
Q  Q:408365330     E-Mail:egojit@qq.com
 

即时聊天IM之四 Android客户端IM帮助类编写的更多相关文章

  1. android 客户端支付宝 php服务器端编写

    生成私钥 输入“genrsa -out rsa_private_key.pem 1024”命令,回车后,在当前 bin 文件目 录中会新增一个 rsa_private_key.pem 文件,其文件为原 ...

  2. [PHP]AES加密----PHP服务端和Android客户端

    本文采取128位AES-CBC模式加密和解密 1.首先对服务端安装mcrypt: sudo apt-get install php5-mcrypt php5-dev sudo php5enmod mc ...

  3. 即时聊天IM之三 XMPP协议客户端库的和Android端框架概述

    合肥程序员群:49313181.    合肥实名程序员群:128131462 (不愿透露姓名和信息者勿加入) Q  Q:408365330     E-Mail:egojit@qq.com smack ...

  4. 【原创】轻量级即时通讯技术MobileIMSDK:Android客户端开发指南

    申明:MobileIMSDK 目前为个人维护的原创开源工程,现陆续整理了一些资料,希望对需要的人有用.如需与作者交流,见文章底签名处,互相学习. MobileIMSDK开源工程的代码托管地址请进入 G ...

  5. 看完这篇包你进大厂,实战即时聊天,一文说明白:聊天服务器+聊天客户端+Web管理控制台。

    一.前言 说实话,写这个玩意儿是我上周刚刚产生的想法,本想写完后把代码挂上来赚点积分也不错.写完后发现这东西值得写一篇文章,授人予鱼不如授人以渔嘛(这句话是这么说的吧),顺便赚点应届学生MM的膜拜那就 ...

  6. Android客户端消息推送原理简介

    首先简单介绍一下Android消息推送的主要三种方式,如果你已经看过类似的文章,请直接忽略三种介绍.    1.使用SMS服务,即服务器端发送短信,然后手机客户端监听短信的广播,然后对数据进行一定的处 ...

  7. 即时聊天 / XMPP

    MQTT是第二个即时聊天协议(了解) 5.即时通讯 即时通讯网上有第三方的解决方案,比如环信,融云等.我们是自己搭的xmpp服务器,服务器使用的tigase,之前写过相关的博客,自己去年也做了对应的w ...

  8. 7.xmpp版即时聊天

    即时聊天的解决方案 socket: xmpp:xmpp+openfire+asmack 环信 常见协议 比较安全,tcp上还加了俩层 简单聊一下socket socket:套接字,连接需要ip和端口, ...

  9. Mina框架的学习笔记——Android客户端的实现

    Apache MINA(Multipurpose Infrastructure for Network Applications) 是 Apache 组织一个较新的项目,它为开发高性能和高可用性的网络 ...

随机推荐

  1. netty 解决TCP粘包与拆包问题(二)

    TCP以流的方式进行数据传输,上层应用协议为了对消息的区分,采用了以下几种方法. 1.消息固定长度 2.第一篇讲的回车换行符形式 3.以特殊字符作为消息结束符的形式 4.通过消息头中定义长度字段来标识 ...

  2. input失去隐藏光标(移动端)

    <input type="text" readonly name='lbinput' onfocus="lbinput.blur()">

  3. spine实现预加载(一)

    前言 本文实现了spine动画的预加载,解决在战斗等大量加载spine动画的时候出现卡顿现象. 这里使用和修改三个类,直接修改的源码,当然你也可以继承LuaSkeletonAnimation,自己封装 ...

  4. IOS 截取图片 部分 并生成新图片

    /** * 从图片中按指定的位置大小截取图片的一部分 * * @param image UIImage image 原始的图片 * @param rect CGRect rect 要截取的区域 * * ...

  5. Android下添加新的自定义键值和按键处理流程

            Android下添加新的自定义键值和按键处理流程     说出来不怕大家笑话,我写这篇博客的原因在于前几天去一个小公司面试Android系统工程师,然后在面试的时候对方的技术总监问了我 ...

  6. 使用JavaScript访问子节点方法elementNode.childNodes时,需要注意的地方

    有这样一个HTML结构 <div> javascript <p>javascript</p> <div>jQuery</div> <h ...

  7. JavascriptExecutor

    Why we use it?To enhance the capabilities of the existing scripts by performing javascript injection ...

  8. Query Designer:公式冲突

    声明:原创作品,转载时请注明文章来自SAP师太技术博客( 博/客/园www.cnblogs.com):www.cnblogs.com/jiangzhengjun,并以超链接形式标明文章原始出处,否则将 ...

  9. Thread

    问题:编写一个能提现多线程的例子?假设有t1,t2两个线程,如何保证t2线程在t1线程执行完后再执行? package cn.changb.thread; public class MyThread ...

  10. JS实现验证码倒计时效果

    通常做注册页面时会有获取验证码按钮,然后点击后过60秒才能重新获取,比如现在项目中遇到的 然后点击后的样式,并且数字是递减的,到0时重新回到最初的状态(上图). 首先构造HTML结构 <butt ...