찾다
php教程php手册PHP SMTP邮件发送类,支持SSL连接
PHP SMTP邮件发送类,支持SSL连接Jun 06, 2016 pm 07:32 PM
phpsmtpssl보내다주소지원하다연결하다우편

上一篇地址:http://www.oschina.net/code/snippet_1182150_25620 我又改了一下子! 测试过了gmail和QQ邮箱的SMTP。 PHP 源码与演示: 源码出处 ?php/*** 邮件发送类* 支持发送纯文本邮件和HTML格式的邮件,可以多收件人,多抄送,多秘密抄送,带附件(单个或

上一篇地址:http://www.oschina.net/code/snippet_1182150_25620
我又改了一下子!
测试过了gmail和QQ邮箱的SMTP。 PHP

源码与演示:源码出处

<?php
/**
* 邮件发送类
* 支持发送纯文本邮件和HTML格式的邮件,可以多收件人,多抄送,多秘密抄送,带附件(单个或多个附件),支持到服务器的ssl连接
* 需要的php扩展:sockets、Fileinfo和openssl。
* 编码格式是UTF-8,传输编码格式是base64
* @example
* $mail = new MySendMail();
* $mail->setServer("smtp@126.com", "XXXXX@126.com", "XXXXX"); //设置smtp服务器,普通连接方式
* $mail->setServer("smtp.gmail.com", "XXXXX@gmail.com", "XXXXX", 465, true); //设置smtp服务器,到服务器的SSL连接
* $mail->setFrom("XXXXX"); //设置发件人
* $mail->setReceiver("XXXXX"); //设置收件人,多个收件人,调用多次
* $mail->setCc("XXXX"); //设置抄送,多个抄送,调用多次
* $mail->setBcc("XXXXX"); //设置秘密抄送,多个秘密抄送,调用多次
* $mail->addAttachment("XXXX"); //添加附件,多个附件,调用多次
* $mail->setMail("test", "<b>test</b>"); //设置邮件主题、内容
* $mail->sendMail(); //发送
*/
class MySendMail {
	/**
	* @var string 邮件传输代理用户名
	* @access protected
	*/
	protected $_userName;

	/**
	* @var string 邮件传输代理密码
	* @access protected
	*/
	protected $_password;

	/**
	* @var string 邮件传输代理服务器地址
	* @access protected
	*/
	protected $_sendServer;

	/**
	* @var int 邮件传输代理服务器端口
	* @access protected
	*/
	protected $_port;

	/**
	* @var string 发件人
	* @access protected
	*/
	protected $_from;

	/**
	* @var array 收件人
	* @access protected
	*/
	protected $_to = array();

	/**
	* @var array 抄送
	* @access protected
	*/
	protected $_cc = array();

	/**
	* @var array 秘密抄送
	* @access protected
	*/
	protected $_bcc = array();

	/**
	* @var string 主题
	* @access protected
	*/
	protected $_subject;

	/**
	* @var string 邮件正文
	* @access protected
	*/
	protected $_body;

	/**
	* @var array 附件
	* @access protected
	*/
	protected $_attachment = array();

	/**
	* @var reource socket资源
	* @access protected
	*/
	protected $_socket;

	/**
	* @var reource 是否是安全连接
	* @access protected
	*/
	protected $_isSecurity;

	/**
	* @var string 错误信息
	* @access protected
	*/
	protected $_errorMessage;


	/**
	* 设置邮件传输代理,如果是可以匿名发送有邮件的服务器,只需传递代理服务器地址就行
	* @access public
	* @param string $server 代理服务器的ip或者域名
	* @param string $username 认证账号
	* @param string $password 认证密码
	* @param int $port 代理服务器的端口,smtp默认25号端口
	* @param boolean $isSecurity 到服务器的连接是否为安全连接,默认false
	* @return boolean
	*/
	public function setServer($server, $username="", $password="", $port=25, $isSecurity=false) {
		$this->_sendServer = $server;
		$this->_port = $port;
		$this->_isSecurity = $isSecurity;
		$this->_userName = empty($username) ? "" : base64_encode($username);
		$this->_password = empty($password) ? "" : base64_encode($password);
		return true;
	}

	/**
	* 设置发件人
	* @access public
	* @param string $from 发件人地址
	* @return boolean
	*/
	public function setFrom($from) {
		$this->_from = $from;
		return true;
	}

