微信公众号打通与登录的实现

news2024/11/26 1:22:44

今天实现一下与微信公众号进行对接,通过扫描二维码的方式来进行注册与登录,获取用户的微信唯一标识作为用户的username,下面我们开始编写。

骨架建立:

建包:

第一步还是先将骨架建好,与网关骨架差不多,我们需要的东西不多,如图:

controller:接收微信请求

handler:进行微信信息处理

redis:redis的工具类和配置文件

utils:一些工具类

依赖引入:

本次依赖如下:每个依赖我都写好注释了

    <dependencies>
        <!-- 启动依赖-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
            <version>2.4.2</version>
            <exclusions>
                <exclusion>
                    <artifactId>spring-boot-starter-logging</artifactId>
                    <groupId>org.springframework.boot</groupId>
                </exclusion>
            </exclusions>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <version>1.18.22</version>
        </dependency>
        <!-- 日志依赖-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-log4j2</artifactId>
            <version>2.4.2</version>
        </dependency>
        <!-- xml解析器-->
        <dependency>
            <groupId>org.dom4j</groupId>
            <artifactId>dom4j</artifactId>
            <version>2.1.3</version>
        </dependency>
        <!-- xml和Java对象的序列化-->
        <dependency>
            <groupId>com.thoughtworks.xstream</groupId>
            <artifactId>xstream</artifactId>
            <version>1.4.18</version>
        </dependency>
        <!-- json序列化-->
        <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-core</artifactId>
            <version>2.12.7</version>
        </dependency>
        <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-databind</artifactId>
            <version>2.12.7</version>
        </dependency>
        <!-- json序列化工具-->
        <dependency>
            <groupId>com.google.code.gson</groupId>
            <artifactId>gson</artifactId>
            <version>2.8.6</version>
        </dependency>
        <!-- 操作Redis的依赖库-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-redis</artifactId>
            <version>2.4.2</version>
        </dependency>
        <!-- 对象池-->
        <dependency>
            <groupId>org.apache.commons</groupId>
            <artifactId>commons-pool2</artifactId>
            <version>2.9.0</version>
        </dependency>
    </dependencies>

配置文件:

没什么好说的,不懂大家问问ai,很简单

server:
  port: 1500

spring:
  # redis配置
  redis:
    # Redis数据库索引(默认为0)
    database: 1
    # Redis服务器地址
    host: 127.0.0.1
    # Redis服务器连接端口
    port: 6379
    # Redis服务器连接密码(默认为空)
    password:
    # 连接超时时间
    timeout: 2s
    lettuce:
      pool:
        # 连接池最大连接数
        max-active: 200
        # 连接池最大阻塞等待时间(使用负值表示没有限制)
        max-wait: -1ms
        # 连接池中的最大空闲连接
        max-idle: 10
        # 连接池中的最小空闲连接
        min-idle: 0


 微信打通初测试:

阅读文档:

接入概述 | 微信开放文档

第一步: 

根据文档的要求,我们第一步需要配置服务器。点击箭头位置进入接口调试工具

 NATAPP-内网穿透 基于ngrok的国内高速内网映射工具

这是我使用的内网穿透工具,大家可以自行选择,因为微信公众号的对接需要443端口,我们需要内网穿透 。进入后购买免费的就可以了

修改你的地址和端口,端口为你服务的端口号 ,并保存你的authtoken

链接:https://pan.baidu.com/s/1Sr13Bea82z4oO_AbpDrKDA?pwd=1111 
提取码:1111

 下载它的客户端,解压后是一个exe文件,我们双击打开,输入命令。authtoken换成你自己的

start natapp.exe -authtoken (authtoken)

 成功会会出现以下界面,我们将Forwarding后的地址复制到测试号的地址中

 如下图所示:

第二步:

阅读文档发现我们需要一个方法接收微信服务器的访问,微信服务器会携带四个参数,我们需要将刚刚定义的token,与其携带的timestamo,nonce进行字典排序,并将其进行sha1加密,将加密后的字符串与signature参数进行比较,如果成功了,就证明这是来自微信的请求

