亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

golang怎么整合antlr語法校驗

發布時間:2023-02-28 11:57:04 來源:億速云 閱讀:261 作者:iii 欄目:開發技術

本篇內容介紹了“golang怎么整合antlr語法校驗”的有關知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠學有所成!

1. 背景

在項目中我們可能會遇到表達式檢索的場景,例如,輸入以下表達式檢索,需要解析表達式并得到檢索結果。

ip="192.168.1.3" && (port="80" || protocol="http")

此時,我們需要對語法進行校驗、解析,應當如何做呢?

下面給大家推薦一種使用語法校驗工具Antlr

Antlr是一個語法分析器,本身是用java實現的,然是Runtime的庫也支持Golang、Java、Python等。

2. goland安裝antlr插件

打開goland,File --> Settings --> Plugins, 搜索antlr,安裝 antlr4

golang怎么整合antlr語法校驗

插件安裝完成后,可以看到ANTLR Preview窗口,一會我們可以在這個窗口進行簡單的語法校驗。

golang怎么整合antlr語法校驗

3. 編寫語法校驗規則

1.創建工程,引入包

go get -u github.com/antlr/antlr4/runtime/Go/antlr/v4

2.在工程中新建一個antlr目錄,創建一個后綴名為 .g4 的文件,作為規則文件。此處我們創建Rule.g4

// 定義語法名稱,需要和文件名匹配
grammar Rule;

