Android之基于XMPP协议即时通讯软件(二)

2013博客之星投票请移步:http://vote.blog.csdn.net/blogstaritem/blogstar2013/weidi1989


昨天给大家粗略的介绍了一下我那个简单的项目:Android之基于XMPP协议即时通讯软件(一)

从今天开始,就详细展开的介绍设计思路,一是给自己做个总结,二是希望能给各位朋友一点点帮助吧!

也许和很多刚刚开始入行程序猿的朋友一样,我最初也是每开始一个新项目,只是稍微思考一下便开始写代码。一个星期后,尽管实现了部分功能,却发现自己有点乱了,因为又突然生出很多新的想法,于是又顺着思路写下去,再持续一段时间,就会发现这个项目做不下去了。代码重复很多,而且bug也是层出不穷。因此不得不推翻重新开始,浪费了很多时间。所以说,开始前的准备工作是相当重要的。

一般说来完成一个项目可以分成三步:

首先:画几天时间好好理清思路,可以在草稿纸上写写画画,列出必须实现的功能已经界面特效等等,做到胸有成竹。

其次:开始搭建框架,定义好所有功能的接口,并且注释好做什么用,这是方便团队合作的项目,个人项目也可以养成写注释的好习惯。

最后:分开实现各部分功能,并且分开调试,可以单独独立成一个应用或者单元测试模块。比如说本应用的所有界面效果都是独立成一个应用测试无误后才最后合入的。


我这个项目呢,一开始是完成了所有的界面模块工作,因为是直接从之前的项目移植过来的,稍微修改了一下,所以这部分省掉了很多时间。

重心就直接放在了功能实现这方面了。功能模块部分就是下面两个包:com.way.service和 com.way.smack


首先是定义通用接口,主要就是将第三方包asmack.jar中需要用到的东西拿出来,这个包是非常庞大的,但是我们这个小应用并不是需要实现所有的功能,所以,抽取自己需要的那部分就可以了,定义成接口,方便以后扩展需要的功能,代码如下:

/**
 * 首先定义一些接口,需要实现一些什么样的功能,
 * 
 * @author way
 * 
 */
public interface Smack {
	/**
	 * 登陆
	 * 
	 * @param account
	 *            账号
	 * @param password
	 *            密码
	 * @return 是否登陆成功
	 * @throws XXException
	 *             抛出自定义异常,以便统一处理登陆失败的问题
	 */
	public boolean login(String account, String password) throws XXException;

	/**
	 * 注销登陆
	 * 
	 * @return 是否成功
	 */
	public boolean logout();

	/**
	 * 是否已经连接上服务器
	 * 
	 * @return
	 */
	public boolean isAuthenticated();

	/**
	 * 添加好友
	 * 
	 * @param user
	 *            好友id
	 * @param alias
	 *            昵称
	 * @param group
	 *            所在的分组
	 * @throws XXException
	 */
	public void addRosterItem(String user, String alias, String group)
			throws XXException;

	/**
	 * 删除好友
	 * 
	 * @param user
	 *            好友id
	 * @throws XXException
	 */
	public void removeRosterItem(String user) throws XXException;

	/**
	 * 修改好友昵称
	 * 
	 * @param user
	 *            好友id
	 * @param newName
	 *            新昵称
	 * @throws XXException
	 */
	public void renameRosterItem(String user, String newName)
			throws XXException;

	/**
	 * 移动好友到新分组
	 * 
	 * @param user
	 *            好友id
	 * @param group
	 *            新组名
	 * @throws XXException
	 */
	public void moveRosterItemToGroup(String user, String group)
			throws XXException;

	/**
	 * 重命名分组
	 * 
	 * @param group
	 *            之前的组名
	 * @param newGroup
	 *            新组名
	 */
	public void renameRosterGroup(String group, String newGroup);

	/**
	 * 请求好友重新授权,用在添加好友失败时,重复添加 再次向对方发出申请
	 * 
	 * @param user
	 *            好友id
	 */
	public void requestAuthorizationForRosterItem(String user);

	/**
	 * 添加新分组
	 * 
	 * @param group
	 */
	public void addRosterGroup(String group);

	/**
	 * 设置当前在线状态
	 */
	public void setStatusFromConfig();

	/**
	 * 发送消息
	 * 
	 * @param user
	 * @param message
	 */
	public void sendMessage(String user, String message);

	/**
	 * 向服务器发送心跳包,保持长连接 通过一个闹钟控制,定时发送,
	 */
	public void sendServerPing();

	/**
	 * 从jid中获取好友名
	 * 
	 * @param jid
	 * @return
	 */
	public String getNameForJID(String jid);
}