下面我们在controller中实现这个逻辑,sha1加密是一个固定的写法,我这里不去细讲了,大家知道它进行了一次加密就好,在结尾我粘贴出来,文章里我不粘贴工具类了

CallBackController:
package com.yizhiliulianta.wx.controller;

import com.yizhiliulianta.wx.utils.SHA1;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;

@RestController
@Slf4j
public class CallBackController {

    private static final String token = "yizhiliulianta";


    @GetMapping("callback")
    public String callback(@RequestParam("signature") String signature,
                           @RequestParam("timestamp") String timestamp,
                           @RequestParam("nonce") String nonce,
                           @RequestParam("echostr") String echostr) {
        log.info("get验签请求参数:signature:{},timestamp:{},nonce:{},echostr:{}",
                signature, timestamp, nonce, echostr);
        String shaStr = SHA1.getSHA1(token, timestamp, nonce, "");
        if (signature.equals(shaStr)) {
            return echostr;
        }
        return "unknown";
    }
}

大家可以阅读一下代码,和刚刚的过程是一样的,接收参数,加密,判断是否合法,最后返回,如果不合法返回unknown。

下面我们用aippost测试一下该方法是否成功:

成功了!我们继续

第三步:

我们将刚刚的方法修改为post请求,新增一个就好,并多添加一个参数为requestBody,因为我们在进行扫码成功时,微信会给我们返回一个请求体,为xml格式,并且我们需要将我们的返回格式也设置为xml格式,而且我们需要将echostr变为不必须的,不然会报错,代码如下:

    @PostMapping(value = "callback", produces = "application/xml;charset=UTF-8")
    public String callback(
            @RequestBody String requestBody,
            @RequestParam("signature") String signature,
            @RequestParam("timestamp") String timestamp,
            @RequestParam("nonce") String nonce,
            @RequestParam(value = "echostr",required = false) String echostr) {
        log.info("接收到微信的请求:requestBody:{},signature:{},timestamp:{},nonce:{},echostr:{}"
                ,requestBody,signature,timestamp,nonce,echostr);
        return "unknown";
    }

下面我们测试一下,首先,我们在测试号的网页下,会有一个二维码,这里我就不截图了,当大家配置好网址,token和端口后,我们扫当前的二维码,并且关注,我们在后台会收到如下信息:

 我来分析一下都是什么,大家也可以看文档,都是有介绍的

ToUserName开发者微信号(本次服务的标识)
FromUserName发送方账号(一个OpenID)(也就是说,谁关注了我们)
CreateTime消息创建时间
MsgType消息类型
Event事件类型,subscribe(订阅)、unsubscribe(取消订阅)(证明该用户进行了关注动作)

下面当我们给该公众号发送一则消息,响应的内容如下:

 唯一改变的是消息的类型,变成了text,Event变成了Content,并包含着本次发的内容:你好

 第四步:

当我们接收到用户的消息,我们需要对其进行回复,因为我们接收的格式为xml,所以我们返回也需要xml。这里我们看一下文档,发现需要返回的内容如下:

 

我们需要知道发送方是谁,接收方是谁,因为我们给用户返回,所以发送方变成后台程序,而接收方变成了用户。

我们清楚了这个逻辑后,只需要将用户关注后,微信给我们推送的xml信息解析然后编写一个xml返回就好了

