023-从零搭建微服务-推送服务(三)

原【短信服务】更名【推送服务】

写在最前

如果这个项目让你有所收获,记得 Star 关注哦,这对我是非常不错的鼓励与支持。

源码地址(后端):https://gitee.com/csps/mingyue

源码地址(前端):https://gitee.com/csps/mingyue-ui

文档地址:https://gitee.com/csps/mingyue/wikis

常用邮箱客户端

  • QQ邮箱:
    • POP3: pop.qq.com,使用 SSL,端口号995
    • IMAP: imap.qq.com,使用 SSL,端口号993
    • SMTP: smtp.qq.com,使用SSL,端口号465或587
  • 网易邮箱:
    • POP3: pop.126.com、pop.yeah.net,使用 SSL,端口号995
    • IMAP: imap.163.com,、imap.yeah.net,使用 SSL,端口号993
    • SMTP: smtp.126.com、smtp.163.com,使用 SSL,端口号465或587
  • 电信189邮箱:
    • POP3: pop.189.cn,使用 SSL,端口号995
    • IMAP: imap.189.cn,使用 SSL,端口号993
    • SMTP: smtp.189.cn,使用 SSL,端口号465或587
  • 微软Outlook邮箱:
    • POP3: outlook.office365.com,使用 TLS,端口号995
    • IMAP: outlook.office365.com,使用 TLS,端口号993
    • SMTP: smtp.office365.com,使用 STARTTLS,端口号587

封装邮件工具

新建模块 mingyue-common-email 邮件模块

新建 Jakarta Mail 配置属性

@Data
@ConfigurationProperties(prefix = "email")
public class EmailProperties {/*** 过滤开关*/private Boolean enabled;/*** SMTP 服务器域名*/private String host;/*** SMTP 服务端口*/private Integer port;/*** 是否需要用户名密码验证*/private Boolean auth;/*** 用户名*/private String user;/*** 密码*/private String pass;/*** 发送方,遵循 RFC-822 标准*/private String from;/*** 使用 STARTTLS 安全连接,STARTTLS 是对纯文本通信协议的扩展。它将纯文本连接升级为加密连接(TLS或SSL), 而不是使用一个单独的加密通信端口。*/private Boolean starttlsEnable;/*** 使用 SSL 安全连接*/private Boolean sslEnable;/*** SMTP 超时时长,单位毫秒,缺省值不超时*/private Long timeout;/*** Socket 连接超时值,单位毫秒,缺省值不超时*/private Long connectionTimeout;}

新建 Java Mail 配置类

@AutoConfiguration
@EnableConfigurationProperties(EmailProperties.class)
public class EmailConfiguration {@Bean@ConditionalOnProperty(value = "email.enabled", havingValue = "true")public MailAccount mailAccount(EmailProperties emailProperties) {MailAccount account = new MailAccount();account.setHost(emailProperties.getHost());account.setPort(emailProperties.getPort());account.setAuth(emailProperties.getAuth());account.setFrom(emailProperties.getFrom());account.setUser(emailProperties.getUser());account.setPass(emailProperties.getPass());account.setSocketFactoryPort(emailProperties.getPort());account.setStarttlsEnable(emailProperties.getStarttlsEnable());account.setSslEnable(emailProperties.getSslEnable());account.setTimeout(emailProperties.getTimeout());account.setConnectionTimeout(emailProperties.getConnectionTimeout());return account;}}

新建邮件工具类

org.springframework.boot.autoconfigure.AutoConfiguration.imports:com.csp.mingyue.common.email.config.EmailConfiguration

