一行正则表达式判断质数的代码

 更新时间:2022年05月26日 16:22:52   作者:jayzou  
这篇文章主要介绍了一行正则表达式判断质数,其实这个正则性能非常差(穷举法),实用性不高,但是思路很让人惊艳,需要的朋友可以参考下

背景

昨天无意中看到一篇大佬的文章Primality regex正则表达式判断质数),惊为天人,正则表达式也能用来判断质数了?立马来研究下

示例

perl -wle 'print "Prime" if (1 x shift) !~ /^1?$|^(11+?)\1+$/' [number]

翻译成JS代码如下

function isPrime(n) {
    return !/^1?$|^(11+?)\1+$/.test("1".repeat(n))
}

代码逻辑非常简单,生成"1" * n长度的字符串,通过/^1?$|^(11+?)\1+$/正则表达式进行判断,再将结果取反

正则分析

/^1?$|^(11+?)\1+$/

上面正则表达式有2个分支,分别是

  • /^1?$
  • ^(11+?)\1+$

分支1 逻辑很简单,就是匹配0或者1个 "1",因为要排除数字1(非质数)

分支2 就有意思了,可以拆成2块来看

  • ^(11+?)
  • \1+$

表达式1,非贪婪模式下匹配 "11" "111" "1111"....,作为一个分组
表达式2,\1代表将表达式1匹配的结果赋值给\1,判断是否结尾,否的话会触发回溯(因为表达式1可能匹配多种情况)

举个例子就更清晰了,比如传入n = 9,分支1不满足可以直接忽略
^(11+?)\1+$

步骤匹配结果说明
step 11 1 1 1 1 1 1 1 1(11+?)匹配到"11"
step 21 1 1 1 1 1 1 1 1分组结果赋值给\1,那么正则就变成 "11"+$,继续匹配剩余的字符(7个"1")
step 31 1 1 1 1 1 1 1 1再重复3轮的匹配,发现剩余一个"1",不满足$,进行回溯
step 41 1 1 1 1 1 1 1 1还是不满足$,继续回溯
step 51 1 1 1 1 1 1 1 1一直回溯到step 1(11+?)匹配到"111"
step 61 1 1 1 1 1 1 1 1分组结果赋值给\1,那么正则就变成 "111"+$,继续匹配剩余的字符(6个"1")
step 71 1 1 1 1 1 1 1 1再重复2轮的匹配,满足$,匹配成功

原理

经过上述的分析,不难发现,其实回溯就是将数字不断除于2、3、4....,最后检查是否有余数,没有的话就匹配成功(非质数),非常简单粗暴的穷举法

优化空间

仔细看正则匹配的过程分析,其实step 3 ~ step 4的回溯完全没有必要,那么正则可以改写成这样/^1?$|^(11+?)\1+?$/,将\1+改成非贪婪模式\1+?,那么就放弃step 4回溯

性能测试

console.time('优化前')
console.log(!/^1?$|^(11+?)\1+$/.test("1".repeat(33331)));
console.timeEnd('优化前')
console.time('优化后')
console.log(!/^1?$|^(11+?)\1+?$/.test("1".repeat(33331)));
console.timeEnd('优化后')
// true
// 优化前: 227.9189453125 ms
// true
// 优化后: 155.797119140625 ms

耗时上减少了接近一半

总结

其实这个正则性能非常差(穷举法),实用性不高,但是思路很让人惊艳

到此这篇关于一行正则表达式判断质数的文章就介绍到这了,更多相关正则表达式判断质数内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 正则表达式模式修正符 比如/esi

    正则表达式模式修正符 比如/esi

    下面列出了当前在 PCRE 中可能使用的修正符。括号中是这些修正符的内部 PCRE 名。修正符中的空格和换行被忽略,其它字符会导致错误。
    2010-07-07
  • 精确查找PHP WEBSHELL木马 修正版

    精确查找PHP WEBSHELL木马 修正版

    上篇提到了关于网上流传查找PHP webshell的python脚本中,不严谨的代码,并且给出了一个python的检测代码,同时,下文里也提到不能检测到反引号的命令执行的地方。今天,我想了下,现在把思路发出来。
    2011-04-04
  • 正则基础之 神奇的转义

    正则基础之 神奇的转义

    不同的语言或应用场景下,正则定义方式、元字符出现的位置不同,转义的方式也是林林总总,不一而同
    2012-10-10
  • 在实际例子中学习正则表达式(高效率)

    在实际例子中学习正则表达式(高效率)

    正则表达式,又称正规表示法、常规表示法。下面小编给大家分享几个例子给大家讲下正则表达式知识,非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2016-08-08
  • RegExp 随笔 JavaScript RegExp 对象

    RegExp 随笔 JavaScript RegExp 对象

    这篇文章主要介绍了RegExp 随笔 JavaScript RegExp 对象,需要的朋友可以参考下
    2016-10-10
  • 正则表达式不区分大小写以及解决思路的探索 .

    正则表达式不区分大小写以及解决思路的探索 .

    今天在写一个正则表达式的时候,因为字符有大小写的问题,多种大小写的组合,这时想到了用正则表达式
    2014-06-06
  • 常用的正则表达式实例整理

    常用的正则表达式实例整理

    这篇文章主要介绍了常用的正则表达式实例整理,非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2018-03-03
  • JavaScript 实现基础 正则表达式

    JavaScript 实现基础 正则表达式

    正则表达式用来从某一段字符串中匹配所需要的字符,这些字符可以非常简单,也可以非常复杂。JavaScript生来就对正则表达式有着良好的支持,在网络的字符搜索匹配中发挥着重要的作用。
    2009-08-08
  • 正则表达式的字符串替换方法

    正则表达式的字符串替换方法

    这篇文章主要介绍了正则表达式的字符串替换方法,用到了一些高级的正则写法,需要的朋友可以参考下
    2016-01-01
  • 详解正则表达式Matcher类中group方法

    详解正则表达式Matcher类中group方法

    这篇文章主要介绍了正则表达式Matcher类中group方法,本文给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下
    2019-08-08

最新评论