这里我使用一个工具类,还是贴在最后,重要的是逻辑和如何对接,代码如下:

    @PostMapping(value = "callback", produces = "application/xml;charset=UTF-8")
    public String callback(
            @RequestBody String requestBody,
            @RequestParam("signature") String signature,
            @RequestParam("timestamp") String timestamp,
            @RequestParam("nonce") String nonce,
            @RequestParam(value = "echostr",required = false) String echostr) {
        log.info("接收到微信的请求:requestBody:{},signature:{},timestamp:{},nonce:{},echostr:{}"
                ,requestBody,signature,timestamp,nonce,echostr);
        Map<String, String> msgMap = MessageUtil.parseXml(requestBody);
        String toUserName = msgMap.get("ToUserName");
        String fromUserName = msgMap.get("FromUserName");

        String mag = "<xml>\n" +
                "  <ToUserName><![CDATA["+fromUserName+"]]></ToUserName>\n" +
                "  <FromUserName><![CDATA["+toUserName+"]]></FromUserName>\n" +
                "  <CreateTime>12345678</CreateTime>\n" +
                "  <MsgType><![CDATA[text]]></MsgType>\n" +
                "  <Content><![CDATA[你好,我是一支榴莲挞]]></Content>\n" +
                "</xml>\n";

        return mag;
    }

我们再来测试一下,给公众号发送一个消息,如图:

成功!到这里对接就已经成功了

思考: 

我们到这里打通了微信的对接,我们对接后需要什么?

一:用户关注后,我们需要给其反馈

二:我们需要获取用户的唯一标识作为用户名

三:我们需要根据用户不同的动作和发的信息做出不同的回应

我们目前实现了一,二也很简单,下面我们需要实现三,我的想法是用户发送验证码,我们给其一个随机的4位数,并将用户的唯一标识与验证码放入到redis里,然后用户进入auth服务,实现登录,需要输入刚刚我们发送的验证码,接着,我们在redis里根据用户发送的验证码进行查询,如果查询到了,登录成功,否则登录失败,这样我们就实现了整个登录的实现。

微信打通功能优化与完善:

本次我们使用工厂模式来进行解耦:

因为用户至少有两个动作,一个是关注的动作,一个是发送消息的动作。

枚举:

首先我们定义动作的枚举:

一个是用户关注事件,一个是消息事件

package com.yizhiliulianta.wx.handler;

public enum WxChatMsgTypeEnum {

    SUBSCRIPE("event.subscribe","用户关注事件"),
    TEXT_MSG("text","接收用户文本消息");

    private String msgType;

    private String desc;

    WxChatMsgTypeEnum(String msgType, String desc) {
        this.msgType = msgType;
        this.desc = desc;
    }

    public static WxChatMsgTypeEnum getByMsgType(String msgType){
        for (WxChatMsgTypeEnum wxChatMsgTypeEnum : WxChatMsgTypeEnum.values()){
            if (wxChatMsgTypeEnum.msgType.equals(msgType)){
                return wxChatMsgTypeEnum;
            }
        }
        return null;
    }
}

 工厂设计:

接口规范:

首先我们需要一个接口规范,告诉每个实现类,需要执行什么方法

一:我需要知道你是干什么的,哪个实现类

二:处理微信的请求

代码如下:

package com.yizhiliulianta.wx.handler;

import java.util.Map;

public interface WxChatMsgHandler {

    WxChatMsgTypeEnum getMsgType();

    String dealMsg(Map<String, String> messageMap);


}

工厂管理: 

package com.yizhiliulianta.wx.handler;

import org.springframework.beans.factory.InitializingBean;
import org.springframework.stereotype.Component;

import javax.annotation.Resource;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

@Component
public class WxChatMsgFactory implements InitializingBean {

    @Resource
    private List<WxChatMsgHandler> wxChatMsgHandlerList;

    private final Map<WxChatMsgTypeEnum,WxChatMsgHandler> handlerMap = new HashMap<>();

    public WxChatMsgHandler getHandlerByMsgType(String msgType){
        WxChatMsgTypeEnum msgTypeEnum = WxChatMsgTypeEnum.getByMsgType(msgType);
        return handlerMap.get(msgTypeEnum);
    }

