uniapp uni实人认证

news2024/11/23 8:43:11

uni实人认证依赖 目前仅支持App平台。

h5端活体人脸检测,使用的是百度云的h5人脸实名认证

使用要求

1、app端

在使用前,请确保您已注册DCloud账号,并已完成实名认证。

然后需要按文档开通服务 业务开通 | uni-app官网

2、h5端

 在使用前,需要根据方案接入要求先准备好 人脸实名认证H5端方案

开发

// index.vue

<template>
	<view class="page">
		<button class="btn" @click="$u.throttle(livenessDetection, 1000)">
            开始刷脸验证
        </button>
	</view>
</template>

<script>
	import { getVerifyToken, h5VerifyIdentifyReturn } from '@/api/user.js'
	import { HTTP_REQUEST_URL, H5_URL } from '@/config.js'
	import permision from '@/utils/js_sdk/permission.js'
	import store from '@/store'
	export default {
		data() {
			return {
                // 终端id
				orgId: 0,
                // 姓名
				legalPerson: '',
                // 身份证
				idCard: '',
			}
		},

		onLoad(options) {
			this.orgId = options.orgId || 0
			this.legalPerson = options.legalPerson || ''
			this.idCard = options.idCard || ''

			// #ifdef H5
			if (options.verify_result && options.verify_result == 1) {
				this.h5VerifyIdentifyReturn()
			}
			// #endif
		},

		methods: {
			// 检查是否开启摄像头权限
			checkPermissionCamera() {
				return new Promise(async (resolve) => {
					switch (uni.getSystemInfoSync().platform) {
						case 'android':
							const val = await permision.requestAndroidPermission('android.permission.CAMERA')
							if (val == 1) {
								// 已获取授权
								resolve()
							} else if (val == -1) {
								// 被永久拒绝授权
								uni.showModal({
									title: '温馨提示',
									content: '摄像头权限已经被您永久拒绝,请前往设置页面打开',
									showCancel: false,
									confirmText: '前往设置',
									success() {
										// 打开当前App的权限设置界面
										permision.gotoAppPermissionSetting()
									}
								})
							}
							break

						case 'ios':
							// 是不是第一次开启相机
							const iosFirstCamera = uni.getStorageSync('iosFirstCamera');
							if (iosFirstCamera) {
								// 判断iOS上是否给予摄像头权限,有权限返回true,否则返回false
								const val2 = await permision.judgeIosPermission('camera')
								if (val2) {
									resolve()
								} else {
									uni.showModal({
										title: '温馨提示',
										content: '摄像头权限已经被您拒绝,请前往设置页面打开',
										showCancel: false,
										confirmText: '前往设置',
										success() {
											// 打开当前App的权限设置界面
											permision.gotoAppPermissionSetting()
										}
									})
								}
							} else {
								uni.setStorageSync('iosFirstCamera', true);
								resolve()
							}
							break
					}
				})
			},

			// 活体认证
			async livenessDetection() {
				// #ifdef MP-WEIXIN
				uni.$u.toast('请使用app进行刷脸验证')
				// #endif

				// #ifdef APP-PLUS
				this.appAuth()
				// #endif

				// #ifdef H5
				await this.checkPermissionCamera()
				this.h5Auth()
				// #endif
			},

			// #ifdef H5
			// h5端实名认证
			async h5Auth() {
				uni.showLoading({
					title: '认证中',
					mask: true
				})

				// 获取verify_token
				const res = await getVerifyToken()
				const verify_token = res.verify_token
				const successUrl = encodeURIComponent(`${ H5_URL }/pages/register/faceDetect?verify_result=1`)
				const failedUrl = encodeURIComponent(`${ H5_URL }/pages/register/faceDetect?verify_result=2`)

				// 跳转到百度云实名认证H5页
				window.location.href = `https://brain.baidu.com/face/print/?token=${ verify_token }&successUrl=${ successUrl }&failedUrl=${ failedUrl }`
				uni.hideLoading()
			},

			// 活体检测成功后的处理
			async h5VerifyIdentifyReturn() {
				uni.showLoading({
					title: '获取认证结果',
					mask: true
				})

				await h5VerifyIdentifyReturn()
				this.successAfter()
			},
			// #endif

			// #ifdef APP-PLUS
			// app端实名认证
			appAuth() {
				uni.showLoading({
					title: '认证中',
					mask: true
				})

				// 1、获取实人认证设备信息
				const metaInfo = uni.getFacialRecognitionMetaInfo()

				// 2、提交姓名、身份证号、metaInfo到云函数获取certifyId
				uniCloud.callFunction({
					name: 'livenessDetection',
					data: {
						step: 1,
						metaInfo,
						legalPerson: this.legalPerson,
						idCard: this.idCard
					}
				}).then(res => {
					if (res.result.code == 500) return uni.$u.toast(res.result.msg)

					const certifyId = res.result.data

					// 3、云函数获取certifyId后,调起实人认证界面
					uni.startFacialRecognitionVerify({
						certifyId,
						// 刷脸圈的颜色 加上这个参数会在ios端会提示 errCode: 10013 刷脸失败
						// progressBarColor: '#4a9f8b;',
						// 认证界面UI朝向 竖屏
						screenOrientation: 'port',
						success: async (e) => {
							if (e.errCode != 0) return uni.$u.toast(e.errMsg)

							// 4、获取认证结果
							const certResult = await uniCloud.callFunction({
								name: 'livenessDetection',
								data: {
									step: 2,
									orgId: this.orgId,
									certifyId,
									serversUrl: HTTP_REQUEST_URL
								}
							})

							if (certResult.result.code == 200) {
								this.successAfter()
							} else {
								uni.hideLoading()
								uni.showModal({
									title: '认证失败',
									content: certResult.result.msg,
									showCancel: false
								})
							}
						},
						fail: (e) => {
							uni.$u.toast(e.errMsg)
						}
					})
				}).catch(err => {
					uni.$u.toast('请求云函数失败:' + JSON.stringify(err))
				})
			},
			// #endif

			// 认证成功后的处理
			successAfter() {
				// 审核状态变更为:待审核
				store.commit('user/set_userStatus', 1)
				uni.showToast({
					title: '认证成功',
					icon: 'success',
					mask: true
				})

				setTimeout(() => {
					uni.switchTab({
						url: '/pages/index/index'
					})
				}, 1200)
			}
		}
	}