@NoArgsConstructor(access = AccessLevel.PRIVATE)
public class EmailUtils {private static final MailAccount ACCOUNT = SpringUtils.getBean(MailAccount.class);/*** 获取邮件发送实例*/public static MailAccount getMailAccount() {return ACCOUNT;}/*** 获取邮件发送实例 (自定义发送人以及授权码)** @param user 发送人* @param pass 授权码*/public static MailAccount getMailAccount(String from, String user, String pass) {ACCOUNT.setFrom(StrUtil.blankToDefault(from, ACCOUNT.getFrom()));ACCOUNT.setUser(StrUtil.blankToDefault(user, ACCOUNT.getUser()));ACCOUNT.setPass(StrUtil.blankToDefault(pass, ACCOUNT.getPass()));return ACCOUNT;}/*** 使用配置文件中设置的账户发送文本邮件,发送给单个或多个收件人<br>* 多个收件人可以使用逗号“,”分隔,也可以通过分号“;”分隔** @param to      收件人* @param subject 标题* @param content 正文* @param files   附件列表* @return message-id* @since 3.2.0*/public static String sendText(String to, String subject, String content, File... files) {return send(to, subject, content, false, files);}/*** 使用配置文件中设置的账户发送HTML邮件,发送给单个或多个收件人<br>* 多个收件人可以使用逗号“,”分隔,也可以通过分号“;”分隔** @param to      收件人* @param subject 标题* @param content 正文* @param files   附件列表* @return message-id* @since 3.2.0*/public static String sendHtml(String to, String subject, String content, File... files) {return send(to, subject, content, true, files);}/*** 使用配置文件中设置的账户发送邮件,发送单个或多个收件人<br>* 多个收件人可以使用逗号“,”分隔,也可以通过分号“;”分隔** @param to      收件人* @param subject 标题* @param content 正文* @param isHtml  是否为HTML* @param files   附件列表* @return message-id*/public static String send(String to, String subject, String content, boolean isHtml, File... files) {return send(splitAddress(to), subject, content, isHtml, files);}/*** 使用配置文件中设置的账户发送邮件,发送单个或多个收件人<br>* 多个收件人、抄送人、密送人可以使用逗号“,”分隔,也可以通过分号“;”分隔** @param to      收件人,可以使用逗号“,”分隔,也可以通过分号“;”分隔* @param cc      抄送人,可以使用逗号“,”分隔,也可以通过分号“;”分隔* @param bcc     密送人,可以使用逗号“,”分隔,也可以通过分号“;”分隔* @param subject 标题* @param content 正文* @param isHtml  是否为HTML* @param files   附件列表* @return message-id* @since 4.0.3*/public static String send(String to, String cc, String bcc, String subject, String content, boolean isHtml, File... files) {return send(splitAddress(to), splitAddress(cc), splitAddress(bcc), subject, content, isHtml, files);}/*** 使用配置文件中设置的账户发送文本邮件,发送给多人** @param tos     收件人列表* @param subject 标题* @param content 正文* @param files   附件列表* @return message-id*/public static String sendText(Collection<String> tos, String subject, String content, File... files) {return send(tos, subject, content, false, files);}/*** 使用配置文件中设置的账户发送HTML邮件,发送给多人** @param tos     收件人列表* @param subject 标题* @param content 正文* @param files   附件列表* @return message-id* @since 3.2.0*/public static String sendHtml(Collection<String> tos, String subject, String content, File... files) {return send(tos, subject, content, true, files);}/*** 使用配置文件中设置的账户发送邮件,发送给多人** @param tos     收件人列表* @param subject 标题* @param content 正文* @param isHtml  是否为HTML* @param files   附件列表* @return message-id*/public static String send(Collection<String> tos, String subject, String content, boolean isHtml, File... files) {return send(tos, null, null, subject, content, isHtml, files);}/*** 使用配置文件中设置的账户发送邮件,发送给多人** @param tos     收件人列表* @param ccs     抄送人列表,可以为null或空* @param bccs    密送人列表,可以为null或空* @param subject 标题* @param content 正文* @param isHtml  是否为HTML* @param files   附件列表* @return message-id* @since 4.0.3*/public static String send(Collection<String> tos, Collection<String> ccs, Collection<String> bccs, String subject, String content, boolean isHtml, File... files) {return send(getMailAccount(), true, tos, ccs, bccs, subject, content, null, isHtml, files);}// ------------------------------------------------------------------------------------------------------------------------------- Custom MailAccount/*** 发送邮件给多人** @param mailAccount 邮件认证对象* @param to          收件人,多个收件人逗号或者分号隔开* @param subject     标题* @param content     正文* @param isHtml      是否为HTML格式* @param files       附件列表* @return message-id* @since 3.2.0*/public static String send(MailAccount mailAccount, String to, String subject, String content, boolean isHtml, File... files) {return send(mailAccount, splitAddress(to), subject, content, isHtml, files);}/*** 发送邮件给多人** @param mailAccount 邮件帐户信息* @param tos         收件人列表* @param subject     标题* @param content     正文* @param isHtml      是否为HTML格式* @param files       附件列表* @return message-id*/public static String send(MailAccount mailAccount, Collection<String> tos, String subject, String content, boolean isHtml, File... files) {return send(mailAccount, tos, null, null, subject, content, isHtml, files);}/*** 发送邮件给多人** @param mailAccount 邮件帐户信息* @param tos         收件人列表* @param ccs         抄送人列表,可以为null或空* @param bccs        密送人列表,可以为null或空* @param subject     标题* @param content     正文* @param isHtml      是否为HTML格式* @param files       附件列表* @return message-id* @since 4.0.3*/public static String send(MailAccount mailAccount, Collection<String> tos, Collection<String> ccs, Collection<String> bccs, String subject, String content, boolean isHtml, File... files) {return send(mailAccount, false, tos, ccs, bccs, subject, content, null, isHtml, files);}/*** 使用配置文件中设置的账户发送HTML邮件,发送给单个或多个收件人<br>* 多个收件人可以使用逗号“,”分隔,也可以通过分号“;”分隔** @param to       收件人* @param subject  标题* @param content  正文* @param imageMap 图片与占位符,占位符格式为cid:$IMAGE_PLACEHOLDER* @param files    附件列表* @return message-id* @since 3.2.0*/public static String sendHtml(String to, String subject, String content, Map<String, InputStream> imageMap, File... files) {return send(to, subject, content, imageMap, true, files);}/*** 使用配置文件中设置的账户发送邮件,发送单个或多个收件人<br>* 多个收件人可以使用逗号“,”分隔,也可以通过分号“;”分隔** @param to       收件人* @param subject  标题* @param content  正文* @param imageMap 图片与占位符,占位符格式为cid:$IMAGE_PLACEHOLDER* @param isHtml   是否为HTML* @param files    附件列表* @return message-id*/public static String send(String to, String subject, String content, Map<String, InputStream> imageMap, boolean isHtml, File... files) {return send(splitAddress(to), subject, content, imageMap, isHtml, files);}/*** 使用配置文件中设置的账户发送邮件,发送单个或多个收件人<br>* 多个收件人、抄送人、密送人可以使用逗号“,”分隔,也可以通过分号“;”分隔** @param to       收件人,可以使用逗号“,”分隔,也可以通过分号“;”分隔* @param cc       抄送人,可以使用逗号“,”分隔,也可以通过分号“;”分隔* @param bcc      密送人,可以使用逗号“,”分隔,也可以通过分号“;”分隔* @param subject  标题* @param content  正文* @param imageMap 图片与占位符,占位符格式为cid:$IMAGE_PLACEHOLDER* @param isHtml   是否为HTML* @param files    附件列表* @return message-id* @since 4.0.3*/public static String send(String to, String cc, String bcc, String subject, String content, Map<String, InputStream> imageMap, boolean isHtml, File... files) {return send(splitAddress(to), splitAddress(cc), splitAddress(bcc), subject, content, imageMap, isHtml, files);}/*** 使用配置文件中设置的账户发送HTML邮件,发送给多人** @param tos      收件人列表* @param subject  标题* @param content  正文* @param imageMap 图片与占位符,占位符格式为cid:$IMAGE_PLACEHOLDER* @param files    附件列表* @return message-id* @since 3.2.0*/public static String sendHtml(Collection<String> tos, String subject, String content, Map<String, InputStream> imageMap, File... files) {return send(tos, subject, content, imageMap, true, files);}/*** 使用配置文件中设置的账户发送邮件,发送给多人** @param tos      收件人列表* @param subject  标题* @param content  正文* @param imageMap 图片与占位符,占位符格式为cid:$IMAGE_PLACEHOLDER* @param isHtml   是否为HTML* @param files    附件列表* @return message-id*/public static String send(Collection<String> tos, String subject, String content, Map<String, InputStream> imageMap, boolean isHtml, File... files) {return send(tos, null, null, subject, content, imageMap, isHtml, files);}/*** 使用配置文件中设置的账户发送邮件,发送给多人** @param tos      收件人列表* @param ccs      抄送人列表,可以为null或空* @param bccs     密送人列表,可以为null或空* @param subject  标题* @param content  正文* @param imageMap 图片与占位符,占位符格式为cid:$IMAGE_PLACEHOLDER* @param isHtml   是否为HTML* @param files    附件列表* @return message-id* @since 4.0.3*/public static String send(Collection<String> tos, Collection<String> ccs, Collection<String> bccs, String subject, String content, Map<String, InputStream> imageMap, boolean isHtml, File... files) {return send(getMailAccount(), true, tos, ccs, bccs, subject, content, imageMap, isHtml, files);}// ------------------------------------------------------------------------------------------------------------------------------- Custom MailAccount/*** 发送邮件给多人** @param mailAccount 邮件认证对象* @param to          收件人,多个收件人逗号或者分号隔开* @param subject     标题* @param content     正文* @param imageMap    图片与占位符,占位符格式为cid:$IMAGE_PLACEHOLDER* @param isHtml      是否为HTML格式* @param files       附件列表* @return message-id* @since 3.2.0*/public static String send(MailAccount mailAccount, String to, String subject, String content, Map<String, InputStream> imageMap, boolean isHtml, File... files) {return send(mailAccount, splitAddress(to), subject, content, imageMap, isHtml, files);}/*** 发送邮件给多人** @param mailAccount 邮件帐户信息* @param tos         收件人列表* @param subject     标题* @param content     正文* @param imageMap    图片与占位符,占位符格式为cid:$IMAGE_PLACEHOLDER* @param isHtml      是否为HTML格式* @param files       附件列表* @return message-id* @since 4.6.3*/public static String send(MailAccount mailAccount, Collection<String> tos, String subject, String content, Map<String, InputStream> imageMap, boolean isHtml, File... files) {return send(mailAccount, tos, null, null, subject, content, imageMap, isHtml, files);}/*** 发送邮件给多人** @param mailAccount 邮件帐户信息* @param tos         收件人列表* @param ccs         抄送人列表,可以为null或空* @param bccs        密送人列表,可以为null或空* @param subject     标题* @param content     正文* @param imageMap    图片与占位符,占位符格式为cid:$IMAGE_PLACEHOLDER* @param isHtml      是否为HTML格式* @param files       附件列表* @return message-id* @since 4.6.3*/public static String send(MailAccount mailAccount, Collection<String> tos, Collection<String> ccs, Collection<String> bccs, String subject, String content, Map<String, InputStream> imageMap,boolean isHtml, File... files) {return send(mailAccount, false, tos, ccs, bccs, subject, content, imageMap, isHtml, files);}/*** 根据配置文件,获取邮件客户端会话** @param mailAccount 邮件账户配置* @param isSingleton 是否单例(全局共享会话)* @return {@link Session}* @since 5.5.7*/public static Session getSession(MailAccount mailAccount, boolean isSingleton) {Authenticator authenticator = null;if (mailAccount.isAuth()) {authenticator = new UserPassAuthenticator(mailAccount.getUser(), mailAccount.getPass());}return isSingleton ? Session.getDefaultInstance(mailAccount.getSmtpProps(), authenticator) //: Session.getInstance(mailAccount.getSmtpProps(), authenticator);}/*** 发送邮件给多人** @param mailAccount      邮件帐户信息* @param useGlobalSession 是否全局共享Session* @param tos              收件人列表* @param ccs              抄送人列表,可以为null或空* @param bccs             密送人列表,可以为null或空* @param subject          标题* @param content          正文* @param imageMap         图片与占位符,占位符格式为cid:${cid}* @param isHtml           是否为HTML格式* @param files            附件列表* @return message-id* @since 4.6.3*/private static String send(MailAccount mailAccount, boolean useGlobalSession, Collection<String> tos, Collection<String> ccs, Collection<String> bccs, String subject, String content,Map<String, InputStream> imageMap, boolean isHtml, File... files) {final Mail mail = Mail.create(mailAccount).setUseGlobalSession(useGlobalSession);// 可选抄送人if (CollUtil.isNotEmpty(ccs)) {mail.setCcs(ccs.toArray(new String[0]));}// 可选密送人if (CollUtil.isNotEmpty(bccs)) {mail.setBccs(bccs.toArray(new String[0]));}mail.setTos(tos.toArray(new String[0]));mail.setTitle(subject);mail.setContent(content);mail.setHtml(isHtml);mail.setFiles(files);// 图片if (MapUtil.isNotEmpty(imageMap)) {for (Map.Entry<String, InputStream> entry : imageMap.entrySet()) {mail.addImage(entry.getKey(), entry.getValue());// 关闭流IoUtil.close(entry.getValue());}}return mail.send();}/*** 将多个联系人转为列表,分隔符为逗号或者分号** @param addresses 多个联系人,如果为空返回null* @return 联系人列表*/private static List<String> splitAddress(String addresses) {if (StrUtil.isBlank(addresses)) {return null;}List<String> result;if (StrUtil.contains(addresses, CharUtil.COMMA)) {result = StrUtil.splitTrim(addresses, CharUtil.COMMA);} else if (StrUtil.contains(addresses, ';')) {result = StrUtil.splitTrim(addresses, ';');} else {result = CollUtil.newArrayList(addresses);}return result;}
}

