llvm數據流分析

llvm數據流分析

  • 1.數據流分析
  • 2.LLVM實現
    • 2.1.常量傳播
    • 2.2.活躍性分析

相關參考文檔:DataFlowAnalysisIntro、ustc編譯原理課程、南大程序分析課程1、南大程序分析課程2。

1.數據流分析

數據流分析在編譯優化等程序分析任務上都有重要應用。通常數據流分析可被抽象為一個IFDS-based worklist算法。核心是給每個CFG結點 s s s (結點表示一個語句或者基本塊)計算兩個集合 IN [ s ] \text{IN}[s] IN[s] OUT [ s ] \text{OUT}[s] OUT[s]

  • meet運算: IN [ s ] = meet s i ∈ prev ( s ) OUT [ s i ] \text{IN}[s] = \text{meet}_{s_i \in \text{prev}(s)} \text{OUT}[s_i] IN[s]=meetsi?prev(s)?OUT[si?] meet [ s ] = ∪ s i ∈ next ( s ) IN [ s i ] \text{meet}[s] = \cup_{s_i \in \text{next}(s)} \text{IN}[s_i] meet[s]=si?next(s)?IN[si?] (反向數據流傳播任務)。

  • update: OUT [ s ] = gen [ s ] ∪ ( IN [ s ] ? kill [ s ] ) \text{OUT}[s] = \text{gen}[s] \cup (\text{IN}[s] - \text{kill}[s]) OUT[s]=gen[s](IN[s]?kill[s]), IN [ s ] = gen [ s ] ∪ ( OUT [ s ] ? kill [ s ] ) \text{IN}[s] = \text{gen}[s] \cup (\text{OUT}[s] - \text{kill}[s]) IN[s]=gen[s](OUT[s]?kill[s]) (反向數據流傳播任務)。通常具體實現中 gen \text{gen} gen kill \text{kill} kill 可由update操作完成。

數據流分析的worklist算法 (前向) 如下。worklist算法會不斷循環直到 IN [ s ] \text{IN}[s] IN[s] OUT [ s ] \text{OUT}[s] OUT[s] 收斂(集合不再變大)為止。 IN [ s ] \text{IN}[s] IN[s] OUT [ s ] \text{OUT}[s] OUT[s] 的元素為fact,fact的具體類型視數據流分析任務而定。需要注意的是這個分析框架是flow-sensitive & path-insensitive的,path-sensitive的分析(符號執行)不會有 meet 操作,而是直接將當前分析狀態 fork 多份然后將每個 fork 后的狀態單獨添加到 worklist中。

def DataFlowAnalysis(cfg: CFG):for stmt in CFG:worklist.append(stmt)Out[stmt].clear()while (!worklist.empty()):cur_stmt = worklist.pop()for prev_stmt in Prev(cur_stmt):meet(In[cur_stmt], Out[prev_stmt])changed = update(cur_stmt, Out[cur_stmt], In[cur_stmt])if (changed):worklist.append(cur_stmt)
數據流分析任務方向meet操作fact示例genkill
到達定值分析 (reaching-define analysis)前向 ∪ \cup 變量定義l: x = ax: lx: prevL, prevL 為上一個定義 x 的語句
常量傳播 (constant propagation)前向 ∪ \cup 變量值x = ax: val(a), val(a) 表示 a 的值x: _, _ 表示之前的值
活躍變量分析 (live variabiles analysis)反向 ∪ \cup 變量x = nnx
可用表達式分析 (available expressions analysis)前向 ∩ \cap 表達式a = x op yx op q所有涉及 a 的表達式
  • 1.reaching-define的結果可以用來構造數據依賴圖(參考Joern),比如 l: x = a,中 OUT [ l ] = { ( a , l 1 ) , ( x , l ) } \text{OUT}[l] = \{(a, l_1), (x, l)\} OUT[l]={(a,l1?),(x,l)},那么可以知道引用的 al1 出定義,添加邊 l 1 → a l l_1 \stackrel{a}{\rightarrow} l l1?al。同時對于循環中的 x = y op z,如果 yz 的定義在循環外,那么可以把 y op z 移動到循環外。

  • 2.常量傳播的結果 OUT [ s ] \text{OUT}[s] OUT[s] 表示 s s s 處為常量的變量情況。可以用來對變量進行常量替換。

  • 3.活躍變量的分析結果中 IN [ s ] \text{IN}[s] IN[s] 表示 s s s 處的活躍變量,可以用來優化寄存器分配。

  • 4.可用表達式的分析結果可以用來刪除公共子表達式減少多余計算。

