GlobalWebsoket.js 的使用,实现获取实时数据

news2024/10/6 13:40:10

在称重小程序是使用 GlobalWebsoket 实现获取实时数据

  • 前言
  • 一、逻辑分析
  • 二、实现方式
    • 1.方法整体流转分析 -- 初始化并绑定
      • 1. onLoad
      • 1. init
      • 2. getDeviceInfo
      • 3. initWebSocket
      • 4. setProperties
    • 2.方法整体流转分析 -- 解除绑定
      • 1. onBackPress
      • 2. remoeSubscribe
  • 三、参数调用分析
  • 四、请求查看
  • 整体代码


前言

这里使用的 GlobalWebsoket 是封装好的,详细见

GlobalWebsoket.js 封装配置分析


一、逻辑分析

现在小程序的功能需求是当用户点击某个设备,进入到设备详情页,测试需要创建 websoket 的连接,并监听当前设备的实时数据,切换设备之后,解除上次监听的内容,重新绑定新的设备的实时数据监听

实现思路 :
设备页面打开,获取设备信息成功之后,初始化 initwebsoket ,并发送需要监听的数据
在页面离开的时候,接触所有监听,但是并不断掉 websoket 的连接,只是接触绑定的监听
切换设备后,获取设备信息成功之后绑定新的设备需要监听的内容

二、实现方式

1.方法整体流转分析 – 初始化并绑定

当页面打开,执行 onLoad()

1. onLoad

		onLoad(option) {
			// console.log(option.id)
			this.init(option.id);
			this.deviceId = option.id
			this.device.id = option.id
		},

1. init

			init(deviceId) {
				this.getDeviceInfo(deviceId);
			},

2. getDeviceInfo

			getDeviceInfo(deviceId) {
				DeviceApi.getDevice({
					id: deviceId
					// id:'2208004'
				}).then(res => {
					console.log(res)
					this.maxCapacity = res.data.maxCapacity
					// 这里获取企业名称
					// this.getCustomerNameById(res.data.customerId)
					this.device = res.data || {}
					this.initWebSocket();
					console.log(this.device);

				})

			},

3. initWebSocket

			// 初始化 WebSocket
			initWebSocket() {
				const app = this
				// 这里解除绑定别忘了 !!! 放到了点击返回按钮的时候,也就是回到列表页面
				// app.remoeSubscribe();
				// debugger
				app.subsProperties = [];
				//把当前设备数据里面的metadata(单据,称重...)转成json
				const metadata = JSON.parse(app.device.metadata);
				app.deviceStatus = getWebsocket(
					`instance-editor-info-status-${app.deviceId}`,
					`/dashboard/device/status/change/realTime`, {
						deviceId: app.deviceId,
					},
				).subscribe(resp => {
					const {
						payload
					} = resp;
					app.device.state =
						payload.value.type === 'online' ? {
							value: 'online',
							text: '在线'
						} : {
							value: 'offline',
							text: '离线',
						};
					if (payload.value.type === 'online') {
						app.device.onlineTime = payload.timestamp;
					} else {
						app.device.offlineTime = payload.timestamp;
					}
					app.$set(app.device, 'timeStamp', moment().unix())
				});
				console.log(app.deviceStatus)
				/**
					这里的 getWebsocket 是引入的
						import {
							getWebsocket
						} from '@/js-sdk/socket/GlobalWebSocket.js'
				*/
				app.eventSubs = getWebsocket(
					`location-info-event-${app.deviceId}-${app.device.productId}`,
					`/dashboard/device/${app.device.productId}/events/realTime`, {
						deviceId: app.deviceId,
					},
				).subscribe(resp => {
					const {
						payload
					} = resp;

				}, );
				// 
				app.propertiesMap = {}
				if (metadata.propertiesGroup && metadata.propertiesGroup.length > 0) {


					//获取时间戳(以秒为单位)
					const timeStamp = moment().unix();

					metadata.propertiesGroup.map(group => {
						let points = []
						let properties = {}
						// 这里把 properties 存了起来,方法在下面
						app.setProperties(group);
						group.properties.map(item => {
							points.push(item.id)
							properties[item.id] = item
						})
						if (!app.propertiesMap.data) {
							app.propertiesMap.data = {}
						}

						// 
						app.propertiesMap.data[group.id] = properties
						//"$BELT_DAYBELT_SENSOR.....传递的都是这些里面的数据
						let str = points.join("-");
						// debugger
						const mSub = getWebsocket(
							`instance-info-property-${app.deviceId}-${app.device.productId}-${group.id}-${str}`,
							`/dashboard/device/${app.device.productId}/properties/realTime`, {
								deviceId: app.deviceId,
								groupId: group.id,
								properties: points,
								history: 0,
							},
						).subscribe(resp => {
							const {
								payload
							} = resp;
							const property = payload.value.property;
							const item = app.propertiesMap.data[group.id][property];
							if (item) {
								item.next(payload);
							}
						}, );

						app.subsProperties.push(mSub)
					})

					app.$set(app.propertiesMap, 'timeStamp', timeStamp)
				}
			},

