【java发送qq邮件问题】Could not connect to SMTP host: smtp.qq.com, port: 465 & No appropriate protocol

背景

  • 使用javax发送qq邮件就出现了问题
  • 我所用的jdk版本是1.8
  • 报错内容:
     javax.mail.MessagingException: Could not connect to SMTP host: smtp.qq.com, port: 465
    
  • 同时还有下面这个报错:
    javax.net.ssl.SSLHandshakeException: No appropriate protocol (protocol is disabled or cipher suites are inappropriate)
    

初探

  • 网上的各种解决方式,大概包括更改另外的端口、开放云服务器端口、新增或者更改javax.mail的相关配置等。
  • 上述方式我都尝试过,都没有解决。

柳暗花明

  • 看到了一篇博客说是与java.security有关,对我来说确实是这个问题。

问题描述

  • 根据大佬的描述,要修改java.security这个文件中的部分代码,这个文件的位置是:jdk1.8/jre/lib/security/java.security
  • 我在jdk1.8版本找到了这个路径,但是在jdk1.7中是另外的路径jdk1.7/conf/security/java.security
  • 确实应该是这个文件的问题。

JDK定位

要修改这个文件,首先要定位jdk的位置。下面说一下怎么定位jdk的位置:

  • window下,cmd命令行输入java -verbose,看最后两行输出:
    在这里插入图片描述
    如上图,我本地的jdk位置是D:\java8,根据这个位置,再确定我是jdk1.8,因此那个问题的最终位置应该是在D:\java8\jre\lib\security\java.security。如果jdk是1.7,那么最终位置可能就是D:\java7\conf\security\java.security了。

  • linux下,命令行输入which java,就可以看到jdk的位置了:
    在这里插入图片描述
    然后其他步骤与window下一样。

终局之战

最后,要打开java.security文件,然后将jdk.tls.disabledAlgorithm参数后面的SSLv3、TLSv1、TLSv1.1都删掉。

  • 以window为例,用记事本打开文件,ctrl+F搜索jdk.tls.disabledAlgorithm,然后删掉上面三个协议即可。
  • 以linux为例,首先进入这个文件所在的目录,然后使用vim java.security命令打开这个文件。\键搜索,输入jdk.tls.disabledAlgorithm然后回车,会自动定位到代码的位置。然后i进入插入模式,将上述三个协议删掉即可。

提示:

  • 建议删除前先备份。
    在这里插入图片描述
### Java发送邮件时SSL握手异常问题分析 当遇到`javax.net.ssl.SSLHandshakeException: No appropriate protocol (protocol is disabled or cipher suites are inappropriate)`错误时,通常是因为使用的协议或加密套件被禁用所致。以下是针对该问题的具体解决方案: #### 1. 配置邮件属性支持SSL 通过配置邮件客户端的Properties对象来启用SSL连接并指定合适的协议版本。 ```java Properties properties = new Properties(); properties.put("mail.smtp.auth", "true"); properties.put("mail.smtp.starttls.enable", "true"); properties.put("mail.smtp.host", "smtp.example.com"); properties.put("mail.smtp.port", "587"); // 启用SSL Socket Factory properties.put("mail.smtp.socketFactory.class", "javax.net.ssl.SSLSocketFactory"); properties.put("mail.smtp.socketFactory.fallback", "false"); properties.put("mail.smtp.socketFactory.port", "465"); // 使用SSL端口 ``` 上述代码片段中启用了SSLSocketFactory以确保使用安全的传输层协议[^1]。 #### 2. 调整JDK的安全策略 如果仍然报错,则可能需要调整JDK中的`java.security`文件。具体操作如下: - **Linux环境**: 编辑位于 `/usr/lib/jvm/java-1.8.0-openjdk-<version>/jre/lib/security/java.security` 的文件。 - **Windows环境**: 编辑位于 `C:\Program Files\Java\<jdk_version>\jre\lib\security\java.security` 的文件。 找到以下参数并修改其值: ```plaintext jdk.tls.disabledAlgorithms=SSLv3, TLSv1, TLSv1.1, RC4, DES, MD5withRSA, \ DH keySize < 1024, EC keySize < 224, 3DES_EDE_CBC, anon, NULL, \ include jdk.disabled.namedCurves ``` 将其改为仅保留必要的算法禁用列表(移除对TLSv1和TLSv1.1的禁用),例如: ```plaintext jdk.tls.disabledAlgorithms=RC4, DES, MD5withRSA, \ DH keySize < 1024, EC keySize < 224, 3DES_EDE_CBC, anon, NULL, \ include jdk.disabled.namedCurves ``` 此更改允许更高版本的TLS协议参与握手过程[^4]。 #### 3. 更新依赖库至最新版 确保所使用的JavaMail API以及相关依赖项均为最新稳定版本。旧版本可能存在兼容性问题或者不支持现代密码学标准。 #### 4. 替代方案——显式定义Cipher Suites 对于某些特殊场景下可以尝试手动设定可用的cipher suites集合: ```java Session session = Session.getInstance(properties); session.setProtocolForAddress(.Protocol SMTP, "smtps"); Transport transport = session.getTransport(); SSLContext sslContext = SSLContext.getInstance("TLSv1.2"); sslContext.init(null, null, null); SSLSocketFactory socketFactory = sslContext.getSocketFactory(); ((SMTPTransport)transport).setSSLSocketFactory(socketFactory); ``` 以上方法强制指定了TLS v1.2作为通信协议[^2]。 --- ### 总结 综合来看,解决此类问题的关键在于合理配置应用层面的网络选项与底层运行环境之间的协调工作。一方面要确认应用程序正确设置了所需的SSL/TLS参数;另一方面也要核查操作系统上安装的JVM实例是否开放了必需的服务功能集。 相关问题
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值