	/**
	* 设置收件人,多个收件人,调用多次.
	* @access public
	* @param string $to 收件人地址
	* @return boolean
	*/
	public function setReceiver($to) {
		$this->_to[] = $to;
		return true;
	}

	/**
	* 设置抄送,多个抄送,调用多次.
	* @access public
	* @param string $cc 抄送地址
	* @return boolean
	*/
	public function setCc($cc) {
		$this->_cc[] = $cc;
		return true;
	}

	/**
	* 设置秘密抄送,多个秘密抄送,调用多次
	* @access public
	* @param string $bcc 秘密抄送地址
	* @return boolean
	*/
	public function setBcc($bcc) {
		$this->_bcc[] = $bcc;
		return true;
	}

	/**
	* 设置邮件附件,多个附件,调用多次
	* @access public
	* @param string $file 文件地址
	* @return boolean
	*/
	public function addAttachment($file) {
		if(!file_exists($file)) {
			$this->_errorMessage = "file " . $file . " does not exist.";
			return false;
		}
		$this->_attachment[] = $file;
		return true;
	}

	/**
	* 设置邮件信息
	* @access public
	* @param string $body 邮件主题
	* @param string $subject 邮件主体内容,可以是纯文本,也可是是HTML文本
	* @return boolean
	*/
	public function setMail($subject, $body) {
		$this->_subject = base64_encode($subject);
		$this->_body = base64_encode($body);
		return true;
	}

	/**
	* 发送邮件
	* @access public
	* @return boolean
	*/
	public function sendMail() {
		$command = $this->getCommand();

		$this->_isSecurity ? $this->socketSecurity() : $this->socket();
		
		foreach ($command as $value) {
			$result = $this->_isSecurity ? $this->sendCommandSecurity($value[0], $value[1]) : $this->sendCommand($value[0], $value[1]);
			if($result) {
				continue;
			}
			else{
				return false;
			}
		}
		
		//其实这里也没必要关闭,smtp命令:QUIT发出之后,服务器就关闭了连接,本地的socket资源会自动释放
		$this->_isSecurity ? $this->closeSecutity() : $this->close();
		return true;
	}

	/**
	* 返回错误信息
	* @return string
	*/
	public function error(){
		if(!isset($this->_errorMessage)) {
			$this->_errorMessage = "";
		}
		return $this->_errorMessage;
	}