4. setProperties

			setProperties(group) {
				group.properties.map(item => {
					item.listener = [];
					item.onSubscribe = function(callback) {
						item.listener.push(callback);
					}
					item.next = function(data) {
						item.listener.forEach(element => {
							element(data);

						});
					}
					return item;
				})
			},

2.方法整体流转分析 – 解除绑定

1. onBackPress

离开当前页面的生命周期钩子

		onBackPress(){
		  	 console.log('页面返回...')
			 // 解除绑定
			this.remoeSubscribe();
		},

2. remoeSubscribe

			remoeSubscribe() {
				console.log("解绑了")
				const app = this
				console.log(app.deviceStatus)
				console.log(app.eventSubs)
				console.log(app.subsProperties)
				if (app.deviceStatus) {
					app.deviceStatus.unsubscribe();
				}
				if (app.eventSubs) {
					app.eventSubs.unsubscribe();
				}
				if (app.subsProperties.length > 0) {
					app.subsProperties.forEach(function(i, index) {
						i.unsubscribe();
					});
				}
			},

三、参数调用分析

这里记录调用过程中参数的输出,之后 再写


四、请求查看

请求成功后,可以看到绑定了(sub)
在这里插入图片描述当点击返回按钮,可以看到解除了绑定(unsub),以及再次进入,重新绑定(sub)
在这里插入图片描述


整体代码

代码如下:

