网络编程——Java Email

本文介绍了电子邮件的基本协议POP3、SMTP和IMAP,详细讲解了Java使用SMTP发送邮件的过程,包括配置SMTP信息、创建Session、发送邮件以及发送HTML邮件和附件的方法。此外,还探讨了如何在邮件中内嵌图片。
摘要由CSDN通过智能技术生成

Email就是电子邮件。当然,使用Java程序也可以收发电子邮件。

一、邮件协议

常见的邮件协议有:POP3、SMTP、IMAP。

1.POP3

        POP3是Post Office Protocol 3的简称,即邮局协议的第3个版本,它规定怎样将个人计算机连接到Internet的邮件服务器和下载电子邮件的电子协议。它是因特网电子邮件的第一个离线协议标准,POP3允许用户从服务器上把邮件存储到本地主机(即自己的计算机)上,同时删除保存在邮件服务器上的邮件,而POP3服务器则是遵循POP3协议的接收邮件服务器,用来接收电子邮件的。

2.SMTP

        SMTP 的全称是“Simple Mail Transfer Protocol”,即简单邮件传输协议。它是一组用于从源地址到目的地址传输邮件的规范,通过它来控制邮件的中转方式。SMTP 协议属于 TCP/IP 协议簇,它帮助每台计算机在发送或中转信件时找到下一个目的地。SMTP 服务器就是遵循 SMTP 协议的发送邮件服务器。SMTP 认证,简单地说就是要求必须在提供了账户名和密码之后才可以登录 SMTP 服务器,这就使得那些垃圾邮件的散播者无可乘之机。增加 SMTP 认证的目的是为了使用户避免受到垃圾邮件的侵扰。

3.IMAP

       IMAP全称是Internet Mail Access Protocol,即交互式邮件存取协议,它是跟POP3类似邮件访问标准协议之一。不同的是,开启了IMAP后,您在电子邮件客户端收取的邮件仍然保留 在服务器上,同时在客户端上的操作都会反馈到服务器上,如:删除邮件,标记已读等,服务器上的邮件也会做相应的动作。所以无论从浏览器登录邮箱或者客户端 软件登录邮箱,看到的邮件以及状态都是一致的。

IMAP和POP3有什么区别?

       POP3协议允许电子邮件客户端下载服务器上的邮件,但是在客户端的操作(如移动邮件、标记已读等),不会反馈到服务器上,比如通过客户端收取了邮箱中的3封邮件并移动到其他文件夹,邮箱服务器上的这些邮件是没有同时被移动的 。IMAP提供webmail 与电子邮件客户端之间的双向通信,客户端的操作都会反馈到服务器上,对邮件进行的操作,服务器上的邮件也会做相应的动作。

二、准备SMTP登陆信息

       连接SMTP服务器时,需要准备一个Properties对象,填入相关信息。最后获取Session实例时,如果服务器需要认证,还需要传入一个Authenticator对象,并返回指定的用户名和口令。当我们获取到Session实例后,打开调试模式可以看到SMTP通信的详细内容,便于调试。

import java.util.Properties;
import javax.mail.Authenticator;
import javax.mail.PasswordAuthentication;
import javax.mail.Session;

public class Demo01 {
	public static void main(String[] args) {
		// SMTP服务器地址
		String smtp = "smtp.163.com";
		
		// 邮箱账号和密码(授权密码)
		String userName = "yummy010419@163.com";
		String password = "***FMZMCZDECYN";
		
		// SMTP服务器的连接信息
		Properties props = new Properties();
		props.put("mail.smtp.host", smtp);
		props.put("mail.smit.port", "25");
		props.put("mail.smtp.auth", "true");
		props.put("mail.smtp.starttls.enable", "true");
		
		// 创建Session
		// 参数1:SMTP服务器的连接信息
		// 参数2:用户认证对象(Authenticator接口的匿名实现类)
		Session session = Session.getInstance(props, new Authenticator() {
			@Override
			protected PasswordAuthentication getPasswordAuthentication() {
				// TODO Auto-generated method stub
				return super.getPasswordAuthentication();
			}
			
		});
		session.setDebug(true);
		System.out.println(session);
	}
}

 三、发送邮件

发送邮件时,我们需要构造一个Message对象,然后调用Transport.send(Message)即可完成发送:绝大多数邮件服务器要求发送方地址和登录用户名必须一致,否则发送将失败。

import javax.mail.MessagingException;
import javax.mail.Session;
import javax.mail.Transport;
import javax.mail.internet.InternetAddress;
import javax.mail.internet.MimeMessage;
import javax.mail.internet.MimeMessage.RecipientType;

public class Demo02 {
	public static void main(String[] args) {
		
		// 1.创建Session会话
		
		try {
			Session session = JavaMaikUtils.creSession();
			
			// 2.创建邮件对象(Message抽象类的子类对象)
			MimeMessage msg = new MimeMessage(session);// 传入session
			msg.setFrom(new InternetAddress("yummy010419@163.com"));// 发件人
			msg.setRecipient(RecipientType.TO, new InternetAddress("3347995330@qq.com"));
			msg.setSubject("这是一份测试邮件", "utf-8");//标题
			
			// 邮件正文中包含有"html"标签(控制文本的格式)
			msg.setText("今天<b>开心</b>吗", "utf-8");
			
			// 3.发送邮件
			Transport.send(msg);
		} catch (MessagingException e) {
			e.printStackTrace();
		}
	}
}

      填入真实的地址,运行上述代码,我们可以在控制台看到JavaMail打印的调试信息,邮箱就可以收到消息了。