	/**
	* 返回mail命令
	* @access protected
	* @return array
	*/
	protected function getCommand() {
		$separator = "----=_Part_" . md5($this->_from . time()) . uniqid(); //分隔符

		$command = array(
				array("HELO sendmail\r\n", 250)
			);
		if(!empty($this->_userName)){
			$command[] = array("AUTH LOGIN\r\n", 334);
			$command[] = array($this->_userName . "\r\n", 334);
			$command[] = array($this->_password . "\r\n", 235);
		}

		//设置发件人
		$command[] = array("MAIL FROM: <" . $this->_from . ">\r\n", 250);
		$header = "FROM: <" . $this->_from . ">\r\n";

		//设置收件人
		if(!empty($this->_to)) {
			$count = count($this->_to);
			if($count == 1){
				$command[] = array("RCPT TO: <" . $this->_to[0] . ">\r\n", 250);
				$header .= "TO: <" . $this->_to[0] .">\r\n";
			}
			else{
				for($i=0; $i<$count; $i++){
					$command[] = array("RCPT TO: <" . $this->_to[$i] . ">\r\n", 250);
					if($i == 0){
						$header .= "TO: <" . $this->_to[$i] .">";
					}
					elseif($i + 1 == $count){
						$header .= ",<" . $this->_to[$i] .">\r\n";
					}
					else{
						$header .= ",<" . $this->_to[$i] .">";
					}
				}
			}
		}

		//设置抄送
		if(!empty($this->_cc)) {
			$count = count($this->_cc);
			if($count == 1){
				$command[] = array("RCPT TO: <" . $this->_cc[0] . ">\r\n", 250);
				$header .= "CC: <" . $this->_cc[0] .">\r\n";
			}
			else{
				for($i=0; $i<$count; $i++){
					$command[] = array("RCPT TO: <" . $this->_cc[$i] . ">\r\n", 250);
					if($i == 0){
					$header .= "CC: <" . $this->_cc[$i] .">";
					}
					elseif($i + 1 == $count){
						$header .= ",<" . $this->_cc[$i] .">\r\n";
					}
					else{
						$header .= ",<" . $this->_cc[$i] .">";
					}
				}
			}
		}

		//设置秘密抄送
		if(!empty($this->_bcc)) {
			$count = count($this->_bcc);
			if($count == 1) {
				$command[] = array("RCPT TO: <" . $this->_bcc[0] . ">\r\n", 250);
				$header .= "BCC: <" . $this->_bcc[0] .">\r\n";
			}
			else{
				for($i=0; $i<$count; $i++){
					$command[] = array("RCPT TO: <" . $this->_bcc[$i] . ">\r\n", 250);
					if($i == 0){
					$header .= "BCC: <" . $this->_bcc[$i] .">";
					}
					elseif($i + 1 == $count){
						$header .= ",<" . $this->_bcc[$i] .">\r\n";
					}
					else{
						$header .= ",<" . $this->_bcc[$i] .">";
					}
				}
			}
		}

		//主题
		$header .= "Subject: =?UTF-8?B?" . $this->_subject ."?=\r\n";
		if(isset($this->_attachment)) {
			//含有附件的邮件头需要声明成这个
			$header .= "Content-Type: multipart/mixed;\r\n";
		}
		elseif(false){
			//邮件体含有图片资源的,且包含的图片在邮件内部时声明成这个,如果是引用的远程图片,就不需要了
			$header .= "Content-Type: multipart/related;\r\n";
		}
		else{
			//html或者纯文本的邮件声明成这个
			$header .= "Content-Type: multipart/alternative;\r\n";
		}

		//邮件头分隔符
		$header .= "\t" . 'boundary="' . $separator . '"';

		$header .= "\r\nMIME-Version: 1.0\r\n";

		//这里开始是邮件的body部分,body部分分成几段发送
		$header .= "\r\n--" . $separator . "\r\n";
		$header .= "Content-Type:text/html; charset=utf-8\r\n";
		$header .= "Content-Transfer-Encoding: base64\r\n\r\n";
		$header .= $this->_body . "\r\n";
		$header .= "--" . $separator . "\r\n";

		//加入附件
		if(!empty($this->_attachment)){
			$count = count($this->_attachment);
			for($i=0; $i<$count; $i++){
				$header .= "\r\n--" . $separator . "\r\n";
				$header .= "Content-Type: " . $this->getMIMEType($this->_attachment[$i]) . '; name="=?UTF-8?B?' . base64_encode( basename($this->_attachment[$i]) ) . '?="' . "\r\n";
				$header .= "Content-Transfer-Encoding: base64\r\n";
				$header .= 'Content-Disposition: attachment; filename="=?UTF-8?B?' . base64_encode( basename($this->_attachment[$i]) ) . '?="' . "\r\n";
				$header .= "\r\n";
				$header .= $this->readFile($this->_attachment[$i]);
				$header .= "\r\n--" . $separator . "\r\n";
			}
		}

		//结束邮件数据发送
		$header .= "\r\n.\r\n";


		$command[] = array("DATA\r\n", 354);
		$command[] = array($header, 250);
		$command[] = array("QUIT\r\n", 221);
		
		return $command;
	}

	/**
	* 发送命令
	* @access protected
	* @param string $command 发送到服务器的smtp命令
	* @param int $code 期望服务器返回的响应吗
	* @return boolean
	*/
	protected function sendCommand($command, $code) {
		echo 'Send command:' . $command . ',expected code:' . $code . '<br />';
		//发送命令给服务器
		try{
			if(socket_write($this->_socket, $command, strlen($command))){

				//当邮件内容分多次发送时,没有$code,服务器没有返回
				if(empty($code))  {
					return true;
				}

				//读取服务器返回
				$data = trim(socket_read($this->_socket, 1024));
				echo 'response:' . $data . '<br /><br />';

				if($data) {
					$pattern = "/^".$code."+?/";
					if(preg_match($pattern, $data)) {
						return true;
					}
					else{
						$this->_errorMessage = "Error:" . $data . "|**| command:";
						return false;
					}
				}
				else{
					$this->_errorMessage = "Error:" . socket_strerror(socket_last_error());
					return false;
				}
			}
			else{
				$this->_errorMessage = "Error:" . socket_strerror(socket_last_error());
				return false;
			}
		}catch(Exception $e) {
			$this->_errorMessage = "Error:" . $e->getMessage();
		}
	}

