因为项目需要。所以要实现信息的推送,于是乎就要用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字符串显示在界面上