10.java程序员必知必会类库之邮件

news/2024/4/20 4:20:14/文章来源:https://blog.csdn.net/wlyang666/article/details/130156916

前言

邮件功能在当前互联网应用中已经是很成熟的功能,也是作为java程序员应该掌握的技能。常见使用场景有:

  1. 电商软件开电子发票,需要发到用户邮箱里面
  2. 生产实时报警,需要发到邮箱里面
  3. 银行软件申请的征信报告,电子账单流水,需要发到邮箱里面

在这里插入图片描述

1. 前置准备

  1. 确定发邮件发送人的邮箱地址,然后基于不同的邮件服务器申请授权码
  2. 些云服务器会对邮箱端口做拦截,会出现本地能发邮件,测试环境发邮件报错,此时考虑切换465端口
  3. 当前主流邮件服务器有QQ和163,当然也有每家公司自己的邮件服务器
  4. 后续测试案例,统一以QQ邮箱为例,发送邮件

2. javax.mail

2.1 介绍

javax.mail是Sun发布的用来处理email的API。它可以方便地执行一些常用的邮件传输。 JavaMail是可选包。

2.2 pom依赖引入

<dependency><groupId>javax.mail</groupId><artifactId>mail</artifactId><version>1.4</version>
</dependency>
<dependency><groupId>junit</groupId><artifactId>junit</artifactId><version>4.9</version>
</dependency>

2.3 工具类

public class MailUtils {private String smtpHost; // 邮件服务器地址private String sendUserName; // 发件人的用户名private String sendUserPass; // 发件人密码private MimeMessage mimeMsg; // 邮件对象private Multipart mp;// 附件添加的组件private void init() {// 创建一个密码验证器Authentication authenticator = null;authenticator = new Authentication(sendUserName, sendUserPass);// 实例化Properties对象Properties props = System.getProperties();props.put("mail.smtp.host", smtpHost);props.put("mail.smtp.auth", "true"); // 需要身份验证props.put("mail.smtp.starttls.enable", "true");// 建立会话Session session = Session.getDefaultInstance(props, authenticator);// 置true可以在控制台(console)上看到发送邮件的过程session.setDebug(true);// 用session对象来创建并初始化邮件对象mimeMsg = new MimeMessage(session);// 生成附件组件的实例mp = new MimeMultipart();}private MailUtils(String smtpHost, String sendUserName, String sendUserPass, String to, String cc, String mailSubject,String mailBody, List<String> attachments) {this.smtpHost = smtpHost;this.sendUserName = sendUserName;this.sendUserPass = sendUserPass;init();setFrom(sendUserName);setTo(to);setCC(cc);setBody(mailBody);setSubject(mailSubject);if (attachments != null) {for (String attachment : attachments) {addFileAffix(attachment);}}}/*** 邮件实体** @param smtpHost     邮件服务器地址* @param sendUserName 发件邮件地址* @param sendUserPass 发件邮箱密码* @param to           收件人,多个邮箱地址以半角逗号分隔* @param cc           抄送,多个邮箱地址以半角逗号分隔* @param mailSubject  邮件主题* @param mailBody     邮件正文* @param attachments  附件路径* @return*/public static MailUtils entity(String smtpHost, String sendUserName, String sendUserPass, String to, String cc,String mailSubject, String mailBody, List<String> attachments) {return new MailUtils(smtpHost, sendUserName, sendUserPass, to, cc, mailSubject, mailBody, attachments);}/*** 设置邮件主题** @param mailSubject* @return*/private boolean setSubject(String mailSubject) {try {mimeMsg.setSubject(mailSubject);} catch (Exception e) {return false;}return true;}/*** 设置邮件内容,并设置其为文本格式或HTML文件格式,编码方式为UTF-8** @param mailBody* @return*/private boolean setBody(String mailBody) {try {BodyPart bp = new MimeBodyPart();bp.setContent("<meta http-equiv=Content-Type content=text/html; charset=UTF-8>" + mailBody,"text/html;charset=UTF-8");// 在组件上添加邮件文本mp.addBodyPart(bp);} catch (Exception e) {System.err.println("设置邮件正文时发生错误!" + e);return false;}return true;}/*** 添加一个附件** @param filename 邮件附件的地址,只能是本机地址而不能是网络地址,否则抛出异常* @return*/public boolean addFileAffix(String filename) {try {if (filename != null && filename.length() > 0) {BodyPart bp = new MimeBodyPart();FileDataSource fileds = new FileDataSource(filename);bp.setDataHandler(new DataHandler(fileds));bp.setFileName(MimeUtility.encodeText(fileds.getName(), "utf-8", null)); // 解决附件名称乱码mp.addBodyPart(bp);// 添加附件}} catch (Exception e) {System.err.println("增加邮件附件:" + filename + "发生错误!" + e);return false;}return true;}/*** 设置发件人地址** @param from 发件人地址* @return*/private boolean setFrom(String from) {try {mimeMsg.setFrom(new InternetAddress(from));} catch (Exception e) {return false;}return true;}/*** 设置收件人地址** @param to 收件人的地址* @return*/private boolean setTo(String to) {if (to == null)return false;try {mimeMsg.setRecipients(Message.RecipientType.TO, InternetAddress.parse(to));} catch (Exception e) {return false;}return true;}/*** 设置抄送** @param cc* @return*/private boolean setCC(String cc) {if (cc == null) {return false;}try {mimeMsg.setRecipients(Message.RecipientType.CC, InternetAddress.parse(cc));} catch (Exception e) {return false;}return true;}/*** no object DCH for MIME type multipart/mixed报错解决*/private void solveError() {MailcapCommandMap mc = (MailcapCommandMap) CommandMap.getDefaultCommandMap();mc.addMailcap("text/html;; x-java-content-handler=com.sun.mail.handlers.text_html");mc.addMailcap("text/xml;; x-java-content-handler=com.sun.mail.handlers.text_xml");mc.addMailcap("text/plain;; x-java-content-handler=com.sun.mail.handlers.text_plain");mc.addMailcap("multipart/*;; x-java-content-handler=com.sun.mail.handlers.multipart_mixed; x-java-fallback-entry=true");mc.addMailcap("message/rfc822;; x-java-content-handler=com.sun.mail.handlers.message_rfc822");CommandMap.setDefaultCommandMap(mc);Thread.currentThread().setContextClassLoader(MailUtils.class.getClassLoader());}/*** 发送邮件** @return*/public boolean send() throws Exception {mimeMsg.setContent(mp);mimeMsg.saveChanges();System.out.println("正在发送邮件....");solveError();Transport.send(mimeMsg);System.out.println("发送邮件成功!");return true;}
}