<template>
	<view>
		<!-- 顶部 -->
		<view class="top">
			<!-- 顶部标题 -->
			<div class="info-title">
				<div>DCL20221025</div>
				<u-button size="mini" type='primary'>在线</u-button>
				<div class="showmore-text" @click="toDeviceInfoPage()">查看设备详情</div>
			</div>
			<!-- 设备信息卡片 -->
			<view class="info-card">
				<!-- info-text -->
				<view class="info-text">所属企业:{{ customerUser }}</view>
				<view class="info-text">物料名称 : xx饲料</view>
				<view class="info-text">罐体容量 : {{maxCapacity}}</view>
				<view class="info-text contact">负责人 :未登记
					<!-- {{contact}} -->
					<u-icon name="phone" color="#60CD9A" size='22px'></u-icon><!-- {{phone}} -->
				</view>
			</view>
		</view>
		<!-- 中间分隔 -->
		<u-gap :bg-color="$u.color.bgColor" height="45"></u-gap>
		<!-- 数据看板 -->
		<view class="data-board">
			<div class="data-box">
				<div class="prop">当前存量</div>
				<div class="data">13.02</div>
			</div>
			<div class="block-box">
				<div class="inner-box show-line"></div>
				<div class="inner-box"></div>
			</div>

			<div class="data-box">
				<div class="prop">满仓率</div>
				<div class="data">10.35%</div>
			</div>

		</view>

		<!-- 中间分隔 -->
		<u-gap :bg-color="$u.color.bgColor" height="45"></u-gap>
		<!-- 运行数据看板 -->
		<view class="running-data-board warpper  my-border_radius">
			<!-- 顶部标题 -->
			<div class="board-title  my-border_radius">
				<div class="title-text">运行数据监控</div>
				<div class="showmore-text" @click="toDeviceRunningDataPage()">查看完整数据</div>
			</div>
			<div class="content">
				<div class="device-img">
					<img src="../../static/img/device.png" style="height: 100%;width: 115px;">
				</div>
				<div class="divice-running-data">
					<div class="data-box">
						<div class="data-item">塔内重量 : 13.02<u-button size="mini" type='error' slot="right"
								style="margin-left:30rpx">缺料</u-button>
						</div>
						<div class="data-item">工作状态 : 运行中<u-icon name="moments" color="#52C08C" size="40"
								style="margin-left:30rpx"></u-icon>
						</div>
						<div class="data-item">倾斜度 : 0.2°<u-icon name="warning" color="#E60707" size="40"
								style="margin-left:30rpx"></u-icon>
						</div>
						<div class="data-item">通讯时间 : 2022-10-25 17:19</div>
					</div>
				</div>
			</div>
			<div class="table">
				<u-table>
					<u-tr>
						<u-td v-for="(val,key) in tableData" style="background-color: #f5f6f8">{{key}}</u-td>
					</u-tr>
					<u-tr>
						<u-td v-for="(val,key) in tableData">{{val}}</u-td>
					</u-tr>
				</u-table>
			</div>
		</view>

		<!-- 中间分隔 -->
		<u-gap :bg-color="$u.color.bgColor" height="45"></u-gap>
		<!-- 喂养数据看板 -->

		<view class="feeding-data-board warpper my-border_radius">
			<!-- 顶部标题 -->
			<div class="board-title  my-border_radius">
				<div class="title-text">喂养数据</div>
				<div class="showmore-text">查看完整数据</div>
			</div>
			<!-- 喂养数据表格部分 -->
			<div class="feeding-data-content">
				<view class="row form-title">
					<view class="time">喂养时间</view>
					<view class="feed-weight">喂养重量</view>
					<view class="staff">操作人</view>
				</view>
				<view class="form-data">
					<u-empty text="数据为空" mode="list" v-if="!hasData"></u-empty>

					<view class="row" v-for="item in feedData" v-if="hasData">
						<view class="time">{{item.time}}</view>
						<view class="feed-weight">{{item.feedWeight}}</view>
						<view class="staff">{{item.staff}}</view>
					</view>
				</view>
			</div>
		</view>
		<span @click="hasData=!hasData" style="background-color: antiquewhite;">临时:点击切换数据是否为空</span>
		<!-- 中间分隔 -->
		<u-gap :bg-color="$u.color.bgColor" height="45"></u-gap>
		<!-- 喂养数据看板 -->

		<view class="running-operating warpper my-border_radius">
			<!-- 顶部标题 -->
			<div class="board-title  my-border_radius">
				<div class="title-text">设备操作</div>
				<!-- <div class="showmore-text">查看完整数据</div> -->
			</div>
			<div class="operation-content">

				<!-- 此处switch的slot为right,如果不填写slot名,也即<u-switch v-model="model.remember"></u-switch>,将会左对齐 -->
				<u-form-item label="开始运行" prop="remember" label-width="170" :border-bottom="false"
					style="font-size: 36rpx;padding-left: 20rpx;padding-right: 100rpx;height:100rpx; display:flex;">
					<u-switch slot="right" v-model="workState" active-color="#7493E9" style="margin-left: 200rpx">
					</u-switch>
				</u-form-item>
				<div class="button-box">
					<u-button size="medium" :customStyle="customStyle" shape="circle">--0-- 置零</u-button>
					<u-button size="medium" :customStyle="customStyle" shape="circle" @click="toParameterSetting()">
						运行参数设置
					</u-button>
				</div>
			</div>
		</view>

		<!-- 中间分隔 -->
		<u-gap :bg-color="$u.color.bgColor" height="45"></u-gap>
	</view>