真正的实现就留给他的子工具类SmackImpl。昨天我说过,本应用只有一个服务:XXService,他时继承自BaseService,BaseService也是完成了一部分通用的功能,比如说通知栏消息处理等。在XXService中,我们主要是处理Activity与Smack中连接工作。 每个Activity在onResume时,就与该XXService绑定,绑定之后,便可调用该Service中的所有公共函数,比如说登陆、发消息等等,然后Service再调用Smack中已经定义好的各种接口函数,即可完成与服务器的交互。没错,其实这就是MVC设计模式。

Service担当这个C的角色,Activity就是V的角色,Smack就是M了。


现在我们的Activity可以调用Service中的函数提交消息给服务器了,这是单方面的,我们同时需要获得服务器返回的信息,比如说是否登陆成功、收消息之类。这个任务就自然的落到了回调接口IConnectionStatusCallback身上了,在绑定服务成功的同时,我们注册此回调接口,下面是此接口定义的唯一一个函数。也许眼尖的朋友会问了,为什么没有收到新消息的回调,其实一开始我是加了的,后面因为将数据库改成了ContentProvider,就可以去掉了,因为收消息可以通过监听数据库动态变化嘛,对吧?这就是ContentProvider的好处所在,这个留待后续详细说,这里只是提及一下。

/**
 * 服务器回调接口
 * 
 * @author way
 * 
 */
public interface IConnectionStatusCallback {
	/**
	 * 连接状态改变
	 * 
	 * @param connectedState
	 *            连接状态,有连接、未连接、链接中三种
	 * @param reason
	 *            连接失败的原因
	 */
	public void connectionStatusChanged(int connectedState, String reason);
}

在绑定服务成功的时候,将此回调传递给Service,比如登陆界面的 ServiceConnection:

	ServiceConnection mServiceConnection = new ServiceConnection() {

		@Override
		public void onServiceConnected(ComponentName name, IBinder service) {
			mXxService = ((XXService.XXBinder) service).getService();
			//注册回调
			mXxService.registerConnectionStatusCallback(LoginActivity.this);
		}

		@Override
		public void onServiceDisconnected(ComponentName name) {
			mXxService.unRegisterConnectionStatusCallback();
			mXxService = null;
		}

	};

就这样,整体与服务器交互的基本框架就完成了。很简单对吧?

OK,今天就到这里了,以上仅是本人的一点心得体会,后续将详细讲解功能接口Smack的实现以及如何处理联系人和聊天记录数据库等功能,非常感谢你看到了文章末尾,话说,你是否忘记投票了呢?O(∩_∩)O哈哈~






  • 45
    点赞
  • 31
    收藏
    觉得还不错? 一键收藏
  • 37
    评论
XMPP协议是一种用于实现即时通讯的开放式协议。下面是使用XMPP协议的简要教程: 1. 首先,你需要搭建一个XMPP服务器,比如Openfire。你可以在网上找到很多关于Openfire搭建的教程。 2. 在前端应用中,你需要导入XMPPframework库。这个库可以帮助你与XMPP服务器进行通信。 3. 在与服务器建立连接之前,你需要建立一个XMPP流(stream)。这个流的建立过程包括协商安全性等步骤。 4. 一旦建立了XMPP流,你可以通过发送XML Stanza给服务器来进行通信。XML Stanza是一种特定格式的XML消息。 5. 服务器会根据你发送的消息和程序逻辑,向客户端发送XML Stanza。这个过程不是一问一答的,任何时候都有可能从一方发信给另外一方。 6. 通信的最后阶段是关闭流和TCP/IP连接。 总的来说,XMPP协议类似于HTTP协议,它的通信过程是一个"解包装-〉包装"的过程。你只需要理解接收到的消息类型,并理解返回的消息类型,就可以很好地利用XMPP进行数据通信。 对于开发客户端聊天工具,你可以使用Smack库。Smack是一个用Java编写的开源XMPP(Jabber)库,支持PC和移动开发。你可以在Android平台上使用Smack库来开发基于XMPP协议的即时聊天功能。 希望这个简要教程对你有所帮助。如果你需要更详细的教程,可以在网上搜索相关资源。 #### 引用[.reference_title] - *1* *2* [基于XMPP协议即时通讯教程附Demo](https://blog.csdn.net/qq_25608527/article/details/48247427)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down28v1,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [即时聊天IM之三 XMPP协议客户端库的和Android端框架概述](https://blog.csdn.net/weixin_35835018/article/details/114100513)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down28v1,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 37
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值