uniapp處理流式請求

在uniapp里面處理流式請求相對于web端來說有點麻煩,下面我將講述幾種處理流式請求的方式。

1.websocket

WebSocket 是處理實時數據流的最佳選擇之一,UniApp 提供了原生的 WebSocket 支持:

<template><view class="container"><scroll-view scroll-y class="data-container" :scroll-top="scrollTop"><view v-for="(item, index) in messages" :key="index" class="message-item">{{ item }}</view></scroll-view><view class="control-panel"><button @click="connectSocket" type="primary" :disabled="isConnected">連接</button><button @click="closeSocket" type="warn" :disabled="!isConnected">斷開</button></view></view>
</template><script>
export default {data() {return {socketTask: null,isConnected: false,messages: [],scrollTop: 0}},methods: {connectSocket() {this.socketTask = uni.connectSocket({url: 'wss://your-websocket-server.com/stream',success: () => {console.log('準備連接...')}})this.socketTask.onOpen(() => {this.isConnected = truethis.addMessage('連接已建立')// 訂閱數據流this.socketTask.send({data: JSON.stringify({action: 'subscribe'})})})this.socketTask.onMessage((res) => {this.addMessage(res.data)})this.socketTask.onClose(() => {this.isConnected = falsethis.addMessage('連接已關閉')})this.socketTask.onError((res) => {this.addMessage('錯誤: ' + JSON.stringify(res))})},closeSocket() {if (this.socketTask && this.isConnected) {this.socketTask.close()}},addMessage(msg) {this.messages.push(typeof msg === 'string' ? msg : JSON.stringify(msg))this.$nextTick(() => {this.scrollTop = 99999 // 滾動到底部})}},onUnload() {// 頁面卸載時關閉連接this.closeSocket()}
}
</script><style>
.container {display: flex;flex-direction: column;height: 100vh;padding: 20rpx;
}
.data-container {flex: 1;border: 1px solid #eee;padding: 20rpx;margin-bottom: 20rpx;background-color: #f9f9f9;
}
.message-item {padding: 10rpx;border-bottom: 1px solid #eee;word-break: break-all;
}
.control-panel {display: flex;justify-content: space-around;padding: 20rpx 0;
}
</style>

2.uni.request 處理流式請求

這種方式適用于h5頁面和小程序,不適用于app

<template><view class="container"><view class="stream-container"><view v-for="(item, index) in streamData" :key="index" class="stream-item">{{ item }}</view></view><button @click="startStream" type="primary">開始接收流數據</button></view>
</template><script>
export default {data() {return {streamData: [],dataBuffer: ''}},methods: {startStream() {uni.request({url: 'https://your-stream-api.com/stream',method: 'GET',enableChunked: true,dataType: 'text',onChunkReceived: (res) => {this.handleChunk(res.data);},success: (res) => {uni.showToast({title: '流數據接收完成',icon: 'success'});},fail: (err) => {uni.showModal({title: '錯誤',content: '流數據接收失敗: ' + JSON.stringify(err),showCancel: false});}});},handleChunk(chunk) {// 將接收到的數據添加到緩沖區this.dataBuffer += chunk;// 處理可能的換行符分隔的數據const lines = this.dataBuffer.split('\n');// 保留最后一個可能不完整的行this.dataBuffer = lines.pop() || '';// 處理完整的行for (const line of lines) {if (line.trim()) {try {// 嘗試解析JSONconst data = JSON.parse(line);this.streamData.push(JSON.stringify(data));} catch (e) {// 非JSON數據直接顯示this.streamData.push(line);}}}}}
}
</script><style>
.container {padding: 20px;
}
.stream-container {border: 1px solid #eee;padding: 10px;margin-bottom: 20px;max-height: 300px;overflow-y: auto;
}
.stream-item {padding: 5px 0;border-bottom: 1px solid #f5f5f5;
}
</style>

3.使用 SSE (Server-Sent Events)

雖然 UniApp 沒有原生的 SSE API,但可以通過封裝 XMLHttpRequest 來實現 SSE:

function createSSEConnection(url) {// 創建一個標準的XMLHttpRequest對象const xhr = new XMLHttpRequest()xhr.open('GET', url, true)xhr.setRequestHeader('Accept', 'text/event-stream')xhr.setRequestHeader('Cache-Control', 'no-cache')// 設置響應類型為文本xhr.responseType = 'text'// 數據緩沖區let buffer = ''// 處理進度事件xhr.onprogress = function(e) {// 獲取新數據const newData = xhr.responseText.substring(buffer.length)if (newData) {buffer += newData// 按行分割數據const lines = newData.split('\n')for (const line of lines) {if (line.startsWith('data:')) {const eventData = line.substring(5).trim()// 觸發數據處理handleSSEData(eventData)}}}}xhr.onerror = function(e) {console.error('SSE連接錯誤:', e)}xhr.onreadystatechange = function() {if (xhr.readyState === 4) {if (xhr.status === 200) {console.log('SSE連接完成')} else {console.error('SSE連接失敗:', xhr.status)}}}// 發送請求xhr.send()return xhr
}// 處理SSE數據
function handleSSEData(data) {try {const parsedData = JSON.parse(data)console.log('收到SSE數據:', parsedData)// 處理數據...} catch (e) {console.log('收到SSE文本:', data)// 處理非JSON數據...}
}// 使用方法
const sseConnection = createSSEConnection('https://your-sse-endpoint.com/events')// 關閉連接
function closeSSE() {if (sseConnection) {sseConnection.abort()}
}

4.使用分頁和輪詢模擬流

對于不支持真正的流式請求的場景,可以使用分頁和輪詢來模擬流式體驗:

// 輪詢獲取數據
let lastId = 0
let isPolling = falsefunction startPolling() {isPolling = truepoll()
}function stopPolling() {isPolling = false
}function poll() {if (!isPolling) returnuni.request({url: 'https://your-api.com/data',data: {last_id: lastId,limit: 10},success: (res) => {const data = res.dataif (data && data.items && data.items.length > 0) {// 處理接收到的數據processItems(data.items)// 更新最后ID用于下次請求lastId = data.items[data.items.length - 1].id}// 如果還有更多數據,繼續輪詢if (data.has_more) {setTimeout(poll, 1000) // 1秒后再次輪詢} else {console.log('所有數據接收完畢')isPolling = false}},fail: (err) => {console.error('輪詢失敗:', err)// 錯誤后延遲重試setTimeout(() => {if (isPolling) poll()}, 3000)}})
}

5.app端適用renderjs

該方式適用于app端進行流式請求,在app端上面的幾種方式我都嘗試過除了websocket沒有一個能用的,經過反復的查詢才找到適用renderjs這種方式。

RenderJS 是 UniApp 提供的一個運行在視圖層的 JavaScript 引擎,允許開發者直接操作 DOM 和使用瀏覽器特有的 API。

<viewclass="":sseValue="sseValue":change:sseValue="renderScript.getSseValue":messagesRenderjs="messagesRenderjs":change:messagesRenderjs="renderScript.getMessage":downSend="downSend":change:downSend="renderScript.getDownSend":modelchangeValue="modelchangeValue":change:modelchangeValue="renderScript.getModel"></view>
通過這種方式來調用renderjs
<script module="renderScript" lang="renderjs">
export default {data() {return {VITE_AIR14B_url: '',VITE_AILOCAL:'',VITE_ANYTHING:'',modelValue: 'qwen2.5:14b',messages: [],downValue: false,loading: false,downSentValue: false,biaoshi: false,model:'qianwen'};},methods: {getMessage(val) {// console.log(val,'message')this.messages = val;},async getSseValue(val) {console.log(val,'val')if(this.model=='qianwen'){await this.changeModel(val);}else if(this.model=='locel'){await this.changeModelLocal(val)}else if(this.model=='water'){await this.changeModelAnything(val)}else if(this.model=='sxyd'){console.log('sxyd')}},// 暫停生成getDownSend(val) {console.log(val, 'valvalval');if (!this.biaoshi) {this.biaoshi = true;} else {this.downSentValue = true;}//停止生成的標志// this.downSentValue=true},// 獲取模型getModel(val) {console.log(val);this.model=val},// 接收流式數據qianwenasync changeModel(val) {let params = {model: this.modelValue,messages: this.messages,stream: true};try {// 發送請求到 APIconst response = await fetch(this.VITE_AIR14B_url + '/api/chat', {method: 'POST',headers: {'Content-Type': 'application/json'},body: JSON.stringify(params)});// 獲取流式響應體const reader = response.body.getReader();const decoder = new TextDecoder('utf-8'); // 解碼器,將字節流轉換為字符串this.downValue = false;let value = ''; // 用于拼接收到的每一部分 response 內容// 獲取響應之后關閉loading// this.$ownerInstance.callMethod('closeLoading', null);this.messages.push({role: 'assistant',content: ''});// 逐塊讀取數據while (!this.downValue) {// 讀取數據塊const { done: isDone, value: chunk } = await reader.read();// if (this.onDown.value) {//     this.downValue = true;//     break;// }if (this.downSentValue) {this.downValue = true;this.downSentValue = false;this.$ownerInstance.callMethod('setSend', true);this.messages[this.messages.length - 1].content += '\n\n[已取消生成]';this.$ownerInstance.callMethod('setMessageValue', this.messages);break;}this.downValue = isDone;if (this.downValue) {this.$ownerInstance.callMethod('setSend', true);} else {this.$ownerInstance.callMethod('setSend', false);}let chunkString = decoder.decode(chunk, { stream: true });// 將字節流轉換為字符串并追加到 valuevalue += chunkString;// 打印已接收的部分數據// console.log('Received chunk:', chunkString);if (chunkString && this.messages[this.messages.length - 1].role === 'assistant') {try {const parsedChunk = JSON.parse(chunkString);const content = parsedChunk?.message?.content; // 使用可選鏈避免屬性訪問錯誤if (content === undefined) {throw new Error('Invalid chunk structure: Missing message.content');}// 確保 messages 數組非空if (this.messages.length === 0) {throw new Error('Messages array is empty');}this.messages[this.messages.length - 1].content += content;if(this.messages[this.messages.length - 1].content){// 獲取響應之后關閉loadingthis.$ownerInstance.callMethod('closeLoading', null);}// 將值傳遞出去this.$ownerInstance.callMethod('setMessageValue', this.messages);} catch (error) {console.error('處理 chunk 時發生錯誤:', error);}}}// console.log('Final response:', value);} catch (error) {console.error('Error during fetch request:', error);this.loading = false;// 顯示錯誤消息if (this.messages.length > 0 && this.messages[this.messages.length - 1].role === 'assistant') {this.messages[this.messages.length - 1].content += '\n\n[生成回答時出現錯誤]';}}},}
};
</script>

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

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

相關文章

低代碼理解

一、低代碼開發的核心定義 低代碼開發是通過可視化界面和聲明式編程替代傳統手工編碼的軟件開發范式&#xff0c;其核心目標是&#xff1a; 降低技術門檻&#xff1a;允許非專業開發者&#xff08;公民開發者&#xff09;參與應用構建提升開發效率&#xff1a;通過復用預制組…

WHAM 人體3d重建部署筆記 vitpose

目錄 視頻結果: docker安裝說明: conda環境安裝說明: 依賴項: 依賴庫: 安裝 mmpose,mmcv 下載模型權重: 算法原理, demo腳本 報錯inference_top_down_pose_model: 測試命令: 視頻結果: wham_smpl預測結果 git地址: GitHub - yohanshin/WHAM WHAM: Recons…

react 大屏根據屏幕分辨率縮放

記錄&#xff0c;以防忘記 const DataLargeScreen () > {const layoutRef useRef<any>();// ui稿寬度const width useRef(1920).current;// ui稿高度const height useRef(1080).current;const [scaleValue, setScaleValue] useState(1);const useWhichScaleValu…

【網絡】網關

【網絡】網關 網關 是計算機網絡中用于連接兩個不同網絡的設備或服務器&#xff0c;它充當著“翻譯器”和“轉發器”的角色&#xff0c;將數據包從一個網絡傳遞到另一個網絡&#xff0c;并在必要時進行協議轉換和數據重包裝。 主要功能 數據轉發&#xff1a;當本地網絡設備發…

Axure大屏可視化模板:賦能多領域,開啟數據展示新篇章

在當今這個數據爆炸的時代&#xff0c;數據已經成為各行各業的核心資產。然而&#xff0c;如何高效、直觀地展示數據&#xff0c;并將其轉化為有價值的決策依據&#xff0c;成為了許多企業和組織面臨的共同挑戰。Axure大屏可視化模板&#xff0c;作為一款強大的數據展示工具&am…

數據不外傳!通過內網穿透實現綠聯NAS遠程訪問的安全配置方案

文章目錄 前言1. 開啟ssh服務2. ssh連接3. 安裝cpolar內網穿透4. 配置綠聯NAS公網地址 前言 大家好&#xff0c;今天要帶給大家一個超級酷炫的技能——如何讓綠聯NAS秒變‘千里眼’&#xff0c;通過簡單的幾步操作就能輕松實現內網穿透。想象一下&#xff0c;無論你身處何地&a…

面試題精選《劍指Offer》:JVM類加載機制與Spring設計哲學深度剖析-大廠必考

一、JVM類加載核心機制 &#x1f525; 問題5&#xff1a;類從編譯到執行的全鏈路過程 完整生命周期流程圖 關鍵技術拆解 編譯階段 查看字節碼指令&#xff1a;javap -v Robot.class 常量池結構解析&#xff08;CONSTANT_Class_info等&#xff09; 類加載階段 // 手動加載…

WordPress分類目錄綁定二級域名插件

一.子域名訪問形式 1.wordpress 分類目錄 轉換為 子域名 &#xff08;綁定二級域名&#xff09;形式 2.wordpress 頁面轉換為 子域名 &#xff08;綁定二級域名&#xff09; 形式 3.wordpress 作者頁轉換為 子域名 &#xff08;綁定二級域名&#xff09;形式 4.為不同子域名…

Shopify Checkout UI Extensions

結賬界面的UI擴展允許應用開發者構建自定義功能&#xff0c;商家可以在結賬流程的定義點安裝&#xff0c;包括產品信息、運輸、支付、訂單摘要和Shop Pay。 Shopify官方在去年2024年使用結賬擴展取代了checkout.liquid&#xff0c;并將于2025年8月28日徹底停用checkout.liquid…

華為HCIE方向那么多應該如何選擇?

在華為認證體系里&#xff0c;HCIE作為最高等級的認證&#xff0c;是ICT領域專業實力的有力象征。HCIE設置了多個細分方向&#xff0c;這些方向宛如不同的專業賽道&#xff0c;為期望在ICT行業深入發展的人提供了豐富的選擇。今天&#xff0c;咱們就來好好聊聊華為HCIE方向的相…

bootstrap介紹(前端框架)(提供超過40種可復用組件,從導航欄到輪播圖,從卡片到彈窗)bootstrap框架

文章目錄 Bootstrap框架全解析起源與發展核心特性與優勢響應式設計組件豐富度一致性與兼容性 柵格系統深度解析柵格系統工作原理斷點設置與響應式策略 組件系統導航組件表單系統 自定義與擴展SASS變量系統構建系統優化 性能優化策略按需加載減少嵌套層級 實踐案例&#xff1a;電…

FastGPT原理分析-數據集創建第二步:處理任務的執行

概述 文章《FastGPT原理分析-數據集創建第一步》已經分析了數據集創建的第一步&#xff1a;文件上傳和預處理的實現邏輯。本文介紹文件上傳后&#xff0c;數據處理任務的具體實現邏輯。 數據集創建總體實現步驟 從上文可知數據集創建總體上來說分為兩大步驟&#xff1a; &a…

el-select下拉框,搜索時,若是匹配后的數據有且只有一條,則當失去焦點時,默認選中該條數據

1、使用指令 當所需功能只能通過直接的 DOM 操作來實現時&#xff0c;才應該使用自定義指令。可使用方法2封裝成共用函數&#xff0c;但用指令他人復用時比較便捷。 <el-tablev-loading"tableLoading"border:data"tableList"default-expand-allrow-key…

vue中keep-alive組件的使用

keep-alive是vue的內置組件&#xff0c;它的主要作用是對組件進行緩存&#xff0c;避免組件在切換時被重復創建和銷毀&#xff0c;從而提高應用的性能和用戶體驗。它自身不會渲染一個 DOM 元素&#xff0c;也不會出現在父組件鏈中。使用時&#xff0c;只需要將需要緩存的組件包…

Kafka攔截器

文章目錄 1.定義2.生產者攔截器2.1 示例 3.消費者攔截器3.1 示例 1.定義 攔截器主要用于實現clients端的定制化需求&#xff0c;包括消息在生產者發送到 Kafka 或者在消費者接收消息之前進行一些定制化的操作。用于在消息發送和接收的關鍵步驟中進行攔截和處理。可以修改消息&…

進程間通信(匿名管道) ─── linux第22課

目錄 進程間通信 進程間通信目的 進程間通信的發展 進程間通信分類 1. 管道 2. System V IPC 3. POSIX IPC 管道 什么是管道 站在文件描述符角度-深度理解管道 站在內核角度-管道本質 ?編輯 匿名管道 測試匿名管道的讀寫 匿名管道的四大現象&#xff1a; 匿…

架構思維:通用系統設計方法論_從復雜度分析到技術實現指南

文章目錄 Question訂單履約原始架構痛點目標架構架構圖說明關鍵設計點優點 設計方法論復雜來源解決方案評估標準從設計原則出發 技術實現 &#xff08;以選型Redis為例&#xff09;Redis消息隊列的實現細節高可用設計 總結 Question 我們經常聊如何設計一個比較完善的系統&…

Excel(實戰):INDEX函數和MATCH函數、INDEX函數實戰題

目錄 經典用法兩者嵌套查值題目解題分析 INDEX巧妙用法讓數組公式&#xff0c;自動填充所有、有數據的行/列INDEX函數和SEQUENCE函數 經典用法兩者嵌套查值 題目 根據左表查詢這三個人的所有數據 解題分析 INDEX函數的參數&#xff1a;第1個參數是選定查找范圍&#xff0c…

ECharts儀表盤-儀表盤25,附視頻講解與代碼下載

引言&#xff1a; ECharts儀表盤&#xff08;Gauge Chart&#xff09;是一種類似于速度表的數據可視化圖表類型&#xff0c;用于展示單個或多個變量的指標和狀態&#xff0c;特別適用于展示指標的實時變化和狀態。本文將詳細介紹如何使用ECharts庫實現一個儀表盤&#xff0c;…

解決安卓so庫異常無法打印堆棧的問題

解決方案&#xff1a; 設置 android:extractNativeLibs"true" 直接在 AndroidManifest.xml 里加上&#xff1a; <applicationandroid:extractNativeLibs"true"> </application>這樣&#xff0c;so 文件會被解壓&#xff0c;崩潰時可以正常打…