背景

PHP校验邮箱地址的方法很多, 比较常用的就是自己写正则了, 不过正则多麻烦, 用PHP自带了方法做校验多方便;

filter_var

filter_var是PHP内置的一个变量过滤的方法,提供了很多实用的过滤器,可以用来校验整数、浮点数、邮箱、ULR、MAC地址等等。使用格式如:
mixed filter_var ( mixed $variable [, int $filter = FILTER_DEFAULT [, mixed $options ]] )

参数说明:
$variable 待过滤的变量
$filter 应用的筛选器的标识。具体可以查看https://php.net/manual/zh/filter.filters.sanitize.php
$options

一个选项的关联数组,或者按位区分的标示。如果过滤器接受选项,可以通过数组的 "flags" 位去提供这些标示。 对于回调型的过滤器,应该传入 callable。这个回调函数必须接受一个参数,即待过滤的值,并且 返回一个在过滤/净化后的值。
filter_var如果返回false, 说明变量无法通过过滤器, 也就是不合法了。

$email = "ningpanyun@@yeah.net";
var_dump(filter_var($email,  FILTER_VALIDATE_EMAIL));
$email = "asb";
var_dump(filter_var($email,  FILTER_VALIDATE_EMAIL));

$email = "1@a.com";
var_dump(filter_var($email,  FILTER_VALIDATE_EMAIL));
输出:

string(21) "lastchiliarch@163.com"
bool(false)
string(7) "1@a.com"

对于asb这种非法邮箱格式返回了false, 但对于1@a.com则通过了,还是略有瑕疵啊。不过一般的正则也通过会认为1@a.com是一个合法的邮箱, 那有啥办法可以更精准的验证呢?

checkdnsrr

checkdnsrr其实是用来查询指定的主机的DNS记录的,我们可以借用它来验证邮箱是否存在。对于1@abc.com 肯定是MX记录不存在的。checkdnsrr函数的使用可以查看https://php.net/manual/zh/function.checkdnsrr.php

$email = "ningpanyun@@yeah.net";
$arr = explode("@", $email);
$str = array_pop( $arr );
var_dump( checkdnsrr($str, 'MX') );

$email = "1@a.com";
$arr = explode("@", $email);
$str = array_pop( $arr );
var_dump( checkdnsrr($str, 'MX') );

输出:
bool(true)
bool(false)
filter_var+checkdnsrr

我们可以接合filter_var 和checkdnsrr做校验, 对于绝大多数的非法邮箱肯定会在filter_var的时候就挂掉了, 剩下的再用checkdnsrr进一步判断。

$email = "ningpanyun@@yeah.net";
if (filter_var($email) === false) 
{
    echo "invalid email: $email \n";
    continue;
}
if(checkdnsrr(array_pop(explode("@",$email)), "MX") === false) 
{
    echo "invalid email: $email \n";
    continue;
}
输出: invalid email: 1@a.com

但要注意的是, 由于只是检查MX记录, 所以只能判断163.com是存在的, 但不能说明ningpanyun这个用户是存在的。想要更精确的判断邮箱存在, 那只能连接到smtp服务器去验证了。