邮件登录

引入依赖

mingyue-push 引入 mingyue-common-email

<!-- 邮件工具 -->
<dependency><groupId>com.csp.mingyue</groupId><artifactId>mingyue-common-email</artifactId>
</dependency>

新建邮箱验证码接口

@Slf4j
@Tag(name = "邮箱服务模块")
@Validated
@RestController
@RequestMapping("email")
@RequiredArgsConstructor
public class EmailController {private final EmailProperties emailProperties;/*** 邮箱验证码** @param email 邮箱*/@GetMapping("/code")@Operation(summary = "邮箱验证码", parameters = { @Parameter(name = "email", description = "邮箱", required = true) })public R<Void> emailCode(@Valid @NotBlank(message = "邮箱不能为空") String email) {if (!emailProperties.getEnabled()) {return R.fail("当前系统没有开启邮箱功能!");}String key = CacheConstants.CAPTCHA_CODE_KEY + email;String code = RandomUtil.randomNumbers(4);RedisUtils.setCacheObject(key, code, Duration.ofMinutes(Constants.CAPTCHA_EXPIRATION));try {EmailUtils.sendText(email, "登录验证码", "您本次验证码为:" + code + ",有效性为" + Constants.CAPTCHA_EXPIRATION + "分钟,请尽快填写。");} catch (Exception e) {log.error("验证码邮箱发送异常 => {}", e.getMessage());return R.fail(e.getMessage());}return R.ok();}}

