一个正则表达式

一个正则表达式就是由普通字符(例如字符 a 到 z)以及特殊字符(称为 元字符)组成的文字模式。该模式描述在查找文字主体时待匹配的一个或多个字符串。正则表达式作为一个模板,将某个字符模式与所搜索的字符串进行匹配。这里有一些可能会遇到的正则表达式示例:
JScriptVBScript匹配
/^/[ /t]*$/"^/[ /t]*$"匹配一个空白行。
//d{2}-/d{5}/"/d{2}-/d{5}"验证一个ID 号码是否由一个2位数字,一个连字符以及一个5位数字组成。
/<(.*)>.*<///1>/"<(.*)>.*<///1>"匹配一个 HTML 标记。
下表是元字符及其在正则表达式上下文中的行为的一个完整列表:
字符描述
/将下一个字符标记为一个特殊字符、或一个原义字符、或一个 向后引用、或一个八进制转义符。例如,'n' 匹配字符 "n"。'/n' 匹配一个换行符。序列 '//' 匹配 "/" 而 "/(" 则匹配 "("。
^匹配输入字符串的开始位置。如果设置了 RegExp 对象的 Multiline 属性,^ 也匹配 '/n' 或 '/r' 之后的位置。
$匹配输入字符串的结束位置。如果设置了RegExp 对象的 Multiline 属性,$ 也匹配 '/n' 或 '/r' 之前的位置。
*匹配前面的子表达式零次或多次。例如,zo* 能匹配 "z" 以及 "zoo"。* 等价于{0,}。
+匹配前面的子表达式一次或多次。例如,'zo+' 能匹配 "zo" 以及 "zoo",但不能匹配 "z"。+ 等价于 {1,}。
?匹配前面的子表达式零次或一次。例如,"do(es)?" 可以匹配 "do" 或 "does" 中的"do" 。? 等价于 {0,1}。
{n}n 是一个非负整数。匹配确定的 n 次。例如,'o{2}' 不能匹配 "Bob" 中的 'o',但是能匹配 "food" 中的两个 o。
{n,}n 是一个非负整数。至少匹配n 次。例如,'o{2,}' 不能匹配 "Bob" 中的 'o',但能匹配 "foooood" 中的所有 o。'o{1,}' 等价于 'o+'。'o{0,}' 则等价于 'o*'。
{n,m}mn 均为非负整数,其中n <= m。最少匹配 n 次且最多匹配 m 次。例如,"o{1,3}" 将匹配 "fooooood" 中的前三个 o。'o{0,1}' 等价于 'o?'。请注意在逗号和两个数之间不能有空格。
?当该字符紧跟在任何一个其拗品?(*, +, ?, {n}, {n,}, {n,m}) 后面时,匹配模式是非贪婪的。非贪婪模式尽可能少的匹配所搜索的字符串,而默认的贪婪模式则尽可能多的匹配所搜索的字符串。例如,对于字符串 "oooo",'o+?' 将匹配单个 "o",而 'o+' 将匹配所有 'o'。
.匹配除 "/n" 之外的任何单个字符。要匹配包括 '/n' 在内的任何字符,请使用象 '[./n]' 的模式。
(pattern)匹配 pattern 并获取这一匹配。所获取的匹配可以从产生的 Matches 集合得到,在VBScript 中使用 SubMatches 集合,在JScript 中则使用 $0$9 属性。要匹配圆括号字符,请使用 '/(' 或 '/)'。
(?:pattern)匹配 pattern 但不获取匹配结果,也就是说这是一个非获取匹配,不进行存储供以后使用。这在使用 "或" 字符 (|) 来组合一个模式的各个部分是很有用。例如, 'industr(?:y|ies) 就是一个比 'industry|industries' 更简略的表达式。
(?=pattern)正向预查,在任何匹配 pattern 的字符串开始处匹配查找字符串。这是一个非获取匹配,也就是说,该匹配不需要获取供以后使用。例如,'Windows (?=95|98|NT|2000)' 能匹配 "Windows 2000" 中的 "Windows" ,但不能匹配 "Windows 3.1" 中的 "Windows"。预查不消耗字符,也就是说,在一个匹配发生后,在最后一次匹配之后立即开始下一次匹配的搜索,而不是从包含预查的字符之后开始。
(?!pattern)负向预查,在任何不匹配 pattern 的字符串开始处匹配查找字符串。这是一个非获取匹配,也就是说,该匹配不需要获取供以后使用。例如'Windows (?!95|98|NT|2000)' 能匹配 "Windows 3.1" 中的 "Windows",但不能匹配 "Windows 2000" 中的 "Windows"。预查不消耗字符,也就是说,在一个匹配发生后,在最后一次匹配之后立即开始下一次匹配的搜索,而不是从包含预查的字符之后开始
x|y匹配 xy。例如,'z|food' 能匹配 "z" 或 "food"。'(z|f)ood' 则匹配 "zood" 或 "food"。
[xyz]字符集合。匹配所包含的任意一个字符。例如, '[abc]' 可以匹配 "plain" 中的 'a'。
[^xyz]负值字符集合。匹配未包含的任意字符。例如, '[^abc]' 可以匹配 "plain" 中的'p'。
[a-z]字符范围。匹配指定范围内的任意字符。例如,'[a-z]' 可以匹配 'a' 到 'z' 范围内的任意小写字母字符。
[^a-z]负值字符范围。匹配任何不在指定范围内的任意字符。例如,'[^a-z]' 可以匹配任何不在 'a' 到 'z' 范围内的任意字符。
/b匹配一个单词边界,也就是指单词和空格间的位置。例如, 'er/b' 可以匹配"never" 中的 'er',但不能匹配 "verb" 中的 'er'。
/B匹配非单词边界。'er/B' 能匹配 "verb" 中的 'er',但不能匹配 "never" 中的 'er'。
/cx匹配由 x 指明的控制字符。例如, /cM 匹配一个 Control-M 或回车符。x 的值必须为 A-Z 或 a-z 之一。否则,将 c 视为一个原义的 'c' 字符。
/d匹配一个数字字符。等价于 [0-9]。
/D匹配一个非数字字符。等价于 [^0-9]。
/f匹配一个换页符。等价于 /x0c 和 /cL。
/n匹配一个换行符。等价于 /x0a 和 /cJ。
/r匹配一个回车符。等价于 /x0d 和 /cM。
/s匹配任何空白字符,包括空格、制表符、换页符等等。等价于 [ /f/n/r/t/v]。
/S匹配任何非空白字符。等价于 [^ /f/n/r/t/v]。
/t匹配一个制表符。等价于 /x09 和 /cI。
/v匹配一个垂直制表符。等价于 /x0b 和 /cK。
/w匹配包括下划线的任何单词字符。等价于'[A-Za-z0-9_]'。
/W匹配任何非单词字符。等价于 '[^A-Za-z0-9_]'。
/xn匹配 n,其中 n 为十六进制转义值。十六进制转义值必须为确定的两个数字长。例如,'/x41' 匹配 "A"。'/x041' 则等价于 '/x04' & "1"。正则表达式中可以使用 ASCII 编码。.
/num匹配 num,其中 num 是一个正整数。对所获取的匹配的引用。例如,'(.)/1' 匹配两个连续的相同字符。
/n标识一个八进制转义值或一个向后引用。如果 /n 之前至少 n 个获取的子表达式,则 n 为向后引用。否则,如果 n 为八进制数字 (0-7),则 n 为一个八进制转义值。
/nm标识一个八进制转义值或一个向后引用。如果 /nm 之前至少有 nm 个获得子表达式,则 nm 为向后引用。如果 /nm 之前至少有 n 个获取,则 n 为一个后跟文字 m 的向后引用。如果前面的条件都不满足,若 nm 均为八进制数字 (0-7),则 /nm 将匹配八进制转义值 nm
/nml如果 n 为八进制数字 (0-3),且 ml 均为八进制数字 (0-7),则匹配八进制转义值 nml。
/un匹配 n,其中 n 是一个用四个十六进制数字表示的 Unicode 字符。例如, /u00A9 匹配版权符号 (?)。

  『1楼』作者:游泳的鱼 时间:2005-04-15 13:20:25