</template>
<script>
	import moment from 'moment'
	import * as DeviceApi from '@/api/device'
	import * as CustomerApi from '@/api/customer/index.js'
	import * as SiteApi from '@/api/site/index.js'
	import {
		getCustomerNameById
	} from '@/utils/util.js'
	import {
		getWebsocket
	} from '@/js-sdk/socket/GlobalWebSocket.js'


	export default {
		components: {

		},
		watch: {
			workState(val) {
				// 这里监听设备操作中开始关闭工作的 按钮切换
				console.log(val);
			}
		},
		onLoad(option) {
			// console.log(option.id)
			this.init(option.id);
			this.deviceId = option.id
			this.device.id = option.id
		},
		onHide(){
			console.log("页面切走了")
		},
		onBackPress(){
		  	 console.log('页面返回...')
			 // 解除绑定
			this.remoeSubscribe();
		},
		data() {
			return {
				device: {
					id: ''
				},
				deviceId: '',
				deviceStatus: null,
				subsProperties: [],
				eventSubs: null,
				propertiesMap: {},
				propertiesRealData: {},








				maxCapacity: '',
				customerUser: '',
				phone: '',
				contact: '',
				tableData: {
					"CH1": 934,
					"CH2": 456,
					"CH3": 25,
					"CH4": 165,
					"CH5": 95,
					"CH6": 15613,
				},
				hasData: false,
				feedData: [{
						time: '2022-05-10  10:02:56',
						feedWeight: '30吨',
						staff: '刘俊杰1'
					},
					{
						time: '2022-06-10  10:02:56',
						feedWeight: '31吨',
						staff: '刘俊杰2'
					},
					{
						time: '2022-09-10  10:02:56',
						feedWeight: '32吨',
						staff: '刘俊杰3'
					}
				],
				workState: false,
				// 按钮颜色,需要这样写才可以解决兼容问题
				customStyle: {
					background: '#7493E9 !important',
					color: '#fff !important',
					border: 'none !important'
				}
			}
		},
		methods: {

			init(deviceId) {
				this.getDeviceInfo(deviceId);
			},

			getDeviceInfo(deviceId) {
				DeviceApi.getDevice({
					id: deviceId
					// id:'2208004'
				}).then(res => {
					console.log(res)
					this.maxCapacity = res.data.maxCapacity
					// 这里获取企业名称
					// this.getCustomerNameById(res.data.customerId)
					this.device = res.data || {}
					this.initWebSocket();
					console.log(this.device);

				})

			},
			// 初始化 WebSocket
			initWebSocket() {
				const app = this
				// 这里接触绑定别忘了 !!! 
				app.remoeSubscribe();
				// debugger
				app.subsProperties = [];
				//把当前设备数据里面的metadata(单据,称重...)转成json
				const metadata = JSON.parse(app.device.metadata);
				app.deviceStatus = getWebsocket(
					`instance-editor-info-status-${app.deviceId}`,
					`/dashboard/device/status/change/realTime`, {
						deviceId: app.deviceId,
					},
				).subscribe(resp => {
					const {
						payload
					} = resp;
					app.device.state =
						payload.value.type === 'online' ? {
							value: 'online',
							text: '在线'
						} : {
							value: 'offline',
							text: '离线',
						};
					if (payload.value.type === 'online') {
						app.device.onlineTime = payload.timestamp;
					} else {
						app.device.offlineTime = payload.timestamp;
					}
					app.$set(app.device, 'timeStamp', moment().unix())
				});
				console.log(app.deviceStatus)
				app.eventSubs = getWebsocket(
					`location-info-event-${app.deviceId}-${app.device.productId}`,
					`/dashboard/device/${app.device.productId}/events/realTime`, {
						deviceId: app.deviceId,
					},
				).subscribe(resp => {
					const {
						payload
					} = resp;

				}, );
				// 
				app.propertiesMap = {}
				if (metadata.propertiesGroup && metadata.propertiesGroup.length > 0) {


					//获取时间戳(以秒为单位)
					const timeStamp = moment().unix();

					metadata.propertiesGroup.map(group => {
						let points = []
						let properties = {}

						app.setProperties(group);
						group.properties.map(item => {
							points.push(item.id)
							properties[item.id] = item
						})
						if (!app.propertiesMap.data) {
							app.propertiesMap.data = {}
						}

						// 
						app.propertiesMap.data[group.id] = properties
						//"$BELT_DAYBELT_SENSOR.....传递的都是这些里面的数据
						let str = points.join("-");
						// debugger
						const mSub = getWebsocket(
							`instance-info-property-${app.deviceId}-${app.device.productId}-${group.id}-${str}`,
							`/dashboard/device/${app.device.productId}/properties/realTime`, {
								deviceId: app.deviceId,
								groupId: group.id,
								properties: points,
								history: 0,
							},
						).subscribe(resp => {
							const {
								payload
							} = resp;
							const property = payload.value.property;
							const item = app.propertiesMap.data[group.id][property];
							if (item) {
								item.next(payload);
							}
						}, );

						app.subsProperties.push(mSub)
					})

					app.$set(app.propertiesMap, 'timeStamp', timeStamp)
				}
			},
			setProperties(group) {
				group.properties.map(item => {
					item.listener = [];
					item.onSubscribe = function(callback) {
						item.listener.push(callback);
					}
					item.next = function(data) {
						item.listener.forEach(element => {
							element(data);

						});
					}
					return item;
				})
			},
			remoeSubscribe() {
				console.log("解绑了")
				const app = this
				console.log(app.deviceStatus)
				console.log(app.eventSubs)
				console.log(app.subsProperties)
				if (app.deviceStatus) {
					app.deviceStatus.unsubscribe();
				}
				if (app.eventSubs) {
					app.eventSubs.unsubscribe();
				}
				if (app.subsProperties.length > 0) {
					app.subsProperties.forEach(function(i, index) {
						i.unsubscribe();
					});
				}
			},
			// async getCustomerNameById(id) {
			// 	const {
			// 		data: res
			// 	} = await CustomerApi.get({
			// 		id: id
			// 	}).catch(err => {})
			// 	this.customerUser = res.name
			// 	this.phone = res.phone
			// 	this.contact = res.contact
			// },
			// async getSiteNameById(id) {
			// 	const {
			// 		data: res
			// 	} = await SiteApi.get({
			// 		id: id
			// 	}).catch(err => {})
			// 	console.log(res.name)
			// 	console.log(res)
			// },
			// 跳转到设备详情页
			toDeviceInfoPage() {
				// this.device.id = 2208004;
				console.log(this.device.id)
				// console.log('/pages_device/device/detail/device-info&id='+this.device.id);
				uni.navigateTo({
					url: '/pages_device/device/detail/device-info?id=' + this.device.id
				})
			},
			toParameterSetting() {
				uni.navigateTo({
					url: '/pages_device/device/setting/parameter-setting?id=' + this.device.id
				})
			},
			toDeviceRunningDataPage() {
				console.log()
				uni.navigateTo({
					url: '/pages_device/device/setting/parameter-setting'
				})
			},
		}
	}
