语法:

  • 正则表达式和待匹配字符串都是一行
  • “^” 标记正则表达式的开始
  • “$” 标记正则表达式的结束
  • “*” 匹配前面的子表达式零次或多次
  • “+” 匹配前面的子表达式一次或多次
  • “?” 匹配前面的子表达式零次或一次, 当该字符紧跟在任何一个其他限制符(*,+,?,{n},{n,},{n,m})后面时,表示该匹配模式是非贪婪的,而不是匹配前面的子表达式
  • “{n}” n是一个非负整数。匹配确定的n次
  • “{n,}” n是一个非负整数。至少匹配n次
  • “{n,m}” m和n均为非负整数,其中n<=m。最少匹配n次且最多匹配m次
  • “.” 匹配除“\n”之外的任何单个字符
  • “x|y” 匹配x或y,其中x和y是两个子表达式,如果是字符就是单个字符
  • “[xyz]”字符合集,匹配其中的任意一个字符,如果两个字符之间有”-”,就表示这三个字符用于匹配一个ASSIC码值在两字符之间的一个字符,如果开头有”^”,表示负值字符集合。匹配未包含的任意字符
  • “(pattern)”一个子表达式,可嵌套
  • \b匹配一个单词边界,即是否后面是非标识符字符,只是检测,不匹配实际内容
  • \B匹配非单词边界,即是否后面不是非标识符字符,只是检测,不匹配实际内容
  • \d 匹配一个数字字符。等价于[0-9]。
  • \D 匹配一个非数字字符。等价于[^0-9]。
  • \f 匹配一个换页符。等价于\x0c
  • \n 匹配一个换行符。等价于\x0a
  • \r 匹配一个回车符。等价于\x0d
  • \s 匹配任何空白字符,包括空格、制表符、换页符等等。等价于[ \f\n\r\t\v]。
  • \S 匹配任何非空白字符。等价于[^ \f\n\r\t\v]。
  • \t 匹配一个制表符。等价于\x09
  • \v 匹配一个垂直制表符。等价于\x0b
  • \w 匹配包括下划线的任何单词字符。等价于“[A-Za-z0-9_]”。
  • \W 匹配任何非单词字符。等价于“[^A-Za-z0-9_]”。
  • \xn匹配n,其中n为十六进制转义值。十六进制转义值必须为确定的两个数字长。例如,“\x41”匹配“A”。“\x041”则等价于“\x04&1”。正则表达式中可以使用ASCII编码。.
  • \num匹配num,其中num是一个正整数。对所获取的匹配的引用。例如,“(.)\1”匹配两
  • 个连续的相同字符。只支持0≤num≤9,并且引用的只能是元表达式,比如(ab)将计数为两个表达式”a”和”b”而忽略括号

策略:

先将正则表达式解析,生成一棵语法树,树枝是”()”“[]”“|”带来的子表达式,节点的exp_id表示这个节点将用于匹配怎样的字符,还有字段记录这个表达式将重复的次数。

“()”的子树是和主树的结构一样的,解析规则也是相同的,但是”[]”和”|”的子树有自己的解析规则

每棵子树的根节点不匹配实际内容,而是用于指示它有子树

再用这棵语法树对字符串进行匹配

Step1:解析语法树

每个函数都只进行一个或特定几个字符的解析,之后递归的调用以递归下降分析

Check:

首先判断是否达到了正则表达式尾

“|”会再当前节点之前插入一个节点,并将当前节点移到插入的节点的孩子上,因为难以只对之后的一个表达式进行特殊处理,所以之后将继续解析。而是在解析完成之后进行再次的处理,遇到”|”的表达式的时候会将其后面的一个表达式放到其孩子树(此时已有一个节点)的后面,这样孩子树将有两个节点

“-”会判断当前的模式,如果在一个中括号中,并且当前节点和后面的都是一个非转义字符,那么就修改当前的节点为字符范围匹配,即便后面的是反斜杠也可以判断是否是非转义字符

“+”“*”会直接修改当前节点的重复次数(即便它有一棵孩子树)

“?”需要判断当前的节点是否已经被限制符修饰过了,如果是就将当前节点设置为非贪婪的,否则就只修改重复次数

