websocket实例(显示文件导入处理进度)
大批量数据导入时,需要即时显示对文件的处理进度。考虑ajax轮询太浪费资源,使用websocket实现。
项目使用Spring MVC(3.1),与websocket结合要求版本4.0以上。所以没有使用Spring提供的websocket。
1.依赖Tomcat 7 或者 J2EE 7
maven导入:
<!-- webscoket start -->
<dependency>
<groupId>org.apache.tomcat</groupId>
<artifactId>tomcat-websocket-api</artifactId>
<version>7.0.47</version>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>javax</groupId>
<artifactId>javaee-api</artifactId>
<version>7.0</version>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>org.java-websocket</groupId>
<artifactId>Java-WebSocket</artifactId>
<version>1.3.0</version>
</dependency>
<!-- webscoket end -->
2.服务端
工具类,存储唯一key和连接
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap; import javax.websocket.Session; public class WebsocketSessionUtils { public static Map<String, Session> clients = new ConcurrentHashMap<>(); public static void put(String batchKey, Session session) {
clients.put(batchKey, session);
} public static Session get(String batchKey) {
return clients.get(batchKey);
} public static void remove(String batchKey) {
clients.remove(batchKey);
} public static boolean hasConnection(String batchKey) {
return clients.containsKey(batchKey);
} }
websocket处理类,不要添加业务逻辑
import javax.websocket.*;
import javax.websocket.server.PathParam;
import javax.websocket.server.ServerEndpoint; import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory; @ServerEndpoint("/websocket.ws/{batchKey}")
public class WebsocketEndPoint {
private static Log log = LogFactory.getLog(WebsocketEndPoint.class); @OnOpen
public void onOpen(@PathParam("batchKey") String batchKey, Session session) {
log.info("Websocket Start Connecting:" + batchKey);
WebsocketSessionUtils.put(batchKey, session);
} @OnMessage
public String onMessage(@PathParam("batchKey") String batchKey,
String message) {
return "Got your message (" + message + ").Thanks !";
} @OnError
public void onError(@PathParam("batchKey") String batchKey,
Throwable throwable, Session session) {
log.info("Websocket Connection Exception:" + batchKey);
log.info(throwable.getMessage(), throwable);
WebsocketSessionUtils.remove(batchKey);
} @OnClose
public void onClose(@PathParam("batchKey") String batchKey, Session session) {
log.info("Websocket Close Connection:" + batchKey);
WebsocketSessionUtils.remove(batchKey);
} }
3.客户端
使用js闭包封装websocket,导入websocket.js
var myWebSocket = (function() {
var mySocket = {};
mySocket.tryTime = 0;
mySocket.webSocketUrl = null;
mySocket.batchKey = null;
mySocket.webSocket = null;
mySocket.initSocket = function() {
if (!window.WebSocket) {
// alert("Your browsers don't support websocket.");
return false;
}
mySocket.webSocket = new WebSocket(mySocket.webSocketUrl + mySocket.batchKey);
mySocket.webSocket.onmessage = function(msg) {
console.log(msg);
};
mySocket.webSocket.onerror = function(event) {
console.log(event);
};
mySocket.webSocket.onopen = function(event) {
console.log(event);
};
mySocket.webSocket.onclose = function() {
if (mySocket.tryTime < 10) {
setTimeout(function() {
webSocket = null;
mySocket.tryTime++;
mySocket.initSocket();
}, 500);
} else {
mySocket.tryTime = 0;
}
};
};
mySocket.closeSocket = function() {
if (mySocket.webSocket) {
mySocket.webSocket.close();
}
};
return mySocket;
})();
在ajax执行时启动websocket,使用timestamp作为唯一key。
var commissionWebsocketUrl = "ws://${remote_websocket}/commission/websocket.ws/";
var timestamp=new Date().getTime();
$.ajax({
type : "POST",
url : trialUrl+"?batchKey="+timestamp,
data : $("#batchUploadForm").serialize(),
dataType : "html",
beforeSend: function(xhr) {
var countDiv = $('#loading #countDiv');
if(countDiv.length==0){
countDiv =$("<div id='countDiv'></span>");
$('#loading').append(countDiv);
}
$('#loading').show();
myWebSocket.webSocketUrl = commissionWebsocketUrl;
myWebSocket.batchKey = timestamp;
myWebSocket.initSocket();
myWebSocket.webSocket.onmessage = function(msg) {
countDiv.html("mTusker is processing date,Please wait... "+msg.data);
};
},
complete: function(){
$('#loading').hide();
myWebSocket.closeSocket();
},
success : function(data) {
$("#trialInfoPopup").html(data);
mydialog = $("#trialInfoPopup").dialog({
position: ['center',100] ,width : 1400,height: 600,
modal : true,
title : "Batch Upload Trial"
});
}
});
java项目作为客户端
import java.io.IOException; import javax.websocket.ClientEndpoint;
import javax.websocket.OnError;
import javax.websocket.OnMessage;
import javax.websocket.OnOpen;
import javax.websocket.Session; @ClientEndpoint
public class WebsocketClient {
@OnOpen
public void onOpen(Session session) {
System.out.println("Connected to endpoint:" + session.getBasicRemote());
try {
session.getBasicRemote().sendText("Hello");
} catch (IOException ex) {
}
} @OnMessage
public void onMessage(String message) {
System.out.println(message);
} @OnError
public void onError(Throwable t) {
t.printStackTrace();
}
}
import java.io.IOException;
import java.net.URI; import javax.websocket.ContainerProvider;
import javax.websocket.DeploymentException;
import javax.websocket.Session;
import javax.websocket.WebSocketContainer; public class WebsocketApp {
private static String webUrl = "localhost"; public static void send(String batchKey, String message) {
WebSocketContainer container = ContainerProvider.getWebSocketContainer();
String uri = "ws://"+webUrl+"/websocket.ws/" + batchKey;
System.out.println("Connecting to" + uri);
try {
Session session = WebsocketSessionUtils.get(batchKey);
if (session == null) {
session = container.connectToServer(WebsocketClient.class, URI.create(uri));
WebsocketSessionUtils.put(batchKey, session);
}
session.getBasicRemote().sendText(message);
} catch (DeploymentException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
} public static String getWebUrl() {
return webUrl;
} public static void setWebUrl(String webUrl) {
WebsocketApp.webUrl = webUrl;
} }
把导入文件处理部分,当做java项目客户端,处理过程中调用websocket
int num = 0;
for(Map<String,String> rowInfo : rows){
WebsocketApp.send(batchKey, num++ + "/" + rows.size());
......
}
由此实现在处理文件过程中即时发送处理进度。
注: 当使用https时需要改为
var commissionWebsocketUrl = "wss://${remote_websocket}/commission/websocket.ws/";
websocket实例(显示文件导入处理进度)的更多相关文章
- Websocket实例
C#版Websocket实例 websocket有java.nodejs.python,Php等等版本,我使用的是C#版本,服务器端是Fleck,github地址:https://github.com ...
- C#版Websocket实例
C#版Websocket实例 Demo地址:www.awbeci.xyz websocket有java.nodejs.python,Php等等版本,我使用的是C#版本,服务器端是Fleck,git ...
- Bootstrap历练实例:动画的进度条
<!DOCTYPE html><html><head><meta http-equiv="Content-Type" content=&q ...
- Bootstrap历练实例:堆叠的进度条
<!DOCTYPE html><html><head><meta http-equiv="Content-Type" content=&q ...
- Bootstrap历练实例:默认的进度条
<!DOCTYPE html><html><head><meta http-equiv="Content-Type" content=&q ...
- requests实现文件下载, 期间显示文件信息&下载进度_python3
requests实现文件下载, 期间显示文件信息&下载进度 """使用模块线程方式实现网络资源的下载 # 实现文件下载, 期间显示文件信息&下载进度 # ...
- WebSocket入门教程(五)-- WebSocket实例:简单多人聊天室
from:https://blog.csdn.net/u010136741/article/details/51612594 [总目录] WebSocket入门教程--大纲 [实例简介] ...
- Java web实时进度条整个系统共用(如java上传进度条、导入excel进度条等)
先上图: 这上文件上传的: 这是数据实时处理的: 1:先说说什么是进度条:进度条即计算机在处理任务时,实时的,以图片形式显示处理任务的速度,完成度,剩余未完成任务量的大小,和可能需要处理时间,显示方式 ...
- HTML5 CSS3 诱人的实例 : 网页载入进度条的实现,下载进度条等
今天给大家带来一个比較炫的进度条,进度条在一耗时操作上给用户一个比較好的体验,不会让用户认为在盲目等待,对于没有进度条的长时间等待,用户会任务死机了,毫不犹豫的关掉应用:一般用于下载任务,删除大量任务 ...
随机推荐
- Lodop6 以上打印控件使用,详参考自带说明文档,打印样式及文字大小要特殊设置一下
<link href="../css/cssprint.css" rel="stylesheet" /> <script src=" ...
- 北京易信软科信息技术有限公司-仓库管理系统V1.0
北京易信软科您可信赖的北京软件研发服务商,公司团队有多年应用软件设计制作及开发经验,为各大企业提供软件设计.制作及维护服务,为用户提供可靠高效的应用服务平台 我们通过专业的项目实施流程,为您提供优质的 ...
- @RequestMapping用法详解
@RequestMapping是一个用来处理请求地址映射的注解,可用于类或方法上.用于类上,表示类中的所有响应请求的方法都是以该地址作为父路径. RequestMapping注解有六个属性,下面我们把 ...
- [转]:Delphi 中的哈希表(1): THashedStringList
unit Unit1; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms ...
- spring-security4.1.2的学习
spring security教程 spring security是什么? Spring Security是一个能够为基于Spring的企业应用系统提供声明式的安全访问控制解决方案的安全框架.它提供了 ...
- 艺术品照片融合到背景墙上效果及DEMO
演示地址: 功能:图片按照角度增加阴影,比较复杂的功能就是当墙面是不规则的时候,贴的艺术品必须按照墙面的角度进行变形处理.
- JPA merge(obj) 方法
JPA中的merge类似Hibernate中的saveOrUpdate方法,当数据库中存在id=2的Person,在em.close()时会发送一条update语句,而当数据库中不存在id=2的Per ...
- 《CoffeeScript应用开发》学习: 第四章-改进应用程序
检查值是否存在 使用存在运算符 CoffeeScript中有一个非常有用的存在运算符?,它能正确地处理值是否存在(存在的意思为变量不为undefined或者null)的情况.在变量后添加?来判断它是否 ...
- ycm添加自定义补全路径
修改~/.vim/bundle/YouCompleteMe/third_party/ycmd/cpp/ycm/.ycm_extra_conf.py的flags变量 未改前如下: flags = [ ...
- linux 学习 14 日志管理
第十四讲 日志管理 14.1 日志管理-简介 .日志服务 在CentOS .x中日志服务已经由rsyslogd取代了原先的syslogd服务.rsyslogd日志服务更加先进,功能更多.但是不论该服 ...