欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

golang 整合antlr語法校驗解析

 更新時間:2023年02月28日 10:53:55   作者:lj907722644  
Antlr是一個語法分析器,本身是用java實現的,然是Runtime的庫也支持Golang、Java、Python等,本文給大家講解使用golang整合antlr進行語法解析,感興趣的朋友一起看看吧

1. 背景

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

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

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

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

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

接下來給大家演示一下使用golang整合antlr進行語法解析。

2. goland安裝antlr插件

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

在這里插入圖片描述

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

在這里插入圖片描述

3. 編寫語法校驗規(guī)則

1.創(chuàng)建工程,引入包

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

2.在工程中新建一個antlr目錄,創(chuàng)建一個后綴名為 .g4 的文件,作為規(guī)則文件。此處我們創(chuàng)建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

在這里插入圖片描述

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

在這里插入圖片描述

4.樹狀圖校驗

在這里插入圖片描述

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.正確表達式

在這里插入圖片描述

2.key不在支持的語法內

在這里插入圖片描述

3.缺少key

在這里插入圖片描述

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

在這里插入圖片描述

5.缺少括號

在這里插入圖片描述

到此這篇關于golang 整合antlr語法校驗的文章就介紹到這了,更多相關go antlr語法校驗內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • 如何讓shell終端和goland控制臺輸出彩色的文字

    如何讓shell終端和goland控制臺輸出彩色的文字

    這篇文章主要介紹了如何讓shell終端和goland控制臺輸出彩色的文字的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-05-05
  • 基于golang時間轉換的問題

    基于golang時間轉換的問題

    下面小編就為大家?guī)硪黄趃olang時間轉換的問題。小編覺得挺不錯的,現在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-08-08
  • 如何在?Go語言中使用日志包

    如何在?Go語言中使用日志包

    這篇文章主要介紹了如何在?Go語言中使用日志包,日志文件就是一種快速找到這些?bug,更好地了解程序工作狀態(tài)的方法,下文基于go語言介紹該詳細需要的小伙伴可以參考一下
    2022-04-04
  • 詳解Go語言的錯誤處理和資源管理

    詳解Go語言的錯誤處理和資源管理

    資源處理是什么?打開文件需要關閉,打開數據庫連接,連接需要釋放。這些成對出現的就是資源管理。有時候我們雖然釋放了,但是程序在中間出錯了,那么可能導致資源釋放失敗。如何保證打開的文件一定會被關閉呢?這就是資源管理與錯誤處理考慮的一個原因
    2021-06-06
  • Go語言實現對XML的讀取和修改

    Go語言實現對XML的讀取和修改

    這篇文章主要為大家詳細介紹了Go語言實現對XML的讀取和修改的相關知識,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學習一下
    2023-12-12
  • Go分布式鏈路追蹤實戰(zhàn)探索

    Go分布式鏈路追蹤實戰(zhàn)探索

    這篇文章主要為大家介紹了Go分布式鏈路追蹤實戰(zhàn)示例探索,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2024-01-01
  • golang數組-----尋找數組中缺失的整數方法

    golang數組-----尋找數組中缺失的整數方法

    這篇文章主要介紹了golang數組-----尋找數組中缺失的整數方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-12-12
  • golang中import cycle not allowed解決的一種思路

    golang中import cycle not allowed解決的一種思路

    這篇文章主要給大家介紹了關于golang中import cycle not allowed解決的一種思路,文中通過示例代碼介紹的非常詳細,需要的朋友可以參考借鑒,下面隨著小編來一起學習學習吧
    2018-08-08
  • 詳解Golang如何比較兩個slice是否相等

    詳解Golang如何比較兩個slice是否相等

    開發(fā)中常會遇到需要比較兩個slice包含的元素是否完全相等的情況,我們通常會通過兩種方法去比較切片是否相等。這里通過幾個示例來看一下這兩種方法,感興趣的小伙伴們可以參考借鑒,希望對大家能有所幫助
    2022-11-11
  • Golang中List的實現方法示例詳解

    Golang中List的實現方法示例詳解

    最近決定復習下Go,所以下面這篇文章主要給大家介紹了關于Golang中List的實現方式,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧。
    2017-09-09

最新評論