正则表达式用于字符串处理,表单验证等场合,实用高效,但用到时总是不太把握,以致往往要上网查一番。我将一些常用的表达式收藏在这里,作备忘之用。本贴随时会更新。

匹配中文字符的正则表达式: [/u4e00-/u9fa5]

匹配双字节字符(包括汉字在内):[^/x00-/xff]

应用:计算字符串的长度(一个双字节字符长度计2,ASCII字符计1)

String.prototype.len=function(){return this.replace([^/x00-/xff]/g,"aa").length;}

匹配空行的正则表达式:/n[/s| ]*/r

匹配HTML标记的正则表达式:/<(.*)>.*<///1>|<(.*) //>/

匹配首尾空格的正则表达式:(^/s*)|(/s*$)

 

String.prototype.trim = function()
{
    return this.replace(/(^/s*)|(/s*$)/g, "");
}

利用正则表达式分解和转换IP地址:

下面是利用正则表达式匹配IP地址,并将IP地址转换成对应数值的javascript程序:

function IP2V(ip)
{
 re=/(/d+)/.(/d+)/.(/d+)/.(/d+)/g  //匹配IP地址的正则表达式
if(re.test(ip))
{
return RegExp.$1*Math.pow(255,3))+RegExp.$2*Math.pow(255,2))+RegExp.$3*255+RegExp.$4*1
}
else
{
 throw new Error("Not a valid IP address!")
}
}

不过上面的程序如果不用正则表达式,而直接用split函数来分解可能更简单,程序如下:

var ip="10.100.20.168"
ip=ip.split(".")
alert("IP值是:"+(ip[0]*255*255*255+ip[1]*255*255+ip[2]*255+ip[3]*1))

匹配Email地址的正则表达式:/w+([-+.]/w+)*@/w+([-.]/w+)*/./w+([-.]/w+)*

匹配网址URL的正则表达式:http://([/w-]+/.)+[/w-]+(/[/w- ./?%&=]*)?

利用正则表达式去除字串中重复的字符的算法程序 :[注:此程序不正确,原因见本贴回复]