除了上述提到的經典問題外,flow-sensitive的指針分析抽象解釋也是一個數據流分析問題。指針分析的fact是指向關系 (比如 x → o x \rightarrow o xo 表示 x x x 可能指向 o o o),meet操作是 ∪ \cup genkill 則需要考慮strong update和weak update。具體可參考blog。

  • 對于賦值語句 s : x = y s: x = y s:x=y gen [ s ] = { x → o ∣ ? o ∈ pts ( y ) } \text{gen}[s] = \{x \rightarrow o \mid \forall o \in \text{pts}(y)\} gen[s]={xo?opts(y)} kill [ s ] = { x → _ } \text{kill}[s] = \{x \rightarrow \_ \} kill[s]={x_},進行strong update。

  • 對于 s : ? x = y s: *x = y s:?x=y gen [ s ] = { z → o ∣ ? z ∈ pts ( x ) ? o ∈ pts ( y ) } \text{gen}[s] = \{z \rightarrow o \mid \forall z \in \text{pts}(x) \; \forall o \in \text{pts}(y) \} gen[s]={zo?zpts(x)?opts(y)} kill \text{kill} kill 情況就比較復雜。

    • 如果 x x x 只指向內存對象 z z z,那么 kill [ s ] = { z → _ } \text{kill}[s] = \{z \rightarrow \_ \} kill[s]={z_},依舊可以進行strong update。

    • 如果 x x x 可能指向好幾塊內存對象,那么 kill [ s ] = { } \text{kill}[s] = \{\} kill[s]={},此時進行的是weak update。

抽象解釋可參考blog,fact為變量值域,SVF的抽象解釋模塊對 p = c (c 為整型常量,p 為整型變量) 生成的fact為 p → ? [ c , c ] , ? ? p \rightarrow \langle [c, c], \top \rangle p?[c,c],??,包括值域 [ c , c ] [c, c] [c,c] 和指向集 ? \top ?,同時進行抽象解釋和指針分析(也就是對應paper提到的Combined Abstract Domains)。和指針分析一樣,抽象解釋中 store 指令可能涉及到 kill 操作,也需要考慮strong update和weak update。

難點:C/C++中別名的存在是數據流分析的一大難點(參考blog),不同變量對同一內存的引用導致多個定義可能指向一個變量,通常數組指針結構體的使用可能帶來別名關系,為了保證soundness,越保守的策略通常越不會 kill 不確定的fact。通常可以先進行一個指針分析來確定大致別名關系。或者參考Dr.Checker和Falcon將指針分析和到達定值一起分析。考慮到二者到達收斂的循環次數可能不一致,Dr.Checker預設了一個循環次數上限,不過這種策略可能會導致指針分析不夠可靠(unsound)。對于Falcon,它用到了兩個集合 E E E S S S 來分別保存top-level pointer的指向集和address-taken object的到達定值, ( π , l , q ) ∈ S ( o ) (\pi, l, q) \in S(o) (π,l,q)S(o) 表示 store 語句 l : ? p = q l: *p = q l:?p=q o ∈ pts ( p ) o \in \text{pts}(p) opts(p) 因此 o o o l l l 處的到達定值為 ( π , q ) (\pi, q) (π,q) ( π \pi π 是路徑條件,Falcon采用路徑敏感的分析策略),Falcon中只有 store 語句會更新 S S S 集合,其它語句(load, gep 等) 會查詢 S S S 并更新top-level variable對應 E E E 集合。

2.LLVM實現

2.1.常量傳播