DEBUG: setDebug: JavaMail version 1.6.2
DEBUG: getProvider() returning javax.mail.Provider[TRANSPORT,smtp,com.sun.mail.smtp.SMTPTransport,Oracle]
DEBUG SMTP: need username and password for authentication
DEBUG SMTP: protocolConnect returning false, host=smtp.163.com, user=lenovo, password=<null>
DEBUG SMTP: useEhlo true, useAuth true
DEBUG SMTP: trying to connect to host "smtp.163.com", port 25, isSSL false
220 163.com Anti-spam GT for Coremail System (163com[20141201])
DEBUG SMTP: connected to host "smtp.163.com", port: 25
EHLO LAPTOP-KTOMC7JR
250-mail
250-PIPELINING
250-AUTH LOGIN PLAIN XOAUTH2
250-AUTH=LOGIN PLAIN XOAUTH2
250-coremail 1Uxr2xKj7kG0xkI17xGrU7I0s8FY2U3Uj8Cz28x1UUUUU7Ic2I0Y2UFTa94RUCa0xDrUUUUj
250-STARTTLS
250-ID
250 8BITMIME
DEBUG SMTP: Found extension "PIPELINING", arg ""
DEBUG SMTP: Found extension "AUTH", arg "LOGIN PLAIN XOAUTH2"
DEBUG SMTP: Found extension "AUTH=LOGIN", arg "PLAIN XOAUTH2"
DEBUG SMTP: Found extension "coremail", arg "1Uxr2xKj7kG0xkI17xGrU7I0s8FY2U3Uj8Cz28x1UUUUU7Ic2I0Y2UFTa94RUCa0xDrUUUUj"
DEBUG SMTP: Found extension "STARTTLS", arg ""
DEBUG SMTP: Found extension "ID", arg ""
DEBUG SMTP: Found extension "8BITMIME", arg ""
STARTTLS
220 Ready to start TLS
EHLO LAPTOP-KTOMC7JR
250-mail
250-PIPELINING
250-AUTH LOGIN PLAIN XOAUTH2
250-AUTH=LOGIN PLAIN XOAUTH2
250-coremail 1Uxr2xKj7kG0xkI17xGrU7I0s8FY2U3Uj8Cz28x1UUUUU7Ic2I0Y2UrqWrgsUCa0xDrUUUUj
250-STARTTLS
250-ID
250 8BITMIME
DEBUG SMTP: Found extension "PIPELINING", arg ""
DEBUG SMTP: Found extension "AUTH", arg "LOGIN PLAIN XOAUTH2"
DEBUG SMTP: Found extension "AUTH=LOGIN", arg "PLAIN XOAUTH2"
DEBUG SMTP: Found extension "coremail", arg "1Uxr2xKj7kG0xkI17xGrU7I0s8FY2U3Uj8Cz28x1UUUUU7Ic2I0Y2UrqWrgsUCa0xDrUUUUj"
DEBUG SMTP: Found extension "STARTTLS", arg ""
DEBUG SMTP: Found extension "ID", arg ""
DEBUG SMTP: Found extension "8BITMIME", arg ""
DEBUG SMTP: protocolConnect login, host=smtp.163.com, user=yummy010419@163.com, password=<non-null>
DEBUG SMTP: Attempt to authenticate using mechanisms: LOGIN PLAIN DIGEST-MD5 NTLM XOAUTH2 
DEBUG SMTP: Using mechanism LOGIN
DEBUG SMTP: AUTH LOGIN command trace suppressed
DEBUG SMTP: AUTH LOGIN succeeded
DEBUG SMTP: use8bit false
MAIL FROM:<yummy010419@163.com>
250 Mail OK
RCPT TO:<3347995330@qq.com>
250 Mail OK
DEBUG SMTP: Verified Addresses
DEBUG SMTP:   3347995330@qq.com
DATA
354 End data with <CR><LF>.<CR><LF>
Date: Sun, 17 Jul 2022 17:49:54 +0800 (CST)
From: yummy010419@163.com
To: 3347995330@qq.com
Message-ID: <1028214719.0.1658051394158@LAPTOP-KTOMC7JR>
Subject: =?utf-8?B?6L+Z5piv5LiA5Lu95rWL6K+V6YKu5Lu2?=
MIME-Version: 1.0
Content-Type: text/html; charset=utf-8
Content-Transfer-Encoding: base64

5LuK5aSpPGI+5byA5b+DPC9iPuWQlw==
.
250 Mail OK queued as smtp12,EMCowAB3Qq5B29NiuePTCA--.64472S3 1658051403
DEBUG SMTP: message successfully delivered to mail server
QUIT
221 Bye

       从上面的调试信息可以看出,SMTP协议是一个请求-响应协议,客户端总是发送命令,然后等待服务器响应。服务器响应总是以数字开头,后面的信息才是用于调试的文本。这些响应码已经被定义在SMTP协议中了,查看具体的响应码就可以知道出错原因。