var s="abacabefgeeii"
var s1=s.replace(/(.).*/1/g,"$1")
var re=new RegExp("["+s1+"]","g")
var s2=s.replace(re,"")
alert(s1+s2)  //结果为:abcefgi

我原来在CSDN上发贴寻求一个表达式来实现去除重复字符的方法,最终没有找到,这是我能想到的最简单的实现方法。思路是使用后向引用取出包括重复的字符,再以重复的字符建立第二个表达式,取到不重复的字符,两者串连。这个方法对于字符顺序有要求的字符串可能不适用。

得用正则表达式从URL地址中提取文件名的javascript程序,如下结果为page1

s="http://www.9499.net/page1.htm"
s=s.replace(/(.*//){0,}([^/.]+).*/ig,"$2")
alert(s)

利用正则表达式限制网页表单里的文本框输入内容:

用正则表达式限制只能输入中文:οnkeyup="value=value.replace(/[^/u4E00-/u9FA5]/g,'')" onbeforepaste="clipboardData.setData('text',clipboardData.getData('text').replace(/[^/u4E00-/u9FA5]/g,''))"

用正则表达式限制只能输入全角字符: οnkeyup="value=value.replace(/[^/uFF00-/uFFFF]/g,'')" onbeforepaste="clipboardData.setData('text',clipboardData.getData('text').replace(/[^/uFF00-/uFFFF]/g,''))"

用正则表达式限制只能输入数字:οnkeyup="value=value.replace(/[^/d]/g,'') "onbeforepaste="clipboardData.setData('text',clipboardData.getData('text').replace(/[^/d]/g,''))"

用正则表达式限制只能输入数字和英文:οnkeyup="value=value.replace(/[/W]/g,'') "onbeforepaste="clipboardData.setData('text',clipboardData.getData('text').replace(/[^/d]/g,''))"

应用:javascript中没有像vbscript那样的trim函数,我们就可以利用这个表达式来实现,如下:

  『2楼』作者:游泳的鱼 时间:2005-04-15 13:20:51
匹配中文字符的正则表达式: [/u4e00-/u9fa5]

匹配双字节字符(包括汉字在内):[^/x00-/xff]

应用:计算字符串的长度(一个双字节字符长度计2,ASCII字符计1)

String.prototype.len=function(){return this.replace([^/x00-/xff]/g,"aa").length;}

匹配空行的正则表达式:/n[/s| ]*/r

匹配HTML标记的正则表达式:/<(.*)>.*<///1>|<(.*) //>/

匹配首尾空格的正则表达式:(^/s*)|(/s*$)

应用:javascript中没有像vbscript那样的trim函数,我们就可以利用这个表达式来实现,如下:

String.prototype.trim = function()
{
return this.replace(/(^/s*)|(/s*$)/g, "");
}

利用正则表达式分解和转换IP地址:

下面是利用正则表达式匹配IP地址,并将IP地址转换成对应数值的Javascript程序:

function IP2V(ip)
{
re=/(/d+)/.(/d+)/.(/d+)/.(/d+)/g //匹配IP地址的正则表达式
if(re.test(ip))
{
return RegExp.$1*Math.pow(255,3))+RegExp.$2*Math.pow(255,2))+RegExp.$3*255+RegExp.$4*1
}
else
{
throw new Error("Not a valid IP address!")
}
}

不过上面的程序如果不用正则表达式,而直接用split函数来分解可能更简单,程序如下:

var ip="10.100.20.168"
ip=ip.split(".")
alert("IP值是:"+(ip[0]*255*255*255+ip[1]*255*255+ip[2]*255+ip[3]*1))

匹配Email地址的正则表达式:/w+([-+.]/w+)*@/w+([-.]/w+)*/./w+([-.]/w+)*

匹配网址URL的正则表达式:http://([/w-]+/.)+[/w-]+(/[/w- ./?%&=]*)?

利用正则表达式去除字串中重复的字符的算法程序:[注:此程序不正确,原因见本贴回复]

var s="abacabefgeeii"
var s1=s.replace(/(.).*/1/g,"$1")
var re=new RegExp("["+s1+"]","g")
var s2=s.replace(re,"")
alert(s1+s2) //结果为:abcefgi

我原来在CSDN上发贴寻求一个表达式来实现去除重复字符的方法,最终没有找到,这是我能想到的最简单的实现方法。思路是使用后向引用取出包括重复的字符,再以重复的字符建立第二个表达式,取到不重复的字符,两者串连。这个方法对于字符顺序有要求的字符串可能不适用。

得用正则表达式从URL地址中提取文件名的javascript程序,如下结果为page1