</script>

<style lang="scss" scoped>
	// 顶部
	.top {
		width: 100%;
		height: 340rpx;
		background-color: #fff;
		padding: 30rpx 20rpx 20rpx 20rpx;

		// 顶部标题
		.info-title {
			display: flex;
			justify-content: space-between;
			font-size: 36rpx;
			height: 50rpx;
			line-height: 50rpx;
			margin-bottom: 20rpx;

			// 顶部标题 查看设备详情
			.showmore-text {
				font-size: 30rpx;
				color: rgba(16, 16, 16, 0.5);
			}
		}

		// 设备信息卡片
		.info-card {
			height: 220rpx;
			width: 80%;
			margin: 0 auto;
			color: rgb(153, 153, 153);

			// 信息文本
			.info-text {
				font-size: 36rpx;
				margin-top: 8rpx;
			}

			// 联系人
			.contact {
				display: flex;
				justify-content: space-between;
				padding-right: 50rpx;
			}

		}
	}

	// 数据看板
	.data-board {
		width: 100%;
		height: 220rpx;
		background-color: #fff;
		border: #BBBBBB 1rpx solid;
		display: flex;
		justify-content: space-around;

		// 左右两侧的数据盒子
		.data-box {
			width: 40%;
			padding: 20rpx 30rpx;

			// 标题
			.prop {
				height: 50%;
				background-color: #fff;
				font-size: 44rpx;
				line-height: 86rpx;
				text-align: center;
				color: rgba(16, 16, 16, 0.5);
			}

			// 数据值
			.data {
				height: 50%;
				font-size: 66rpx;
				line-height: 86rpx;
				text-align: center;
				color: rgb(82, 192, 140);
			}
		}

		// 中间为了显示淑竖线了两个盒子
		.block-box {
			height: 220rpx;
			width: 10%;
			display: flex;
			justify-content: center;
			align-items: center;

			.inner-box {
				height: 60%;
				width: 50%;
			}

			// 竖线的样式
			.show-line {
				border-right: 1px solid #BBBBBB
			}
		}

	}

	// 上面两个圆角的样式
	.my-border_radius {
		border-top-left-radius: 16rpx;
		border-top-right-radius: 16rpx;
	}

	.warpper {
		width: 95%;
		background-color: #fff;
		border: #BBBBBB 1rpx solid;
		margin: 0 auto;


	}

	.running-data-board {
		height: 670rpx;
		position: relative;

		.content {
			width: 100%;
			height: 450rpx;
			// background-color: red;
			display: flex;
			flex-direction: row;

			.device-img {
				// width: 100%;
				height: 100%;
				// background-color: red;
				// background: url('')  fixed center;
				flex: 3;
				text-align: center;
			}

			.divice-running-data {
				height: 100%;
				// background-color: #666;
				flex: 5;
				display: flex;
				justify-content: center;
				align-items: center;
				padding-right: 10rpx;

				.data-box {
					width: 100%;

					.data-item {
						width: 98%;
						height: 70rpx;
						font-size: 32rpx;
						padding-left: 10rpx;
						line-height: 70rpx;

						border-left: 1px solid #DBDBDB;
						border-bottom: 1px solid #DBDBDB;
						border-right: 1px solid #DBDBDB;
					}

					.data-item:first-child {
						border-top: 1px solid #DBDBDB;
					}
				}
			}
		}

		.table {
			width: 100%;
			position: absolute;
			bottom: 0;
		}
	}


	.feeding-data-board {
		height: 400rpx;
		position: relative;

		.feeding-data-content {
			height: 320rpx;
			width: 100%;
			position: absolute;
			bottom: 0;

			.feedbox {
				width: 90%;
				margin-left: 40rpx;
			}

			.form-data {
				width: 100%;
				height: 120px;
				background-color: #fff;

				display: flex;
				flex-direction: column
			}

			.row {
				width: 100%;
				height: 80rpx;
				background-color: #f4f4f4;
				text-align: center;
				line-height: 80rpx;
			}

			.time {

				height: 80rpx;
				background-color: #fff;
				flex: 4;
				border: 1rpx solid #f4f4f4;
			}

			.feed-weight {
				height: 80rpx;
				background-color: #fff;
				flex: 2;
				border: 1rpx solid #f4f4f4;
			}

			.staff {
				height: 80rpx;
				background-color: #fff;
				flex: 2;
				border: 1rpx solid #f4f4f4;
			}
		}
	}


	.running-operating {
		height: 300rpx;
		border-radius: 16rpx;

		.operation-content {
			padding-left: 40rpx;
			padding-right: 40rpx;
			font-size: 52rpx;
		}

		.button-box {
			display: flex;
			justify-content: space-between;
		}
	}


	// 每一个卡片的标题样式
	.board-title {
		display: flex;
		// flex: ;
		justify-content: space-between;
		height: 80rpx;
		background-color: rgba(63, 106, 225, 0.72);
		font-size: 36rpx;
		color: #fff;
		font-weight: 500;
		line-height: 80rpx;
		padding: 0 16rpx;

		.showmore-text {
			font-size: 30rpx;
		}
	}