llvm提供了source code level和LLVM IR level的常量傳播算法,source code level通過ConstantPropagationAnalysis類實現,不過由于只是簡單實現沒法適用到real-world環境下,因此只是放到unittests下。在這個實現中變量值有上界 ? \top ?, 下屆 ⊥ \bot , 常量值 c c c 3種類型。用 a a a 表示任意值,其中有 ⊥ ∪ a = a \bot \cup a = a a=a ? ∪ a = ? \top \cup a = \top ?a=?, c ∪ c = c c \cup c = c cc=c, c 1 ∪ c 2 = ? c_1 \cup c_2 = \top c1?c2?=?。對于給定 Stmt* S, ConstantPropagationAnalysis 類通過AST分析識別其中是否存在: 1.變量聲明并初始化 (int x = 42)、2.賦值語句 (x = 24)、3.復合賦值 (x += 4)。對于前兩種語句 ConstantPropagationAnalysis 會嘗試計算右表達式的常量值,如果不是常量則設置左表達式對應變量為 ? \top ?,反之設置為具體值。對于復合賦值直接將左變量設置為 ? \top ?。這里會調用 Expr::EvaluateAsInt 對表達式進行求值。x = a 中,如果 a 是常量那么返回常量值,如果 a 是變量似乎不會再遞歸求值直接返回 ? \top ?

LLVM IR level的通過SCCPPass來進行函數內Sparse Conditional Constant Propagation (SCCP)。跨函數傳播通過IPSCCPPass實現。SCCP相比普通常量傳播的改進在于增加對分支條件的處理。普通常量傳播不會考慮分支條件為常量 (truefalse),SCCP則會基于分支條件的常量值刪除不可達分支。根據這個stackoverflow post,啟用常量傳播pass的前提條件應該是先使用mem2reg優化。

函數內常量傳播的入口為runSCCP函數,lattice定義為ValueLatticeElement,變量值通常通常有以下狀態。

狀態含義
unknown該值尚無已知信息,可以轉換為任何其他狀態。通常作為起始狀態
undef該值是 UndefValue 或產生 undef,可以與 constantconstantrange meet后轉化為 constant。可以轉換為 constantconstantrange_including_undefoverdefined
constant該值是一個特定的常量,不能是 undef
notconstant該值已知不是某個特定常量(適用于非整數類型)。
constantrange該值屬于某個范圍(僅適用于整數類型)。
constantrange_including_undef該值屬于某個范圍,但也可能是 undefundefconstantrange meet后為 constantrange_including_undef。與其他fact meet后仍為 constantrange_including_undef
overdefined該值無法精確建模,即可能具有多個動態值,不再進行任何狀態轉換。相當于 ⊥ \bot

涉及到merge (meet)操作時,狀態轉移關系定義在ValueLatticeElement::mergeIn,狀態轉移關系如下。對于 res = lhs op rhs,運算結果如下:

lhs\rhsunknownoverdefinedundefconstantconstantrangenotconstantconstantrange_including_undef
unknownunknownoverdefinedundefconstantconstantrangenotconstantconstantrange_including_undef
overdefinedoverdefinedoverdefinedoverdefinedoverdefinedoverdefinedoverdefinedoverdefined
undefoverdefinedundefconstantconstantrangeoverdefinedoverdefined
constantoverdefinedoverdefinedconstantconstant (lhs == rhs), overdefined (lhs != rhs)overdefinedoverdefinedoverdefined
notconstantoverdefinedoverdefinedoverdefinedoverdefinedoverdefinednotconstantoverdefined
constantrangeoverdefinedoverdefinedoverdefinedoverdefinedconstantrange (值域會進行union)overdefinedoverdefined

其中還有一個輔助類SCCPInstVisitor來收集中間結果。

成員變量類型作用
BBExecutableSmallPtrSet<BasicBlock *, 8>記錄可執行的基本塊
ValueStateDenseMap<Value *, ValueLatticeElement>記錄 Value 的lattice狀態, Value 通常對應1個llvm指令
StructValueStateDenseMap<std::pair<Value *, unsigned>, ValueLatticeElement>記錄結構體字段的 lattice 狀態
TrackedGlobalsDenseMap<GlobalVariable *, ValueLatticeElement>記錄全局變量的lattice狀態
TrackedRetValsMapVector<Function *, ValueLatticeElement>記錄單返回值函數的返回值狀態
TrackedMultipleRetValsMapVector<std::pair<Function *, unsigned>, ValueLatticeElement>記錄多返回值函數的返回值狀態
MRVFunctionsTrackedSmallPtrSet<Function *, 16>存儲 TrackedMultipleRetVals 中的函數,方便查找
MustPreserveReturnsInFunctionsSmallPtrSet<Function *, 16>存儲返回值不可修改的函數
TrackingIncomingArgumentsSmallPtrSet<Function *, 16>存儲需要分析參數的函數
OverdefinedInstWorkListSmallVector<Value *, 64>記錄即將 overdefined 的指令,加速SCCP收斂
InstWorkListSmallVector<Value *, 64>記錄待 SCCP 處理的指令,加速SCCP收斂
BBWorkListSmallVector<BasicBlock *, 64>記錄待 SCCP 處理的基本塊 ,主worklist
KnownFeasibleEdgesDenseSet<Edge>記錄已確認的CFG邊,避免重復計算
AnalysisResultsDenseMap<Function *, AnalysisResultsForFn>存儲函數SCCP分析結果

