服务端部署
在服务端src/websocket目录下新建一个.java文件,命名为chat.java,
注意事项:
1、利用@ServerEndpoint(“/ExampleSeverAndroid”)设定访问URL,此处websocket连接访问url为Http://localhost:8080/ExampleSeverAndroid.
2、重写websocket的@OnOpen、@OnClose、@OnMessage、@OnError方法
3、在与websocket.java同目录下新建Democonfig.java以实现websocket的配置调用方法。
chat.java源码(巨乱,很多无用代码,按需索取):
package websocket;
import bean.MyMessage;
import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONObject;
import dao.UserDao;
import util.StringUtil;
import javax.websocket.*;
import javax.websocket.server.PathParam;
import javax.websocket.server.ServerEndpoint;
import java.io.IOException;
import java.util.List;
import java.util.Objects;
import java.util.concurrent.CopyOnWriteArraySet;
//import net.sf.json.JSONObject;
//import net.sf.json.JSONObject;
//该注解用来指定一个URI,客户端可以通过这个URI来连接到WebSocket。类似Servlet的注解mapping。无需在web.xml中配置。
@ServerEndpoint("/ExampleSeverAndroid")
public class Chat {
//静态变量,用来记录当前在线连接数。应该把它设计成线程安全的。
private static int onlineCount = 0;
//concurrent包的线程安全Set,用来存放每个客户端对应的MyWebSocket对象。若要实现服务端与单一客户端通信的话,可以使用Map来存放,其中Key可以为用户标识
private static CopyOnWriteArraySet<Chat> webSocketSet = new CopyOnWriteArraySet<Chat>();
//与某个客户端的连接会话,需要通过它来给客户端发送数据
private Session session;
/**
* 连接建立成功调用的方法
* @param session 可选的参数。session为与某个客户端的连接会话,需要通过它来给客户端发送数据
*/
@OnOpen
public void onOpen(Session session, @PathParam("userId") String userId){
this.session = session;
webSocketSet.add(this); //加入set中
addOnlineCount(); //在线数加1
System.out.println("OPEN!" + getOnlineCount());
// JSONObject msg = new JSONObject();
// try {
// msg.put("msg", "连接成功");
// msg.put("status", "SUCCESS");
// msg.put("userId", userId);
// sendMessage(JSON.toJSONString(msg));
// } catch (Exception e) {
// //logger.debug("IO异常");
// }
}
/**
* 连接关闭调用的方法
*/
@OnClose
public void onClose( @PathParam("userId") String userId){
webSocketSet.remove(this); //从set中删除
subOnlineCount(); //在线数减1
System.out.println("Close!" + getOnlineCount());
}
/**
* 收到客户端消息后调用的方法
* @param message 客户端发送过来的消息
*/
@OnMessage
public void onMessage(String message, @PathParam("userId") String userId) {
System.out.println("Message From Client:" + message);
try {
MyMessage myMessage = JSON.parseObject(message, MyMessage.class);
String messageType = myMessage.getMessageType();
//"MessageType"
//"
//朋友圈上传
if(messageType.equals("post"))
{ //朋友圈文案
String messageContent = myMessage.getMessage();
//图片base64转码字符串
String messageImage = myMessage.getImage();
//用户名
//System.out.println("11111111111111");
String messageusername = myMessage.getUsername();
UserDao dao = new UserDao();
dao.ImageAndText(messageusername,messageImage,messageContent);
//评论上传
}else if(messageType.equals("comment"))
{
String comment = myMessage.getComment();
String time = myMessage.getTimestamp();
UserDao dao = new UserDao();
dao.Comment(myMessage.getUsername(),myMessage.getTimestamp(),myMessage.getComment());
}
// System.out.println("");
//广播当前朋友圈
for(Chat item: webSocketSet){
try {
UserDao dao = new UserDao();
List<String> share = dao.GetShare();
String str = share.toString();
// System.out.println(str);
item.sendMessage(share.toString());
} catch (IOException e) {
e.printStackTrace();
continue;
}
}
// if("1".equals(messageType)){ //单聊
// String recUser = myMessage.getUserId();//recUser:消息接收者
// // sendInfo(messageContent,recUser,userId);//messageContent:输入框实际内容 recUser:消息接收者 userId 消息发送者
// }else{ //群聊
// // sendGroupInfo(messageContent,userId);//messageContent:输入框实际内容 userId 消息发送者
// }
} catch (Exception e) {
// logger.error("解析失败:{}", e);
}
//群发消息
}
/**
* 发生错误时调用
* @param session
* @param error
*/
@OnError
public void onError(Session session, Throwable error){
System.out.println("ERROR");
error.printStackTrace();
}
/**
* 这个方法与上面几个方法不一样。没有用注解,是根据自己需要添加的方法。
* @param message
* @throws IOException
*/
public synchronized void sendMessage(String message) throws IOException{
// this.session.getBasicRemote().sendText(message);
this.session.getAsyncRemote().sendText(message);
}
/**
* 群聊
* message : 消息内容,输入的实际内容,不是拼接后的内容
* sendUser : 消息发送者
*/
public void sendGroupInfo(String message,String sendUser) throws IOException {
JSONObject msgObject = new JSONObject();//msgObject 包含发送者信息的消息
if (StringUtil.isNotEmpty(webSocketSet)) {
for (Chat item : webSocketSet) {
// logger.info("回送消息:" + message);
//拼接返回的消息,除了输入的实际内容,还要包含发送者信息
msgObject.put("message",message);
msgObject.put("sendUser",sendUser);
item.sendMessage(JSON.toJSONString(msgObject));
}
}
}
/**
* 单聊
* message : 消息内容,输入的实际内容,不是拼接后的内容
* recUser : 消息接收者
* sendUser : 消息发送者
*/
// public void sendInfo( String message , String recUser,String sendUser) {
// JSONObject msgObject = new JSONObject();//msgObject 包含发送者信息的消息
// for (Chat item : webSocketSet) {
// if (StringUtil.equals(item.userId, recUser)) {
// // logger.info("给用户" + recUser + "传递消息:" + message);
// //拼接返回的消息,除了输入的实际内容,还要包含发送者信息
// msgObject.put("message",message);
// msgObject.put("sendUser",sendUser);
// item.sendMessage(JSON.toJSONString(msgObject));
// }
// }
// }
/**
* 群聊
* message : 消息内容,输入的实际内容,不是拼接后的内容
* sendUser : 消息发送者
*/
// public void sendGroupInfo(String message,String sendUser) {
// JSONObject msgObject = new JSONObject();//msgObject 包含发送者信息的消息
// if (StringUtil.isNotEmpty(webSocketSet)) {
// for (Chat item : webSocketSet) {
// if(!StringUtil.equals(item.userId, sendUser)) { //排除给发送者自身回送消息,如果不是自己就回送
// logger.info("回送消息:" + message);
// //拼接返回的消息,除了输入的实际内容,还要包含发送者信息
// msgObject.put("message",message);
// msgObject.put("sendUser",sendUser);
// item.sendMessage(JSON.toJSONString(msgObject));
// }
// }
// }
// }
/**
* Map/Set的key为自定义对象时,必须重写hashCode和equals。
* 关于hashCode和equals的处理,遵循如下规则:
* 1)只要重写equals,就必须重写hashCode。
* 2)因为Set存储的是不重复的对象,依据hashCode和equals进行判断,所以Set存储的对象必须重写这两个方法。
* 3)如果自定义对象做为Map的键,那么必须重写hashCode和equals。
*
* @param o
* @return
*/
@Override
public boolean equals(Object o) {
if (this == o) {
return true;
}
if (o == null || getClass() != o.getClass()) {
return false;
}
Chat that = (Chat) o;
return Objects.equals(session, that.session);
}
@Override
public int hashCode() {
return Objects.hash(session);
}
public static synchronized int getOnlineCount() {
return onlineCount;
}
public static synchronized void addOnlineCount() {
Chat.onlineCount++;
}
public static synchronized void subOnlineCount() {
Chat.onlineCount--;
}
}
democonfig.java源码:
package websocket;
import javax.websocket.Endpoint;
import javax.websocket.server.ServerApplicationConfig;
import javax.websocket.server.ServerEndpointConfig;
import java.util.Set;
public class DemeConfig implements ServerApplicationConfig{
//注解的方式 启动
public Set<Class<?>> getAnnotatedEndpointClasses(Set<Class<?>> scan) {
System.out.println("-------------websoket start--------00---------");
System.out.println("scan.size() = " + scan.size());
return scan; //必须要返回scan,否则会造成连接失败
}
//接口方式启动
public Set<ServerEndpointConfig> getEndpointConfigs(
Set<Class<? extends Endpoint>> arg0) {
return null;
}
}
服务端完毕!
客户端(Android)
private static final String wsurl = "ws://192.168.0.115:8080/ExampleSeverAndroid";
public static WebSocketConnection mConnect = new WebSocketConnection();
//发送函数
public static void sendMessage(String msg1) {
if (mConnect.isConnected()) {
mConnect.sendTextMessage(msg1);
} else {
Log.i(TAG, "no connection!!");
}
}
WebSocketOptions options = new WebSocketOptions();
options.setMaxMessagePayloadSize(1000000); //max size of message
options.setMaxFramePayloadSize(1000000); //max size of frame
mConnect.connect(wsurl, new WebSocketHandler() {
@Override
public void onOpen() {
System.out.println("open方法调用");
Log.i(TAG, "Status:Connect to " + wsurl);
}
@Override
public void onTextMessage(String payload) {
System.out.println("接受到消息");
System.out.println(payload);
JSONArray array = JSONArray.parseArray(payload);
mRecyclerView = findViewById(R.id.recyclerview);
//每次刷新前清空
mNewsList.clear();
for (int i = 0; i < array.size(); i++) {
News news = new News();
JSONObject jo = array.getJSONObject(i);
news.username = jo.getString("username");
//同时可以等于自己
if(!friends.contains(news.username)&&!login_username.equals(news.username))
continue;
news.time = jo.getString("time");
news.text = jo.getString("text");
news.image = jo.getString("image");
news.comment = jo.getString("comment");
mNewsList.add(news);
}
mMyAdapter = new MyAdapter();
mRecyclerView.setAdapter(mMyAdapter);
LinearLayoutManager layoutManager = new LinearLayoutManager(MomentActivity.this);
mRecyclerView.setLayoutManager(layoutManager);
}
@Override
public void onClose(int code, String reason) {
Log.i(TAG, "Connection lost..");
System.out.println("关闭");
}
},options);
Andriod 中发送时,需要另外开启一个线程,不能再主线程中执行 ,否则会报错:
newThread1 = new Thread(new Runnable() {
@Override
public void run() {
JSONObject msg = new JSONObject();
try {
msg.put("messageType", "get");
//线程睡眠,等websocketwriter创建后再执行sendMessage
Thread.sleep(500);
sendMessage(JSON.toJSONString(msg));
System.out.println("json 数据发送成功");
} catch (Exception e) {
//logger.debug("IO异常");
System.out.println("异常"+e);
}
}
});
newThread1.start();
移动端完毕!
文章及其草率与混乱,,需要摘取,,不要误导读者思路