// DECIMAL, IDENTIFIER, COMMENTS, WS are set using regular expressions
// key 為表達式中可支持的檢索字段,可以是固定值(每個值中間用 | 隔開,是”或“的意思),也可以是正則表達式
// value 使用正則表達式
KEY : 'ip' | 'port' | 'protocol';
//VALUE :'"' ( '""' | ~["\r\n] )* '"' ;
//KEY : ('A' .. 'Z' | 'a' .. 'z' |  '_') + ;
VALUE :'"' ( '\\"' | ~["] )* '"' ;

// COMMENT and WS are stripped from the output token stream by sending
// to a different channel 'skip'

COMMENT : '//' .+? ('\n'|EOF) -> skip ;

WS : [ \r\t\u000C\n]+ -> skip ;


/* Parser rules */
// 語法校驗的入口
start : logicalExpr* EOF;

// 語法支持的結構
logicalExpr
    : comparisonExpr // 示例: key == value 表示支持 == 和 != 的表達式
    | logicalExpr operator logicalExpr // 示例: key1 == value1 && key2 != value2 表示支持 && 和 || 運算符連接表達式
    | lparen logicalExpr rparen // 示例: (key1 == value1 && key2 != value2) 表示支持 () 連接表達式
    ;

comparisonExpr
    : KEY compare VALUE
    ;
compare
    : '='
    | '!='
    ;
operator
    : '&&'
    | '||'
    ;
lparen
    :  '('
    ;
rparen
    :  ')'
    ;

3.初始化校驗語法

1.選中Rule.g4 文件,鼠標右鍵,選擇 Configure ANTLR Tool…

2.配置輸出路徑,和Rule.g4 同目錄;配置語言,使用Go

golang怎么整合antlr語法校驗

3.選中Rule.g4 文件,鼠標右鍵,選擇 Generate ANTLR Recognizer,完成規則初始化

golang怎么整合antlr語法校驗

4.樹狀圖校驗

golang怎么整合antlr語法校驗

4. 語法校驗

1.自定義listener

package parser

import (
	"github.com/antlr/antlr4/runtime/Go/antlr/v4"
	"strings"
)

type MyRuleListener struct {
	*BaseRuleListener
	Queue    []interface{}
	QueueStr []string
}

// 注意:方法名必須是這個名字
func (s *MyRuleListener) EnterComparisonExpr(ctx *ComparisonExprContext) {
	key := ctx.GetChild(0).(antlr.ParseTree).GetText()
	operator := ctx.GetChild(1).(antlr.ParseTree).GetText()
	value := ctx.GetChild(2).(antlr.ParseTree).GetText()
	if strings.HasPrefix(value, "\"") {
		value = value[1:]
	}
	if strings.HasSuffix(value, "\"") {
		value = value[:len(value)-1]
	}
	keyValue := map[string]string{}
	keyValue["key"] = key
	keyValue["operator"] = operator
	keyValue["value"] = value

	s.PushStr(ctx.GetText())
	s.Push(keyValue)
}

// EnterKeyValue is called when production KeyValue is entered.
func (s *MyRuleListener) ExitOperator(ctx *OperatorContext) {
	s.Push(ctx.GetText())
	s.PushStr(ctx.GetText())
}

// EnterKeyValue is called when production KeyValue is entered.
func (s *MyRuleListener) ExitLparen(ctx *LparenContext) {
	s.Push(ctx.GetText())
	s.PushStr(ctx.GetText())
}

// EnterKeyValue is called when production KeyValue is entered.
func (s *MyRuleListener) ExitRparen(ctx *RparenContext) {
	s.Push(ctx.GetText())
	s.PushStr(ctx.GetText())
}

func (s *MyRuleListener) Push(i interface{}) {
	s.Queue = append(s.Queue, i)
}

func (s *MyRuleListener) PushStr(i string) {
	s.QueueStr = append(s.QueueStr, i)
}

2.獲取解析異常的錯誤信息

package parser

import "github.com/antlr/antlr4/runtime/Go/antlr/v4"

type RuleErrorListener struct {
	antlr.ErrorListener
	Msg string
}

func (l *RuleErrorListener) SyntaxError(recognizer antlr.Recognizer, offendingSymbol interface{}, line, column int, msg string, e antlr.RecognitionException) {
	l.Msg = msg
}

3.校驗

package main

import (
	parser "antlr-demo/antlr"
	"errors"
	"fmt"
	"github.com/antlr/antlr4/runtime/Go/antlr/v4"
)

func main() {
	expre := "ip=\"192.168.1.3\" && (port=\"80\" || protocol=\"http\")"
	err := checkExpre(expre)
	if err != nil {
		fmt.Println(err)
	}
}

func checkExpre(expre string) error {
	input := antlr.NewInputStream(expre)
	var lexerErr parser.RuleErrorListener
	lexer := parser.NewRuleLexer(input)
	lexer.AddErrorListener(&lexerErr)
	stream := antlr.NewCommonTokenStream(lexer, 0)
	ruleParser := parser.NewRuleParser(stream)
	ruleParser.BuildParseTrees = true
	var ruleErr parser.RuleErrorListener
	ruleParser.AddErrorListener(&ruleErr)
	tree := ruleParser.Start()
	listener := new(parser.MyRuleListener)
	antlr.ParseTreeWalkerDefault.Walk(listener, tree)

	if lexerErr.Msg != "" || ruleErr.Msg != "" {
		return errors.New("輸入的語法不正確")
	}
	expreList := listener.QueueStr
	fmt.Println("expreList--->", expreList)
	expreMap := listener.Queue
	fmt.Println("expreMap--->", expreMap)
	return nil
}

4.結果驗證

1.正確表達式

golang怎么整合antlr語法校驗

2.key不在支持的語法內

golang怎么整合antlr語法校驗

3.缺少key

golang怎么整合antlr語法校驗

4.運算符不在支持的語法內

golang怎么整合antlr語法校驗

5.缺少括號

golang怎么整合antlr語法校驗

“golang怎么整合antlr語法校驗”的內容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業相關的知識可以關注億速云網站,小編將為大家輸出更多高質量的實用文章!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

托克逊县| 绍兴县| 饶河县| 积石山| 桐乡市| 元氏县| 武山县| 汝阳县| 静宁县| 慈溪市| 彭州市| 历史| 浙江省| 屏东市| 霍山县| 马山县| 班玛县| 阿瓦提县| 贵溪市| 大余县| 威海市| 湘潭县| 通渭县| 图木舒克市| 喀喇沁旗| 沽源县| 镶黄旗| 塘沽区| 绥阳县| 高台县| 资溪县| 抚顺市| 普定县| 吴江市| 宜君县| 宁海县| 陇西县| 沙河市| 九龙坡区| 永和县| 海口市|