    @Override
    public void afterPropertiesSet() {
        for (WxChatMsgHandler wxChatMsgHandler : wxChatMsgHandlerList) {
            handlerMap.put(wxChatMsgHandler.getMsgType(),wxChatMsgHandler);
        }
    }
}

 首先我们需要一个工厂管理类,通过这个类来获取和管理每个实现接口的实现类。

首先我们通过Spring的自动注入,列出实现接口的所有类,并注入到一个集合中

我们应该怎么通过对应的枚举,来获取到对应的实现类呢,这里我使用实现InitializingBean接口,重写afterPropertiesSet方法,让我们在bean初始化后,进行处理的一个方法。

首先我们构建一个map集合,键为枚举值,值为实现类

在afterPropertiesSet方法中,循环所有实现类,并将实现类的枚举值,和实现类放入map集合,这样我们就可以通过枚举来获取对应的实现类了,那么getHandlerByMsgType也就理解了,就是通过传入的枚举值获取对应的枚举,并在map表中取到对应的实现类。

关注实现:

package com.yizhiliulianta.wx.handler;

import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Component;

import java.util.Map;

@Component
@Slf4j
public class SubscribeMsgHandler implements WxChatMsgHandler{
    @Override
    public WxChatMsgTypeEnum getMsgType() {
        return WxChatMsgTypeEnum.SUBSCRIPE;
    }

    @Override
    public String dealMsg(Map<String, String> messageMap) {
        log.info("触发用户关注事件");
        String fromUserName = messageMap.get("FromUserName");
        String toUserName = messageMap.get("ToUserName");
        String subscribeContent = "谢谢您的关注,我是一支榴莲挞!";
        String replyContent = "<xml>\n" +
                "  <ToUserName><![CDATA["+fromUserName+"]]></ToUserName>\n" +
                "  <FromUserName><![CDATA["+toUserName+"]]></FromUserName>\n" +
                "  <CreateTime>12345678</CreateTime>\n" +
                "  <MsgType><![CDATA[text]]></MsgType>\n" +
                "  <Content><![CDATA["+subscribeContent+"]]></Content>\n" +
                "</xml>";
        return replyContent;
    }
}

这个代码应该不难理解,实现刚刚的接口,返回对应的枚举和实现处理请求。

处理请求的方式和刚刚的是一样的,不再去详细说了,相信大家可以看懂

验证码实现:

package com.yizhiliulianta.wx.handler;

import com.yizhiliulianta.wx.redis.RedisUtil;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Component;

import javax.annotation.Resource;
import java.util.Map;
import java.util.Random;
import java.util.concurrent.TimeUnit;

@Component
@Slf4j
public class ReceiveTextMsgHandler implements WxChatMsgHandler {

    private static final String KEY_WORD = "验证码";

    private static final String LOGIN_PREFIX = "loginCode";

    @Resource
    private RedisUtil redisUtil;

    @Override
    public WxChatMsgTypeEnum getMsgType() {
        return WxChatMsgTypeEnum.TEXT_MSG;
    }

    @Override
    public String dealMsg(Map<String, String> messageMap) {
        log.info("接收到文本消息事件");
        String content = messageMap.get("Content");
        if (!KEY_WORD.equals(content)) {
            return "";
        }
        //发送方 --》 接收方
        String fromUserName = messageMap.get("FromUserName");
        //接收方 --》 发送方
        String toUserName = messageMap.get("ToUserName");
        //随机四位数验证码
        Random r = new Random();
        int one = r.nextInt(10);  //产生0到9的随机数
        int two = r.nextInt(10);
        int three = r.nextInt(10);
        int four = r.nextInt(10);
        String num = "" + one + two + three + four;//将4个随机数组成一个字符串
        //存入redis
        String numKey = redisUtil.buildKey(LOGIN_PREFIX,num);
        redisUtil.setNx(numKey, fromUserName, 5L, TimeUnit.MINUTES);
        //返回内容
        String numContent = "您当前的验证码是:" + num + "(5分钟内有效)";
        String replyContent = "<xml>\n" +
                "  <ToUserName><![CDATA[" + fromUserName + "]]></ToUserName>\n" +
                "  <FromUserName><![CDATA[" + toUserName + "]]></FromUserName>\n" +
                "  <CreateTime>12345678</CreateTime>\n" +
                "  <MsgType><![CDATA[text]]></MsgType>\n" +
                "  <Content><![CDATA[" + numContent + "]]></Content>\n" +
                "</xml>";
        return replyContent;
    }
}