這里worklist算法大致如下,相比原版worklist算法再次做了些優化:

  • 1.visit(BB) 表示對basic block BB 下的所有指令 (Value) 進行 meetupdate

  • 2.markUsersAsChanged 會對 I 的所有 users 進行 visit,也就是優先處理 OverdefinedInstWorkList 中的元素和非 overdefined 的結構體元素。

  • 3.優先處理 OverdefinedInstWorkList 的原因是如果變量 (Value) I 的狀態為 Overdefined,它的 user 狀態多半為 Overdefined,沒法再收斂了,減少這部分的迭代次數能加速SCCP收斂。

  • 4.InstWorkList 主要用于處理值從 undef 變成 constant 的情況。盡早傳播 constant,讓更多值變成 constant,提高優化效果。

def SCCP_Worklist():while !BBWorkList.empty() or !OverdefinedInstWorkList.empty() or !InstWorkList.empty():# 優先處理 OverdefinedInstWorkList,盡快傳播 overdefined 狀態while !OverdefinedInstWorkList.empty():I = OverdefinedInstWorkList.pop()markUsersAsChanged(I)# 處理普通指令工作列表while !InstWorkList.empty():I = InstWorkList.pop()# 只處理未 overdefined 的值if isStructType(I) or not isOverdefined(I):markUsersAsChanged(I)# 處理基本塊工作列表while !BBWorkList.empty():BB = BBWorkList.pop()visit(BB)

具體的update規則可以參考 SCCP::visitXXInst 函數。這里有意思的是:

  • 1.SCCPInstVisitor::visitStoreInst,其中只對全局變量進行處理,也就是局部變量的 store 都不操作。

  • 2.SCCPInstVisitor::visitLoadInst中如果加載的是結構體變量直接為 overdefined ( ⊥ \bot ),SCCPInstVisitor::resolvedUndefsIn中對于其它 load 將值設為 undef。這可能是考慮到別名的一種保守處理。根據這個stackoverflow post,對于下面代碼,llvm編譯器壓根沒做優化。原因是 c = a + b 中存在 load aload bstore 1, a, store 2, b 操作。而 SCCPPass 沒有詳細地處理,因此,沒能優化。也是因此開啟該優化的一個前提是先用 mem2reg

int a,b,c;
a = 1;
b = 2;
c = a + b;
  • 3.SCCPInstVisitor::visitCmpInst中調用ValueLatticeElement::getCompare對 r = cmp op1, op2 計算 r 的值,最終結果有 overdefined, constant(1), constant(0) 3種。

  • 4.SCCPInstVisitor::visitSelectInst中會根據 cmp 的計算結果計算當前值。如果 cond 的常量值為 1,則選 true branch的值,反之亦然;如果為 overdefined,則合并兩個分支的常量值。

runSCCP結尾會調用SCCPSolver::simplifyInstsInBlock對變量進行常量替換以及刪除無用指令并調用removeNonFeasibleEdges刪除infeasible CFG edge以及隨后刪除無用基本塊。

2.2.活躍性分析

llvm提供兩個level的活躍性(包括活躍變量和可用表達式)分析:source code和machine code,source code level可通過clang static analyzer的LiveVariablesDumper (CSA參數為 debug.DumpLiveVars) 以及LiveExpressionsDumper(CSA參數為 debug.DumpLiveExprs)打印活躍變量信息。在machine code level會通過LiveVariablesAnalysis進行活躍變量分析。

source code level的活躍性分析這位大佬的兩篇blog(clang中的活躍性分析講解源碼,clang中的活躍性分析(續)給出示例)對clang的源代碼進行了具體說明。負責活躍性分析的包括LiveVariables和RelaxedLiveVariables類,后者相比前者分析結果更不精確,不過source code level的活躍性分析主要是輔助clang static analyzer而不是編譯優化,因此在analyzer中反而應用了 RelaxedLiveVariables,而 LiveVariables 純粹只是用來dump source code level的活躍變量信息。transfer function的定義在TransferFunctions中,活躍性分析的dataflow fact分別用VarDecl*(變量分析)和Expr*(表達式分析)表示。

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

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

