百度下邮箱格式的正则表示,能够搜索到各式各样,五花八门的表示。如果没有仔细甄别,错误使用其中的一些代码,则很可能造成在遇到一些特殊的邮箱格式时无法识别。这里就分析下邮件相关的RFC标准,可详见RFC 5322, Internet Message Format或[2-RFC5322], 但在此之前需要先学习下[1-RFC5234]中关于ABNF的核心规则。
[1-RFC5234] 中Appendix B. Core ABNF of ABNF
B.1. Core Rules
ALPHA = %x41-5A / %x61-7A ; A-Z / a-z BIT = "0" / "1" CHAR = %x01-7F ; any 7-bit US-ASCII character, excluding NUL CR = %x0D ; carriage return CRLF = CR LF ; Internet standard newline CTL = %x00-1F / %x7F ; controls DIGIT = %x30-39 ; 0-9 DQUOTE = %x22 ; " (Double Quote) HEXDIG = DIGIT / "A" / "B" / "C" / "D" / "E" / "F" HTAB = %x09 ; horizontal tab LF = %x0A ; linefeed LWSP = *(WSP / CRLF WSP) ; Use of this linear-white-space rule ; permits lines containing only white ; space that are no longer legal in ; mail headers and have caused ; interoperability problems in other ; contexts. ; Do not use when defining mail ; headers and use with caution in ; other contexts. OCTET = %x00-FF ; 8 bits of data SP = %x20 ; space VCHAR = %x21-7E ; visible (printing) characters WSP = SP / HTAB ; white space
以下内容见[2-RFC5322], 章节编号与原文保持一致。
2 Message词法分析(Lexical Analysis of Messages)
取值范围在1-127的字符被称为US-ASCII字符.
Messages可分为许多行字符,一行是用两个相连的字符carriage-return(CR: ASCII码值13)与line-feed(LF: ASCII码值10)分隔, 通常写作CRLF.
Message是由header字段, 可选的body字段组成。
header section是由一些特定语法的许多行组成, header section后面是一个空行,接下来是body字段。
本规范中使用header field描述单个字段,用header section描述所有header字段。
每一行字符数务必(MUST)不能超过998个,不包括CRLF以外通常不应当超过78个字符。
998加上CRLF达到1000个字符,有许多接收方实现限定一行不能超过1000个字符。
78加上CRLF达到80个字符,有许多显示界面一行超过80个字符后被截断或显示到下一行。
Header Fields先是一个field name, 接下来是一个冒号, 接下来是一个field body, 后面是CLRF.
field name必须是可打印US-ASCII字符组成,即值为[33,126]之间除冒号以外的字符。
field body必须是可打印US-ASCII字符, 空格(SP:ASCII码值32), horizontal tab(HTAB:
ASCII码值9)字符组成。SP与HTAB组合即为WSP(white space characters).
field body务必不能(MUST NOT)包含CRLF, 除了用于"folding"与"unfolding"以外。
一些field body被称为"unstructured", 表示无需进一步处理的单行字符。
一些field body被称为"structured", 由一些特殊token组成, 这些token后面可以有comments与空格字符.
每个header field通常是由一行字符,由field name, 冒号和field body组成。
考虑到需要处理每行有998/78字符, header field中的field body部分可用多行来表示,这被称为"folding".
例如下面header field:
Subject: This is a test
可以被表示为
Subject: This
is a test
从folded多行变为单行的过程被称为"unfolding", 即去掉CRLF并立即跟一个WSP.
每个header field应当用其unfolded格式来做进一步的语法与此法分析。
unfolded header field没有长度限制。
3. 语法(Syntax)
3.2 词法(Lexical)tokens
3.2.1. Quoted characters
quoted-pair = ("\" (VCHAR / WSP)) / obs-qp
3.2.2. Folding White Space and Comments
header字段bodies中, 许多elements之间可以存在空白字符(White space characters), 这里的字符包含用于folding的空格;
圆括号(parentheses)中的字符串被视为comments。
下面定义了folding white space (FWS)和comment结构。
圆括号(parentheses)中的字符串只要不是在引号串内就被视为comments, Comments可以嵌套(nest).
有一些地方可随意插入comments与FWS, 为此新定义一个"CFWS" token.
但是,在一个folded header字段的任意一行,不能都是由WSP字符组成。
FWS = ([*WSP CRLF] 1*WSP) / obs-FWS ; Folding white space ctext = %d33-39 / ; 可打印US-ASCII字符, %d42-91 / ; 取值在[33-126]范围内但不包含 %d93-126 / ; 40="(", 41=")", or 92="\" obs-ctext %d33-39: 33: "!" 34: """ 35: "#" 36: "$" 37: "%" 38: "&" 39: "'" 42 "*" 43 "+" 44 "、" 45 "-" 46 "." 47 "/" 48-57: "0"-"9" 58: ":" 59: ";" 60: "<" 61: "=" 62: ">" 63: "?" 64: "@" 65-90: "A-Z" 91: "[" 93: "]" 94: "^" 95: "-" 96: "'" 97-122: "a-z" 123: "{" 124: "|" 125: "}" 126: "~" ccontent = ctext / quoted-pair / comment comment = "(" *([FWS] ccontent) [FWS] ")" CFWS = (1*([FWS] comment) [FWS]) / FWS
3.2.3. Atom
在structured header field bodies中有一些productions是由一些基本字符组成的简单串,
这些productions被称为atoms.
但有一些productions允许句点字符(".", ASCII值为46), 因此引入"dot-atom" token来表示这种情况。
atext = ALPHA / DIGIT / ; 不包含specials的可打印US-ASCII字符 "!" / "#" / ; 用于atoms. "$" / "%" / "&" / "'" / "*" / "+" / "-" / "/" / "=" / "?" / "^" / "_" / "`" / "{" / "|" / "}" / "~" atom = [CFWS] 1*atext [CFWS] dot-atom-text = 1*atext *("." 1*atext) dot-atom = [CFWS] dot-atom-text [CFWS] specials = "(" / ")" / ; atext中未出现的特殊字符 "<" / ">" / "[" / "]" / ":" / ";" / "@" / "\" / "," / "." / DQUOTE
3.2.4. Quoted Strings
用双引号(DQUOTE, ASCII值34)包含的串.
qtext = %d33 / ; 不包含双引号"""与反斜杠"\"的可打印字符。 %d35-91 / %d93-126 / obs-qtext qcontent = qtext / quoted-pair quoted-string = [CFWS] DQUOTE *([FWS] qcontent) [FWS] DQUOTE [CFWS]
用双引号包含的串被称为一个unit, 即双引号串在语法上与atom一致。
因为双引号串可包含FWS, 即允许folding.
而且注意双引号串中可以有双引号串, 引起可以有双引号字符与反斜杠(backslash)字符。
从语法上讲, 双引号串中的"\"与FWS/CFWS中的CRLF都是不可见的,因此它们不是双引号串的一部分。
3.2.5. 其他Tokens
定义三个token: word与phrase用于atoms和/或双引号串的组合, unstructured 用于
unstructured header字段, 以及structured header字段中的一些地方.
word = atom / quoted-string phrase = 1*word / obs-phrase unstructured = (*([FWS] VCHAR) *WSP) / obs-unstruct
3.3. Date与Time规范
date-time = [ day-of-week "," ] date time [CFWS] day-of-week = ([FWS] day-name) / obs-day-of-week day-name = "Mon" / "Tue" / "Wed" / "Thu" / "Fri" / "Sat" / "Sun" date = day month year ; 应当表示本地时间 day = ([FWS] 1*2DIGIT FWS) / obs-day ; 一个月的第几天 month = "Jan" / "Feb" / "Mar" / "Apr" / "May" / "Jun" / "Jul" / "Aug" / "Sep" / "Oct" / "Nov" / "Dec" year = (FWS 4*DIGIT FWS) / obs-year ; 四位数字 time = time-of-day zone time-of-day = hour ":" minute [ ":" second ] ; 应当表示本地时间, 一天中的时:分[:秒], ; 范围00:00:00 - 23:59:60 hour = 2DIGIT / obs-hour minute = 2DIGIT / obs-minute second = 2DIGIT / obs-second zone = (FWS ( "+" / "-" ) 4DIGIT) / obs-zone ; date与time-of-day偏离UTC或GMT的偏差\ ; +表示ahead of(即east of)UTC, -表示behind(即west of) UTC ; 前两个数字表示hours偏差, 后两个数字表示minutes偏差 ; +hhmm表示 +(hh * 60 + mm) 分钟, -hhmm表示 -(hh * 60 + mm) 分钟 ; "+0000"用于表示UTC的时区 ; "-0000"用于表示本地时区生成的时间, date-time不包含本地时区的信息
3.4. Address规范
Addresses表示messages的接收与发送方.
一个address可以是单个邮箱,也可以是一组邮箱。
address = mailbox / group mailbox = name-addr / addr-spec name-addr = [display-name] angle-addr angle-addr = [CFWS] "<" addr-spec ">" [CFWS] / obs-angle-addr group = display-name ":" [group-list] ";" [CFWS] display-name = phrase mailbox-list = (mailbox *("," mailbox)) / obs-mbox-list address-list = (address *("," address)) / obs-addr-list group-list = mailbox-list / CFWS / obs-group-list
一个mailbox通常由两部分组成:(1) 一个可选的display-name, (2) 用<>封装的addr-spec地址。
mailbox的一岁简化形式是只有addr-spec地址,没有接收方名称, 也没有<>。
3.4.1. Addr-Spec规范
addr-spec = local-part "@" domain local-part = dot-atom / quoted-string / obs-local-part domain = dot-atom / domain-literal / obs-domain domain-literal = [CFWS] "[" *([FWS] dtext) [FWS] "]" [CFWS] dtext = %d33-90 / ; 可打印US-ASCII字符, %d94-126 / ; 不包含 "[", "]", or "\" obs-dtext
3.5. Overall Message Syntax
message由header fields, 接下来是一个可选的message body.
message中的一行最大998个字符,推荐最大为78个字符, 这里都不包含CRLF.
在message body中, 虽然在text rule中列出的所有字符都可以用, 但不鼓励使用US-ASCII控制字符(值1到8, 11, 12, 14-31),
因为无法保证接收方如何来显示它们。
message = (fields / obs-fields) [CRLF body] body = (*(*998text CRLF) *998text) / obs-body text = %d1-9 / ; Characters excluding CR %d11 / ; and LF %d12 / %d14-127
其他略。。。
邮箱地址格式主要参见上面的3.4.1. Addr-Spec规范, 建议参照这里的规范来编写正则表达式,如果自己没有能力编写正则表达式则建议直接采用PHP、JAVA、C#、C++、C等语言中现成的库来判断。
参考资料:
[1-RFC5234] RFC 5234, Augmented BNF for Syntax Specifications: ABNF, Standards Track, January 2008, http://www.rfc-editor.org/rfc/rfc5234.txt
[2-RFC5322] RFC 5322, Internet Message Format, Standards Track, October 2008, http://www.rfc-editor.org/rfc/rfc5322.txt