	/**
	* 安全连接发送命令
	* @access protected
	* @param string $command 发送到服务器的smtp命令
	* @param int $code 期望服务器返回的响应吗
	* @return boolean
	*/
	protected function sendCommandSecurity($command, $code) {
		echo 'Send command:' . $command . ',expected code:' . $code . '<br />';
		try {
			if(fwrite($this->_socket, $command)){
				//当邮件内容分多次发送时,没有$code,服务器没有返回
				if(empty($code))  {
					return true;
				}
				//读取服务器返回
				$data = trim(fread($this->_socket, 1024));
				echo 'response:' . $data . '<br /><br />';

				if($data) {
					$pattern = "/^".$code."+?/";
					if(preg_match($pattern, $data)) {
						return true;
					}
					else{
						$this->_errorMessage = "Error:" . $data . "|**| command:";
						return false;
					}
				}
				else{
					return false;
				}
			}
			else{
				$this->_errorMessage = "Error: " . $command . " send failed";
				return false;
			}
		}catch(Exception $e) {
			$this->_errorMessage = "Error:" . $e->getMessage();
		}
	} 

	/**
	* 读取附件文件内容,返回base64编码后的文件内容
	* @access protected
	* @param string $file 文件
	* @return mixed
	*/
	protected function readFile($file) {
		if(file_exists($file)) {
			$file_obj = file_get_contents($file);
			return base64_encode($file_obj);
		}
		else {
			$this->_errorMessage = "file " . $file . " dose not exist";
			return false;
		}
	}

	/**
	* 获取附件MIME类型
	* @access protected
	* @param string $file 文件
	* @return mixed
	*/
	protected function getMIMEType($file) {
		if(file_exists($file)) {
			$mime = mime_content_type($file);
			/*if(! preg_match("/gif|jpg|png|jpeg/", $mime)){
				$mime = "application/octet-stream";
			}*/
			return $mime;
		}
		else {
			return false;
		}
	}

	/**
	* 建立到服务器的网络连接
	* @access protected
	* @return boolean
	*/
	protected function socket() {
		//创建socket资源
		$this->_socket = socket_create(AF_INET, SOCK_STREAM, getprotobyname('tcp'));
		
		if(!$this->_socket) {
			$this->_errorMessage = socket_strerror(socket_last_error());
			return false;
		}

		socket_set_block($this->_socket);//设置阻塞模式

		//连接服务器
		if(!socket_connect($this->_socket, $this->_sendServer, $this->_port)) {
			$this->_errorMessage = socket_strerror(socket_last_error());
			return false;
		}
		$str = socket_read($this->_socket, 1024);
		if(!preg_match("/220+?/", $str)){
			$this->_errorMessage = $str;
			return false;
		}
		
		return true;
	}

	/**
	* 建立到服务器的SSL网络连接
	* @access protected
	* @return boolean
	*/
	protected function socketSecurity() {
		$remoteAddr = "tcp://" . $this->_sendServer . ":" . $this->_port;
		$this->_socket = stream_socket_client($remoteAddr, $errno, $errstr, 30);
		if(!$this->_socket){
			$this->_errorMessage = $errstr;
			return false;
		}

		//设置加密连接,默认是ssl,如果需要tls连接,可以查看php手册stream_socket_enable_crypto函数的解释
		stream_socket_enable_crypto($this->_socket, true, STREAM_CRYPTO_METHOD_SSLv23_CLIENT);

		stream_set_blocking($this->_socket, 1); //设置阻塞模式
		$str = fread($this->_socket, 1024);
		if(!preg_match("/220+?/", $str)){
			$this->_errorMessage = $str;
			return false;
		}

		return true;
	}

	/**
	* 关闭socket
	* @access protected
	* @return boolean
	*/
	protected function close() {
		if(isset($this->_socket) && is_object($this->_socket)) {
			$this->_socket->close();
			return true;
		}
		$this->_errorMessage = "No resource can to be close";
		return false;
	}

	/**
	* 关闭安全socket
	* @access protected
	* @return boolean
	*/
	protected function closeSecutity() {
		if(isset($this->_socket) && is_object($this->_socket)) {
			stream_socket_shutdown($this->_socket, STREAM_SHUT_WR);
			return true;
		}
		$this->_errorMessage = "No resource can to be close";
		return false;
	}
}
성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
php怎么把负数转为正整数php怎么把负数转为正整数Apr 19, 2022 pm 08:59 PM

