Vue3實現視頻播放彈窗組件,支持全屏播放,音量控制,進度條自定義樣式,適配瀏覽器小窗播放,視頻大小自適配,緩沖loading,代碼復制即用

效果圖

?

?組件所需VUE3代碼

<template><div class="video-dialog" :class="fullScreen && 'video-dialog-full-screen'"><el-dialogv-model="props.visible"draggable:show-close="false"title=""centeralign-center@close="changeVisible"><!-- 視頻容器 --><divclass="video-content":style="{backgroundImage: firstFrameUrl ? `url(${firstFrameUrl})` : 'none',}"><!-- 背景磨砂層 --><div class="video-backdrop" v-if="firstFrameUrl"></div><!-- 視頻加載中狀態 --><div class="video-loading" v-if="isLoading"><div class="loading-spinner"></div><p v-if="loadingText">{{ loadingText }}</p></div><!-- 視頻標簽 --><videoref="videoRef":src="videoSrc"@timeupdate="updateProgress"@ended="handleEnded"@click="togglePlay"@volumechange="updateVolume"@seeked="updateBackground"@loadedmetadata="handleMetadataLoaded"@error="handleVideoError"@waiting="handleVideoBuffering"@playing="handleVideoPlaying"></video><!-- 播放錯誤提示 --><div class="video-error" v-if="videoError"><img :src="DataUrl.ErrorIcon || 'https://picsum.photos/48/48'" alt="播放錯誤" class="error-icon" /><p class="error-text">{{ errorMessage }}</p><p class="error-tips">建議使用MP4格式(H.264編碼)重新上傳</p></div><!-- 視頻數據 --><ul class="video-data" v-if="previewInfo"><li><img :src="DataUrl.ZanIcon" alt="" class="data-icon i-1" /><p>760</p></li><li><img :src="DataUrl.PingIcon" alt="" class="data-icon i-2" /><p>670</p></li><li><img :src="DataUrl.LookEyeIcon" alt="" class="data-icon i-3" /><p>890</p></li></ul><!-- 視頻基本信息 --><div class="video-base-info" v-if="previewInfo"><div class="title flex"><h1>網紅名稱</h1><span></span><p>2025.03.30</p></div><p class="note">風掠過耳際時突然懂了:原來旅行不是趕路,是讓山川湖海,把心里的褶皺慢慢燙平。</p></div><img@click="changeVisible":src="DataUrl.CloseRadiuIcon"alt=""class="close-icon"/></div><!-- 視頻進度條 --><div class="video-progress"><div class="progress-bg" @click="seekToPosition"><div class="progress-value" :style="{ width: progress + '%' }"></div><div class="progress-pointer" :style="{ left: progress + '%' }"></div></div></div><!-- 視頻控制欄 --><div class="video-footer-tool"><div class="flex"><!-- 播放按鈕 --><img:src="isPlaying ? DataUrl.PauseIcon : DataUrl.PlayZIcon"alt=""class="icon-1"@click="togglePlay"/><!-- 下一個按鈕 --><img:src="DataUrl.PlayNextIcon"alt=""class="icon-2"@click="playNextVideo"/><!-- 音量控制 --><div class="volume-container"><img:src="isMuted ? DataUrl.MuteIcon : DataUrl.YlIcon"alt=""class="icon-3"@click="toggleMute"/><div class="hover-mute-content" @mouseenter="showVolumeSlider"><divclass="hover-mute-content-value":style="{ width: volumePercent + '%' }"@mousedown="startVolumeAdjust"></div></div></div><!-- 視頻時間 --><p>{{ currentTime }} /<span>{{ durationTime }}</span></p></div><!-- 全屏按鈕 --><div><img:src="fullScreen ? DataUrl.UnFdIcon : DataUrl.FdIcon"alt=""class="icon-4"@click="fullScreen = !fullScreen"/></div></div></el-dialog></div>
</template><script setup lang="ts">
import { ref, onMounted, watch, nextTick } from "vue"
import DataUrl from "@/config/data-url.js"// 定義props
const props = defineProps({visible: {type: Boolean,default: false,},previewInfo: {type: Boolean,default: true,},videoInfo: {type: Object,default: {},},
})// 視頻源
const videoSrc = ref(props.videoInfo?.url ||"https://umi-rise.oss-ap-southeast-1.aliyuncs.com/20250707/WeChat_20250707174726.mp4"
)// 組件狀態
const videoRef = ref(null)
const isPlaying = ref(false)
const isMuted = ref(false)
const progress = ref(0)
const currentTime = ref("00:00")
const durationTime = ref("00:00")
const showControls = ref(true)
const timer = ref(null)
const fullScreen = ref(false)// 音量控制狀態
const volumePercent = ref(70)
const isAdjustingVolume = ref(false)
const isVolumeSliderVisible = ref(false)// 背景控制狀態
const showLeftBackground = ref(false)
const showRightBackground = ref(false)
const videoAspectRatio = ref(0)
const firstFrameUrl = ref("")
const isFirstFrameLoaded = ref(false)// 播放錯誤狀態
const videoError = ref(false)
const errorMessage = ref("")// 新增:加載狀態
const isLoading = ref(true)
const loadingText = ref("準備播放...")const emit = defineEmits(["update-visible", "confim"])// 關閉對話框
const changeVisible = () => {if (videoRef.value) {videoRef.value.pause();isPlaying.value = false; // 重置播放狀態}emit("update-visible", false)
}// 播放/暫停視頻
const togglePlay = () => {if (videoRef.value) {if (isPlaying.value) {videoRef.value.pause()} else {videoRef.value.play().catch((err) => {console.log("自動播放失敗,需要用戶交互后才能播放", err)})}isPlaying.value = !isPlaying.value}
}// 更新進度條
const updateProgress = () => {if (videoRef.value) {const percent = (videoRef.value.currentTime / videoRef.value.duration) * 100progress.value = Math.min(100, percent)currentTime.value = formatTime(videoRef.value.currentTime)if (videoRef.value.duration > 0 && durationTime.value === "00:00") {durationTime.value = formatTime(videoRef.value.duration)}}
}// 格式化時間
const formatTime = (seconds: number) => {const minutes = Math.floor(seconds / 60)const secs = Math.floor(seconds % 60)return `${minutes.toString().padStart(2, "0")}:${secs.toString().padStart(2, "0")}`
}// 進度條點擊跳轉
const seekToPosition = (e: MouseEvent) => {if (videoRef.value) {const progressBar = e.currentTarget as HTMLElementconst rect = progressBar.getBoundingClientRect()const clickX = e.clientX - rect.leftconst percent = (clickX / rect.width) * 100videoRef.value.currentTime = (percent / 100) * videoRef.value.duration}
}// 音量控制
const toggleMute = () => {if (videoRef.value) {isMuted.value = !isMuted.valuevideoRef.value.muted = isMuted.valuevolumePercent.value = isMuted.value ? 0 : volumePercent.value}
}const showVolumeSlider = () => {isVolumeSliderVisible.value = true
}const startVolumeAdjust = (e: MouseEvent) => {isAdjustingVolume.value = trueadjustVolume(e)document.addEventListener("mousemove", adjustVolume)document.addEventListener("mouseup", endVolumeAdjust)e.preventDefault()
}const adjustVolume = (e: MouseEvent) => {if (!isAdjustingVolume.value || !videoRef.value) returnconst volumeBar = document.querySelector(".hover-mute-content") as HTMLElementif (!volumeBar) returnconst rect = volumeBar.getBoundingClientRect()const clickX = e.clientX - rect.leftlet percent = (clickX / rect.width) * 100percent = Math.max(0, Math.min(100, percent))volumePercent.value = percentvideoRef.value.volume = percent / 100if (isMuted.value) {isMuted.value = falsevideoRef.value.muted = false}
}const endVolumeAdjust = () => {isAdjustingVolume.value = falsedocument.removeEventListener("mousemove", adjustVolume)document.removeEventListener("mouseup", endVolumeAdjust)
}// 更新音量顯示
const updateVolume = () => {if (videoRef.value) {if (videoRef.value.muted !== isMuted.value) {isMuted.value = videoRef.value.muted}if (!isMuted.value) {volumePercent.value = Math.round(videoRef.value.volume * 100)}}
}// 更新背景顯示
const updateBackground = () => {if (!videoRef.value || !videoRef.value.videoWidth) return// 計算視頻寬高比videoAspectRatio.value =videoRef.value.videoWidth / videoRef.value.videoHeight// 獲取容器尺寸const container = videoRef.value.parentElementif (!container) returnconst containerWidth = container.offsetWidthconst containerHeight = container.offsetHeight// 計算容器寬高比const containerAspectRatio = containerWidth / containerHeight// 根據寬高比差異調整視頻顯示邏輯const isVideoWider = videoAspectRatio.value > containerAspectRatioshowLeftBackground.value = isVideoWidershowRightBackground.value = isVideoWider
}// 加載第一幀
const loadFirstFrame = async () => {if (!videoRef.value) returntry {// 嘗試獲取視頻編碼信息(部分瀏覽器支持)const videoTracks = videoRef.value.videoTracksif (videoTracks && videoTracks.length > 0) {const codec = videoTracks[0].codec || videoTracks[0].kindif (codec && !codec.includes("avc1") && !codec.includes("h264")) {console.warn("檢測到非H.264編碼,可能無法播放:", codec)// 可在這里顯示警告提示}}// 嘗試直接使用視頻幀await captureFrame()} catch (error) {console.error("直接捕獲失敗:", error)// 嘗試通過Fetch和Blob URL繞過跨域try {const response = await fetch(videoSrc.value)const blob = await response.blob()const blobUrl = URL.createObjectURL(blob)const tempVideo = document.createElement("video")tempVideo.crossOrigin = "anonymous"tempVideo.src = blobUrlawait new Promise((resolve, reject) => {tempVideo.onloadedmetadata = resolvetempVideo.onerror = reject})tempVideo.currentTime = 0.1await new Promise((resolve, reject) => {tempVideo.onseeked = resolvetempVideo.onerror = reject})const canvas = document.createElement("canvas")canvas.width = tempVideo.videoWidthcanvas.height = tempVideo.videoHeightconst ctx = canvas.getContext("2d")ctx.drawImage(tempVideo, 0, 0, canvas.width, canvas.height)firstFrameUrl.value = canvas.toDataURL("image/jpeg", 0.8)isFirstFrameLoaded.value = true// 清理資源URL.revokeObjectURL(blobUrl)tempVideo.remove()} catch (error) {console.error("Blob方法失敗:", error)// 回退到默認占位圖firstFrameUrl.value ="https://umi-rise.oss-ap-southeast-1.aliyuncs.com/20250707/cover-video-00008.png"isFirstFrameLoaded.value = true}}
}const captureFrame = () => {return new Promise((resolve, reject) => {if (!videoRef.value || !videoRef.value.videoWidth) {reject(new Error("視頻未加載"))return}const video = videoRef.valuevideo.currentTime = 0.5const handleSeeked = () => {const canvas = document.createElement("canvas")canvas.width = video.videoWidthcanvas.height = video.videoHeightconst ctx = canvas.getContext("2d")ctx.drawImage(video, 0, 0, canvas.width, canvas.height)try {firstFrameUrl.value = canvas.toDataURL("image/jpeg", 0.8)isFirstFrameLoaded.value = trueresolve()} catch (error) {reject(error)} finally {video.removeEventListener("seeked", handleSeeked)}}video.addEventListener("seeked", handleSeeked)})
}// 處理視頻元數據加載完成
const handleMetadataLoaded = () => {loadingText.value = "加載中..."// 視頻元數據加載完成,但可能還需要緩沖// 不立即隱藏loading,等待canplay或playing事件
}// 處理視頻緩沖
const handleVideoBuffering = () => {if (!isLoading.value) {isLoading.value = trueloadingText.value = "緩沖中..."}
}// 處理視頻開始播放
const handleVideoPlaying = () => {// 視頻真正開始播放時,隱藏loadingisLoading.value = falseisPlaying.value = true
}// 處理視頻播放錯誤
const handleVideoError = () => {if (!videoRef.value) returnconst error = videoRef.value.errorif (!error) return// 根據錯誤碼判斷原因switch (error.code) {case error.MEDIA_ERR_ABORTED:errorMessage.value = "視頻加載被中斷"breakcase error.MEDIA_ERR_NETWORK:errorMessage.value = "網絡錯誤,無法加載視頻"breakcase error.MEDIA_ERR_DECODE:errorMessage.value = "視頻編碼不支持,無法播放"breakcase error.MEDIA_ERR_SRC_NOT_SUPPORTED:errorMessage.value = "視頻格式不支持"breakdefault:errorMessage.value = "播放失敗,請重試"}// 顯示錯誤提示,隱藏第一幀背景和loadingvideoError.value = truefirstFrameUrl.value = ""isLoading.value = falseisPlaying.value = false
}// 播放下一個視頻
const playNextVideo = () => {console.log("播放下一個視頻")
}// 視頻播放結束
const handleEnded = () => {isPlaying.value = falseconsole.log("視頻播放結束")
}// 顯示/隱藏控制欄
const toggleControls = () => {showControls.value = !showControls.valueif (timer.value) clearTimeout(timer.value)timer.value = setTimeout(() => {showControls.value = false}, 3000)
}const playVideo = () => {videoError.value = falseisLoading.value = true // 開始加載loadingText.value = "準備播放..."videoRef.value.removeEventListener("click", toggleControls)videoRef.value.addEventListener("click", toggleControls)videoRef.value.play().catch((err) => {console.log("自動播放失敗", err)isLoading.value = false // 加載失敗,隱藏loading})videoRef.value.addEventListener("click", toggleControls)nextTick(updateProgress)nextTick(updateBackground)// 延遲加載第一幀,確保視頻元素已初始化setTimeout(loadFirstFrame, 100)
}// 生命周期鉤子
onMounted(() => {if (videoRef.value) {playVideo()}// 監聽窗口大小變化window.addEventListener("resize", updateBackground)
})// 監聽全屏狀態變化
watch(() => fullScreen.value,() => {nextTick(updateBackground)}
)// 監聽視頻源變化
watch(() => props.videoInfo?.url,(newUrl, oldUrl) => {if (newUrl && newUrl !== oldUrl) {// 更新視頻源videoSrc.value = newUrl// 重置視頻狀態videoError.value = falseisPlaying.value = falseprogress.value = 0currentTime.value = "00:00"// 加載新視頻時顯示loadingisLoading.value = trueloadingText.value = "準備播放..."// 重新加載并播放新視頻if (videoRef.value) {videoRef.value.src = newUrlvideoRef.value.load()videoRef.value.play().catch((err) => {console.log("切換視頻播放失敗:", err)isPlaying.value = falseisLoading.value = false // 加載失敗,隱藏loading})// 重新加載第一幀loadFirstFrame()}}},{ immediate: true }
)// 組件卸載時清理
onUnmounted(() => {if (timer.value) clearTimeout(timer.value)window.removeEventListener("resize", updateBackground)
})
</script><style lang="scss" scoped>
.video-dialog,
.video-dialog-full-screen {:deep(.el-overlay) {background: rgba(0, 0, 0, 0.66);}:deep(.el-dialog) {width: 900px;--el-dialog-border-radius: 20px;--el-dialog-padding-primary: 0;background: transparent;border-radius: 12px;padding: 0;overflow: hidden;.video-content {width: 100%;height: 503px;position: relative;overflow: hidden;background-size: cover;background-position: center;// 背景磨砂層.video-backdrop {position: absolute;top: 0;left: 0;right: 0;bottom: 0;background: rgba(0, 0, 0, 0.4);backdrop-filter: blur(10px);z-index: 1;}video {max-width: 100%;max-height: 100%;height: 100%;position: absolute;top: 50%;left: 50%;transform: translate(-50%, -50%);object-fit: contain;z-index: 2;}// 加載狀態樣式.video-loading {position: absolute;top: 0;left: 0;right: 0;bottom: 0;z-index: 5; // 高于視頻但低于錯誤提示display: flex;flex-direction: column;justify-content: center;align-items: center;background: rgba(0, 0, 0, 0.6);.loading-spinner {width: 48px;height: 48px;border: 4px solid rgba(255, 255, 255, 0.3);border-radius: 50%;border-top-color: #ff3f81;animation: spin 1s linear infinite;margin-bottom: 16px;}p {color: white;font-size: 16px;font-family: 'PingFang SC', sans-serif;}}@keyframes spin {to { transform: rotate(360deg); }}// 播放錯誤提示.video-error {position: absolute;top: 50%;left: 50%;transform: translate(-50%, -50%);z-index: 10; // 最高層級text-align: center;width: 80%;.error-icon {width: 48px;height: 48px;margin-bottom: 16px;}.error-text {font-size: 18px;color: #ff4d4f;margin-bottom: 8px;}.error-tips {font-size: 14px;color: rgba(255, 255, 255, 0.7);}}// 視頻數據.video-data {position: absolute;right: 24px;bottom: 49px;z-index: 3;li {margin-bottom: 17px;&:last-child {margin-bottom: 0;}}.data-icon {width: 24px;height: 24px;display: block;margin: auto;}p {font-family: DIN, DIN;color: #ffffff;text-shadow: 1px 1px 2px rgba(0, 0, 0, 0.6);margin-top: 15px;font-weight: 400;font-size: 16px;}}.close-icon {position: absolute;right: 24px;top: 35px;width: 50px;height: 50px;z-index: 2;cursor: pointer;}.video-base-info {position: absolute;left: 24px;bottom: 12px;z-index: 3;.title {align-items: center;h1 {font-family: PingFang SC, PingFang SC;font-weight: 500;font-size: 18px;color: #ffffff;text-shadow: 1px 1px 2px rgba(0, 0, 0, 0.6);}span {margin: 0 8px;display: inline-block;width: 2px;height: 2px;background: #ffffff;box-shadow: 1px 1px 2px 0px rgba(0, 0, 0, 0.6);}p {font-family: DIN, DIN;font-weight: 400;font-size: 12px;color: #eae9e8;text-shadow: 1px 1px 2px rgba(0, 0, 0, 0.6);}}.note {margin-top: 2px;width: 234px;font-family: PingFang SC, PingFang SC;font-weight: 400;font-size: 14px;color: #ffffff;text-shadow: 1px 1px 2px rgba(0, 0, 0, 0.6);line-height: 1.5;}}}.video-content::before {content: "";position: absolute;top: 0;left: 0;right: 0;bottom: 0;backdrop-filter: blur(8px);background: rgba(255, 255, 255, 0.1);z-index: 2;pointer-events: none;}// 進度條樣式.video-progress {position: absolute;bottom: 56px;left: 0;right: 0;height: 4px;z-index: 3;.progress-bg {width: 100%;height: 100%;background: rgba(255, 255, 255, 0.2);border-radius: 2px;cursor: pointer;.progress-value {height: 100%;background: #ff3f81;border-radius: 2px;width: 0;transition: width 0.1s;position: relative;}.progress-value::after {content: "";position: absolute;right: 0;top: 50%;transform: translateY(-50%);width: 12px;height: 12px;background: #ff3f81;border-radius: 50%;}.progress-pointer {position: absolute;top: 50%;transform: translateY(-50%);width: 12px;height: 12px;background: #ff3f81;border-radius: 50%;margin-left: -6px;box-shadow: 0 0 8px rgba(255, 63, 129, 0.5);display: none;}}}// 底部控制欄樣式.video-footer-tool {height: 56px;width: 100%;display: flex;justify-content: space-between;align-items: center;background: #16110c;padding: 0 24px;.icon-1,.icon-2,.icon-3 {width: 14.22px;height: 16px;margin-right: 24px;cursor: pointer;object-fit: contain;}.icon-3 {width: 16px;margin-right: 0;}.icon-4 {width: 18px;height: 18px;cursor: pointer;}p {font-family: PingFang SC, PingFang SC;font-weight: 500;font-size: 14px;color: #ffffff;span {color: rgba($color: #ffffff, $alpha: 0.65);}}// 音量控制樣式.volume-container {display: flex;align-items: center;margin-right: 24px;.hover-mute-content {width: 68px;height: 6px;border-radius: 24px;background: rgba(255, 255, 255, 0.37);cursor: pointer;position: relative;display: none;margin-left: 24px;.hover-mute-content-value {position: absolute;top: 0;left: 0;width: 50%;height: 100%;background: #ffffff;border-radius: 24px;transition: width 0.1s;&::after {content: "";position: absolute;right: 0;top: 50%;transform: translateY(-50%);width: 12px;height: 12px;background: #ffffff;border-radius: 50%;box-shadow: 0 0 8px rgba(255, 255, 255, 0.5);}}}}.volume-container:hover {width: calc(72px + 68px - 24px);max-width: calc(72px + 68px - 24px);.hover-mute-content {display: block;}}}}
}// 全屏模式樣式
.video-dialog-full-screen {:deep(.el-dialog) {width: 100%;height: 100%;--el-dialog-border-radius: 0;background: transparent;border-radius: 0;.video-content {width: 100%;height: calc(100vh - 56px);video {max-width: 100%;max-height: 100%;height: 100%;position: absolute;top: 50%;left: 50%;transform: translate(-50%, -50%);object-fit: contain;z-index: 2;}}}
}
</style>

DataUrl? Icon文件
新建或者自定義一個相關文件,將以下ICON.base64代碼引入,這邊用的是js文件

const dataUrl = {"ErrorIcon": ``,    "ZanIcon":``,"PingIcon": ``,"LookEyeIcon": ``,"CloseRadiuIcon": ``,"PauseIcon": ``,"PlayZIcon": ``,"PlayNextIcon": ``,"MuteIcon": ``,"YlIcon": ``,"UnFdIcon": ``,"FdIcon": ``,
}export default dataUrl

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/bicheng/90386.shtml
繁體地址,請注明出處:http://hk.pswp.cn/bicheng/90386.shtml
英文地址,請注明出處:http://en.pswp.cn/bicheng/90386.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

LLM層歸一化:γβ與均值方差的協同奧秘

LLM層歸一化參數均值和方差;縮放和平移參數是什么 層歸一化(Layer Normalization,LN)是深度學習中用于穩定神經網絡訓練的一種歸一化技術 均值和方差參數用于對輸入數據進行標準化處理,即將輸入數據轉換為均值為0、方差為1的標準正態分布 縮放因子γ\gammaγ:標準化后…

智慧場景:定制開發開源AI智能名片S2B2C商城小程序賦能零售新體驗

摘要&#xff1a;智慧場景作為零售行業創新發展的關鍵載體&#xff0c;正深刻改變著消費者的生活方式。本文聚焦智慧零售模式下智慧場景的構建&#xff0c;以定制開發開源AI智能名片S2B2C商城小程序為切入點&#xff0c;深入探討其在零售企業選址布局、商業模式創新、經營理念轉…

QML WorkerScript

WorkerScript是QML中實現多線程編程的關鍵組件&#xff0c;它允許開發者將耗時操作移至后臺線程執行&#xff0c;避免阻塞主UI線程&#xff0c;從而提升應用響應速度和用戶體驗。本文將全面介紹WorkerScript的核心機制、使用方法和最佳實踐。WorkerScript核心機制WorkerScript通…

銳浪報表 Grid++Report 表頭表尾的隱藏

設計銳浪表格的模板時&#xff0c;可以通過設計多個表頭、表尾&#xff0c;表頭、表尾中放入打印控件&#xff0c;可以打印相關的數據。在真實打印時&#xff0c;可以通過打印時讓表頭、表尾隱藏或顯示&#xff0c;實現用戶的表格樣式。一、表頭的指定1、 表頭可以多個&#xf…

低速信號設計之 QSPI 篇

一、引言? 在服務器技術不斷演進的當下,對高效、穩定的數據存儲和傳輸需求日益增長。QSPI(Quad Serial Peripheral Interface)總線作為一種高速、串行的外圍設備接口,在服務器領域中發揮著關鍵作用。它為服務器中的各類存儲設備及部分外圍芯片與主處理器之間提供了快速可…

別只知道暴力循環!我從用戶名校驗功能中領悟到的高效字符集判斷法(1684. 統計一致字符串的數目)

別只知道暴力循環&#xff01;我從用戶名校驗功能中領悟到的高效字符集判斷法 &#x1f60e; 大家好&#xff0c;日常開發中&#xff0c;我們經常會遇到一些看似不起眼&#xff0c;卻能成為性能瓶頸的小模塊。今天&#xff0c;我想和大家分享一個我親身經歷的故事&#xff0c;…

力扣面試150題--在排序數組中查找元素的第一個和最后一個位置

Day 85 題目描述思路 當 nums[mid] < target 時&#xff0c;說明目標值在右側&#xff0c;移動左指針 left mid 1 當 nums[mid] > target 時&#xff0c;說明目標值可能在當前位置或左側&#xff0c;移動右指針 right mid - 1 循環結束后&#xff0c;left 指針會指向第…

C++實戰:人臉識別7大核心實例

計算機視覺實例應用 基于C++的人臉識別實例 以下是一些基于C++的人臉識別實例的示例和實現方法,涵蓋了多種技術和庫的應用。這些例子可以幫助開發者快速上手并實現人臉識別功能。 OpenCV 基礎人臉檢測 使用OpenCV的預訓練模型進行人臉檢測是入門級示例。OpenCV自帶Haar級聯…

Uniapp中使用vue3語法

在setup語法糖中調用uniapp的頁面生命周期 <script setup>import { onShow } from "dcloudio/uni-app"onShow(() > {//hanlder...}) </script>vue2混入在vue3中建議使用組合式API 新建baseHook.js import { ref } from "vue"; export fu…

C++vector(2)

2.vector深度剖析及模擬實現 2.1std::vector的核心框架接口的模擬實現bit::vector vector的模擬實現 2.2 使用memcpy拷貝問題 假設模擬實現的vector中的reserve接口中&#xff0c;使用memcpy進行的拷貝&#xff0c;以下代碼會發生什么問題&#xff1f; int main() {gxl::ve…

IPSec VPN -- 野蠻模式

一、野蠻模式簡介野蠻模式VPN是指IPsec VPN中IKE協商采用野蠻模式&#xff08;Aggressive Mode&#xff09;的虛擬專用網絡。它是IKE第一階段協商的一種方式&#xff0c;與主模式相對&#xff0c;具有協商速度快但安全性稍低的特點。以下是具體介紹&#xff1a;1、工作原理&…

rk3588開發板使用硬件編碼處理視頻

開發板默認下載的ffmpeg是通用版&#xff0c;無法調用rk3588的硬件編碼器&#xff0c;視頻編碼效率低。 nyanmisaka開發了用于jellyfin的ffmpeg&#xff0c;支持rk3588硬件編碼器&#xff0c;編譯方法&#xff1a; https://github.com/nyanmisaka/ffmpeg-rockchip/wiki/Compil…

`neutron router-gateway-set` 操作失敗的可能原因及解決方案

根據提供的錯誤信息和搜索結果&#xff0c;neutron router-gateway-set 操作失敗的可能原因及解決方案如下&#xff1a;一、常見錯誤原因數據庫字符集配置問題&#xff08;中文名支持&#xff09; 表現&#xff1a;若路由器名稱包含中文字符&#xff0c;可能因數據庫字符集非UT…

(一)ZooKeeper 發展歷史

?博客主頁&#xff1a; https://blog.csdn.net/m0_63815035?typeblog &#x1f497;《博客內容》&#xff1a;.NET、Java.測試開發、Python、Android、Go、Node、Android前端小程序等相關領域知識 &#x1f4e2;博客專欄&#xff1a; https://blog.csdn.net/m0_63815035/cat…

OpenCV快速入門之CV寶典

文章目錄OpenCV的基礎應用一、OpenCV簡介&#xff1a;1.1 OpenCV 優勢1.2 OpenCV-Python二、環境安裝2.1 環境導入三、圖像表示3.1 顏色空間&#xff08;Color Space&#xff09;3.2 具體說明3.3 圖像在計算機中的表示四、基本圖像操作4.1 創建窗口**1. 核心窗口行為控制**cv.W…

LangChain4j 兩種類型API

LangChain4j operates on two levels of abstraction: &#xfeff;LangChain4j 提供了兩種類型API抽象Low level. At this level, you have the most freedom and access to all the low-level components such as ChatModel, UserMessage, AiMessage, EmbeddingStore, Embedd…

CLI 與 IDE 編碼代理比較:提升開發效率的兩種路徑

引言 在當今快速發展的軟件開發領域&#xff0c;人工智能編碼助手已成為開發者工具箱中不可或缺的一部分。根據行業報告&#xff0c;使用AI編碼助手可以將開發速度提高55%以上&#xff0c;同時顯著提升代碼質量。目前市場上主要有兩種類型的編碼代理&#xff1a;集成在IDE中的代…

【STM32】FreeRTOS 任務的創建(二)

這篇文章在于 詳細解釋 FreeRTOS 中任務的創建過程&#xff0c;包括任務創建的本質過程、API 詳解、兩種創建方式&#xff08;動態/靜態&#xff09;、任務函數規范、常見錯誤及實踐建議。 這里參照&#xff1a;RTOS官方文檔&#xff1a;https://www.freertos.org/zh-cn-cmn-s…

軟考 系統架構設計師系列知識點之面向服務架構設計理論與實踐(9)

接前一篇文章:軟考 系統架構設計師系列知識點之面向服務架構設計理論與實踐(8) 所屬章節: 第15章. 面向服務架構設計理論與實踐 第3節 SOA的參考架構 15.3 SOA的參考架構 IBM的Websphere業務集成參考架構(如圖15-2所示,以下簡稱參考架構)是典型的以服務為中心的企業集…

分區域材料設計:主承重區 / 次承重區 / 足弓區的彈性參數與刺激強度匹配

你是否總在為足部酸痛、膝蓋不適或腰背僵硬煩惱&#xff1f;穿了昂貴的緩震跑鞋&#xff0c;用了定制矯形器&#xff0c;問題卻反復出現&#xff1f;今天&#xff0c;我們要顛覆一個流傳百年的“常識”——腳不是脆弱的“需要被保護的對象”&#xff0c;而是被錯誤的設計“慣壞…