- 浏览: 541402 次
- 性别:
- 来自: 北京
文章分类
最新评论
-
tangyunliang:
大哥你太历害了谢谢
Android基于XMPP Smack Openfire开发IM【四】初步实现两个客户端通信 -
u013015029:
LZ,请问下,在// 添加消息到聊天窗口 , 这里获取Ed ...
Android基于XMPP Smack Openfire开发IM【四】初步实现两个客户端通信 -
endual:
怎么保持会话,我搞不懂啊
Android基于XMPP Smack Openfire开发IM【一】登录openfire服务器 -
donala_zq:
显示:[2013-11-30 11:50:36 - Andro ...
android-----------新浪微博 -
donala_zq:
哥,运行不了啊
android-----------新浪微博
本博客要介绍的内容:
Openfire做服务器端,两个客户端:Spark、android模拟器。实现两个客户端之间的通信。
第一步:启动openfire服务器。(这里需要用到两个用户登录,前面的博客中已经说明如何添加用户了。)
[img]
[/img]
第二步:启动Spark客户端,如果您还没有Spark,下载请到:http://www.igniterealtime.org/downloads/index.jsp;安装简单。
我电脑的ip为:192.168.0.124,根据自己的修改。
安装后运行如下下图:
[img]
[/img]
第三步:启动eclipse中的项目(另一个客户端),代码如下:
package com.example.openfiretest; import org.jivesoftware.smack.Chat; import org.jivesoftware.smack.ChatManager; import org.jivesoftware.smack.ChatManagerListener; import org.jivesoftware.smack.ConnectionConfiguration; import org.jivesoftware.smack.MessageListener; import org.jivesoftware.smack.XMPPConnection; import org.jivesoftware.smack.packet.Message; import android.os.Bundle; import android.os.Handler; import android.app.Activity; import android.content.Intent; import android.view.Menu; import android.view.View; import android.view.View.OnClickListener; import android.view.Window; import android.widget.CheckBox; import android.widget.EditText; import android.widget.Toast; public class MainActivity extends Activity { private EditText accountEditText; private EditText passwordEditText; public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); requestWindowFeature(Window.FEATURE_NO_TITLE); setContentView(R.layout.activity_main); accountEditText = (EditText) findViewById(R.id.username); passwordEditText = (EditText) findViewById(R.id.password); findViewById(R.id.login).setOnClickListener(new OnClickListener() { public void onClick(View v) { String account = accountEditText.getText().toString(); String password = passwordEditText.getText().toString(); if (account.equals("") || password.equals("")) { Toast.makeText(MainActivity.this, "账号或密码不能为空!", Toast.LENGTH_SHORT).show(); } else { ClientConServer ccs = new ClientConServer(MainActivity.this); boolean b = ccs.login(account, password); // 如果登录成功 if (b) { Toast.makeText(MainActivity.this, "登陆成功!", Toast.LENGTH_SHORT).show(); Intent intent = new Intent(MainActivity.this,ChatActivity.class); startActivity(intent); } else { Toast.makeText(MainActivity.this, "登陆失败!", Toast.LENGTH_SHORT).show(); } } } }); } }
package com.example.openfiretest; import java.util.Collection; import org.jivesoftware.smack.Chat; import org.jivesoftware.smack.ChatManager; import org.jivesoftware.smack.ChatManagerListener; import org.jivesoftware.smack.ConnectionConfiguration; import org.jivesoftware.smack.MessageListener; import org.jivesoftware.smack.Roster; import org.jivesoftware.smack.RosterEntry; import org.jivesoftware.smack.RosterGroup; import org.jivesoftware.smack.XMPPConnection; import org.jivesoftware.smack.XMPPException; import org.jivesoftware.smack.packet.Message; import android.content.Context; import android.content.Intent; import android.os.Handler; import android.util.Log; import android.widget.Toast; public class ClientConServer { private static int PORT=5222; private Context context; public ClientConServer(Context context){ this.context=context; } //这里收到消息后,通过广播将消息发送到需要的地方.哈哈,既然收到了服务器发送来的信息,如何处理自己决定。 private Handler handler = new Handler(){ public void handleMessage(android.os.Message m) { Message msg=new Message(); msg=(Message) m.obj; //把从服务器获得的消息通过广播发送 Intent intent = new Intent("org.yhn.mes"); String[] message=new String[]{ msg.getFrom(), msg.getBody()}; System.out.println("==========收到服务器消息 From==========="+message[0].toString()); System.out.println("==========收到服务器消息 Body==========="+message[1].toString()); intent.putExtra("message", message); context.sendBroadcast(intent); }; }; public boolean login(String a,String p){ //ConnectionConfiguration config = new ConnectionConfiguration("192.168.0.124", PORT); /** 是否启用安全验证 */ //config.setSASLAuthenticationEnabled(false); /** 是否启用调试 */ //config.setDebuggerEnabled(true); /** 创建connection链接 */ //XMPPConnection connection = new XMPPConnection(config); XMPPConnection connection = ConnUtil.getConnection(); try { /** 建立连接 */ connection.connect(); /** 登录*/ connection.login(a, p); /** 开启读写线程,并加入到管理类中*/ //ClientSendThread cst=new ClientSendThread(connection); //cst.start(); //ManageClientThread.addClientSendThread(a, cst); //获取用户组、成员信息。 System.out.println("======开始获取组及用户=========="); Roster roster = connection.getRoster(); Collection<RosterGroup> entriesGroup = roster.getGroups(); System.out.println("组的个数:"+entriesGroup.size()); for(RosterGroup group: entriesGroup){ Collection<RosterEntry> entries = group.getEntries(); System.out.println("=========groupName==="+group.getName()); for (RosterEntry entry : entries) { //Presence presence = roster.getPresence(entry.getUser()); //Log.i("---", "user: "+entry.getUser()); System.out.println("组成员的名字:"+entry.getName()); System.out.println("组成员的user:"+entry.getUser()); //Log.i("---", "tyep: "+entry.getType()); //Log.i("---", "status: "+entry.getStatus()); //Log.i("---", "groups: "+entry.getGroups()); } } System.out.println("======结束获取组及用户=========="); //在登陆以后应该建立一个监听消息的监听器,用来监听收到的消息: ChatManager chatManager = connection.getChatManager(); chatManager.addChatListener(new MyChatManagerListener()); return true; } catch (XMPPException e) { e.printStackTrace(); } return false; } /** message listener*/ class MyChatManagerListener implements ChatManagerListener { public void chatCreated(Chat chat, boolean arg1) { chat.addMessageListener(new MessageListener(){ public void processMessage(Chat arg0, Message msg) { /**通过handler转发消息*/ android.os.Message m=handler.obtainMessage(); m.obj=msg; m.sendToTarget(); } }); } } }
package com.example.openfiretest; import org.jivesoftware.smack.ConnectionConfiguration; import org.jivesoftware.smack.XMPPConnection; public class ConnUtil { private static XMPPConnection connection; public static synchronized XMPPConnection getConnection(){ if(connection!=null){ return connection; }else{ ConnectionConfiguration config = new ConnectionConfiguration("192.168.0.124", 5222); /** 是否启用安全验证 */ config.setSASLAuthenticationEnabled(false); /** 是否启用调试 */ //config.setDebuggerEnabled(true); /** 创建connection链接 */ connection = new XMPPConnection(config); } return connection; } }
package com.example.openfiretest; import org.jivesoftware.smack.Chat; import org.jivesoftware.smack.ChatManager; import org.jivesoftware.smack.ChatManagerListener; import org.jivesoftware.smack.ConnectionConfiguration; import org.jivesoftware.smack.MessageListener; import org.jivesoftware.smack.XMPPConnection; import org.jivesoftware.smack.XMPPException; import org.jivesoftware.smack.packet.Message; import android.app.Activity; import android.content.Intent; import android.os.Bundle; import android.util.Log; import android.view.View; import android.view.View.OnClickListener; import android.view.Window; import android.widget.Button; import android.widget.EditText; import android.widget.TextView; import android.widget.Toast; public class ChatActivity extends Activity { String account; String chatNick; public static String chatContents; TextView chatTextView; public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); requestWindowFeature(Window.FEATURE_NO_TITLE); setContentView(R.layout.activity_chat); /** 设置top面板信息*/ //chatNick=getIntent().getStringExtra("nick"); //account=getIntent().getStringExtra("account"); //这里的用户昵称和账号是手动添加的。 chatNick = "忍者无敌笑哈哈"; account = "number1@admin-pc"; TextView nick_tv=(TextView) findViewById(R.id.chat_top_nick); nick_tv.setText(chatNick); Button send_btn=(Button) findViewById(R.id.chat_btn_send); send_btn.setOnClickListener(new OnClickListener(){ public void onClick(View arg0) { EditText input=(EditText) findViewById(R.id.chat_input); String content=input.getText().toString(); try { XMPPConnection connection = ConnUtil.getConnection(); ChatManager cm = connection.getChatManager(); Chat chat=cm.createChat(account, new MessageListener(){ public void processMessage(Chat arg0, Message msg) { Log.i("---", msg.getFrom()+"说:"+msg.getBody()); //添加消息到聊天窗口 , } }); Message m=new Message(); m.setBody(content); chat.sendMessage(m); } catch (XMPPException e) { e.printStackTrace(); } } }); } }
启动后如下图:
[img]
[/img]
第四步:这时,Spark客户端的篮球组中多了一个人。(不清楚组添加、用户添加的请看前面的博客);
[img]
[/img]
第五步:手机模拟器发送消息给Spark,输入信息后点击发送,
[img]
[/img]
第六步:Spark给手机模拟器发送消息,输入消息后按Ctrl+Enter发送,手机端把收到的消息,通过控制台打印。
[img]
[/img]
第七步:openfire服务器端发送消息给所有的客户端:
[img]
[/img]
over
菜鸟学习openfire,如有不妥之处,多多批评。
评论
4 楼
tangyunliang
2014-05-14
大哥
你太历害了
谢谢
3 楼
u013015029
2014-05-08
LZ,请问下,在// 添加消息到聊天窗口 , 这里获取EditText的内容 msg.getBody() 为什么是空的呢?
2 楼
Silence222
2013-09-22
LZ有没有发现,客户端搁置时间长的话会收不到消息?怎么解决的?
1 楼
wangyezidong
2013-09-15
请问LZ有demo吗发给我看看
发表评论
-
Android基于XMPP Smack Openfire开发IM【三】客户端接收服务器发送的消息
2013-03-26 16:12 8912Demo需求:android客户端接收服务器发送来的消息。 ... -
Android基于XMPP Smack Openfire开发IM 【二】获取好友列表
2013-03-26 11:28 3665第一步,在openfire后台中创建一些用户,并添加两个组。我 ... -
Android基于XMPP Smack Openfire开发IM【一】登录openfire服务器
2013-03-25 17:41 5942第一,安装好openfire后,启动openfire进入后台。 ... -
Openfire 的安装和配置
2013-02-05 13:07 14621、 下载最新的openfire安装文件 官方下载站点:ht ... -
本文介绍AndroidPn项目实现推送的使用过程
2013-02-05 12:41 1761在上一篇文章中提到在A ...
相关推荐
基于风光负荷不确定性的微电网两阶段鲁棒优化模型:投资与运行成本的综合考虑及CCG算法的Matlab+Yamilp+CPLEX求解方法,考虑风光负荷的不确定性,构建了微电网两阶段鲁棒优化容量配置模型,第一阶段目标函数为微电网投资成本,第二阶段目标函数为微电网的运行成本。 采用CCG算法求解。 包含非线性项的线性化处理。 程序代码matlab+yamilp+CPLEX ,核心关键词: 微电网; 两阶段鲁棒优化; 容量配置模型; CCG算法; 线性化处理; MATLAB; YAMILP; CPLEX 用分号分隔每个关键词的结果为: 微电网; 两阶段鲁棒优化; 容量配置模型; CCG算法; 线性化处理; MATLAB; Yamilp; CPLEX,微电网鲁棒优化模型:两阶段成本最小化与线性化处理
西门子S7-200PLC在Z35摇臂钻床控制系统中的应用与组态王组态设计研究,98西门子S7-200PLC和组态王组态Z35摇臂钻床控制系统组态设计PLC设计 ,西门子S7-200PLC; 组态王组态; 摇臂钻床控制系统; PLC设计,西门子S7-200PLC与组态王Z35摇臂钻床控制系统组态设计
烟花代码编程python满屏-10.排队接水——我直接把水吸过来.py
基于三菱PLC的喷灌控制系统设计及组态画面实现,No.393 三菱PLC组态王组态画面基于PLC的喷灌控制设计灌溉控制 ,基于三菱PLC; 喷灌控制设计; 组态王组态画面; 灌溉控制。,基于三菱PLC的喷灌控制设计:组态王组态画面应用
基于Dijkstra算法的路径规划Matlab代码:读取黑白色地图并实现图像上的路径规划示例,读取黑白色地图,在图片上进行路径规划matlab代码 路径规划算法基于读图识别的Dijkstra算法 随意设置坐标,规划路径 示例如下 ,读取地图; 路径规划算法; Dijkstra算法; 坐标设置; 规划路径,基于Dijkstra算法的Matlab黑白色地图路径规划代码
融合空间特征的债券图表数据文本检测方法研究.pdf
基于粒子群优化与遗传算法的高速列车横向悬挂模糊PID控制技术研究:s函数建模与仿真分析详解文件,基于粒子群 遗传算法的高速列车横向悬挂模糊PID控制-s函数搭建。 赠word详解文件 ,基于粒子群; 遗传算法; 高速列车横向悬挂; 模糊PID控制; s函数搭建,基于遗传算法的高速列车悬挂PID模糊控制的S函数构建方案
一种基于Retinex的非线性彩色图像增强算法.pdf
基于因式分解的逆合成孔径雷达三维成像技术及序列因子分解法研究,逆合成孔径雷达三维成像,因式分解法,序列因子分解法。 此部分只做了因式分解的工作,假设散射点已经准确提取并得到散射点坐标矩阵,利用因式分解恢复目标三维结构 ,逆合成孔径雷达三维成像; 因式分解法; 序列因子分解; 散射点提取; 散射点坐标矩阵; 恢复目标三维结构。,利用因式分解恢复三维结构:逆合成孔径雷达三维成像技术
MATLAB R2021b环境下深度学习驱动的车道线检测算法研究与应用,MATLAB环境下一种基于深度学习的车道线检测方法 算法运行环境为matlab r2021b,执行基于深度学习的车道线检测。 if batch ~= numBatches lastFrameIdx = miniBatchSize*batch; else 压缩包=数据+程序 ,核心关键词如下: MATLAB环境; 深度学习; 车道线检测; 算法运行环境; 批处理; 压缩包 以上关键词用分号分隔为: MATLAB环境; 深度学习; 车道线检测; 算法运行环境r2021b; 批处理(batch~numBatches); 压缩包(数据+程序) 注意:最后一个关键词“压缩包”并不是直接从您提供的信息中提炼出来的,而是根据您提到的“压缩包=数据+程序”这一句推断出来的。如果这不是您想要的核心关键词,请告诉我,我会进行相应的调整。,MATLAB深度学习车道线检测算法压缩包
西门子S7-200PLC在物料自动称量控制系统中的应用:基于称重配料混合程序的组态设计与王设计PLC程序设计,74基西门子S7-200PLC的物料自动称量控制系统称重配料物料混合程序组态设计组态设计组态王设计plc程序设计 ,核心关键词:西门子S7-200PLC;自动称量控制系统;物料混合程序;组态设计;组态王设计;PLC程序设计。,西门子S7-200PLC物料自动称量控制系统称重配料程序组态设计
优化无人机侦查与目标搜索路径规划:对未知区域进行覆盖并精确寻靶的Matlab代码解析与实践指南。,无人机覆盖搜索路径规划 无人机搜索目标路径规划 无人机侦查路径规划 对未知区域进行覆盖搜索,并且寻找目标 matlab代码有详细注释,可快速上手。 ,核心关键词:无人机; 覆盖搜索路径规划; 搜索目标路径规划; 侦查路径规划; 未知区域覆盖搜索; MATLAB代码注释。,**无人机未知区域覆盖搜索及目标路径规划Matlab代码详解**
shp格式,可直接导入arcgis使用
基于博途软件编程的11层电梯控制系统:文档齐全,优质售后保障的智能运行体验,基于PLC的单部11层电梯控制系统,采用博途软件编写提供画面,文档(含接线图,流程图,IO分配表)及优质的后服务。 最终运行效果,详见上方演示视频 ,核心关键词:PLC控制;11层电梯;博途软件;画面编写;文档资料;接线图;流程图;IO分配表;售后服务;演示视频。,"博途软件控制下的11层电梯系统,高效运行效果详见演示视频"
"飞剪追剪程序:PLC与伺服同步控制完整指南,适合新手学习与参考,包含PLC程序、触摸屏程序及CAD电路图纸",飞剪追剪程序plc程序伺服程序 同步控制 适合新手学习参考 包含PLC程序+触摸屏程序+CAD电路图纸。 ,飞剪追剪程序; PLC程序; 伺服程序; 同步控制; 新手学习参考; PLC程序+触摸屏程序; CAD电路图纸,适合新手的飞剪追剪程序全套教程:PLC+伺服同步控制
使用Matlab编程:无迹卡尔曼滤波算法(UKF)的编写及其在电池SOC估计中的应用,噪声系数自适应优化方案的研究。,使用matlab编写m脚本,编写无迹卡尔曼滤波算法(UKF)估计电池SOC,注释清晰。 卡尔曼滤波算法(EKF)锂电池SOC估计,噪声系数自适应 Matlab ,matlab; 无迹卡尔曼滤波算法(UKF); 电池SOC估计; 注释清晰; 噪声系数自适应。,Matlab无迹卡尔曼滤波(UKF)用于电池SOC估计的代码实现
Ansys LS-DYNA多孔延时起爆与重复起爆模拟的完整过程解析,Ansys ls_dyna多孔延时起爆,重复起爆模拟 全过程 ,Ansys;LS_dyna;多孔延时起爆;重复起爆模拟;全过程,Ansys LS-DYNA多孔延时重复起爆模拟全过程
FPGA驱动双目视觉系统:立体匹配、视差图与深度图生成技术的研究与应用,FPGA双目视觉 立体视觉 视差图 深度图 双目立体匹配sgm sgbm tang 20k 高云fpga usb摄像头采集图像 ,FPGA双目视觉;立体视觉;视差图;深度图;双目立体匹配sgm;sgbm;高云FPGA;USB摄像头图像采集。,基于FPGA的双目视觉系统:立体匹配与深度图生成
整个手套大盘的指数图如下.docx
基于AGV全覆盖移动避障的扫地机器人路径规划算法研究:深度优先搜索与随机碰撞对比,AGV全覆盖移动避障路径规划matlab代码 扫地机器人路径规划 第一类算法 全覆盖智能算法 %% 基于深度优先搜索算法的路径规划—扫地机器人移动仿真 % 返回深度优先搜索实现全覆盖的运行次数 % 将栅格模型的每一个栅格看成一个点 % 实际中栅格模型是连续的,在计算机处理时看作离散的 % 将栅格模型抽象为标识矩阵,矩阵对应位置的标记表示栅格对应位置的状态 第二对比算法 %% 随机碰撞的路径规划—扫地机器人移动仿真 % 返回深度优先搜索实现全覆盖的运行次数 % 将栅格模型的每一个栅格看成一个点 % 实际中栅格模型是连续的,在计算机处理时看作离散的 % 将栅格模型抽象为标识矩阵,矩阵对应位置的标记表示栅格对应位置的状态 ,核心关键词: 1. AGV全覆盖移动避障 2. 路径规划 3. Matlab代码 4. 扫地机器人路径规划 5. 第一类算法 6. 全覆盖智能算法 7. 深度优先搜索算法 8. 栅格模型 9. 标识矩阵 10. 随机碰撞的路径规划,"AGV全覆盖避障路径规划:Matlab代码实现与对比"