怎樣取得JS函數參數名 ?用AST取得js函數參數名的方法分析
本篇文章帶給大家的內容是關於怎樣取得JS函數參數名稱 ?用AST取得js函數參數名稱的方法分析,有一定的參考價值,有需要的朋友可以參考一下,希望對你有幫助。
寫在最前面
最近專案有個需求,取得函數參數名,聽起來很簡單,但有了ES6,參數和函數寫法千奇百怪,在github上大概看了幾個庫,基本上都是正則,
對通用的寫法能夠覆蓋,稍微越過邊界,往往無法正確匹配。
於是就有了使用AST去進行覆蓋查找的想法。
概念
抽象語法樹(abstract syntax tree或縮寫為AST),或語法樹(syntax tree),是原始碼的抽象語法結構的樹狀表現形式
為什麼要用AST
通過AST,我們可以對程式碼進行查找,看起來好像正規表示式也可以做到,那麼為什麼要用AST而不用正規表示?
就說從函數取得參數名,誇張點,如果有以下表達式:
function x(a=5,b="a",c=function(x=1,y){console.log(x=function(i=8,j){})},d={x:1,y:2,z:'x=6'},e=x=>7,f=['3=5','x.1','y,2',1],g=(x,y)=>{let z=(i,j=6)=>{}},h){}
參數是[a,b,c,d,e,f,g,h]
你確定還想用正規去匹配參數名稱嗎...
AST是從程式碼的意義去編輯,而正則只能從程式碼的字面去編輯。
以上誇張的函數,使用AST去分析,可以很輕鬆地取得它的參數名稱
Esprima
我們使用esprima,一個可以將Javascript程式碼解析成抽象樹的庫。
首先我們要安裝它:
npm install esprima
#接著呼叫:
const esprima=require('require'')
接下來就是分析的時候了
一個簡單的AST例子
先來個簡單的例子:
function a(b){}
透過esprima解析後,產生結構圖如下:
{ "type": "Program", "body": [ { // 这个type表示这是一个函数表达式 "type": "FunctionDeclaration", "id": { "type": "Identifier", "name": "a" }, "params": [ { // 参数数组内的Identifier代表参数 "type": "Identifier", "name": "b" } ], "body": { "type": "BlockStatement", "body": [] }, "generator": false, "expression": false, "async": false } ], "sourceType": "script" }
想法:
1、FunctionDeclaration說明是一個函數表達式,進入params屬性。
2、判斷params中每一個的type是否為Identifier,在params屬性下的Identifier就代表是參數。
3、找出name屬性的值,結果為['b']。
根據上述思路,我們可以寫出一個簡單的取得參數的方法了。
function getParams(fn){ // 此处分析的代码必须是字符串 let astEsprima=esprima.parseScript(fn.toString()) let funcParams = [] let node = astEsprima.body[0] // 找到type,进入params属性 if (node.type === "FunctionDeclaration") funcParams = node.params let validParam=[] funcParams.forEach(obj=>{ if(obj.type==="Identifier") validParam.push(obj.name) }) return validParam }
測試一番,取得結果["b"],慶祝收工。
好吧,別高興太早了,要知道函數的創建方法不下10種,而參數寫法又有好幾種...
以下是一部分的函數創建方法和參數寫法
function a(x){} // 注意:第二条和第三条在AST中意义不同 let a=function(x=1){} a=function(...x){} let a=([x]=[1])=>{} async function a(x){} function *a(x){} class a{ constructor(x){} } new Function ('x','console.log(x)') (function(){return function(x){}})() eval("(function(){return function(a,b){}})()")
有什麼想法?如果你有發出"我K"的想法,那說明我這個裝逼還算成功- -...
其實只需要分幾種情況(很多寫法的type都是一致的),就可以完全滲入以上所有的參數物件內部,再進行參數取得就是循環判斷解決的事了。
由於篇幅問題,這裡不一一分析,只是將AST分析樹所用的type和一些注意點。
函數結構
變數宣告語句和表達式語句
上面註解中let a=function(x=1){}和a=function(...x ){}是兩種意義。
其中let a=function(x=1){}指的是變數宣告語句,
對應的type是VariableDeclaration,需要進入它的初始值init就可以取得到函數所在的語法對象,它的type是FunctionExpression函數表達式,再去params中查找即可。
變數宣告語句:
├──VariableDeclaration....init ├──FunctionExpression.params
而a=function(...x){}是表達式語句,
對應的type是ExpressionStatement,需要進入它的表達式expression取得到表達式內部,這時我們要進入賦值表達式(type為AssignmentExpression)的右邊(right屬性),
取得函數所在的語法對象,它的type同樣也是FunctionExpression函數表達式。
表達式語句:
├──ExpressionStatement.expression ├──AssignmentExpression.right ├──FunctionExpression.params
class宣告和Function建構子
class宣告對應的type有ClassDeclaration(class xx{...})或ClassExpression(let x =class{...}),他們一個是宣告一個是表達式,處理方式是相同的,
進入物件內部,找到kind為constructor的對象,取得參數資料。
class宣告語句:
├──ClassDeclaration...body... ├──{kind:constructor} ├──FunctionExpression.params
Function建構子對應的type是NewExpression或ClassExpression,參數在屬性arguments內部,但是Function的參數都是字串,
而且最後一個參數一定是函數內部語句,因此對於Function建構函數,就是對字串進行處理。
Function建構子
├──NewExpression.arguments ├──{value:<String>} ---->对字符串进行处理,分割参数
箭頭函數
箭頭函式type是ArrowFunctionExpression,也只是名稱不同,內部結構幾乎一致。
函數結構的type就到此。
參數結構
參數的type有以下:
Identifier:最終我們需要取得的參數值的type
Property:當存在解構參數,例如[a,b] 或 {x,y}
ArrayPattern:存在解構參數且是數組,例如[a,b]
ObjectPattern:存在解構參數且是對象,例如{ x,y}
RestElement:存在擴充運算符,例如(...args)
#我們只需要設定一個遞歸循環,想法和上面一樣,一層進入另一層,在內部進行查找。
總結
篇幅有限,就寫這麼多,接著做一個總結。
這篇講的主題只有1個,透過對AST樹中每個物件的type分析,type表示的是對應的程式碼的意義,也是程式碼的語義,例如
VariableDeclaration內部一定會有init,為什麼,因為變數宣告是有初始值的,如果你不設置,那麼就為undefined
type遠不止這次說的這麼多,官網(或者Google)上有詳細介紹。
以上是怎樣取得JS函數參數名 ?用AST取得js函數參數名的方法分析的詳細內容。更多資訊請關注PHP中文網其他相關文章!

熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

AI Hentai Generator
免費產生 AI 無盡。

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

熱門話題

如何使用WebSocket和JavaScript實現線上語音辨識系統引言:隨著科技的不斷發展,語音辨識技術已成為了人工智慧領域的重要組成部分。而基於WebSocket和JavaScript實現的線上語音辨識系統,具備了低延遲、即時性和跨平台的特點,成為了廣泛應用的解決方案。本文將介紹如何使用WebSocket和JavaScript來實現線上語音辨識系

WebSocket與JavaScript:實現即時監控系統的關鍵技術引言:隨著互聯網技術的快速發展,即時監控系統在各個領域中得到了廣泛的應用。而實現即時監控的關鍵技術之一就是WebSocket與JavaScript的結合使用。本文將介紹WebSocket與JavaScript在即時監控系統中的應用,並給出程式碼範例,詳細解釋其實作原理。一、WebSocket技

如何利用JavaScript和WebSocket實現即時線上點餐系統介紹:隨著網路的普及和技術的進步,越來越多的餐廳開始提供線上點餐服務。為了實現即時線上點餐系統,我們可以利用JavaScript和WebSocket技術。 WebSocket是一種基於TCP協定的全雙工通訊協議,可實現客戶端與伺服器的即時雙向通訊。在即時線上點餐系統中,當使用者選擇菜餚並下訂單

如何使用WebSocket和JavaScript實現線上預約系統在當今數位化的時代,越來越多的業務和服務都需要提供線上預約功能。而實現一個高效、即時的線上預約系統是至關重要的。本文將介紹如何使用WebSocket和JavaScript來實作一個線上預約系統,並提供具體的程式碼範例。一、什麼是WebSocketWebSocket是一種在單一TCP連線上進行全雙工

JavaScript和WebSocket:打造高效的即時天氣預報系統引言:如今,天氣預報的準確性對於日常生活以及決策制定具有重要意義。隨著技術的發展,我們可以透過即時獲取天氣數據來提供更準確可靠的天氣預報。在本文中,我們將學習如何使用JavaScript和WebSocket技術,來建立一個高效的即時天氣預報系統。本文將透過具體的程式碼範例來展示實現的過程。 We

JavaScript教學:如何取得HTTP狀態碼,需要具體程式碼範例前言:在Web開發中,經常會涉及到與伺服器進行資料互動的場景。在與伺服器進行通訊時,我們經常需要取得傳回的HTTP狀態碼來判斷操作是否成功,並根據不同的狀態碼來進行對應的處理。本篇文章將教你如何使用JavaScript來取得HTTP狀態碼,並提供一些實用的程式碼範例。使用XMLHttpRequest

用法:在JavaScript中,insertBefore()方法用於在DOM樹中插入一個新的節點。這個方法需要兩個參數:要插入的新節點和參考節點(即新節點將要插入的位置的節點)。

JavaScript中的HTTP狀態碼取得方法簡介:在進行前端開發中,我們常常需要處理與後端介面的交互,而HTTP狀態碼就是其中非常重要的一部分。了解並取得HTTP狀態碼有助於我們更好地處理介面傳回的資料。本文將介紹使用JavaScript取得HTTP狀態碼的方法,並提供具體程式碼範例。一、什麼是HTTP狀態碼HTTP狀態碼是指當瀏覽器向伺服器發起請求時,服務