Nacos 放行接口

# 安全配置
security:# 不校验白名单ignore:whites:# 放行邮箱验证码- /push/email/code

mingyue-push-biz.yml

email:enabled: falsehost: smtp.qq.comport: 465# 是否需要用户名密码验证auth: true# 发送方,遵循RFC-822标准from: xxxxxx@qq.com# 用户名(注意:如果使用foxmail邮箱,此处user为qq号)user: xxxxxx@qq.com# 密码(注意,某些邮箱需要为SMTP服务单独设置密码,详情查看相关帮助)pass: xxxxxx# 使用 STARTTLS安全连接,STARTTLS是对纯文本通信协议的扩展。starttlsEnable: true# 使用SSL安全连接sslEnable: true# SMTP超时时长,单位毫秒,缺省值不超时timeout: 0# Socket连接超时值,单位毫秒,缺省值不超时connectionTimeout: 0

启动项目发送测试

curl -X 'GET' \'http://mingyue-gateway:9100/push/email/code?email=xxx@qq.com' \-H 'accept: */*'

推送邮箱收到以下信息即可

您本次验证码为:6508,有效性为2分钟,请尽快填写。

短信登录

邮箱登录接口

/*** 邮箱登录*/
@PostMapping("/emailLogin")
@Operation(summary = "邮箱登录")
public R<String> emailLogin(@RequestBody @Valid EmailLoginDto dto) {log.info("------- 进入【邮箱登录】请求: " + SaHolder.getRequest().getUrl());// 用户登录SaTokenInfo login = sysLoginService.emailLogin(dto);if (Objects.isNull(login)) {return R.fail("登录失败");}return R.ok("登录成功", login.getTokenValue());
}