2.4 测试类

public class TestMail {@Testpublic void testSend() throws Exception {//QQ邮箱测试String userName = "xxx@qq.com"; // 发件人邮箱String password = "rlhcuxponcichbf"; // 发件人密码,其实不一定是邮箱的登录密码,对于QQ邮箱来说是SMTP服务的授权文本String smtpHost = "smtp.qq.com"; // 邮件服务器//163邮箱测试
//         String userName = "xxxxxxx@163.com"; // 发件人邮箱
//         String password = "wdedwe"; // 发件人密码,其实不一定是邮箱的登录密码,对于QQ邮箱来说是SMTP服务的授权文本
//         String smtpHost = "smtp.163.com"; // 邮件服务器String to = "xxx@qq.com"; // 收件人,多个收件人以半角逗号分隔String cc = "xxx@qq.com"; // 抄送,多个抄送以半角逗号分隔String subject = "这是邮件的主题 163"; // 主题String body = "这是邮件的正文163"; // 正文,可以用html格式List<String> attachments = Arrays.asList("E:\\weixinData\\WeChat Files\\wxid_gv8xbkloz0wc22\\FileStorage\\File\\2023-03\\test\\src\\main\\resources\\书籍统计表.xls", "E:\\weixinData\\WeChat Files\\wxid_gv8xbkloz0wc22\\FileStorage\\File\\2023-03\\test\\src\\main\\resources\\书籍统计表.xls"); // 附件的路径,多个附件也不怕MailUtils emailUtils = MailUtils.entity(smtpHost, userName, password, to, cc, subject, body, attachments);emailUtils.send();// 发送!}
}

收到邮件
在这里插入图片描述

2. Apache Commons Email

2.1 介绍

commons-email是apache提供的一个开源的API,是对javamail的封装。主要包括:SimpleEmail,MultiPartEmail,HtmlEmail,EmailAttachment四个类。

