安卓用websocket实现推送

本文介绍了在Android项目中如何使用WebSocket实现信息推送,详细讲述了遇到的坑,包括WebSocket标准未最终确定、Tomcat7与Tomcat8的区别、WebSocket连接URL的特殊性以及在Chrome、Firefox、IE浏览器中的表现。此外,提供了服务端使用Java注解创建WebSocket Endpoint的示例,以及Android客户端集成WebSocket库并进行通信的代码示例。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

因为项目需要。所以要实现信息的推送,于是乎就要用websocket。

关于websocket还有几个坑。
1websocket现在(2015/12)还没有正式确定。所以有可能还会更改
2tomcat7 和tomcat8中websocket使用方式有很大区别。我们应该使用tomcat8当中的实现方式。
3websocket的链接方式是ws://ip:prot/servlet
但是不要妄想把这个链接放到浏览器地址栏里面,chrome会说url格式不允许,firefox没有反应,ie是直接变成搜索这里面的东西
4我在网上找的教程,服务端都是使用注解。所以让我很不舒服,不过对于使用spring的同学应该感觉很不错

搭建服务端


import javax.websocket.*;
import javax.websocket.server.ServerEndpoint;
import java.io.IOException;

/**
 * Created by shanl on 14-3-2.
 */
@ServerEndpoint(value = "/Websocket1Action")
public class Websocket1Action extends Endpoint {
    private Session session;
    //private static final Logger sysLogger = Logger.getLogger("sysLog");
    @Override
    public void onClose(Session session, CloseReason closeReason) {
        System.out.println("onClose");
    }

    @Override
    public void onError(Session session, java.lang.Throwable throwable) {
        System.out.println("onError");
    }

    @Override
    public void onOpen(Session session, EndpointConfig config) {
        //sysLogger.info("*** WebSocket closed from sessionId " + this.session.getId());
        RemoteEndpoint.Basic remote =  session.getBasicRemote();

        System.out.println("pathParams:"+session.getPathParameters());
        System.out.println("requestParams"+session.getRequestParameterMap());
        session.addMessageHandler(new MyMessageHandle(remote));

    }

    private class MyMessageHandle implements MessageHandler.Whole<String> {
        RemoteEndpoint.Basic remote =  null;
        public MyMessageHandle(RemoteEndpoint.Basic remote){
            this.remote = remote;
        }

        @Override
        public void onMessage(String s) {
            try {
                System.out.println(s);
                remote.sendText("success");
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
}

代码中使用注解让这个类自动变成一个可以被url访问的servlet(他并不是servlet,好像叫endpoint什么的)。因为我项目处ParkTest。所以我想访问这个类的格式

ws://localhost:8080/ParkTest/Websocket1Action

不过注意。如果你在安卓虚拟机里面运行不是这个url。你要把localhost变成你自己的ip地址。

可以看到这端代码中所有的websocket包都是javax里面的。所以用的java自带的东西。比较通用

如果你想测试一下这个代码可以使用下面的html文件

<!DOCTYPE html>
<html>
<head>
    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
    <title>websocket1.html</title>
    <script type="text/javascript">
        var wsuri = "ws://localhost:8080/ParkTest/Websocket1Action";
        var ws = null;
        function startWebSocket() {
            if ('WebSocket' in window)
                ws = new WebSocket(wsuri);
            else if ('MozWebSocket' in window)
                ws = new MozWebSocket(wsuri);
            else
                alert("not support");


            ws.onmessage = function(evt) {
                alert(evt.data);
            };

            ws.onclose = function(evt) {
                alert("close");
            };

            ws.onopen = function(evt) {
                alert("open");
            };
        }

        function sendMsg() {
            ws.send(document.getElementById('writeMsg').value);
        }
    </script>
</head>
<body onload="startWebSocket();">
<input type="text" id="writeMsg"/>
<input type="button" value="send" onclick="sendMsg()"/>
</body>
</html>

安卓端
安卓端我是自己引用了一个包。名字叫做
这里写图片描述

你可以把它添加在项目的依赖文件中,或者百度一个加进来。

下面是安卓的代码

这段代码是自己定义个类,用于和服务器通信。这里面有一个handler是对应activity的handler。用于显示一个toast信息

package com.example.bleuesprit.parkAndroidClient.net;



import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.util.Log;

import java.net.URI;
import java.net.URISyntaxException;

import org.java_websocket.client.WebSocketClient;
import org.java_websocket.drafts.Draft;
import org.java_websocket.drafts.Draft_10;
import org.java_websocket.drafts.Draft_17;
import org.java_websocket.framing.Framedata;
import org.java_websocket.handshake.ServerHandshake;

public class ExampleClient extends WebSocketClient {

    Handler handler;


    public ExampleClient( URI serverUri , Draft draft,Handler handler) {
        super( serverUri, draft );
        this.handler=handler;
    }

    public ExampleClient( URI serverURI ,Handler handler) {
        super(serverURI);this.handler=handler;
    }

    @Override
    public void onOpen( ServerHandshake handshakedata ) {
        Log.i("websocket","opened connection");
        // if you plan to refuse connection based on ip or httpfields overload: onWebsocketHandshakeReceivedAsClient
    }

    @Override
    public void onMessage( String message ) {
        Log.i("websocket", "received: " + message);
        Message handlerMessage=new Message();
        handlerMessage.what=0x123;
        Bundle data=new Bundle();
        data.putString("message",message);
        handlerMessage.setData(data);
        handler.sendMessage(handlerMessage);
    }


    public void onFragment( Framedata fragment ) {
        System.out.println( "received fragment: " + new String( fragment.getPayloadData().array() ) );
    }

    @Override
    public void onClose( int code, String reason, boolean remote ) {
        // The codecodes are documented in class org.java_websocket.framing.CloseFrame
        Log.i("websocket","Connection closed by " + ( remote ? "remote peer" : "us" ));

    }

    @Override
    public void onError( Exception ex ) {
        ex.printStackTrace();
        // if the error is fatal then onClose will be called additionally
    }


}

下面是activity

package com.example.bleuesprit.parkAndroidClient;

import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;

import com.example.bleuesprit.parkAndroidClient.net.ExampleClient;

import org.java_websocket.drafts.Draft_17;

import java.net.URI;
import java.net.URISyntaxException;


/**
 * Created by bleuesprit on 2015/10/28.
 */



public class TestActivity extends Activity {



    Handler handler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            if (msg.what == 0x123) {
                Bundle data=msg.getData();
                String message=data.getString("message");
                Toast.makeText(TestActivity.this,message,Toast.LENGTH_LONG).show();
            }
        }

    };


    Button sendBnt;
    private ExampleClient c=null;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_test);
        sendBnt=(Button)findViewById(R.id.send);


        ExampleClient c = null;
        try {
            c = new ExampleClient( new URI( "ws://219.1,1.1:8080/ParkTest/Websocket1Action" ), new Draft_17(),handler );

        } catch (URISyntaxException e) {
            e.printStackTrace();
        }
        final ExampleClient finalC = c;
        sendBnt.setOnClickListener(
                new View.OnClickListener() {
                    @Override
                    public void onClick(View v) {
                        try {
                            if (finalC != null) {

                                finalC.connectBlocking();
                                finalC.send("handshake");
                            }
                            }catch(InterruptedException e){
                                e.printStackTrace();
                            }


                    }


                }
        );
    }




}

当按钮点下会想服务器发送一个handshake字符串。服务器返回success字符串。handler负责把success字符串显示在界面上

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值