这部分代码就是实现了给用户返回一个四位数的验证码,并将用户的id和验证码存储到redis的一个操作,首先取出content也就是用户发送的信息,如果是验证码这个信息就继续执行,然后生成一个4位随机数

定义redis的主键,与随机数进行构建一个放入redis的键,然后将用户唯一标识作为值放入redis,然后将处理的内容返回 

controller使用:

那该如何在controller中使用呢,重写后的代码如下:

    @PostMapping(value = "callback", produces = "application/xml;charset=UTF-8")
    public String callback(
            @RequestBody String requestBody,
            @RequestParam("signature") String signature,
            @RequestParam("timestamp") String timestamp,
            @RequestParam("nonce") String nonce,
            @RequestParam(value = "echostr",required = false) String echostr) {
        log.info("接收到微信的请求:requestBody:{},signature:{},timestamp:{},nonce:{},echostr:{}"
                ,requestBody,signature,timestamp,nonce,echostr);
        Map<String, String> msgMap = MessageUtil.parseXml(requestBody);
        String msgType = msgMap.get("MsgType");
        String event = msgMap.get("Event");
        log.info("msgType:{},event:{}",msgType,event);

        StringBuilder sb = new StringBuilder();
        sb.append(msgType);
        if (event!=null){
            sb.append(".");
            sb.append(event);
        }
        String msgTypeKey = sb.toString();

        WxChatMsgHandler wxChatMsgHandler = wxChatMsgFactory.getHandlerByMsgType(msgTypeKey);
        
        if (Objects.isNull(wxChatMsgHandler)){
            return "unknown";
        }
        
        String replyContent = wxChatMsgHandler.dealMsg(msgMap);
        log.info("replyContent:{}",replyContent);
        return replyContent;
    }

前面没有什么变化,但是我们需要先将微信的xml解析,通过分析msgtype来判断是什么动作,这里我使用了一个字符串的拼接,因为如果是信息而不是关注动作,那么event是为空的,所以经过判断我们就可以直接将msgTypeKey传入,获取到对应的实现类,然后调用其dealMsg,也就是处理方法,获取返回值,最后将其返回。

测试:

下面我们来测试一下整个流程

重启项目后,我们给公众号发送,验证码。

 成功接收!然后我们去redis里看看

 成功啦!到此为止,我们的微信服务彻底打通,并且与redis进行了集成

登录实现:

auth服务:

下面我们进行auth服务的登录验证逻辑,首先我们进行登录需要用户给我们传一个验证码,然后我们判断成功后,返回给其一个token,代码如下:

auth的controller层

    // 会话登录接口
    @RequestMapping("doLogin")
    public Result<SaTokenInfo> doLogin(@RequestParam("validCode") String validCode) {
        try {
            Preconditions.checkArgument(!StringUtils.isBlank(validCode),"验证码不能为空");
            SaTokenInfo tokenInfo = authUserDomainService.doLogin(validCode);
            return Result.ok(tokenInfo);
        }catch (Exception e){
            log.error("UserController.doLogin.error:{}", e.getMessage(), e);
            return Result.fail("用户登录失败");
        }
    }

auth的domain层:

下面就是执行doLogin的逻辑实现,代码如下:

    @Override
    public SaTokenInfo doLogin(String validCode) {
        String loginKey = redisUtil.buildKey(LOGIN_PREFIX, validCode);
        String openId = redisUtil.get(loginKey);
        if (StringUtils.isBlank(openId)){
            return null;
        }
        AuthUserBO authUserBO = new AuthUserBO();
        authUserBO.setUserName(openId);
        this.register(authUserBO);
        StpUtil.login(openId);
        SaTokenInfo tokenInfo = StpUtil.getTokenInfo();
        return tokenInfo;
    }