</script>

<style lang="scss" scoped>
	.btn {
		width: 440rpx;
		height: 88rpx;
		background: $linear-color;
		border-radius: 44rpx;
		font-size: 32rpx;
		color: #FFFFFF;
		line-height: 44rpx;
		margin-top: 102rpx;
        display: flex;
		flex-direction: row;
		align-items: center;
		justify-content: center;
	}
</style>

这边也贴出页面中有 import 的几个文件,可以参考下

1、改成你们自己服务端的接口地址

// @/api/user.js

// 获取verify_token(百度云实名认证)
export function getVerifyToken() {
	return uni.$u.http.get('/szg-ynf/api/app/userDetail/verifyIdentify')
}

// 实名认证成功
export function h5VerifyIdentifyReturn() {
	return uni.$u.http.get('/szg-ynf/api/app/userDetail/h5VerifyIdentifyReturn')
}

 2、改成你们自己的接口地址

// @/config.js

module.exports = {
	// 请求域名
	HTTP_REQUEST_URL: 'https://xxx.xxx.com',

	// h5地址
	H5_URL: 'https://xxx.xxx.com/h5',
}

3、该文件可以直接复制去使用,不用修改。这也是uniapp官方封装的

// @/utils/js_sdk/permission.js

/**
 * 本模块封装了Android、iOS的应用权限判断、打开应用权限设置界面、以及位置系统服务是否开启
 */