s="http://www.9499.net/page1.htm"
s=s.replace(/(.*//){0,}([^/.]+).*/ig,"$2")
alert(s)

利用正则表达式限制网页表单里的文本框输入内容:

用正则表达式限制只能输入中文:οnkeyup="value=value.replace(/[^/u4E00-/u9FA5]/g,'')" onbeforepaste="clipboardData.setData('text',clipboardData.getData('text').replace(/[^/u4E00-/u9FA5]/g,''))"

用正则表达式限制只能输入全角字符: οnkeyup="value=value.replace(/[^/uFF00-/uFFFF]/g,'')" onbeforepaste="clipboardData.setData('text',clipboardData.getData('text').replace(/[^/uFF00-/uFFFF]/g,''))"

用正则表达式限制只能输入数字:οnkeyup="value=value.replace(/[^/d]/g,'') "onbeforepaste="clipboardData.setData('text',clipboardData.getData('text').replace(/[^/d]/g,''))"

用正则表达式限制只能输入数字和英文:οnkeyup="value=value.replace(/[/W]/g,'') "onbeforepaste="clipboardData.setData('text',clipboardData.getData('text').replace(/[^/d]/g,''))"

  『3楼』作者:游泳的鱼 时间:2005-04-15 13:21:47
[前言:]在我们编写WEB程序时,经常会判断一个字符串的有效性,如;一个串是否是数字、是否是有效的Email地址等等。如果不使用

正则表达式,那么判断的程序会很长,并且容易出错,如果使用正则表达式,这些判断就是一件很轻松的工作了。本文全面介绍正则表达式的

慨念、格式。并以在PHP、ASP中的应用实例增加读者的感性认识。正则表达式的应用很广,需要大家在学习和实践中不断的总结。
  正则表达式简介 

  简单的说,正则表达式是一种可以用于模式匹配和替换的强有力的工具。在网络编程中应用广泛,如PHP脚本语言或是JavaScript、VBScri

pt这样的客户端脚本语言都提供了对正则表达式的支持。由此可见,正则表达式已经超出了某种语言或某个系统的局限,成为人们广为接受的

概念和功能。 

  正则表达式可以让用户通过使用一系列的特殊字符构建匹配模式,然后把匹配模式与数据文件、程序输入以及WEB页面的表单输入等目标对

象进行比较,根据比较对象中是否包含匹配模式,执行相应的程序。 

  举例来说,正则表达式的一个最为普遍的应用就是用于验证用户在线输入的邮件地址的格式是否正确,如果通过正则表达式验证用户邮件

地址的格式正确,用户所填写的表单信息将会被正常处理;反之,如果用户输入的邮件地址与正则表达的模式不匹配,将会弹出提示信息,要

求用户重新输入正确的邮件地址。由此可见正则表达式在WEB应用的逻辑判断中具有举足轻重的作用。在后面我们会举例详细介绍。 

  正则表达式形式一般如:/love/,其中位于"/"定界符之间的部分就是将要在目标对象中进行匹配的模式。用户只要把希望查找匹配对象的

模式内容放入"/"定界符之间即可。为了能够使用户更加灵活的定制模式内容,正则表达式提供了专门的"元字符"。所谓元字符就是指那些在正

则表达式中具有特殊意义的专用字符,可以用来规定其前导字符(即位于元字符前面的字符)在目标对象中的出现模式。较为常用的元字符包

括:  "+,"*,?以及{}",或者"/s,/S,/d,/w和/W"等等。为了能够方便用户更加灵活的设定匹配模式,正则表达式允许使用者在匹配模式中利

用[]界定匹配于某一个范围的字符而不局限于具体的字符。 

  除了我们以上的元字符之外,正则表达式中还具有另外一种较为独特的专用字符,即定位符。定位符用于规定匹配模式在目标对象中的出

现位置。较为常用的定位符包括:  "^",  "$",  "/b"  以及  "/B"。 

  如果我们希望在正则表达式中实现类似编程逻辑中的"或"运算,在多个不同的模式中任选一个进行匹配的话,可以使用管道符  "|"。例如

: 

  正则表达式中还有一个较为常用的运算符,即否定符  "[^]"。与我们前文所提到的定位符  "^"  不同,否定符  "[^]"规定目标对象中不能存

在模式中所规定的字符串。一般来说,当"^"出现在  "[]"内时就被视做否定运算符;而当"^"位于"[]"之外,或没有"[]"时,则应当被视做定位

符。 

  最后,当用户需要在正则表达式的模式中加入元字符,并查找其匹配对象时,可以使用转义符"/"。例如:/Th/*/,该正则表达式将会与目

标对象中的"Th*"而非"The"等相匹配。

  正则表达式的语法规则和标记

  现在我们正式进入则表达式的学习,我会根据实例结合讲解正则表达式的用法,看完后你就会觉得写UBB代码如此简单了,只要你一步一步

的跟着我学  看完本文章后你就成为UBB高手了。激动人心的就是你能写出自已的UBB标签来了,再也不用到别人那里去拷贝现成的代码和模板了

。  还好VBScritp5.0给我们提供了"正则表达式"对象,只要你的服务器安装了IE5.x,就可以运行了.

  字符描述:

      ^符号匹配字符串的开头。例如:

    ^abc 与"abc  xyz"匹配,而不与"xyz  abc"匹配

  $符号匹配字符串的结尾。例如:

    abc$ 与"xyz  abc"匹配,而不与"abc  xyz"匹配。

    注意:如果同时使用^符号和$符号,将进行精确匹配。例如:

       ^abc$ 只与"abc"匹配   

  *符号匹配0个或多个前面的字符。例如:

    ab* 可以匹配"ab"、"abb"、"abbb"等

  +符号匹配至少一个前面的字符。例如:

    ab+ 可以匹配"abb"、"abbb"等,但不匹配"ab"。 

  ?符号匹配0个或1个前面的字符。例如:

    ab?c? 可以且只能匹配"abc"、"abbc"、"abcc"和"abbcc"

  .符号匹配除换行符以外的任何字符。例如:

    (.)+ 匹配除换行符以外的所有字符串

  x|y匹配"x"或"y"。例如:

    abc|xyz 可匹配  "abc"或  "xyz",而"ab(c|x)yz"匹配  "abcyz"和"abxyz"

  {n}匹配恰好n次(n为非负整数)前面的字符。例如:

    a{2} 可以匹配"aa",但不匹配"a"

  {n,}匹配至少n次(n为非负整数)前面的字符。例如:

    a{3,} 匹配"aaa"、"aaaa"等,但不匹配"a"和"aa"。

    注意:a{1,}等价于a+

       a{0,}等价于a*

  {m,n}匹配至少m个,至多n个前面的字符。例如:

    a{1,3} 只匹配"a"、"aa"和"aaa"。

    注意:a{0,1}等价于a?

  [xyz]表示一个字符集,匹配括号中字符的其中之一。例如:

    [abc] 匹配"a"、"b"和"c"

  [^xyz]表示一个否定的字符集。匹配不在此括号中的任何字符。例如:

    [^abc] 可以匹配除"a"、"b"和"c"之外的任何字符

  [a-z]表示某个范围内的字符,匹配指定区间内的任何字符。例如:

    [a-z] 匹配从"a"到"z"之间的任何一个小写字母字符

  [^m-n]表示某个范围之外的字符,匹配不在指定范围内的字符。例如:

    [m-n] 匹配除从"m"到"n"之间的任何字符

  /符号是转义操作符。例如:

    /n 换行符

    /f 分页符

    /r 回车

    /t 制表符

    /v 垂直制表符 

    // 匹配"/"

    // 匹配"/"

    /s 任何白字符,包括空格、制表符、分页符等。等价于"[  /f/n/r/t/v]"

    /S 任何非空白的字符。等价于"^/f/n/r/t/v]"

    /w 任何单词字符,包括字母和下划线。等价于"[A-Za-z0-9_]"

    /W 任何非单词字符。等价于"[^A-Za-z0-9_]"

    /b匹配单词的结尾。例如:

      ve/b 匹配单词"love"等,但不匹配"very"、"even"等

    /B匹配单词的开头。例如:

      ve/B 匹配单词"very"等,但不匹配"love"等

    /d匹配一个数字字符,等价于[0-9]。例如:

      abc/dxyz 匹配"abc2xyz"、"abc4xyz"等,
      但不匹配"abcaxyz"、"abc-xyz"等

    /D匹配一个非数字字符,等价于[^0-9]。例如:

      abc/Dxyz 匹配"abcaxyz"、"abc-xyz"等,
      但不匹配"abc2xyz"、"abc4xyz"等

    /NUM匹配NUM个(其中NUM为一个正整数),引用回到记住的匹配。例如:

      (.)/1 匹配两个连续相同的字符。 

    /oNUM匹配n(其中n为一个小于256的八进制换码值)。例如:

      /o011 匹配制表符

    /xNUM匹配NUM(其中NUM为一个小于256的十六进制换码值)。例如:

      /x41 匹配字符"A"
 
  应用实例 

  在对正则表达式有了较为全面的了解之后,就可以在Perl,PHP,以及ASP等程式中使用正则表达式了。

  下面以PHP语言为例,使用验证用户在线输入的邮件地址以及网址的格式是否正确。PHP  提供了eregi()或ereg()资料处理函数实现字串比

对剖析的模式匹配操作ereg()函数的使用格式如下: 

        ereg  (pattern,  string) 

  其中,pattern代表正则表达式的模式;而string则是执行查找替换操作的目标对象,如Email地址值。本函式以  pattern  的规则来剖析比

对字串  string,找到则传回值为  true。函式ereg()与eregi()的区别就是前者区分大小写,后者与大小写无关。使用PHP编写的程序代码如下

: 

      <?php 
    if  (ereg("^([a-z0-9_-])+@([a-zZ0-9_-])+(/.[a-z0-9_-])+[a-z]{2,3}$",$email)) 
   {  echo  "您的  E-Mail  通过初步检查!";} 
    else 
   {  echo  "不是合法的E-Mail  地址,请重新输入!";} 
    ?> 


  这个例子是可对使用者输入的  E-Mail  作简单的检查,检查使用者的  E-Mail  字串是否有  @  字元,在  @  字元前有小写英文字母、数字或

下"_",在  @  之后有数节字串,最后的小数点后只能有二个或三个小写英文字母。如webmaster@mail.sever.net,  hello_2001@88new.cn就可以

通过检查,而New99@253.com(出现大写字母)和new99@253.comn(最后的小数点后只能超过3个英文字母)就不能通过检查。 

  我们通过调用自定义正规则判别函式也可以进行检查操作,如下面的网址检验函式: 

function  VerifyWebSiteAddr  ($strWebSiteAddr){ 
return  (eregi  ("^([_0-9a-z-]+.)+([0-9a-z-]+.)+[a-z]{2,3}$",  $strWebSiteAddr)); 


  我们知道,PHP程式的运行必须有服务器支持,如果您在自己的主页上想实现以上功能, 

嵌入式脚本语言Javascript或许是好的选择。JavaScript中带有一个功能强大的RegExp()对象,可以用来进行正则表达式的匹配操作。其中的t

est()方法可以检验目标对象中是否包含匹配模式,并相应的返回true或false。只须在HTML文档的<head>区域添加一段Javascript代码。 

<  language="Javascript1.2"> 
  function  verifyAddress(obj){ 
   var  email  =  obj.email.value; 
   var  pattern  =  /^([a-zA-Z0-9_-])+@([a-zA-Z0-9_-])+(/.[a-zA-Z0-9_-])+/; 
   flag  =  pattern.test(email); 
   if(flag){ 
    alert("您的  E-Mail  通过初步检查!"); 
    return  true;} 
   else{ 
    alert("不是合法的E-Mail  地址,请重新输入!"); 
    return  false;} 
  } 
 </script> 


  然后在网页中输入信息的表单域<form>标签区域内中加入一行如下代码: 

   <onSubmit="return  verifyAddress(this);"> 

  当按下提交按钮后,首先运行verifyAddress()函式,进行匹配识别,如果满足条件则发送表单信息到目标页面,否则返回错误信息。 

  事实上,正则表达式的功能远非本文提到的这一点,下次,给大家介绍一种使用正则表达式从任意指定网页中析取任意种类文本信息(如

网页中所有的图片文件名)的技巧。  


  Html源文件中image标签的析取 

  上篇,我们介绍了正则表达式的概念及其在网络编程中使用正则表达式验证用户在线输入的邮件地址以及网址的格式是否正确的应用实例

,今天介绍一种从指定网页源文件中析取image标签的的编程技巧,即从网页源文件中,解析出所有的插图文件名(包括图片路径),也就是标

签<img  src=".../.../abc.jpg">  中的文件名".../.../abc.jpg"(有的可能是gif格式)。编程环境:PHP+Apache  for  Win98。 

  首先, 

用文本编辑器新建一个PHP类型的文件:abstractSRCfrompage.php3。为了方便讲解,我们打算是在浏览器表单域中输入需要析取image标签的

网页的URL(或本机文档),提交后执行析取操作,所以在该文件中,我们要建立一个用于输入网址的表单,举例如下: 


<form  action="  abstractSRCfrompage.php3"  method="post">
  输入网址<input  type=text  name=filename><br>
  <input  type=submit  name=submit  value="提交">
</form> 


  输入正确的网址,提交后表单信息被送到abstractSRCfrompage.php3页面,由于表单本身就在该页面,所以相当于被送到自身页面,下面

我们需要编写析取处理的PHP代码,紧接着表单代码段后写入如下代码:


<?php
if  ($filename!=""){
$fp  =  fopen($filename,  "r");  file://若输入不为空,开启本地或者远程档案;
while  ($buffer  =  fgets($fp,  1024))  {
$source  .=$buffer;  }
fclose($fp);
file://查找$source中是否有<img  ...src=".../...gif  |  jpg">  这样的标记
if(eregi("(<img)+[^<>]+(src=/")+[^/*/"<>|]+(/.)+((gif)|(jpg))+(/")",$source))  {
echo  "找到图片标签:)<br>";}
else{
echo  "未发现图片标签:(<br>";}
file://拆分,第一次用标签,<img  ...src=拆分,得到了以图形文件名开头的数组,
$splitres=split("((/">)|())+(<img)+[^<>]+(src=/")",$source);
echo  "找到:  $imagenums-1个图片<br>分别为:<br>";
for($i=1;$i<sizeof($splitres);$i++){
file://二次拆分,用"拆分。因为文件名能含有",得到的拆分数组的第一个元素就是路径+文件名了;
unset($imgname);  //  再次使用前删除imgname变量;
$imgname=spliti("/"",$splitres[$i]);//将析取的图片信息依次赋给imgname变量
echo  "$i=>".$imgname[0]."<br>";  file://输出析取的图片信息
}
}
?>


  该段程序的设计思路是:PHP程式判断是否输入了档案名称(网址URL或本机档案名),若不为空则以只读方式打开该档案;接着使用函式fge