大家记得把LOGIN_PREFIX这个redis的键复制过来。

首先还是先把redis里的主键构建处理,然后通过封装好的工具类,获取该键的值,也就是用户的唯一标识,我们新建一个authUserBO,将唯一标识作为用户名放入,并调用register也就是注册方法,将其插入数据库中

然后使用satoken的login方法,进行登录,并获取其对应的token进行返回。

这里我对register注册方法进行完善:代码如下

@Override
    public Boolean register(AuthUserBO authUserBO) {
        AuthUser existAuthUser = new AuthUser();
        existAuthUser.setUserName(authUserBO.getUserName());
        //校验用户是否存在
        List<AuthUser> existUser = authUserService.queryByCondition(existAuthUser);
        if (existUser.size() > 0){
            return true;
        }
        AuthUser authUser = AuthUserBOConverter.INSTANCE.convertBOToAuth(authUserBO);
        if (StringUtils.isNotBlank(authUser.getPassword())){
            authUser.setPassword(SaSecureUtil.md5BySalt(authUser.getPassword(),salt));
        }
        //插入用户
        AuthUser user = authUserService.insert(authUser);
        AuthPermission authPermission = new AuthPermission();
        //获取对应权限信息
        authPermission.setPermissionKey(AuthConstant.PERMISSION_KEY);
        AuthPermission permissionResult = authPermissionService.queryByCondition(authPermission);
        //用户与权限做关联
        Long userId = user.getId();
        Long permissionResultId = permissionResult.getId();
        AuthUserPermission authUserPermission = new AuthUserPermission();
        authUserPermission.setUserId(userId);
        authUserPermission.setPermissionId(permissionResultId);
        authUserPermissionService.insert(authUserPermission);
        //放入redis
        String buildKey = redisUtil.buildKey(userPermissionPrefix, authUser.getUserName());
        List<AuthPermission> authPermissionList = new LinkedList<>();
        authPermissionList.add(permissionResult);
        redisUtil.set(buildKey,new Gson().toJson(authPermissionList));

        return true;
    }

就是在开头新增两个判断,首先在数据库查询有没有该用户,如果有的话直接返回,如果没有再进行注册,并对密码进行判空处理,因为我们使用的是微信登录,所以是没有密码的,不然会出现空指针问题

测试:

那么到这里,整个登录的逻辑完成了,下面让我们测试一下,重启服务并请出apipost

看一眼nacos,ok,都正常

获取一个验证码 

 这里还是通过网关来进行分配登录,将验证码作为参数传入,成功了!我们可以看到下方成功返回了token

 看一下redis 

 

成功!

这里有一个点需要注意,我们的redis配置是不同的索引,所以我们需要将wx服务的redis配置文件 和auth服务的redis配置文件索引相同,不然是无法获取的

 

到这里我们的登录服务彻底结束了,下一篇是微服务直接的调用,我们该如何在其他服务里知道,本次登录的用户是谁呢?这需要使用feign了

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

树莓派4B_OpenCv学习笔记9:图片的腐蚀与膨胀

今日继续学习树莓派4B 4G&#xff1a;&#xff08;Raspberry Pi&#xff0c;简称RPi或RasPi&#xff09; 本人所用树莓派4B 装载的系统与版本如下: 版本可用命令 (lsb_release -a) 查询: Opencv 版本是4.5.1&#xff1a; 图像的膨胀与腐蚀一般用于灰度图或者二值图,今日便来学习…

基于python-CNN卷积网络训练识别牛油果和猕猴桃-含数据集+pyqt界面