var isIos
// #ifdef APP-PLUS
isIos = (plus.os.name == 'iOS')
// #endif
// #ifdef H5
// 通过webview打开的h5中也是有plus环境的
isIos = uni.getSystemInfoSync().platform == 'ios'
// #endif

// 判断推送权限是否开启
function judgeIosPermissionPush() {
	var result = false
	var UIApplication = plus.ios.import('UIApplication')
	var app = UIApplication.sharedApplication()
	var enabledTypes = 0
	if (app.currentUserNotificationSettings) {
		var settings = app.currentUserNotificationSettings()
		enabledTypes = settings.plusGetAttribute('types')
		console.log('enabledTypes1:' + enabledTypes)
		if (enabledTypes == 0) {
			console.log('推送权限没有开启')
		} else {
			result = true
			console.log('已经开启推送功能!')
		}
		plus.ios.deleteObject(settings)
	} else {
		enabledTypes = app.enabledRemoteNotificationTypes()
		if (enabledTypes == 0) {
			console.log('推送权限没有开启!')
		} else {
			result = true
			console.log('已经开启推送功能!')
		}
		console.log('enabledTypes2:' + enabledTypes)
	}
	plus.ios.deleteObject(app)
	plus.ios.deleteObject(UIApplication)
	return result
}

// 判断定位权限是否开启
function judgeIosPermissionLocation() {
	var result = false
	var cllocationManger = plus.ios.import('CLLocationManager')
	var status = cllocationManger.authorizationStatus()
	result = (status != 2)
	console.log('定位权限开启:' + result)
	// 以下代码判断了手机设备的定位是否关闭,推荐另行使用方法 checkSystemEnableLocation
	/* var enable = cllocationManger.locationServicesEnabled();
	var status = cllocationManger.authorizationStatus();
	console.log("enable:" + enable);
	console.log("status:" + status);
	if (enable && status != 2) {
		result = true;
		console.log("手机定位服务已开启且已授予定位权限");
	} else {
		console.log("手机系统的定位没有打开或未给予定位权限");
	} */
	plus.ios.deleteObject(cllocationManger)
	return result
}

// 判断麦克风权限是否开启
function judgeIosPermissionRecord() {
	var result = false
	var avaudiosession = plus.ios.import('AVAudioSession')
	var avaudio = avaudiosession.sharedInstance()
	var permissionStatus = avaudio.recordPermission()
	console.log('permissionStatus:' + permissionStatus)
	if (permissionStatus == 1684369017 || permissionStatus == 1970168948) {
		console.log('麦克风权限没有开启')
	} else {
		result = true
		console.log('麦克风权限已经开启')
	}
	plus.ios.deleteObject(avaudiosession)
	return result
}

// 判断相机权限是否开启
function judgeIosPermissionCamera() {
	var result = false
	var AVCaptureDevice = plus.ios.import('AVCaptureDevice')
	var authStatus = AVCaptureDevice.authorizationStatusForMediaType('vide')
	console.log('authStatus:' + authStatus)
	if (authStatus == 3) {
		result = true
		console.log('相机权限已经开启')
	} else {
		console.log('相机权限没有开启')
	}
	plus.ios.deleteObject(AVCaptureDevice)
	return result
}

// 判断相册权限是否开启
function judgeIosPermissionPhotoLibrary() {
	var result = false
	var PHPhotoLibrary = plus.ios.import('PHPhotoLibrary')
	var authStatus = PHPhotoLibrary.authorizationStatus()
	console.log('authStatus:' + authStatus)
	if (authStatus == 3) {
		result = true
		console.log('相册权限已经开启')
	} else {
		console.log('相册权限没有开启')
	}
	plus.ios.deleteObject(PHPhotoLibrary)
	return result
}