</style>

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/30456.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

“大数据分析”相比“传统数据分析”优势明显,体现在哪些方面

一、大数据和数据分析的定义&#xff1a; 数据分析&#xff1a;指使用适当的统计分析方法来收集数据&#xff0c;以进行大量数据分析。 大数据分析&#xff1a;指在可承受的时间范围内无法使用常规软件工具捕获&#xff0c;管理和处理的数据集合&#xff1b; 数据分析的核心…

【web渗透思路】任意账号的注册、登录、重置、查看

目录 一、任意用户注册 1.未验证邮箱/手机号 2、不安全验证邮箱/手机号 3.批量注册 4.个人信息伪造 5.前端验证审核绕过 6.用户名覆盖 二、任意用户登录 1、万能密码 2、验证码、密码回显 3、登录检测不安全 三、任意账号重置 1、重置账号名 2、验证码 3、MVC数…

2022年第三季度泛出行行业洞察:泛出行行业正在经历数智化升级的关键时期,用户规模保持平稳增长,行业整体良性发展

易观分析&#xff1a;泛出行行业涵盖综合车主服务、车辆加油充电、网约车、旅游预定、酒店预定、户外出行等领域。当前泛出行领域正在经历传统模式向数智化新模式的转变&#xff0c;智能化升级和服务品质提升在该领域变革中正发挥着积极的作用。未来泛出行领域将在数智化、电动…