ts(fp,length)取得档案指标fp所指的行并传回该行内长度为length-1  的字串,上例中就是1024-1=1023;然后利用字串比对剖析函式ereg()查

找$source中是否含有<img  ...src=".../...gif  |  jpg"> 

这样的标记(关于该函式在上篇中有详细的介绍);假如找到的话,则利用split()函式按一定的规则执行两次拆分,去掉标记中的<img…src

=字符和"字符,结果得到数组splitres,其中的每一个元素都是以图形路径+文件名开头的数组;用for循环在屏幕上输出每个数组的第一个元

素值,即我们所需要的所有图形路径+文件名。

  其中函式sizeof($splitres)返回个数;在for循环中,依次将数组splitres的每个元素(也是数组)赋给数组变量imgname,并输出imgnam

e的第一个元素值(为找到的一个图形路径+文件名),当执行下次循环时,删除变量imgname,达到重复使用的目的。大家可以仔细研究其中的

奥妙。

  好了,写好后,将abstractSRCfrompage.php3存到你的服务器指定目录下,启动Apache服务器,在浏览器中打开它,随便输入一个存在的

网页名称或是远程URL,看看效果如何。

  如果有兴趣,你可以尝试析取HTML文档中的任意感兴趣的信息,如果稍加改装,做一个网站文本搜索引擎岂不更妙?


  正则表达式在UBB论坛中的应用

  一、UBB代码的概念

  什么是UBB代码?

  UBB代码是HTML的一个变种。一般情况下,UBB论坛不允许你使用HTML代码,而只能用UBB代码替代HTML代码。

  UBB代码是一套由流行的UBB标签组成了固定代码,代码有统一的格式。用户只要遵循代码规则就可以实现用户想要的功能。如:

  想要显示粗体的how  are  you  字样,就应该输入  how  are  you而不是输入<b>how  are  you</b>

  你也许会问:ASP是怎样把  how  are  you转换为<b>how  are  you</b>的呢?

  回答这个问题就是:用正则表达式。

  二、实例分析

  1)在字符串中精确查找链接地址 