// 判断通讯录权限是否开启
function judgeIosPermissionContact() {
	var result = false
	var CNContactStore = plus.ios.import('CNContactStore')
	var cnAuthStatus = CNContactStore.authorizationStatusForEntityType(0)
	if (cnAuthStatus == 3) {
		result = true
		console.log('通讯录权限已经开启')
	} else {
		console.log('通讯录权限没有开启')
	}
	plus.ios.deleteObject(CNContactStore)
	return result
}

// 判断日历权限是否开启
function judgeIosPermissionCalendar() {
	var result = false
	var EKEventStore = plus.ios.import('EKEventStore')
	var ekAuthStatus = EKEventStore.authorizationStatusForEntityType(0)
	if (ekAuthStatus == 3) {
		result = true
		console.log('日历权限已经开启')
	} else {
		console.log('日历权限没有开启')
	}
	plus.ios.deleteObject(EKEventStore)
	return result
}

// 判断备忘录权限是否开启
function judgeIosPermissionMemo() {
	var result = false
	var EKEventStore = plus.ios.import('EKEventStore')
	var ekAuthStatus = EKEventStore.authorizationStatusForEntityType(1)
	if (ekAuthStatus == 3) {
		result = true
		console.log('备忘录权限已经开启')
	} else {
		console.log('备忘录权限没有开启')
	}
	plus.ios.deleteObject(EKEventStore)
	return result
}

// Android权限查询
function requestAndroidPermission(permissionID) {
	return new Promise((resolve, reject) => {
		plus.android.requestPermissions(
			[permissionID], // 理论上支持多个权限同时查询,但实际上本函数封装只处理了一个权限的情况。有需要的可自行扩展封装
			function(resultObj) {
				var result = 0
				for (var i = 0; i < resultObj.granted.length; i++) {
					var grantedPermission = resultObj.granted[i]
					console.log('已获取的权限:' + grantedPermission)
					result = 1
				}
				for (var i = 0; i < resultObj.deniedPresent.length; i++) {
					var deniedPresentPermission = resultObj.deniedPresent[i]
					console.log('拒绝本次申请的权限:' + deniedPresentPermission)
					result = 0
				}
				for (var i = 0; i < resultObj.deniedAlways.length; i++) {
					var deniedAlwaysPermission = resultObj.deniedAlways[i]
					console.log('永久拒绝申请的权限:' + deniedAlwaysPermission)
					result = -1
				}
				resolve(result)
				// 若所需权限被拒绝,则打开APP设置界面,可以在APP设置界面打开相应权限
				// if (result != 1) {
				// gotoAppPermissionSetting()
				// }
			},
			function(error) {
				console.log('申请权限错误:' + error.code + ' = ' + error.message)
				resolve({
					code: error.code,
					message: error.message
				})
			}
		)
	})
}

// 使用一个方法,根据参数判断权限
function judgeIosPermission(permissionID) {
	if (permissionID == 'location') {
		return judgeIosPermissionLocation()
	} else if (permissionID == 'camera') {
		return judgeIosPermissionCamera()
	} else if (permissionID == 'photoLibrary') {
		return judgeIosPermissionPhotoLibrary()
	} else if (permissionID == 'record') {
		return judgeIosPermissionRecord()
	} else if (permissionID == 'push') {
		return judgeIosPermissionPush()
	} else if (permissionID == 'contact') {
		return judgeIosPermissionContact()
	} else if (permissionID == 'calendar') {
		return judgeIosPermissionCalendar()
	} else if (permissionID == 'memo') {
		return judgeIosPermissionMemo()
	}
	return false
}