  1. SimpleEmail:发送简单的email,不能添加附件
  2. MultiPartEmail:文本邮件,可以添加多个附件
  3. HtmlEmail:HTML格式邮件,同时具有MultiPartEmail类所有“功能”
  4. EmailAttchment:附件类,可以添加本地资源,也可以指定网络上资源,在发送时自动将网络上资源下载发送。

2.2 pom依赖引入

<dependency><groupId>org.apache.commons</groupId><artifactId>commons-email</artifactId><version>1.5</version>
</dependency>

2.3 工具类

package com.wanlong.mail.javax;import org.apache.commons.mail.DefaultAuthenticator;
import org.apache.commons.mail.EmailAttachment;
import org.apache.commons.mail.ImageHtmlEmail;
import org.springframework.util.CollectionUtils;import java.net.URL;
import java.util.List;/*** @author wanlong* @version 1.0* @description:* @date 2023/4/24 13:44*/
public class MailUtil {/*** @param smtpHost:     邮件服务器地址* @param sendUserName: 发件人邮箱地址* @param sendUserPass: 发件人邮箱授权码* @param mailSubject:  邮件主题* @param mailBody:     邮件正文* @param attachments:  附件文件地址* @param ccList:       抄送收件人列表* @param bccList:      加密收件人列表* @param to:           收件人列表,可以一个,可以多个* @return void* @Description:发送邮件* @Author: wanlong* @Date: 2023/4/24 14:59**/public static void sendMail(String smtpHost, String sendUserName, String sendUserPass, String mailSubject,String mailBody, List<String> attachments, List<String> ccList,List<String> bccList, String... to) throws Exception {//创建邮件对象ImageHtmlEmail email = new ImageHtmlEmail();// 邮件服务器域名email.setHostName(smtpHost);// 邮件服务器smtp协议的SSL端口 这里正常默认是25端口,有的云服务器默认会禁止25端口,可以替换为465email.setSmtpPort(465);// 用户名和密码为邮箱的账号和密码email.setAuthenticator(new DefaultAuthenticator(sendUserName, sendUserPass));// SSL安全连接email.setSSLOnConnect(true);// 设置字符编码方式email.setCharset("UTF-8");// 发件人email.setFrom(sendUserName);// 收件人,可以发送给多人email.addTo(to);// 抄送if (!CollectionUtils.isEmpty(ccList)) {for (String cc : ccList) {email.addCc(cc);}}// 密送if (!CollectionUtils.isEmpty(bccList)) {for (String bcc : bccList) {email.addBcc(bcc);}}// 邮件主题email.setSubject(mailSubject);// 邮件正文email.setMsg(mailBody);//附件if (!CollectionUtils.isEmpty(attachments)) {for (String path : attachments) {// 附件类,可以添加本地资源,也可以指定网络上资源,在发送时自动将网络上资源下载发送EmailAttachment attachment = new EmailAttachment();//这里判断不一定严谨,只是单纯根据地址还是路径判断的,可以方法区分参数,分别遍历添加if (path.startsWith("http")) {//附件为网上资源attachment.setURL(new URL(path));} else {// 本地路径attachment.setPath(path);}// 定义附件attachment.setDisposition(EmailAttachment.ATTACHMENT);//添加附件email.attach(attachment);}}// 发送String send = email.send();}
}

2.4 测试类

@Test
public void testSendApache() throws Exception {String userName = "qwe@qq.com"; // 发件人邮箱String password = "rlhcuxponcichbf"; // 发件人密码,其实不一定是邮箱的登录密码,对于QQ邮箱来说是SMTP服务的授权文本String smtpHost = "smtp.qq.com"; // 邮件服务器String to = "xxx@newhope.cn"; // 收件人,多个收件人以半角逗号分隔String cc = "xxx@newhope.cn"; // 抄送,多个抄送以半角逗号分隔String subject = "这是邮件的主题 163"; // 主题String body = "这是邮件的正文163"; // 正文,可以用html格式List<String> ccList = Arrays.asList(cc);List<String> pathList = Arrays.asList("E:\\weixinData\\WeChat Files\\wxid_gv8xbkloz0wc22\\FileStorage\\File\\2023-03\\test\\src\\main\\resources\\书籍统计表.xls", "E:\\weixinData\\WeChat Files\\wxid_gv8xbkloz0wc22\\FileStorage\\File\\2023-03\\test\\src\\main\\resources\\书籍统计表.xls");List<String> bccList = new ArrayList<>();MailUtil.sendMail(smtpHost,userName,password,subject,body,pathList,ccList,bccList,to);
}

正常接收邮件:
在这里插入图片描述

3.springboot结合

3.1 介绍

springboot 提供了简洁,方便的调用方式,如果项目本身是springboot,强烈推荐使用这种实现邮箱功能

3.2 pom坐标引入

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd"><modelVersion>4.0.0</modelVersion><parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>2.1.3.RELEASE</version><relativePath/> <!-- lookup parent from repository --></parent><groupId>com.example</groupId><artifactId>demo</artifactId><version>0.0.1-SNAPSHOT</version><name>demo</name><description>Demo project for Spring Boot</description><properties><java.version>8</java.version></properties><dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter</artifactId></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency><!-- Spring Boot 邮件依赖 --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-mail</artifactId></dependency></dependencies><build><plugins><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId></plugin></plugins></build>
</project>

3.3 使用

3.3.1 yml配置邮箱信息

server:port: 8081
spring:mail:host: smtp.qq.com#默认端口号465port: 465username: xxx@qq.compassword: rlhcuxponcichbfprotocol: smtptest-connection: truedefault-encoding: UTF-8properties:mail.smtp.auth: truemail.smtp.starttls.enable: truemail.smtp.starttls.required: truemail.smtp.ssl.enable: truemail.display.sendmail: spring-boot-demo

3.3.2 创建service

package com.wanlong.learn;import javax.mail.MessagingException;public interface MailService {/*** 发送文本邮件** @param to      收件人地址* @param subject 邮件主题* @param content 邮件内容* @param cc      抄送地址*/void sendSimpleMail(String to, String subject, String content, String... cc);/*** 发送HTML邮件** @param to      收件人地址* @param subject 邮件主题* @param content 邮件内容* @param cc      抄送地址* @throws MessagingException 邮件发送异常*/void sendHtmlMail(String to, String subject, String content, String... cc) throws MessagingException;/*** 发送带附件的邮件** @param to       收件人地址* @param subject  邮件主题* @param content  邮件内容* @param filePath 附件地址* @param cc       抄送地址* @throws MessagingException 邮件发送异常*/void sendAttachmentsMail(String to, String subject, String content, String filePath, String... cc) throws MessagingException;/*** 发送正文中有静态资源的邮件** @param to      收件人地址* @param subject 邮件主题* @param content 邮件内容* @param rscPath 静态资源地址* @param rscId   静态资源id* @param cc      抄送地址* @throws MessagingException 邮件发送异常*/void sendResourceMail(String to, String subject, String content, String rscPath, String rscId, String... cc) throws MessagingException;
}

3.3.3 创建实现类

@Service
public class MailServiceImpl implements MailService {@Autowiredprivate JavaMailSender mailSender;@Value("${spring.mail.username}")private String from;/*** 发送文本邮件** @param to      收件人地址* @param subject 邮件主题* @param content 邮件内容* @param cc      抄送地址*/@Overridepublic void sendSimpleMail(String to, String subject, String content, String... cc) {SimpleMailMessage message = new SimpleMailMessage();message.setFrom(from);message.setTo(to);message.setSubject(subject);message.setText(content);if (!StringUtils.isEmpty(cc)) {message.setCc(cc);}mailSender.send(message);}/*** 发送HTML邮件** @param to      收件人地址* @param subject 邮件主题* @param content 邮件内容* @param cc      抄送地址* @throws MessagingException 邮件发送异常*/@Overridepublic void sendHtmlMail(String to, String subject, String content, String... cc) throws MessagingException {MimeMessage message = mailSender.createMimeMessage();MimeMessageHelper helper = new MimeMessageHelper(message, true);helper.setFrom(from);helper.setTo(to);helper.setSubject(subject);helper.setText(content, true);if (!StringUtils.isEmpty(cc)) {helper.setCc(cc);}mailSender.send(message);}/*** 发送带附件的邮件** @param to       收件人地址* @param subject  邮件主题* @param content  邮件内容* @param filePath 附件地址* @param cc       抄送地址* @throws MessagingException 邮件发送异常*/@Overridepublic void sendAttachmentsMail(String to, String subject, String content, String filePath, String... cc) throws MessagingException {MimeMessage message = mailSender.createMimeMessage();MimeMessageHelper helper = new MimeMessageHelper(message, true);helper.setFrom(from);helper.setTo(to);helper.setSubject(subject);helper.setText(content, true);if (!StringUtils.isEmpty(cc)) {helper.setCc(cc);}FileSystemResource file = new FileSystemResource(new File(filePath));String fileName = filePath.substring(filePath.lastIndexOf(File.separator));helper.addAttachment(fileName, file);mailSender.send(message);}/*** 发送正文中有静态资源的邮件** @param to      收件人地址* @param subject 邮件主题* @param content 邮件内容* @param rscPath 静态资源地址* @param rscId   静态资源id* @param cc      抄送地址* @throws MessagingException 邮件发送异常*/@Overridepublic void sendResourceMail(String to, String subject, String content, String rscPath, String rscId, String... cc) throws MessagingException {MimeMessage message = mailSender.createMimeMessage();MimeMessageHelper helper = new MimeMessageHelper(message, true);helper.setFrom(from);helper.setTo(to);helper.setSubject(subject);helper.setText(content, true);if (!StringUtils.isEmpty(cc)) {helper.setCc(cc);}FileSystemResource res = new FileSystemResource(new File(rscPath));helper.addInline(rscId, res);mailSender.send(message);}
}

3.3.4 注入service调用

@Controller
public class EmailController {@Autowiredprivate MailService mailService;/*** 测试发送简单邮件*/@GetMapping("/sendEmail")@ResponseBodypublic String sendSimpleMail() {mailService.sendSimpleMail("xxx@xx.xx", "这是一封简单邮件", "这是一封普通的SpringBoot测试邮件");return "ok";}
}

接收方正常收到邮件:
在这里插入图片描述

注意事项:

  1. 邮件服务器默认端口是25,有些云服务器会禁用25,此时发送邮件会发不出去,可以调整端口为465
  2. 邮件内容可以通过html展示,如果内容不复杂,可以自己手动拼接html报文格式,直接写入正文
  3. 如果邮件正文想标准化,但是报文格式又很复杂,此时不适宜通过手动拼接html展示,可以调用模板引擎 thymeleaf 生成。关键信息参数化,到时候基于每个邮件,生成定制化的文本
  4. 上面的工具类只是参考,因为发件人地址,邮箱服务器,授权码,是长期不会变又比较重要的信息,可以放到配置文件中,应用启动读取配置文件获取
  5. 邮箱一般是通用功能,可以考虑封装统一方法,封装到公司底层框架,其他人使用只要通过调用微服务或者调用本地方法一样,不需要考虑具体实现细节
  6. 因为邮箱要访问外网,一般放在公司网关层实现

参考文档:

javax发送邮件
springBoot发送邮件

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.luyixian.cn/news_show_103077.aspx

如若内容造成侵权/违法违规/事实不符,请联系dt猫网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

《C++ Primer Plus》(第6版)第17章编程练习

《C Primer Plus》&#xff08;第6版&#xff09;第17章编程练习 《C Primer Plus》&#xff08;第6版&#xff09;第17章编程练习1. 计算输入流中第一个\$之前的字符数目2. 将键盘输入&#xff08;直到模拟的文件尾&#xff09;复制到通过命令行指定的文件中3. 将一个文件复制…

【难学易用c++ 之 继承】

目录&#xff1a; 前言一、继承的概念及定义&#xff08;一&#xff09;概念&#xff08;二&#xff09;继承定义继承关系和访问限定符继承基类成员访问方式的变化 二、基类和派生类对象赋值转换三、继承中的作用域四、派生类的默认成员函数五、继承与友元六、继承与静态成员七…

如何使用阿里云短信服务实现登录页面,手机验证码登录?

1&#xff1a;个人如何使用阿里云短信服务&#xff1f; 2022如何使用个人阿里云短信服务&#xff1f;_linxiMY的博客-CSDN博客添加完成之后&#xff0c;等待审核&#xff01;一般2个小时就会出来审核结果了&#xff0c;这里我因为注册申请时填写规则有误&#xff0c;足足审核了…

pytest 学习三(前置后置操作)

pytest测试框架_pytest框架-CSDN博客 一、常用的操作 一、setup/teardown 每个用例之前、之后执行 二、setup_class/teardown_class 在每个类之前、之后执行一次 二、pytest.fixture 设置前置后置操作范围 pytest.fixture(scope"",params,autouse,ids,name) 其中 sc…

JavaScript模块化开发

目录&#xff1a; 1 认识模块化开发 2 CommonJS和Node 3 require函数解析 4 AMD和CMD&#xff08;了解&#xff09; 5 ESModule用法详解 6 ESModule运行原理 模块化不是两个不同的js文件直接导入到某个页面中的&#xff0c;因为这两个文件只要有相同的变量或函数&#xf…

R基础函数概览(一)

rep 函数形式&#xff1a;rep(x, time , length , each ,) 参数说明&#xff1a; x&#xff1a;代表的是你要进行复制的对象&#xff0c;可以是一个向量或者是一个因子。 times&#xff1a;代表的是复制的次数&#xff0c;只能为正数。负数以及NA值都会为错误值。复制是指的…

云原生(docker+k8s+阿里云)-Docker

Gitee-Kubernetes学习 kubectl备忘清单 k8s官方文档-task [云原生-kubectl命令详解] ingress详解 ingress官方文档 云原生-语雀-架构师第一课 从Docker到Kubernetes进阶-社区 云计算学习路线-阿里云大学 如上图&#xff0c;服务器有公网ip和私网ip&#xff0c;公网ip是外部访问…

Ubuntu20.04使用多卡训练HyperNetwork模型和LoRA模型全流程及疑难问题解决方案

目录 一. LoRA模型多卡训练1.1 安装xformer等库1.2 设置路径1.3 多卡训练 二. LoRA模型多卡训练疑难报错解决方案多卡训练报错 软硬件配置&#xff1a; CPU: AMD 5800 8core 16Thread GPU: NVIDIA RTX 3090 *1 NVIDIA TITAN RTX *1 OS: Ubuntu20.04 一. LoRA模型多卡训练 1.1 …

一篇终结synchronized

一&#xff1a;基本原理 Java对象在内存中由两部分组成 &#xff1a; 1 是成员变量 2 是对象头&#xff0c;以32位虚拟机介绍&#xff1a;此时对象头是64位&#xff0c;即8字节 其中32个字节代表 mark word 另外32个字节代表klass word分别是什么意思呢&#xff1f; 1 klass …

写题总结1

先把自己写完的总结一下&#xff1a; 题目一&#xff1a; 猫儿园的告示牌上贴着 ab 大小的矩形广告纸。猫猫对广告不感兴趣&#xff0c;她想知道能否用 cd 的矩形白纸完全覆盖这个广告。猫猫可以对白纸进行平移、旋转&#xff0c;但不能折叠或撕开等。如果可以完全覆盖输出 YE…

滴水逆向3期笔记与作业——01汇编

防止OneNote丢失。 海哥牛逼。 01汇编笔记 01进制进制定义10-2进制转换八进制 02数据宽度/逻辑运算数据宽度与存储逻辑运算计算机做加法的本质作业 03通用寄存器_内存读写通用寄存器表通用寄存器图内存读写计算机操作系统位数意义 04内存地址_堆栈寻址公式PUSH指令POP指令作业 …

【IAR工程】STM8S基于ST标准库读取DHT11数据

【IAR工程】STM8S基于ST标准库读取DHT11数据 ✨申明&#xff1a;本文章仅发表在CSDN网站&#xff0c;任何其他网站&#xff0c;未注明来源&#xff0c;见此内容均为盗链和爬取&#xff0c;请多多尊重和支持原创!&#x1f341;对于文中所提供的相关资源链接将作不定期更换。&…

4月有8本SCIE期刊被剔除(附MDPI/Frontiers/Hindawi最新在检期刊)

2023年4月SCI、SSCI期刊目录更新 2023年4月18日&#xff0c;科睿唯安更新了WOS期刊目录&#xff0c;继上次3月WOS期刊目录剔除50本SCIE&SSCI期刊之后&#xff0c;此次4月更新又有8本SCIE期刊发生变动&#xff0c;其中有4本期刊被踢出SCIE数据库&#xff0c;4本期刊更改了名…

流程图拖拽视觉编程--概述

一般的机器视觉平台采用纯代码的编程方式&#xff0c;如opencv、halcon&#xff0c;使用门槛高、难度大、定制性强、开发周期长&#xff0c;因此迫切需要一个低代码开发的视觉应用平台。AOI缺陷检测的对象往往缺陷种类多&#xff0c;将常用的图像处理算子封装成图形节点,如抓直…

Android 系统架构大图

android的系统架构和其操作系统一样&#xff0c;采用了分层的架构。从架构图看&#xff0c;android分为四个层&#xff0c;从高层到低层分别是应用程序层、应用程序框架层、系统运行库层和Linux核心层。 1.应用程序 Android会同一系列核心应用程序包一起发布&#xff0c;该应用…

确定因果随机森林的树木数量 the number of trees

前言 推断因果性和分析异质性是统计学家在处理混杂任务中的圣杯。传统且主流的方法有:倾向性评分、分层分享、比例风险模型等。新的方法也有很多,代表就是:因果随机森林。这种算法,浅看难度一般,深入探索发现坑还是很多的。这篇博客不对算法做深入探讨,仅仅是我在阅读文…

Nautilus Chain :基于模块化架构的Layer3正在走向成熟

Nautilus Chain 是一个基于 Eclipse 和 Celestia 构建的模块化 Layer3 链。作为定位在 Layer0 的链基建概念&#xff0c;Eclipse 和 Celestia 为面向未来的区块链扩容技术提供了一套开发工具和基础框架。尽管这种前沿技术过去一直处于概念验证阶段&#xff0c;尚未推出适用于大…

Java并发(三)----创建线程的三种方式及查看进程线程

一、直接使用 Thread // 创建线程对象 Thread t new Thread() {public void run() {// 要执行的任务} }; // 启动线程 t.start(); 例如&#xff1a; // 构造方法的参数是给线程指定名字&#xff0c;推荐 Thread t1 new Thread("t1") {Override// run 方法内实现…

手把手教你PXE高效网络装机、Kickstart无人值守安装(详细版)

目录 一、部署PXE远程安装服务1.1PXE定义1.2PXE服务优点1.3搭建网络体系前提条件1.4 搭建PXE远程安装服务器 二. 实验2.1 服务器操作2.2 安装启动TFTP服务并修改TFTP服务的配置文件2.3 安装并启用DHCP服务2.4 准备linux内核&#xff0c;初始化镜像文件2.5 准备PXE引导程序2.6 安…

22、Tweak原理及部分逆向防护

一、Tweak原理 1.1 Tweak产物.dylib 执行make命令时,在 .theos的隐藏目录中,编译出obj/debug目录,包含 arm64、arm64e两种架构,同时生成readbadges.dylib动态库 在arm64、arm64e目录下,有各自架构的readbadges.dylib,而debug目录下的readbadges.dylib,是一个胖二进制文件 fi…