在遇到”(““{““[““\”时会进入相应的函数进行特定的解析

递归调用自己进行递归下降解析

Check “\”:

\x将会有一个辅助函数将其后的两个字符解析为十进制整数,并插入为char

\+数字,将会有一个函数寻找对应位置的表达式指针,因为找的只是元表达式,所以不会有孩子节点,而且因为表达式只会被使用而不会被改变,所以引用也可以直接复制,不用深复制

Check”(“:

因为用了全局变量存储表达式树的头节点和当前节点,所以只用备份当前的值,并将值设置为新节点的孩子节点,之后再调用check就可以为新节点的孩子节点生成一个子树

Check”[“:

会和”()”进行相同的操作,差异是调用时指定的模式,而且”[]”再开头会判断是否有”^”

Check”{}”:

识别模式并修改当前的节点重复次数

Step2:检查字符串

Match pattern将会遍历表达式链表,并调用match expression检查每个单独的表达式。子树的遍历将会在match expression中通过调用match pattern进行

源码:

https://github.com/biaoJM/Simple-RegExp-C

C语言实现简化的正则表达式的更多相关文章

  1. 如何设计一门语言(十)——正则表达式与领域特定语言(DSL)

    几个月前就一直有博友关心DSL的问题,于是我想一想,我在gac.codeplex.com里面也创建了一些DSL,于是今天就来说一说这个事情. 创建DSL恐怕是很多人第一次设计一门语言的经历,很少有人一 ...

  2. go语言之进阶篇正则表达式

    正则表达式是一种进行模式匹配和文本操纵的复杂而又强大的工具.虽然正则表达式比纯粹的文本匹配效率低,但是它却更灵活.按照它的语法规则,随需构造出的匹配模式就能够从原始文本中筛选出几乎任何你想要得到的字符 ...

  3. JavaScript语言精粹 笔记05 正则表达式

    正则表达式 正则表达式以方法的形式被用于对字符串中的信息进行查找.替换画图提取操作.可处理正则表达式的方法有:regexp.exec, regexp.test,string.match, string ...

  4. C#语言之字符串和正则表达式

    本文将完成以下两个目标: 一.创建字符串: 二.正则表达式: 首先,我先来介绍一下System.String类: System.String是一个类,专门用于存储字符串,允许对字符串进行许多操作. 使 ...

  5. 表达式语言之java对正则表达式的处理

    正则表达式用于字符串匹配,字符串查找,字符串替换等.例如注册email格式的验证等.java中处理正则表达式相关的类主要有java.lang.String,java.util.regex.Patter ...

  6. Perl语言学习笔记 9 正则表达式处理文本

    1.更换 s/PATTERN/REPLACE/; #返回是否更换成功布尔值 能够使用捕获变量,如:s/(\w)/$1/ 匹配失败则不做不论什么处理 2.定界符 对于没有左右之分的定界符.反复三次就可以 ...

  7. 正则表达式与领域特定语言(DSL)

    如何设计一门语言(十)——正则表达式与领域特定语言(DSL) 几个月前就一直有博友关心DSL的问题,于是我想一想,我在gac.codeplex.com里面也创建了一些DSL,于是今天就来说一说这个事情 ...

  8. [Java] 简化正则表达式的使用

    使用 RegexString.with(string).pattern(pattern).start() + 后续操作(matches,find或者是replace) 源码 package com; ...

  9. Java语言中的正则表达式

    正则表达式是什么? 正则表达式是一种强大而灵活的文本处理工具.初学正则表达式时,其语法是一个难点,但它确实是一种简洁.动态的语言.正则表达式提供了一种完全通用的方式,能够解决各种字符串处理相关的问题: ...

随机推荐

  1. Eclipse中使用GIT更新项目

    GIT更新项目: 右击项目——Team——Pull:

  2. 路飞学城Python-Day48

    49-清除浮动1:给父盒子设置高度 给父盒子设置高度,这种方式不灵活,公司的产品修改的时候,要求父盒子高度变大, 不可能去手动修改 尽量不要给父元素去修改高度,不建议这样的方式 <!DOCTYP ...

  3. 并发编程——全局解释器锁GIL

    1.全局解释器锁GIL GIL其实就是一把互斥锁(牺牲了效率但是保证了数据的安全). 线程是执行单位,但是不能直接运行,需要先拿到python解释器解释之后才能被cpu执行 同一时刻同一个进程内多个线 ...

  4. 使用jq把js代码封装一个自己的插件

    为什么要把js功能封装成插件呢?我觉得有以下几点吧 1.最基本的原因就是便于代码复用. 2.便于维护和管理. 3.提升自身的能力. 4.避免各个相同功能组件的干扰,以及一些作用域会相互影响的问题. j ...

  5. easyUI combobox的使用

    1.需要用到的方法 设置组合框(combobox)值的数组. $('#cc').combobox('setValues', ['001','002']); 设置组合框(combobox)的值. $(' ...

  6. PAT 天梯赛练习集 L2-004. 这是二叉搜索树吗?

    题目链接: https://www.patest.cn/contests/gplt/L2-004 一棵二叉搜索树可被递归地定义为具有下列性质的二叉树:对于任一结点, 其左子树中所有结点的键值小于该结点 ...

  7. docker 镜像的导入导出

    今天使用docker部署asp.net core应用程序时,发现当我们做好基础镜像之后需要把镜像导出到正式环境,因此学习了一下如何从docker中导出镜像: 1.首先通过docker images命令 ...

  8. 紫书 习题7-13 UVa 817(dfs+栈求表达式的值)

    题目链接  点击打开链接 这道题分为两个部分, 一用搜索枚举每种可能, 二计算表达式的值, 有挺多细节需要注意 特别注意我的代码中在计算表达式的值中用到了一个!(代码枚举中的!表示不加符号, 我现在说 ...

  9. jsonp实现原理

    jquery 封装在 ajax方法 里面的jsonp jsonp跨域的原理       1:使用script 标签发送请求,这个标签支持跨域访问       2:在script 标签里面给服务器端传递 ...

  10. 高级聚合函数rollup(),cube(),grouping sets()

       rollup(),cube(),grouping sets()   上面这几个函数,是对group by分组功能做的功能扩展. a.rollup()   功能:在原结果基础上追加一行总合计记录 ...