// 跳转到**应用**的权限页面
function gotoAppPermissionSetting() {
	if (isIos) {
		var UIApplication = plus.ios.import('UIApplication')
		var application2 = UIApplication.sharedApplication()
		var NSURL2 = plus.ios.import('NSURL')
		// var setting2 = NSURL2.URLWithString("prefs:root=LOCATION_SERVICES");
		var setting2 = NSURL2.URLWithString('app-settings:')
		application2.openURL(setting2)

		plus.ios.deleteObject(setting2)
		plus.ios.deleteObject(NSURL2)
		plus.ios.deleteObject(application2)
	} else {
		// console.log(plus.device.vendor);
		var Intent = plus.android.importClass('android.content.Intent')
		var Settings = plus.android.importClass('android.provider.Settings')
		var Uri = plus.android.importClass('android.net.Uri')
		var mainActivity = plus.android.runtimeMainActivity()
		var intent = new Intent()
		intent.setAction(Settings.ACTION_APPLICATION_DETAILS_SETTINGS)
		var uri = Uri.fromParts('package', mainActivity.getPackageName(), null)
		intent.setData(uri)
		mainActivity.startActivity(intent)
	}
}

// 检查系统的设备服务是否开启
// var checkSystemEnableLocation = async function () {
function checkSystemEnableLocation() {
	if (isIos) {
		var result = false
		var cllocationManger = plus.ios.import('CLLocationManager')
		var result = cllocationManger.locationServicesEnabled()
		console.log('系统定位开启:' + result)
		plus.ios.deleteObject(cllocationManger)
		return result
	} else {
		var context = plus.android.importClass('android.content.Context')
		var locationManager = plus.android.importClass('android.location.LocationManager')
		var main = plus.android.runtimeMainActivity()
		var mainSvr = main.getSystemService(context.LOCATION_SERVICE)
		var result = mainSvr.isProviderEnabled(locationManager.GPS_PROVIDER)
		console.log('系统定位开启:' + result)
		return result
	}
}

module.exports = {
	judgeIosPermission: judgeIosPermission,
	requestAndroidPermission: requestAndroidPermission,
	checkSystemEnableLocation: checkSystemEnableLocation,
	gotoAppPermissionSetting: gotoAppPermissionSetting
}

因为uniapp 的实人认证需要配合 uniCloud服务空间使用,所以这边贴上服务空间上云函数的代码

目录结构:

// index.js 云函数

'use strict'
const crypto = require('crypto')
exports.main = async (event, context) => {
	// 获取实人认证实例
	const frvManager = await uniCloud.getFacialRecognitionVerifyManager({ requestId: context.requestId })
	if (event.step == 1) {
		try {
			/**
			 * 获取certifyId
			 * @return certifyId 认证id,用于客户端调用认证接口及云函数获取认证结果
			 */
			const result = await frvManager.getCertifyId({
				realName: event.legalPerson,
				idCard: event.idCard,
				metaInfo: event.metaInfo
			})

			if (result.errCode == 0) return { code: 200, data: result.certifyId, msg: '获取certifyId成功' }

			return { code: 500, data: '', msg: `获取certifyId失败:${ result.errMsg }` }
		} catch (err) {
			return { code: 500, data: '', msg: '请检查您填写的姓名与身份证号是否正确' }
		}
	} else {
		// 获取认证结果
		const result = await frvManager.getAuthResult({ certifyId: event.certifyId })

		// 认证失败
		if (result.authState == 'FAIL') return { code: 500, data: result.certifyId, msg: `认证失败:${ result.errMsg }` }

		// 认证成功
        // 这里改成你们服务端的认证成功后的接口地址...
        // 服务端的接口为了安全起见,所以进行了加密验签
		const url = event.serversUrl + '/szg-ynf/oauth/userDetail/verifyIdentify'

		// md5加密方式:orgId 时间戳 私钥
		const orgId = event.orgId
		const timestamp = new Date().getTime()
		const signKey = '2bba3152-0b82-bd8e-2afd-453b0d925ea4'
		const sign = crypto.createHash('md5').update(orgId + timestamp + signKey).digest('hex')

		const res = await uniCloud.httpclient.request(url, {
			method: 'POST',
			data: {
				orgId,
				timestamp,
				sign
			},
			contentType: 'json',
			dataType: 'json',
			rejectUnauthorized: false
		})
		return { code: res.data.code, data: res.data.data, msg: res.data.msg }
	}
}
// package.json 云函数