Web3:价值投资的范式转移

​潜力博主推荐&#xff0c;点上面关注博主 ↑↑↑ 进化是宇宙中最强大的力量&#xff0c;是唯一永恒的东西&#xff0c;是一切的驱动力。———桥水基金 雷.达利奥 时间拉长&#xff0c;进化才是人类的主旋律。过去&#xff0c;环境的变化是进化的主因。 现在&#xff0c;技…

Servlet | 深度剖析转发和重定向

一&#xff1a;深度剖析转发和重定向 &#xff08;1&#xff09;在一个web应用中通过两种方式可以完成资源的跳转 第一种方式&#xff1a;转发方式 第二种方式&#xff1a;重定向方式 &#xff08;2&#xff09;转发和重定向的区别 区别一&#xff1a;代码上的区别 ①转发 &a…

阿里资深专家撰写出的 Nginx 底层与源码分析手册,GitHub 已爆赞

NGINX 发展史&#xff1a; 过去最主流的服务器是 1995 年发布的 Apache 1.0。Apache 源于 NCSAHTTPd 服务器&#xff0c;是一个多进程模型的 Web 服务器。但运行到后期&#xff0c;Apache 渐渐出现很多问题&#xff0c;比如内存占用很大、扩展需挂接第三方库、并发能力受限等。…

高效的股票数据接口工具有哪些?

我们已经知道了量化投资是是通过数量化方式及计算机程序化发出买卖指令&#xff0c;以获取稳定收益为目的的交易方式&#xff0c;而其中最重要的载体是数据。在金融领域中量化的应用让金融分析师、外汇交易员、产品研发员等技术人员又有了新的用武之地&#xff0c;转型成为量化…

【微信小程序】saveFile:fail tempFilePath file not exist

开发微信小程序尝试保存文件时&#xff0c;会提示saveFile:fail tempFilePath file not exist错误&#xff0c;是什么问题呢&#xff0c;接下来带你如何分析和解决问题 文章目录1. 定位问题2. 解决问题1. 定位问题 首先&#xff0c;看一下代码怎么写得&#xff0c;如下所示 w…

数据结构之线性表中的顺序表【详解】

前言 现在是北京时间11月24号0点2分&#xff0c;天气有一些冷&#xff0c;我现在很困&#xff0c;但是博客还没写&#xff0c;我很想睡觉&#xff0c;假如我现在放弃的码字&#xff0c;往床上一趟&#xff0c;其实也不怎么样&#xff0c;但是我们不能有拖延症&#xff0c;所以…

关于元宇宙的六七八你知道多少?

&#x1f3e0;个人主页&#xff1a;黑洞晓威 &#x1f9d1;个人简介&#xff1a;大家好&#xff0c;我是晓威&#xff0c;一名普普通通的大二在校生&#xff0c;希望在CSDN中与大家一起成长。&#x1f381;如果你也在正在学习Java&#xff0c;欢迎各位大佬来到我的博客查漏补缺…