代码下载地址&#xff1a; https://download.csdn.net/download/qq_34904125/89383066 本代码是基于python pytorch环境安装的。 下载本代码后&#xff0c;有个requirement.txt文本&#xff0c;里面介绍了如何安装环境&#xff0c;环境需要自行配置。 或可直接参考下面博文…

目前市面上DIY高端空心耳机壳使用的透明原材料是什么?

目前市面上DIY高端空心耳机壳使用的透明原材料是什么&#xff1f; DIY制作耳机壳的UV树脂胶是一种单组份、通过紫外线光固化的胶粘剂&#xff0c;具有低能量固化、收缩低、发热量低、高透明、耐盐酸、耐黄变好、高硬度、韧性好、成型好等特点。这种胶粘剂非常适合用于制作耳机壳…

Redis 主从集群 哨兵原理

一. Redis 主从集群 1.1 基本概念 主从架构&#xff1a;Redis主从集群采用“一主多从”的架构模式&#xff0c;其中主节点&#xff08;Master&#xff09;负责处理客户端的读写请求&#xff0c;而从节点&#xff08;Slave&#xff09;则负责处理读请求。这种读写分离的设计使…

Pytest 读取excel文件参数化应用

本文是基于Pytest框架&#xff0c;读取excel中的文件&#xff0c;传入页面表单中&#xff0c;并做相应的断言实现。 1、编辑媒体需求 首先明确一下需求&#xff0c;我们需要对媒体的表单数据进行编辑&#xff0c;步骤如下&#xff1a; 具体表单如下图所示 1、登录 2、点击我…

618洗地机全网热门推荐,跟着买错不了

步入酷热夏天&#xff0c;家中的清洁工作也迎来了新的挑战。天气炎热&#xff0c;细菌、异味滋生的困扰让日常打扫变得不再轻松&#xff0c;这时一台高性能的洗地机就成了提升生活品质的必备良品。不同于洗地机的技术与类别繁多&#xff0c;洗地机虽原理不复杂&#xff0c;但在…

堆栈溢出的攻击 -fno-stack-protector stack smash 检测

在程序返回的一条语句堆栈项目处&#xff0c;用新函数的起始地址覆盖&#xff0c;将会跳转到执行新函数。 现在系统对这个行为做了判断&#xff0c;已经无法实施这类攻击或技巧。 1&#xff0c;测试代码 #include <stdio.h> void cc() {printf("I am cc( )\n"…

【AI+编程】工作日常场景随时可以AI编程,记一个问答SQL快速导出数据日常示例

今天有个场景&#xff0c;我们有个老项目&#xff0c;由于历史原因差不多1年多没使用了&#xff0c;恰巧客户紧急情况要使用。因为当时没有需求&#xff0c;所以V1.0上线后 就没做更新。 需求很简单&#xff1a;我们假定 项目问题表、客户表、问题答案表&#xff0c; 实时查询…

VisionOS的未来愿景:苹果VisionPro创业者的愿望清单

随着苹果公司在增强现实(AR)领域的不断探索,VisionPro作为其前沿产品,已经开始展现出改变我们与数字世界互动方式的潜力。作为一名VisionPro创业者,对未来VisionOS的更新充满了期待,并提出了一系列愿望清单,这些愿望不仅代表了个人的需求,也反映了用户社区对苹果AR生态的…

【ETAS CP AUTOSAR基础软件】DET、Bfx、CRC、ComStack、rba_ArxmlGen模块详解

文章包含了AUTOSAR基础软件&#xff08;BSW&#xff09;中DET、Bfx、CRC、ComStack、rba_ArxmlGen模块相关的内容详解。本文从AUTOSAR规范解析&#xff0c;ISOLAR-AB配置以及模块相关代码分析三个维度来帮读者清晰的认识和了解DET、Bfx、CRC、ComStack、rba_ArxmlGen。文中涉及…

logTrick