四·、发送HTML邮件

发送HTML邮件和文本邮件是类似的,只需要把:

message.setText(body, "UTF-8");

改为:

message.setText(body, "UTF-8", "html");

传入的body是类似<h1>Hello</h1><p>Hi, xxx</p>这样的HTML字符串即可。

五、发送附件

要在电子邮件中携带附件,我们就不能直接调用message.setText()方法,而是要构造一个Multipart对象:

// 3.邮件内容"复合"对象
			Multipart multipart = new MimeMultipart();
			
			// 正文
			BodyPart textPart = new MimeBodyPart();
			textPart.setContent("等风来不如追风去 <b>追逐</b>的过程就是人生的意义","text/html;charset=utf-8");
			
			// 附件
			BodyPart imagePart = new MimeBodyPart();
			imagePart.setFileName("young.jpg");
			// 数据处理对象(读取附件文件从本地磁盘进行读取)
			imagePart.setDataHandler(
					new DataHandler(
							new ByteArrayDataSource(
									Files.readAllBytes(Paths.get("E:\\test\\gdg.jpg")), 
									"application/octet-stream")));
			
			// 添加至邮件内容
			multipart.addBodyPart(textPart);// 添加正文
			multipart.addBodyPart(imagePart);// 添加附件

一个Multipart对象可以添加若干个BodyPart,其中第一个BodyPart是文本,即邮件正文,后面的BodyPart是附件。BodyPart依靠setContent()决定添加的内容,如果添加文本,用setContent("...", "text/plain;charset=utf-8")添加纯文本,或者用setContent("...", "text/html;charset=utf-8")添加HTML文本。如果添加附件,需要设置文件名(不一定和真实文件名一致),并且添加一个DataHandler(),传入文件的MIME类型。二进制文件可以用application/octet-stream,Word文档则是application/msword。最后,通过setContent()把Multipart添加到Message中,即可发送。

六、发送内嵌图片的HTML邮件

       如果需要在HTML邮件中内嵌图片,可以选择在邮件中加入

<img src="http://example.com/test.jpg">,这样的外部图片链接通常会被邮件客户端过滤,并提示用户显示图片并不安全。只有内嵌的图片才能正常在邮件中显示。所以,这种方式并不推荐。推荐将内嵌图片作为一个附件嵌入邮件,即邮件本身也是Multipart,但需要做一点额外的处理:

// 邮件正文部分
			BodyPart textBodyPart = new MimeBodyPart();
			StringBuilder body = new StringBuilder();
			body.append("<h>郭德纲</h>");
			body.append("<img src = 'cid:gangzi'/>");// 通过内容ID引用附件照片
			textBodyPart.setContent(body.toString(),"text/html;charset= utf-8");
			
			// 邮件附件部分
			BodyPart imageBodyPart = new MimeBodyPart();
			imageBodyPart.setFileName("gdg.jpg");
			imageBodyPart.setDataHandler(
					new DataHandler(
							new ByteArrayDataSource(
									Files.readAllBytes(Paths.get("E:\\test\\gdg.jpg")),"application/octet-stream")));
			imageBodyPart.setHeader("Content-ID", "<gangzi>");//设置内容ID
			
			// 组合正文和邮件
			Multipart multipart = new MimeMultipart();
			multipart.addBodyPart(textBodyPart);
			multipart.addBodyPart(imageBodyPart);

AOP(Aspect-Oriented Programming)是面向切面编程的一种编程范式,通过在程序的不同阶段植入通用的切面逻辑,使得这些逻辑可重用并可以在不同的模块中引用。其中,aop returning是AOP的一个关键方面,用于在方法执行成功返回之后执行的操作。 aop returning是在被通知的方法成功返回结果之后,执行一些附加操作的一种机制。通常,这些附加操作包括日志记录、性能监测、事务管理等。通过aop returning,我们可以将这些通用操作与原始方法逻辑解耦,并将它们封装在切面中。 在使用aop returning时,我们首先需要定义一个切面(Aspect),其中包含切入点(Pointcut)和通知(Advice)。切入点决定了哪些方法会被植入切面逻辑,通知则定义了方法执行前、后以及返回时的具体操作。 在aop returning中,通知类型为返回通知(AfterReturning Advice)。当被通知的方法成功执行并返回结果时,返回通知即会触发。我们可以在返回通知中编写所需的逻辑,比如记录返回结果、日志输出或是检查业务逻辑等。 通过使用aop returning,我们可以集中处理各个方法执行成功后的附加操作,避免了在每个方法中显式编写这些逻辑的重复性工作。这样可以提高代码的可维护性、可读性和代码复用程度。 总而言之,aop returning是AOP编程中用于在方法执行成功返回之后执行的一种机制。通过使用aop returning,我们可以在方法返回后执行一些附加操作,提高代码的可维护性和代码复用程度,并有效地解耦原始方法和通用操作。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

不想早起.

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值