{
	"name": "livenessDetection",
	"dependencies": {},
	"extensions": {
		"uni-cloud-verify": {}
	}
}

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

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

相关文章

工程系统管理 工程项目管理系统源码 工程项目各模块及其功能点清单

工程项目各模块及其功能点清单 一、系统管理 1、数据字典&#xff1a;实现对数据字典标签的增删改查操作 2、编码管理&#xff1a;实现对系统编码的增删改查操作 3、用户管理&#xff1a;管理和查看用户角色 4、菜单管理&#xff1a;实现对系统菜单的增删改查操…

FFmpeg、x264以及fdk-aac 编译整合

接上文 FFMPEG 编译流程(极客版) 编译 fdk-aac libfdk-aac version:0.1.5 下载 wget http://jaist.dl.sourceforge.net/project/opencore-amr/fdk-aac/fdk-aac-0.1.5.tar.gz#!/bin/bash NDK/home/maqi/Desktop/android-ndk-r20b # 这里需要替换成你本地的 NDK 路径&#x…

【来不及刷题之】43、最小栈(PriorityQueue)

因为要在常量时间内查询出最小值&#xff0c;所以需要有另外的数据结构维护最小值&#xff0c;很自然地想到了“堆”这个结构&#xff0c;“最小堆”的堆顶元素刚好是最小值因此出栈和入栈的同时也要维护好最小堆 class MinStack {PriorityQueue<Integer> heap;LinkedLi…

软件测试前途如何?要学吗?

1.前言 当我们面临择业问题的时候&#xff0c;因为我们本身对自己认知的不清晰和对现有自己能选择的岗位的不了解&#xff0c;往往不知道如何选择才是最优解。这个博客就专门来解答大家对于软件测试这个岗位的疑惑&#xff0c;让大家对软件测试这个岗位有更广义的了解。 本博…

观察者模式(下):如何实现一个异步非阻塞的EventBus框架?

上一节课中&#xff0c;我们学习了观察者模式的原理、实现、应用场景&#xff0c;重点介绍了不同应用场景下&#xff0c;几种不同的实现方式&#xff0c;包括&#xff1a;同步阻塞、异步非阻塞、进程内、进程间的实现方式。 同步阻塞是最经典的实现方式&#xff0c;主要是为了…

C++之std::is_same用法(一百五十八)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 人生格言&#xff1a; 人生…

Flink HA方案介绍

1.Flink HA方案介绍 每个Flink集群只有单个JobManager&#xff0c;存在单点失败的情况。Flink有YARN、Standalone和Local三种模式&#xff0c;其中YARN和Standalone是集群模式&#xff0c;Local是指单机模式。但Flink对于YARN模式和Standalone模式提供HA机制&#xff0c;使集群…

【历史上的今天】7 月 12 日:世界上第一台商用数字计算机;Python 之父卸任 BDFL;Wacom 成立

整理 | 王启隆 透过「历史上的今天」&#xff0c;从过去看未来&#xff0c;从现在亦可以改变未来。 今天是 2023 年 7 月 12 日&#xff0c;在 1854 年的今天&#xff0c;伊士曼柯达公司的创始人、胶卷的发明者乔治伊斯曼&#xff08;George Eastman &#xff09;出生&#xf…

NC55 最长公共前缀