相關文章

C++ MySQL 常用接口(基于 MySQL Connector/C++)

C MySQL 常用接口&#xff08;基于 MySQL Connector/C&#xff09; 1. 數據庫連接 接口&#xff1a; sql::mysql::MySQL_Driver *driver; sql::Connection *con;作用&#xff1a; 用于創建 MySQL 連接對象。 示例&#xff1a; driver sql::mysql::get_mysql_driver_insta…

C++藍橋杯基礎篇(十一)

片頭 嗨~小伙伴們&#xff0c;大家好&#xff01;今天我們來學習C藍橋杯基礎篇&#xff08;十一&#xff09;&#xff0c;學習類&#xff0c;結構體&#xff0c;指針相關知識&#xff0c;準備好了嗎&#xff1f;咱們開始咯~ 一、類與結構體 類的定義&#xff1a;在C中&#x…

css中實現border距離視圖左右兩側有距離

首先看效果圖 再看css是如何實現 <!DOCTYPE html> <html><head><meta charset"utf-8"><title></title><style>.main {background-color: aqua;display: block;width: 300px;padding: 0px 32px;box-sizing: border-box;}/…

Ubuntu 22.04 無法進入圖形界面的解決方法

Ubuntu 22.04 無法進入圖形界面&#xff0c;只能進入 tty&#xff0c;可能是由于圖形界面相關的配置或驅動程序出現了問題。以下是一些常見的解決方法&#xff1a; 1. 檢查圖形界面服務狀態 首先&#xff0c;檢查圖形界面服務&#xff08;通常是 gdm 或 lightdm&#xff09;的…

Tweak Power:全方位電腦系統優化的高效工具

在日常使用電腦時&#xff0c;系統性能的下降、垃圾文件的堆積以及硬盤的老化等問題常常困擾著用戶。為了提升電腦性能、優化系統運行&#xff0c;許多人會選擇系統優化工具。然而&#xff0c;國內一些系統優化軟件常常因為廣告過多或功能冗雜而讓人望而卻步。此時&#xff0c;…

深入淺出Bearer Token:解析工作原理及其在Vue、Uni-app與Java中的實現Demo

目錄 前言1. 基本知識2. Demo3. 實戰 前言 &#x1f91f; 找工作&#xff0c;來萬碼優才&#xff1a;&#x1f449; #小程序://萬碼優才/r6rqmzDaXpYkJZF 1. 基本知識 Bearer Token是一種基于Token的認證機制&#xff0c;用于在HTTP請求中傳遞用戶的身份信息 應用于RESTful A…

kubernetes——part3-5 核心概念 Service

一、 service作用 使用kubernetes集群運行工作負載時&#xff0c;由于Pod經常處于用后即焚狀態&#xff0c;Pod經常被重新生成&#xff0c;因此Pod對應的IP地址也會經常變化&#xff0c;導致無法直接訪問Pod提供的服務&#xff0c;Kubernetes中使用了Service來解決這一問題&am…

從零開始 | C語言基礎刷題DAY1

?個人主頁&#xff1a;折枝寄北的博客 DAY1[2025.3.11] 1. 求兩個數的較大值2.從鍵盤輸入的兩個數的大小關系3.一個整數的奇偶性&#xff0c;請判斷4. 考試分數是否通過5.考試成績是否完美&#xff0c;請判斷 1. 求兩個數的較大值 題目&#xff1a; 寫一個函數求兩個整數的較…

開源模型時代的 AI 開發革命:Dify 技術深度解析

開源模型時代的AI開發革命&#xff1a;Dify技術深度解析 引言&#xff1a;AI開發的開源新紀元 在生成式AI技術突飛猛進的2025年&#xff0c;開源模型正成為推動行業創新的核心力量。據統計&#xff0c;全球超過80%的AI開發者正在使用開源模型構建應用&#xff0c;這一趨勢不僅…

Dify Web 前端獨立部署指南(與后端分離,獨立部署)

背景:單獨拆分前端出來部署,二開前后端 本文檔專注于 Dify Web 前端的部署流程和配置,適用于需要將項目部署到各種環境的運維人員和開發者。 1. 環境準備 1.1 部署環境要求 Node.js >= 18.17.0Nginx 或其他Web服務器(生產環境推薦)Docker(可選,用于容器化部署)1.…