php把负数转为正整数的方法:1、使用abs()函数将负数转为正数,使用intval()函数对正数取整,转为正整数,语法“intval(abs($number))”;2、利用“~”位运算符将负数取反加一,语法“~$number + 1”。

php怎么实现几秒后执行一个函数php怎么实现几秒后执行一个函数Apr 24, 2022 pm 01:12 PM

实现方法:1、使用“sleep(延迟秒数)”语句,可延迟执行函数若干秒;2、使用“time_nanosleep(延迟秒数,延迟纳秒数)”语句,可延迟执行函数若干秒和纳秒;3、使用“time_sleep_until(time()+7)”语句。

php怎么除以100保留两位小数php怎么除以100保留两位小数Apr 22, 2022 pm 06:23 PM

php除以100保留两位小数的方法:1、利用“/”运算符进行除法运算,语法“数值 / 100”;2、使用“number_format(除法结果, 2)”或“sprintf("%.2f",除法结果)”语句进行四舍五入的处理值,并保留两位小数。

php怎么根据年月日判断是一年的第几天php怎么根据年月日判断是一年的第几天Apr 22, 2022 pm 05:02 PM

判断方法:1、使用“strtotime("年-月-日")”语句将给定的年月日转换为时间戳格式;2、用“date("z",时间戳)+1”语句计算指定时间戳是一年的第几天。date()返回的天数是从0开始计算的,因此真实天数需要在此基础上加1。

php字符串有没有下标php字符串有没有下标Apr 24, 2022 am 11:49 AM

php字符串有下标。在PHP中,下标不仅可以应用于数组和对象,还可应用于字符串,利用字符串的下标和中括号“[]”可以访问指定索引位置的字符,并对该字符进行读写,语法“字符串名[下标值]”;字符串的下标值(索引值)只能是整数类型,起始值为0。

php怎么替换nbsp空格符php怎么替换nbsp空格符Apr 24, 2022 pm 02:55 PM

方法:1、用“str_replace("&nbsp;","其他字符",$str)”语句,可将nbsp符替换为其他字符;2、用“preg_replace("/(\s|\&nbsp\;||\xc2\xa0)/","其他字符",$str)”语句。

php怎么判断有没有小数点php怎么判断有没有小数点Apr 20, 2022 pm 08:12 PM

php判断有没有小数点的方法:1、使用“strpos(数字字符串,'.')”语法,如果返回小数点在字符串中第一次出现的位置,则有小数点;2、使用“strrpos(数字字符串,'.')”语句,如果返回小数点在字符串中最后一次出现的位置,则有。

php怎么读取字符串后几个字符php怎么读取字符串后几个字符Apr 22, 2022 pm 08:31 PM

在php中,可以使用substr()函数来读取字符串后几个字符,只需要将该函数的第二个参数设置为负值,第三个参数省略即可;语法为“substr(字符串,-n)”,表示读取从字符串结尾处向前数第n个字符开始,直到字符串结尾的全部字符。

See all articles

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

AI Hentai Generator

AI Hentai Generator

AI Hentai를 무료로 생성하십시오.

뜨거운 도구

Dreamweaver Mac版

Dreamweaver Mac版

시각적 웹 개발 도구

SublimeText3 중국어 버전

SublimeText3 중국어 버전

중국어 버전, 사용하기 매우 쉽습니다.

SublimeText3 Mac 버전

SublimeText3 Mac 버전

신 수준의 코드 편집 소프트웨어(SublimeText3)

SublimeText3 영어 버전

SublimeText3 영어 버전

권장 사항: Win 버전, 코드 프롬프트 지원!

DVWA

DVWA

DVWA(Damn Vulnerable Web App)는 매우 취약한 PHP/MySQL 웹 애플리케이션입니다. 주요 목표는 보안 전문가가 법적 환경에서 자신의 기술과 도구를 테스트하고, 웹 개발자가 웹 응용 프로그램 보안 프로세스를 더 잘 이해할 수 있도록 돕고, 교사/학생이 교실 환경 웹 응용 프로그램에서 가르치고 배울 수 있도록 돕는 것입니다. 보안. DVWA의 목표는 다양한 난이도의 간단하고 간단한 인터페이스를 통해 가장 일반적인 웹 취약점 중 일부를 연습하는 것입니다. 이 소프트웨어는