短信登录逻辑处理

public SaTokenInfo emailLogin(EmailLoginDto dto) {// 远程调用用户服务R<LoginUser> userInfoResp = remoteUserService.userInfoByEmail(dto.getEmail());if (userInfoResp.getCode() == Constants.FAIL) {throw new UserException(userInfoResp.getMsg());}// 校验验证码是否正确if (!checkSmsOrEmailCode(dto.getEmail(), dto.getEmailCode())) {throw new UserException("验证码错误");}LoginUser userInfo = userInfoResp.getData();if (dto.getEmail().equals(userInfo.getEmail())) {// 第1步,先登录上LoginHelper.login(userInfo);// 第2步,获取 Token 相关参数SaTokenInfo tokenInfo = StpUtil.getTokenInfo();return tokenInfo;}return null;
}

mingyue-ui 添加 email 模块

<el-tab-pane :label="$t('message.label.two3')" name="email"><Email @signInSuccess="signInSuccess"/>
</el-tab-pane>

启动测试

邮件登录需要启动 MingYueGatewayApplication 网关服务MingYueAuthApplication 认证服务MingYueSystemApplication 系统服务MingYuePushApplication 推送服务 以及 mingyue-ui

image-20230904144333676

小结

邮箱登录的功能也加上了,当然不止可以通过邮件发送验证码呦,自己去拓展一下吧,具体看一下 EmailUtils 邮件工具类。