《蒼穹外賣》SpringBoot后端開發項目核心知識點整理(DAY1 to DAY3)

目錄 一、在本地部署并啟動Nginx服務1. 解壓Nginx壓縮包2. 啟動Nginx服務3. 驗證Nginx是否啟動成功&#xff1a; 二、導入接口文檔1. 黑馬程序員提供的YApi平臺2. YApi Pro平臺3. 推薦工具&#xff1a;Apifox 三、Swagger1. 常用注解1.1 Api與ApiModel1.2 ApiModelProperty與Ap…

大數據hadoop課程筆記

1.課程導入 柯潔 Alpha Go是人工智能領域的里程碑。 深度學習 大模型deepseek chatgpt 大模型 和 大數據 之間有著非常緊密的關系。可以說&#xff0c;大數據是大模型發展的基石&#xff0c;而大模型是大數據價值挖掘的重要工具。 https://youtu.be/nN-VacxHUH8?sifj7Ltk…

架構學習第八周--Kubernetes博客搭建

目錄 一、整體架構 二、部署MySQL主從 三、部署Redis哨兵 四、部署WordPress 五、注意事項 一、整體架構 本項目為在一主三從的Kubernetes集群上部署WordPress博客。因為WordPress部分容器版本自行集成Apache和PHP服務&#xff0c;因此在Kubernetes上部署WordPress只需提供…

Application.OnTime如何引用帶參數的過程

Application.OnTime方法本身并不直接支持傳遞參數給被調用的過程。不過&#xff0c;有幾種方法可以間接實現這個需求。 方法1&#xff1a;使用單引號表達式 使用單引號表達式來傳遞參數時&#xff0c;不能在表達式中使用變量&#xff0c;需要把參數值直接寫到表達中&am…

網絡安全之tcpdump工具

引言 wireshark是一款非常不錯的抓包軟件&#xff0c;在圖形化界面占絕對統治地位&#xff1b;盡管其在字符界面下有些許選項可供使用&#xff0c;但終究不太方便&#xff0c;下面我再介紹一款NB的終端抓包工具 tcpdump 1、混雜模式 linux的網卡有混雜模式一說&#xff0c;當開…

VC++ 獲取目的IP的路由

GetBestRoute 函數獲取到目的IP的最佳匹配路由。 第一個參數為&#xff1a;destination&#xff08;目的IP&#xff09; 第二個參數為&#xff1a;source&#xff08;源IP&#xff09; 通常不需要指定第二個source&#xff0c;這個一般用來匹配具體某一個網卡接口路由的&…

JavaScript 模塊 vs C# 類:封裝邏輯的兩種哲學

引言 在現代軟件開發中&#xff0c;模塊化和面向對象設計是代碼組織的核心課題。本文通過對比 JavaScript 模塊&#xff08;ES6 Module&#xff09;與 C# 類&#xff08;Class&#xff09;的實現方式&#xff0c;探討兩種語言在封裝邏輯時的不同哲學&#xff0c;并給出實際應用…

大模型在甲狀腺癌診療全流程預測及方案制定中的應用研究

目錄 一、引言 1.1 研究背景與意義 1.2 研究目的與創新點 1.3 國內外研究現狀 二、大模型預測甲狀腺癌的理論基礎 2.1 甲狀腺癌相關醫學知識 2.2 大模型技術原理與特點 2.3 大模型在醫療領域的應用潛力 三、術前預測方案 3.1 預測模型構建 3.1.1 數據收集與預處理 …

electron+vue+webview內嵌網頁并注入js

vue內嵌網頁可以使用iframe實現內嵌網頁&#xff0c;但是只能通過postMessage間接通信&#xff0c;在electron環境下&#xff0c;vue可以直接使用webview來內嵌網頁&#xff0c;支持 executeJavaScript、postMessage、send 等豐富的通信機制。 使用 webview的優勢 性能更佳&…

leetcode日記(95)將有序數組轉換為二叉搜索樹

很簡單&#xff0c;感覺自己越來越適應數據結構題目了…… /*** Definition for a binary tree node.* struct TreeNode {* int val;* TreeNode *left;* TreeNode *right;* TreeNode() : val(0), left(nullptr), right(nullptr) {}* TreeNode(int x) : va…