import java.util.*;public class Solution {/*** 代码中的类名、方法名、参数名已经指定&#xff0c;请勿修改&#xff0c;直接返回方法规定的值即可*** param strs string字符串一维数组* return string字符串*/public String longestCommonPrefix (String[] strs) {int n s…

类和对象(—)

今天&#xff0c;我带来类和对象的基础篇。 目录 面向对象和面向过程类类的概念类的定义类的访问限定符c中struct和class的区别封装类的作用域类的实例化类的存储结构体内存对齐规则【面试题】this指针this指针的概念this指针的特性 【面试题】 面向对象和面向过程 C语言是面向…

[PyTorch][chapter 44][时间序列表示方法3]

简介: word2vec 是 Google 于 2013 年开源推出的一个用于获取 word vector 的工具包&#xff0c;它简单、高效&#xff0c;因此引起了很多人的关注。由于 word2vec 的作者 Tomas Mikolov 其主要知识点 目录&#xff1a; word2vec 基本思想 Skip-gram cbow Hierarchical sof…

MedNeXt的一些问题集锦

归纳偏差是一种关于机器学习算法的目标函数的假设&#xff0c;也就是目标函数评分的标准。 归纳偏差是指模型更容易学习到训练数据中的局部和表面特征&#xff0c;而较难捕捉全局和抽象特征。 scalable 可扩展的 network-wide优势&#xff1f;&#xff1f;&#xff1f; 深度监…

(论文精读)PRUNING FILTER IN FILTER《滤波器中的剪枝滤波器》

论文地址&#xff1a;原文 代码实现 中文翻译 一、精读论文 论文题目 PRUNING FILTER IN FILTER 论文作者 Fanxu Meng 孟繁续 刊物名称 NeurIPS 2020 出版日期 2020 摘要 剪枝已成为现代神经网络压缩和加速的一种非常有效的技术。现有的剪枝方法可分为两大类:滤波器…

MVC三层架构

1.MVC三层架构 MVC&#xff08;Model-View-Controller&#xff09;是一种常见的软件设计模式&#xff0c;用于组织和管理应用程序的代码和逻辑。它将应用程序分为三个主要部分&#xff1a;模型&#xff08;Model&#xff09;、视图&#xff08;View&#xff09;和控制器&#…

营销同质化,博鱼 sports牵手那不勒斯打开新大门

体育营销是企业进入新市场的经典方式&#xff0c;特别是对当今寻求高质量发展的国产品牌而言&#xff0c;从产品出海升级为品牌出海&#xff0c;体育营销可谓是一条必经之路。海信、OPPO、华为等中国品牌通过持续的体育营销不断拉近自身与海外消费者的距离&#xff0c;成功在海…

反常积分定义

目录 反常积分的定义 判断敛散性的方法 方法2&#xff1a; 例题 无界函数的反常积分 判断敛散性的方法 例题 反常积分的定义 该极限存在就表示该反常积分收敛 对于定义3&#xff0c;只有两个都收敛的情况下&#xff0c;原反常积分才收敛。 判断敛散性的方法 始终大的函数形成…

走进USB的U1模式

综述&#xff1a; PCIE有PM和ASPM两种功耗管理模式&#xff0c;USB只有一种 USB有U1/U2/U3三种低功耗模式 本文只针对U1进行分析 如下图所示&#xff0c;为主要状态变换 背景知识 U1是一种低功耗模式&#xff0c;定义的是link的状态不是设备的状态发送LGO_X进入低功耗模式&a…

C++图形开发(12):随机方块的速度和高度

文章目录 1.随机高度2.随机速度3.整段代码4.总结 1.随机高度 那既然是随机&#xff0c;自然少不了随机函数rand()咯~ 详见&#xff1a;C爱好者的自我修养&#xff08;17&#xff09;:rand()随机函数 那么随机速度就可以是&#xff1a; rect_height rand() % int(height / 4)…

【在 WSL2 GUI 中使用 pulseaudio 播放声音】

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、pulseaudio 是什么&#xff1f;二、部署步骤1.下载PulseAudio2.前提条件3.开始配置1.配置pulseaudio2.配置麦克风权限3.配置WSL2外放4.配置WSL2麦克风 总结…

hbase之引入Phoenix

Phoenix简介 Phoenix是HBase的开源SQL皮肤。可以使用标准JDBC API代替HBase客户端API来创建表&#xff0c;插入数据和查询HBase数据。 1&#xff09;容易集成&#xff1a;如Spark&#xff0c;Hive&#xff0c;Pig&#xff0c;Flume和Map Reduce&#xff1b; 2&#xf…