查看: 1400|回复: 0

Flex和java的socket通信(五)聊天室的补充-在线列表-私聊

[复制链接]
发表于 2009-6-9 03:36:34 | 显示全部楼层 |阅读模式
目的:通过实现聊天室的一些功能加深对socket以及自定义协议的理解<br /><br />代码:<br />/*<br />* 自定义的协议 <br />* 收到消息: 11开头表示新加入了聊天用户;22开头表示公聊;33开头表示私聊<br />* 发送消息:11开头表示更新用户列表;22开头表示发送到屏幕上;44发送在线人数<br />* <br />*/<br /><br />import java.net.*;<br />import java.io.*;<br />import java.util.*;<br />public class Server5 {<br /><br />private ServerSocket server;<br />private BManager bMan=new BManager();   //消息广播者<br />//Map接口的HashMap类,元素拥有固定key值,key值不重复,这里用来存放在线用户<br />Map&lt;Socket,String&gt; clientList = new HashMap&lt;Socket,String&gt;();<br />public Server5(){}   //构造函数<br />void startServer()   //启动服务器<br />{<br />try{<br />server=new ServerSocket(8888);   //创建服务器套接字<br />System.out.println(&quot;服务器套接字建立完毕&quot;);<br />while(true)<br />{<br />Socket socket=server.accept();   //若客户机提出请求,使用socket进行连接<br />//String strIP = socket.getInetAddress().toString();//登陆者的ip<br />Chat_Thread ct=new Chat_Thread(socket);<br />ct.start();   //启动线程<br />bMan.add(socket);   //添加套接字<br />bMan.sendClientInfo();//使用套接字输出当前聊天人数<br />//funList(clientList);<br />//bMan.sendToAll(strIP+&quot;/加入聊天室&quot;);<br />}<br />}catch(Exception e){<br />System.out.println(e);<br />}<br />}<br />public static void main(String[] args) {<br />Server5 server=new Server5();<br />server.startServer();<br />}<br />class Chat_Thread extends Thread   //与客户机进行通讯的线程类<br />{<br />Socket socket;//x1<br />private BufferedReader reader;   //套接字输入流;<br />private PrintWriter writer; //套接字输出流<br />Chat_Thread(Socket socket)<br />{<br />this.socket=socket;//this.socket就是x1处的socket<br />}<br />public void run()<br />{<br /><br />try<br />{<br />reader=new BufferedReader(new InputStreamReader(socket.getInputStream(),&quot;utf8&quot;));<br />writer=new PrintWriter(socket.getOutputStream(),true);<br />String msg;<br />//msg获取消息<br />while((msg=reader.readLine())!=null)<br />{<br />System.out.println(msg);//服务器屏幕输出消息<br />String str=msg.substring(0, 2);//截取前两个个字符<br />int a=Integer.parseInt(str);//强制转换成int<br />String[] arrMsg=msg.split(&quot;--&quot;);//将获取的消息以&quot;--&quot;符号为标志分解成数组<br />switch(a)<br />{<br />case 11 : //当消息以11开头的时候,将登陆者的信息储存到hashmap之中,并向客户端发送新的在线列表<br />String strName=msg.substring(2);//获取登陆者名字,消息格式“11eko”<br />System.out.println(strName+&quot;登陆了&quot;);//服务器屏幕输出谁登陆了<br />bMan.sendToAll(&quot;22&quot;+strName+&quot;登陆了&quot;);//广播谁登陆了<br />clientList.put(this.socket,strName);//加入到HashMap中<br />funList(clientList);//广播在线列表<br />break;<br />case 22://当消息以22开头的时候,内容为“22--eko--内容”<br />System.out.println(&quot;公聊&quot;);<br />//构造消息,arrMsg[0]=消息头,arrMsg[1]消息发送者,arrMsg[2]消息内容<br />msg=arrMsg[0]+arrMsg[1]+&quot;说:&quot;+arrMsg[2];<br />bMan.sendToAll(msg);//向所有人广播消息<br />break;<br />case 33://消息以33开头时候,内容为“33--sandal--eko--内容”<br />//arrMsg[1]为说话对象,arrMsg[2]为说话人,arrMsg[3]为消息内容<br />if(arrMsg[1].equals(&quot;所有人&quot;))//当说话对象为"所有人"的时候<br />{<br />//构造消息"22eko说:内容"<br />msg=&quot;22&quot;+arrMsg[2]+&quot;说:&quot;+arrMsg[3];<br />//向所有人发送消息<br />bMan.sendToAll(msg);<br />}else //其他情况就是向具体的某个人发送消息了<br />{<br />Socket socketOne;<br />System.out.println(&quot;私聊&quot;);<br />Set set = clientList.keySet();//使用keySet方法获取所有key值<br />Iterator it = set.iterator();//使用Iterator(迭代器)来遍历数据<br />while (it.hasNext()) { //返回是否还有没被访问过的对象<br />Object ok=it.next();//返回下一个没被访问过的对象<br />Object ov=clientList.get(ok);//get方法返回拥有key的元素<br />if(ov.equals(arrMsg[1]))//如果在client中找到&quot;消息发给谁&quot;的时候,发给对方<br />{<br />socketOne=(Socket)ok;//强制转换成key值类型;<br />bMan.sendToONE(socketOne,&quot;22(悄悄话)&quot;+arrMsg[2]+&quot;对你说:&quot;+arrMsg[3]);<br />}else if(ov.equals(arrMsg[2]))//如果在client中找到&quot;发消息的人&quot;的时候,发给他自己<br />{<br />socketOne=(Socket)ok;<br />bMan.sendToONE(socketOne,&quot;22(悄悄话)你对&quot;+arrMsg[1]+&quot;说:&quot;+arrMsg[3]);<br />}<br />}<br />}<br /><br />break;<br />}<br />//bMan.sendToAll(msg);<br />}<br />}catch(Exception e)<br />{<br /><br />}finally<br />{<br />try {<br />bMan.remove(socket);<br />if(reader !=null) reader.close();<br />if(writer !=null) writer.close();<br />if(socket !=null) socket.close();<br />if(clientList.containsKey(socket))<br />{<br />bMan.sendToAll(&quot;22&quot;+clientList.get(socket)+&quot;离开了...&quot;);//广播消息,谁离开了<br />clientList.remove(socket);//删除socket<br />funList(clientList);//广播在线列表<br /><br />}<br />reader=null;<br />writer=null;<br />socket=null;<br />System.out.println(&quot;客户机离开&quot;);<br />bMan.sendClientInfo();//广播在线人数<br />} catch (Exception e) {}<br />}<br /><br />}<br />}<br />void funList(Map clientList) // 广播在线列表<br />{<br />String strList=&quot;&quot;;//在线列表<br />Set set = clientList.keySet();//使用keySet方法获取所有key值<br />System.out.println(set);<br />Iterator it = set.iterator();//使用Iterator(迭代器)来遍历数据<br />System.out.println(it);<br />while (it.hasNext()) {//把用户名称发给在线所有客户端 <br />//构造在线列表格式strList=11--one--two--three<br />strList+=&quot;--&quot;;<br />strList+=clientList.get(it.next());<br />}<br />bMan.sendToAll(&quot;11&quot;+strList);<br />}<br />}<br /><br />class BManager extends Vector<br />{<br />BManager (){}<br />void add(Socket sock)<br />{<br />super.add(sock);<br />}<br />void remove(Socket sock)<br />{<br />super.remove(sock);<br />}<br />synchronized void sendToAll(String msg)//给所有人广播函数<br />{<br />rintWriter writer=null;<br />Socket sock;<br />for(int i=0;i&lt;size();i++) //执行循环<br />{<br />sock=(Socket)elementAt(i);//获取第i个套接字<br />try<br />{<br />//获取第i个套接字输出流<br />writer=new PrintWriter(sock.getOutputStream(),true);<br />}catch(Exception ie){}<br />//使用第i各套接字输出流,输出消息<br />if(writer!=null)writer.println(msg);<br />}<br />}<br />synchronized void sendToONE(Socket socket,String msg)//私聊函数<br />{<br />rintWriter writer=null;<br />Socket sock;<br />for(int i=0;i&lt;size();i++)<br />{<br />sock=(Socket)elementAt(i);<br />if(socket==sock)//与给所有人广播函数类似,仅加入了判断,只有当socket管理器中的socket等于传入的socket的时候才发送消息<br />{<br /><br />try<br />{<br />writer=new PrintWriter(sock.getOutputStream(),true);<br />}catch(Exception ie){}<br />if(writer!=null)writer.println(msg);<br />}<br />}<br /><br />}<br />synchronized void sendClientInfo()<br />{<br />String info=&quot;44当前聊天人数:&quot;+size();<br />//System.out.println(info);<br />sendToAll(info);<br />}<br />}知识点:<br />实现在线列表的思路:每当有新的用户登陆的时候就像服务器发送用户名,服务器收到用户明后就会存在clientList中,然后再提取所有的用户名广播出去。<br />实现私聊的思路,利用hashmap来储存数据,通过name找到socket,然后把消息发给找到的socket<br /><br />关于map,首先就要说说Collection接口,它位于与数据结构有关的API的最上部。构成Collection的单位,我们称之为元素(element),此接口提供了添加,删除元素等管理数据的功能,根据管理方法的不同,可将Collection接口分为Set,List,Map三种接口。<br />实现Map接口的类也有3个,分别是HashMap类,TreeMap类和Hashtable类。这些类具有如下几个特征:元素拥有固定的key值;key值不允许重复。<br />下面就说说这次说用道德HashMap类的常用方法。<br />添加一个拥有key的元素:put(Object key,Object value);<br />删除拥有key的元素:remove(Object key);<br />返回拥有key的元素:get(Object key);<br /><br />通过get方法可以通过key查找value,那反过来呢?如何通过HashMap中的value找到key。似乎并没有现成的方法。这里提供一个思路。关于关于Iterator和Set大家可以自行查查java文档,这里就浪费论坛的空间了。 复制内容到剪贴板 代码:<br />pritnkey(HashMap hm,String value)<br />{<br />ArrayList a=new Array();<br />Set key =hm.keySet();<br />Iterator it=key.iterator();<br />while(it.hasNext())<br />{<br />Object ok=it.next();<br />Object ov=hm.get(ok);<br />if(ov.equals(value))<br />{<br />a.add(ok);<br />}<br />if(a.size()!=0)<br />{<br />System.out.println(a);<br />}<br />}<br />}客户端:myLogin.mxml;Client5.mxml<br />没什么好讲的,知识点在前四节中已经反复的提过了,这里就不再累叙了。<br />所要注意的无非就是通过字符串操作来分割服务器传来的消息,经过判断然后进行归类。<br />所要注意的一点是,在登陆的时候要做好验证工作,确保用户列表不重名。
<br /><br /><blockquote class="blockquote">From: http://www.ia56.com/read-htm-tid-31.html  Powered by PHPWind.com</blockquote>
回复

使用道具 举报

本版积分规则

关注公众号

相关侵权、举报、投诉及建议等,请发 E-mail:admin@discuz.vip

Powered by Discuz! X5.0 © 2001-2026 Discuz! Team.

在本版发帖
关注公众号
QQ客服返回顶部