接下来想给 mingyue-ui 写一个增删改查的前后端交互示例,就编写用户管理吧!

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

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

相关文章

免费小程序商城搭建之b2b2c o2o 多商家入驻商城 直播带货商城 电子商务b2b2c o2o 多商家入驻商城 直播带货商城 电子商务

1. 涉及平台 平台管理、商家端&#xff08;PC端、手机端&#xff09;、买家平台&#xff08;H5/公众号、小程序、APP端&#xff08;IOS/Android&#xff09;、微服务平台&#xff08;业务服务&#xff09; 2. 核心架构 Spring Cloud、Spring Boot、Mybatis、Redis 3. 前端框架…

Android发布依赖到 Jitpack

前言 我们在日常开发中&#xff0c;经常会用到第三方开源的库文件&#xff0c;有的来自JCenter&#xff0c;Maven Central&#xff0c;google等。但是随着JCenter的弃用&#xff0c;现在用的最多的还是Maven Central&#xff0c;google。今天我们就自己亲自发布一个依赖。 现…

IDEA新建的Moudle失效显示为灰色

现象&#xff1a;IDEA新建的Moudle失效显示为灰色&#xff01;&#xff01;&#xff01; 解决方案&#xff1a; 1. 右键点击父模块&#xff0c;选择Open Moudle Settings&#xff1a; 2. 点击加号&#xff0c;选择Import Moudle - 导入模块&#xff1a; 3. 找到对应模块的po…

Fultter学习日志(2)-构建第一个flutter应用

依照上一篇中我们新建的flutter应用 让我们更改pubspec.yaml中的内容为 name: namer_app description: A new Flutter project.publish_to: none # Remove this line if you wish to publish to pub.devversion: 0.0.11environment:sdk: >2.19.4 <4.0.0dependencies:fl…

SQL 注入漏洞攻击

文章目录 1. 介绍2. 无密码登录3. 无用户名无密码登录4. 合并表获取用户名密码 1. 介绍 假设你用自己的用户名和密码登录了一个付费网站&#xff0c;网站服务器就会查询一下你是不是 VIP 用户&#xff0c;而用户数据都是放在数据库中的&#xff0c;服务器通常都会向数据库进行查…

最新IDE流行度最新排名(每月更新)

2023年09月IDE流行度最新排名 顶级IDE排名是通过分析在谷歌上搜索IDE下载页面的频率而创建的 一个IDE被搜索的次数越多&#xff0c;这个IDE就被认为越受欢迎。原始数据来自谷歌Trends 如果您相信集体智慧&#xff0c;Top IDE索引可以帮助您决定在软件开发项目中使用哪个IDE …

【初阶C语言】操作符2---表达式求值

前言&#xff1a;本节重点介绍操作符的使用&#xff0c;如&#xff0c;优先级高低、类型转换等 一、逻辑操作符 前言&#xff1a;逻辑操作符包括逻辑与&#xff08;&&&#xff09;和逻辑或&#xff08;||&#xff09;&#xff0c;操作对象&#xff1a;两个 1.逻辑与&…

一文了解数据科学Notebook

编者按&#xff1a; 主要介绍什么是Notebook&#xff0c;Notebook在数据科学领域的应用的重要性与优势&#xff0c;以及数据科学家/算法团队在选择Notebook时需考虑哪些关键因素。同时&#xff0c;基于Notebook的筛选考量维度&#xff0c;对常见的Notebook进初步对比分析&#…

2023-9-11 台阶-Nim游戏

题目链接&#xff1a;台阶-Nim游戏 #include <iostream> #include <algorithm>using namespace std;int main() {int n;cin >> n;int res 0;for(int i 1;i < n; i){int x;cin >> x;if(i % 2) res ^ x; }if(res) cout << "Yes" &l…

MyBatis-Plus深入 —— 条件构造器与插件管理