((http|https|ftp):(|)((/w)+[.]){1,}(net|com|cn|org|cc|tv|[0-9]{1,3})(((//[/~]*|//[/~]*)

  (/w)+)|[.](/w)+)*(((([?](/w)+){1}[=]*))*((/w)+){1}([/&](/w)+[/=](/w)+)*)*)

  我们知道,链接地址一般以http或者https或者ftp等形式出现。初步总结一下就是,链接地址必须符合如下条件:

  条件1

   以http://或者https://或者ftp://等开头(当然还有其它形式,这里只列出主要的)

  条件2

   http://后面必须跟一个单词字符,紧接着单词字符后面的是"."(这样的组合必须出现一次或多次)。紧跟着"."后面的是域名后缀(如

net或者com或者cn等,如果是以IP地址的形式出现就可以是数字)

  条件3

   出现完整的链接地址后,还可以出现下一级或者更多级的目录(还要注意个人主页的地址有可能出现"~"符号)

  条件4

   链接地址末尾可以带参数。如典型的页数?PageNo=2&action=display等

  现在我们用下面的代码来逐个匹配上面的条件--

  1、((http|https|ftp):(|)  满足条件1

  表示http://  http://  https://  https://  ftp://  ftp://都匹配(在这里考虑了某些用户可能把"//"输成"//"的易发性错误)

  注意:"|"表示"或者","/"是转义字符。""表示"//",""表示"//"

  2、((/w)+[.]){1,}(net|com|cn|org|cc|tv|[0-9]{1,3})  满足条件2

   

"((/w)+[.]){1,}"表示一个单词字符加一个点号可以出现1次或者多次(这里考虑了某些用户喜欢省略www而将http://www.w3c.com写成http://

w3c.com)

  "(net|com|cn|org|cc|tv|[0-9]{1,3})"表示必须要以net或者com或者cn或者org或者cc或者tv或者三位以下的数字结束

  [0-9]{1,3}表示三位以下的数字,因为ip地址的任何段不能超过255

  3、(((//[/~]*|//[/~]*)(/w)+)|[.](/w)+)*  满足条件3

  "(//[/~]*|//[/~]*)"表示可以出现"/~"或者是"/~",(其中"[/~]*"表示  ~  可以出现也可以不出现),因为不是每个链接地址都有下一级

目录

  "(/w)+)|[.](/w)+)"表示必须出现一个单词字符(即目录或者是一个带有扩展名的文件)

  注意:最后还有一个"*"表示上面括号内的可以出现也可以不出现,否则就只能匹配有下一级目录的链接地址了。

  4、(((([?](/w)+){1}[=]*))*((/w)+){1}([/&](/w)+[/=](/w)+)*)*)满足条件4

  "((([?](/w)+){1}[=]*))*((/w)+){1}"表示形如"?PageNo=2"的字符串可以出现也可以不出现,如果出现则只能出现一次(因为不可能有两

个"?"号出现)。

  "([/&](/w)+[/=](/w)+)*)"表示形如"&action=display"的字符串可以出现也可以不出现(因为并不是每个网页都带有两个以上的参数。

  整个"((([?](/w)+){1}[=]*))*((/w)+){1}([/&](/w)+[/=](/w)+)*)*"表示形如"?PageNo=2&action=display"的字符串可以出现也可以不出

现(即链接地址可以有参数也可以没有参数)

  把上面的组合起来,我们就可以匹配一个比较全面的链接地址了。比用简单的"(http:/S+)"来匹配一个链接地址要好,读者可以自行

行测试比较。当然,这段代码还有很多不足之处,希望大家能够继续改进。

  2)替代典型的UBB标签:

  我们的目的就是要把成对的替换成<b></b>下面来看我们实现它的模板

    (/[b/])(.+)(/[//b/])

  这里用了"(.+)"来配匹到之间的整个字符串,在替代的时候我们要写成这样

    str=checkexp(re,str,"<b>$2</b>")

  (注意:checkexp是我自定义的函数,将在后面给出。这个函数将把按照我们提供的模板进行替代。)

  也许你会问这里出现一个"$2"是什么东东,呵注意了这个$2可是很重要的,它代表了"(.+)"所配匹的整个字符串。

  为什么是$2而不是$1、$3呢?因为$1代表(/[b/])所匹配的""字符串,$3代表(/[//b/])所匹配的""字符串,显然这里我们需要的是$2而不

是$1$3。


  三、UBB正则表达模板实例

  下面是我写的一个UBB函数,这个函数基本上能使你的论坛成为一个优秀的UBB代码论坛了。当然,通过改进后,你可以得到一个更强大的U

BB论坛。

Function  ReThestr(face,str)
 dim  re,str

 re="/>"
 str=checkexp(re,str,">")

 re="/<"
 str=checkexp(re,str,"<")

 re="/n/r/n/"
 str=checkexp(re,str,"<P>")

 re=chr(32)
 str=checkexp(re,str,"  ") 

 re="/r"
 str=checkexp(re,str,"  ")

 re="/[img/]((http:(|)){1}((/w)+[.]){1,3}_
(net|com|cn|org|cc|tv)(((//[/~]*|//[/~]*)
(/w)+)|[.](/w)+)*(/w)+[.]{1}(gif|jpg|png))/[//img/]"  ''查找图片地址
 str=checkexp(re,str,"  <img  src=''$1''>  ")

 re="/[w/](http:(|)((/w)+[.]){1,}_
(net|com|cn|org|cc|tv)(((//[/~]*|//[/~]*)(/w)+)|[.](/w)+)*
(((([?](/w)+){1}[=]*))*((/w)+){1}([/&](/w)+[/=](/w)+)*)*)/[//w/]"  ''查找帧地址
 str=checkexp(re,str,"<iframe  width=''300''  height=''300''  src=''$1''></iframe>")

 re="([^(''>)])(<br>)*((http|https|ftp):_
(|)((/w)+[.]){1,}(net|com|cn|org|cc|tv|_
([0-9]{1,3}))(((//[/~]*|//[/~]*)(/w)+)|[.](/w)+)*_
(((([?](/w)+){1}[=]*))*((/w)+){1}([/&](/w)+[/=](/w)+)*)*)"  ''查找链接地址
 str=checkexp(re,str,"$1$2  <a  href=''$3''  target=_blank>$3</a>  ")
 re="([^(http://|http://)])((www|cn)[.](/w)+[.]{1,}_
(net|com|cn|org|cc)(((//[/~]*|//[/~]*)(/w)+)|[.](/w)+)*
(((([?](/w)+){1}[=]*))*((/w)+){1}([/&](/w)+[/=](/w)+)*)*)
"  ''查找不以http://开头的地址
 str=checkexp(re,str,"$1  <a  href=''http://$2''  target=_blank>$2</a>  ")
 re="([^(=)])((/w)+[@]{1}((/w)+[.]){1,3}(/w)+)"  ''查找邮件地址
 str=checkexp(re,str,"  <a  href=''mailto:$2''>$2</a>  ")
 re="/[color=(((/w)+)|][#][0-F]{6})/]((.)+)/[//color/]"  ''替换字体色彩
 str=checkexp(re,str,"<font  color=''$1''>$4</font>")
 re="/[size=(][0-9]{1})/]((.)+)/[//size/]"  ''替换字体大小
 str=checkexp(re,str,"<font  size=''$1''>$2</font>")
 re="/[font=((.)+){1,3}/]((.)+)/[//font/]"  ''替换字体
 str=checkexp(re,str,"<font  face=''$1''>$3</font>")
 re="(/[b/])(.+)(/[//b/])"  ''加粗字体
 str=checkexp(re,str,"<b>$2</b>")
 re="(/[u/])(.+)(/[//u/])"  ''下画线
 str=checkexp(re,str,"<u>$2</u>")
 re="(/[li/])(.+)(/[//li/])"  ''列表
 str=checkexp(re,str,"<li>$2</li>")
 re="(/[QUOTE/])(.+)(/[//QUOTE/])"  ''引用
 str=checkexp(re,str,"_
<BLOCKQUOTE>引用:<HR  SIZE=1>$2<HR  SIZE=1></BLOCKQUOTE>")
 re="/[email=((/w)+][@]{1}((/w)+[.]){1,3}(/w)+)/](.+)(/[//email/])"  ''邮件
 str=checkexp(re,str,"<a  href=mailto:$1>$6</a>")
 re="(/[center/])(.+)(/[//center/])"  ''居中
 str=checkexp(re,str,"<center>$2</center>")

 re="fuck"
 str=checkexp(re,str,"***")

 re="操"
 str=checkexp(re,str,"***")

 re="sex"
 str=checkexp(re,str,"***") 

 re="TMD"
 str=checkexp(re,str,"***")

 re="shit"
 str=checkexp(re,str,"***")

 ReThestr=str
end  function

UBB代码如下:

[ center]  [ / center]  [ email=]  [ / email]    
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值