贴一下灵神的题解里面的解释~ 就是一种优化策略&#xff0c;logtrick class Solution { public:int minimumDifference(vector<int>& nums, int k) {int res 0x3f3f3f3f;int n nums.size();for(int i0;i<n;i){res min(res,abs(nums[i]-k));for(int j i-1;j&g…

2024年6月15日 (周六) 叶子游戏新闻

期刊杂志: 聚合读者、意林、知音、故事会、花火以及国内各大知名报纸电子版&#xff0c;无需付费即可观看各种免费资源 中医自学宝典: 集合了中医医案&#xff0c;医经&#xff0c;方剂 药材知识的app&#xff0c;更方便的免费学习中医知识 《赛博朋克2077》被取消DLC泄露&…

20240615在WIN11下的串口调试助手的下载安装以及使用

20240615在WIN11下的串口调试助手的下载安装以及使用 2024/6/15 18:06 百度&#xff1a;串口调试助手 blob:https://apps.microsoft.com/df934d29-fd7a-4873-bb6b-a4ab5a7934c9 串口调试助手 Installer.exe 收发的LOG&#xff1a; rootok3588:/# ./uart_test /dev/ttyS0 11520…

springboot+vue+mybatis家电系统+PPT+论文+讲解+售后

随着信息互联网购物的飞速发展&#xff0c;一般企业都去创建属于自己的电商平台以及购物管理系统。本文介绍了家电销售系统的开发全过程。通过分析企业对于家电销售系统的需求&#xff0c;创建了一个计算机管理家电销售系统的方案。文章介绍了家电销售系统的系统分析部分&#…

cs与msf权限传递,以及mimikatz抓取明文密码

cs与msf权限传递&#xff0c;以及mimikatz抓取win10明文密码 1、环境准备2、Cobalt Strike ------> MSF2.1 Cobalt Strike拿权限2.2 将CS权限传递给msf 3、MSF ------> Cobalt Strike3.1 msf拿权限3.2 将msf权限传递给CS 4、使用mimikatz抓取明文密码 1、环境准备 攻击&…

【Apache Doris】Compaction 原理 | 实践全析

【Apache Doris】Compaction 原理 | 实践全析 一、Compaction 前文概要二、Compaction 版本策略三、Compaction 类型说明四、Compaction 工程实现五、Compaction 生产实践 作者 &#xff5c; 俞剑波 一、Compaction 前文概要 LSM-Tree 简介 LSM-Tree&#xff08; Log Structu…

Three.js做了一个网页版的我的世界

前言 笔者在前一阵子接触到 Three.js 后, 发现了它能为前端 3D 可视化 / 动画 / 游戏方向带来的无限可能, 正好最近在与朋友重温我的世界, 便有了用 Three.js 来仿制 MineCraft 的想法, 正好也可以通过一个有趣的项目来学习一下前端 3D 领域 介绍 游戏介绍 相信大家对我的世…

Swift开发——循环执行方式

本文将介绍 Swift 语言的循环执行方式 01、循环执行方式 在Swift语言中,主要有两种循环执行控制方式: for-in结构和while结构。while结构又细分为当型while结构和直到型while结构,后者称为repeat-while结构。下面首先介绍for-in结构。 循环控制方式for-in结构可用于区间中的…

推荐这两款非常良心的录屏和文字转语音工具,很是让人心动,不要错过

VPot FREE 吾爱大神制作的文字转音频工具&#xff0c;免费使用。 支持英语、韩语、法语、日语等语言&#xff0c;还是支持男声、女声和儿童声音。 支持将以导入文本的格式转换成音频&#xff0c;并保存为MP3、WAV等常见的音频格式。 VPot FREE提供智能断句的功能&#xff0…

深度学习(八)——神经网络:卷积层

一、卷积层Convolution Layers函数简介 官网网址&#xff1a;torch.nn.functional — PyTorch 2.0 documentation 由于是图像处理&#xff0c;所以主要介绍Conv2d。 class torch.nn.Conv2d(in_channels, out_channels, kernel_size, stride1, padding0, dilation1, groups1, b…