前言 在前面的文章中&#xff0c;荔枝梳理了一个MyBatis-Plus的基本使用、配置和通用Service接口&#xff0c;我们发现在MyBatis-Plus的辅助增强下我们不再需要通过配置xml文件中的sql语句来实现基本的sql操作了&#xff0c;不愧是最佳搭档&#xff01;在这篇文章中&#xff0c…

Linux:工具(vim,gcc/g++,make/Makefile,yum,git,gdb)

目录 ---工具功能 1. vim 1.1 vim的模式 1.2 vim常见指令 2. gcc/g 2.1 预备知识 2.2 gcc的使用 3.make,Makefile make.Makefile的使用 4.yum --yum三板斧 5.git --git三板斧 --Linux下提交代码到远程仓库 6.gdb 6.1 gdb的常用指令 学习目标&#xff1a; 1.知道…

[构建自己的 Vue 组件库] 小尾巴 UI 组件库

文章归档于&#xff1a;https://www.yuque.com/u27599042/row3c6 组件库地址 npm&#xff1a;https://www.npmjs.com/package/xwb-ui?activeTabreadme小尾巴 UI 组件库源码 gitee&#xff1a;https://gitee.com/tongchaowei/xwb-ui小尾巴 UI 组件库测试代码 gitee&#xff1a…

2023年世界机器人大会回顾

1、前记&#xff1a; 本次记录是我自己去世界机器人博览会参观的一些感受&#xff0c;所有回顾为个人感兴趣部分的机器人产品分享。整个参观下来最大的感受就是科学技术、特别是机器人技术和人工智能毫无疑问地、广泛的应用在我们日常生活的方方面面&#xff0c;在安全巡检、特…

Vue 报错error:0308010C:digital envelope routines::unsupported 解决方案(三种)

新换的电脑&#xff0c;系统装的win11&#xff0c;node也是18的版本。 跑了一下老项目&#xff0c;我用的是HbuilderX&#xff0c;点击运行和发行时&#xff0c;都会报错&#xff1a; Error: error:0308010C:digital envelope routines::unsupported 出现这个错误是因为 node.j…

数学建模B多波束测线问题B

数学建模多波束测线问题 完整思路和代码请私信~~~~ 1.问题重述&#xff1a; 单波束测深是一种利用声波在水中传播的技术来测量水深的方法。它通过测量从船上发送声波到声波返回所用的时间来计算水深。然而&#xff0c;由于它是在单一点上连续测量的&#xff0c;因此数据在航…

从 算力云 零开始部署ChatGLM2-6B 教程

硬件最低需求&#xff0c;显存13G以上 基本环境&#xff1a; 1.autodl-tmp 目录下 git clone https://github.com/THUDM/ChatGLM2-6B.git然后使用 pip 安装依赖&#xff1a; pip install -r requirements.txtpip 使用pip 阿里的 再执行git clone之前&#xff0c;要先在命令行…

【笔试强训选择题】Day40.习题(错题)解析

作者简介&#xff1a;大家好&#xff0c;我是未央&#xff1b; 博客首页&#xff1a;未央.303 系列专栏&#xff1a;笔试强训选择题 每日一句&#xff1a;人的一生&#xff0c;可以有所作为的时机只有一次&#xff0c;那就是现在&#xff01;&#xff01;&#xff01; 文章目录…

Unity Animation、Animator 的使用(超详细)

文章目录 1. 添加动画2. Animation2.1 制作界面2.2 制作好的 Animation 动画2.3 添加和使用事件 3. Animator3.1 制作界面3.2 一些参数解释3.3 动画参数 4. Animator中相关类、属性、API4.1 类4.2 属性4.3 API4.4 几个关键方法 5. 动画播放和暂停控制 1. 添加动画 选中待提添加…

【赠书活动】考研备考书单推荐

&#x1f449;博__主&#x1f448;&#xff1a;米码收割机 &#x1f449;技__能&#x1f448;&#xff1a;C/Python语言 &#x1f449;公众号&#x1f448;&#xff1a;测试开发自动化【获取源码商业合作】 &#x1f449;荣__誉&#x1f448;&#xff1a;阿里云博客专家博主、5…

javaweb04-vue基础

话不多说&#xff0c;参考官网地址Vue官网集成Vue应用。 一、Vue快速入门 &#xff08;1&#xff09;新建HTML页面&#xff0c;引入Vue.js 我这里用的是CDN方式 <script src"https://unpkg.com/vue3/dist/vue.global.js"></script> &#xff08;2&am…