websocket 服务器端(tomcat)与 客户端(Android)配置搭建实现通信详解

服务端部署
在服务端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();

移动端完毕!

文章及其草率与混乱,,需要摘取,,不要误导读者思路

  • 1
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值