【MySQL篇】第三篇——表的操作

创建表 创建表案例 查看表结构 修改表 删除表 创建表 在创建数据库之后&#xff0c;接下来就要在数据库中创建数据表了。所谓创建数据表&#xff0c;指的是在已经创建数据库中建立新表。 创建数据表的过程是规定数据列的属性的过程&#xff0c;同时也是实施数据完整性&am…

JAVA实训第二天

目录 JDK8新特性 Java8介绍 JAVA 8主要新特性 Java 8接口增强-默认方法 接口 接口的应用 Lambda表达式介绍 Lambda表达式的写法 功能性接口Lambda表达式的应用 函数接口 JDK8新特性 Java8介绍 •Java8是Java发布以来改动最大的一个版本&#xff0c;其中主要添加了函数式…

自定义数据类型——结构体

我们今天来简单介绍一下结构体。 目录 1. 结构体的声明 2. 结构体成员的访问 3. 结构体传参 首先我们要知道为什么会有结构体的存在&#xff0c;我们的生活里有很多东西&#xff0c;比如一只猫&#xff0c;一本书&#xff0c;一个人&#xff0c;我们如果要用程序来描述他们…

C语言 指针

C语言 指针引言1. 什么是指针2. 简单认识指针3. 取地址符 & 和解引用 * 符一、指针与内存二、指针类型的存在意义1. 指针变量的大小2. 指针移动3. 不同指针类型的解引用三、指针运算1. 指针加减整数程序清单1程序清单22. 指针 - 指针3. 指针关系运算四、二级指针五、野指针…

这个双11,我薅了华为云会议的羊毛

文章目录前言华为云会议悄然助力各行各业和其他云会议产品相比&#xff0c;华为云会议优势是什么&#xff1f;云端一体线下会议室和云会议互通专业会管与会控能力更安全华为云会议有哪些 AI 能力&#xff1f;华为云会议入门有多简单&#xff1f;下载步骤如下安装加入会议预约会…

原生js 之 (DOM操作)

Web API Web API是浏览器提供的一套操作浏览器功能和页面元素的API(BOM和DOM) JavaScipt由三部分构成 ECMAScript、DOM和BOM。 BOM浏览器对象模型&#xff0c;提供了与浏览器的交互方法和接口&#xff1b; DOM 文档对象模型&#xff0c;提供了处理网页内容、结构 、样式的方法…

【数据结构】图的遍历

深度优先遍历 深度优先遍历思想 对于图&#xff1a;选中一个结点&#xff0c;访问和其相邻的、未被访问过的点&#xff0c;全部访问完毕后回退到上一个结点&#xff0c;直至全部结点访问完毕&#xff0c;类似于图的先序遍历&#xff0c;如有邻接表&#xff0c;则按邻接矩阵的顺…

一文熟悉 Go 的基础语法和基本数据类型

一文熟悉 Go 的基础语法和基本数据类型前言Hello&#xff0c;World&#xff01;有关 main 函数的一些要点关键字package声明引入基本数据类型整形数据类型有符号整数类型无符号整数类型其他整数类型浮点数据类型字符类型布尔类型字符串类型基本数据类型的默认值常量和变量声明结…

swift指针内存管理-指针类型使用

为什么说指针不安全 我们在创建一个对象的时候&#xff0c;是需要在堆上开辟内存空间的 但是这个内存空间的声明周期是有限的 也就意味着如果使用指针指向这块内存空间&#xff0c;当这块内存空间的生命周期结束&#xff08;引用计数为0&#xff09;&#xff0c;那么当前的指针…

mac m1 配置goland debbug

大概率无法使用goland的debug功能&#xff0c;如果自己安装没选对路径&#xff0c;也无法使用。原因是&#xff1a; go env 配置不对&#xff0c;需要指向 ARM64&#xff1b; dlv版本不对&#xff0c;需要使用 arm64 系列&#xff1b; dlv路径不对&#xff0c;需要使用 macarm…