700字范文,内容丰富有趣,生活中的好帮手!
700字范文 > 结合SSE实现实时位置展示与轨迹展示

结合SSE实现实时位置展示与轨迹展示

时间:2022-09-10 14:34:55

相关推荐

结合SSE实现实时位置展示与轨迹展示

概述

实时位置与实时轨迹的展示是webgis中非常常见的一个功能,本文结合SSE来实现实现此功能。

SSE简介

SSE是Sever-Sent Event的首字母缩写,它是基于HTTP协议的,在服务器和客户端之间打开一个单向通道,服务端响应的不再是一次性的数据包,而是text/event-stream类型的数据流信息,在有数据变更时从服务器流式传输到客户端。

websockSSE都可以实现在有数据变更时从服务器主动推送到到客户端,他们的比较如下:

SSE 是基于HTTP协议的,它们不需要特殊的协议或服务器实现即可工作;WebSocket需单独服务器来处理协议。SSE单向通信,只能由服务端向客户端单向通信;webSocket全双工通信,即通信的双方可以同时发送和接受信息。SSE 实现简单开发成本低,无需引入其他组件;WebSocket传输数据需做二次解析,开发门槛高一些。SSE 默认支持断线重连;WebSocket则需要自己实现。SSE 只能传送文本消息,二进制数据需要经过编码后传送;WebSocket默认支持传送二进制数据。

由于SSE单向通信的特性,所以很适合“实时位置与实时轨迹的展示”这样的场景。

实现效果

实现代码

服务端代码

服务端使用Node通过expressstream实现SSE,实现代码如下:

const express = require('express')const stream = require("stream");const app = express()let sse = null, contentId = 0function toDataString(data) {if (typeof data === 'object') return toDataString(JSON.stringify(data));return data.split(/\r\n|\r|\n/).map(line => `data: ${line}\n`).join('');}function random(n, m, is = false) {let res = Math.random() * (m - n) + nconst arr = [-res, res]if(is) res = arr[Math.round(Math.random())]return res}let carDict = {}for (let i = 0; i < 20; i++) {const [xmin, ymin, xmax, ymax] = [114.0422270397205295, 22.5261218968098547, 114.0854819347529912, 22.5488829187520494]carDict['car'+i] = {id: 'car'+i,lon: random(xmin, xmax),lat: random(ymin, ymax),dx: random(0.0001, 0.0005, true),dy: random(0.0001, 0.0005, true)}}app.use(express.static(__dirname + '/web'))app.get("/sse", (req, res) => {res.writeHead(200, {"Content-Type": "text/event-stream; charset=utf-8"});sse = new stream.Transform({objectMode: true });sse._transform = (message, encoding, callback) => {if (ment) sse.push(`: ${ment}\n`);if (message.event) sse.push(`event: ${message.event}\n`);if (message.id) sse.push(`id: ${message.id}\n`);if (message.retry) sse.push(`retry: ${message.retry}\n`);if (message.data) sse.push(toDataString(message.data));sse.push("\n");callback();};sse.write(':ok\n\n');sse.pipe(res);});// 触动定时触发setInterval(() => {for (const carid in carDict) {const {dx, dy} = carDict[carid]carDict[carid].lon += dxcarDict[carid].lat += dy}const message = {data: carDict,event: "dynamicUpdate", // 事件类型,需要客户端添加对应的事件监听id: ++contentId,retry: 2000,};sse?.write(message);}, 1000)app.listen(18888, () => {console.log('express server running at http://127.0.0.1:18888')})

客户端代码

<!DOCTYPE html><html><head><title>XYZ</title><meta charset="utf-8"><link rel="stylesheet" href="/en/v4.6.5/css/ol.css" type="text/css"><style>#map,body,html {margin: 0;padding: 0;width: 100%;height: 100%;overflow: hidden;font-size: 16px;}.tools {position: absolute;top: 20px;right: 20px;z-index: 99;}</style><script src="/en/v4.6.5/build/ol.js"></script></head><body><div id="map" class="map"><div class="tools"><button id="start">start</button><button id="stop">stop</button></div></div><script>let es, mapconst navLayer = new ol.layer.Tile({source: new ol.source.XYZ({url: 'http://webrd01./appmaptile?x={x}&y={y}&z={z}&lang=zh_cn&size=1&scale=1&style=8'})});const styleFunction = (feat) => {const rotation = feat.get('rotation')const stroke = new ol.style.Stroke({color: '#04991f',width: 5});const imageData = ''let style = {}if(rotation) {const carId = feat.get('carId')style = {image: new ol.style.Icon({src: imageData,rotation: rotation}),}} else {style = {stroke}}return new ol.style.Style(style)}let vectorSource = new ol.source.Vector({features: []})let vectorLayer = new ol.layer.Vector({source: vectorSource,style: styleFunction});map = new ol.Map({target: 'map',layers: [ navLayer, vectorLayer ],view: new ol.View({minZoom: 0,maxZoom: 18,center: [132921716.5468307, 2575606.3396163424],zoom: 14})});let carFeature = {},carCoords = {},carRoute = {};function getRotation([x1, y1], [x2, y2]) {let res = Math.atan2(y1 - y2, x1 - x2)res = y2 > y1 ? res + Math.PI / 2 : res - Math.PI / 2return res}const openServerPush = () => {es = new EventSource("/sse");es.addEventListener("dynamicUpdate", (e) => {let carsDara = JSON.parse(e.data)for (const carId in carsDara) {const {lon, lat} = carsDara[carId]const coord = ol.proj.fromLonLat([lon, lat])if(!carCoords[carId]) carCoords[carId] = []carCoords[carId].push(coord)if(carCoords[carId].length > 1) {const geomLine = new ol.geom.LineString(carCoords[carId])if(carRoute[carId]) {carRoute[carId].setGeometry(geomLine)} else {const feature = new ol.Feature({geometry: geomLine,carId});vectorSource.addFeature(feature)carRoute[carId] = feature}}const geom = new ol.geom.Point(coord)if(carFeature[carId]) {carFeature[carId].setGeometry(geom)const prevCoord = carCoords[carId][carCoords[carId].length - 2]carFeature[carId].set('rotation', getRotation(prevCoord, coord))} else {const feature = new ol.Feature({geometry: geom,carId,rotation: 0});vectorSource.addFeature(feature)carFeature[carId] = feature}}});es.onopen = () => {console.log("已开启。。。");};es.onmessage = (e, me) => {console.log("默认推送:" + e.data);};es.onerror = (err) => {console.log(err);};};const closeServerPush = () => {if (es) {es.close();}};document.getElementById('start').onclick = openServerPushdocument.getElementById('stop').onclick = closeServerPush</script></body></html>

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。