记录debug分析nacos源码

news2024/9/20 7:56:43

nacos源码分析

  • 一、环境构建
  • 二、源码分析
    • 1. 启动类
    • 2. 源码的中的案例
    • 3.. 服务订阅流程梳理
      • 3.1. 从 NamingFactory.createNamingService(properties);说起
      • 3.2. 服务订阅总结
    • 4. 服务注册流程梳理
      • 4.1. 从 naming.registerInstance("nacos.test.3", "11.11.11.11", 8888);说起
      • 4.2. 服务注册流程总结

一、环境构建

源码需要 jdk1.8 进行构建

  1. 代码下载地址:https://github.com/alibaba/nacos/tree/develop
  2. git 下载:git clone https://github.com/alibaba/nacos.git
  3. 找到 console --> src --> main --> resources --> META-INF 下的derby-schema.sql 导入到本地数据库
  4. 修改 console --> src --> main --> resources --> 下的application.properties 配置文件

  1. 先在 idea 安装 Protobuf 插件,安装成功后将consistency工程进行编译 mvn compile
  2. 将所有的工程进行编译
  3. 配置 vm 启动参数 -Dnacos.standalone=true

二、源码分析

客户端与注册中心服务端的交互,主要集中在服务注册、服务下线、服务发现、订阅某个服务,其实使用最多的就是服务注册和服务发现

1. 启动类

启动类可以看到@EnableScheduling 注解,说明有很多东西都是由定时任务完成的

@SpringBootApplication
@ComponentScan(basePackages = "com.alibaba.nacos", excludeFilters = {
        @Filter(type = FilterType.CUSTOM, classes = {NacosTypeExcludeFilter.class}),
        @Filter(type = FilterType.CUSTOM, classes = {TypeExcludeFilter.class}),
        @Filter(type = FilterType.CUSTOM, classes = {AutoConfigurationExcludeFilter.class})})
@ServletComponentScan
@EnableScheduling
public class Nacos {
    
    public static void main(String[] args) {
        SpringApplication.run(Nacos.class, args);
    }
}

2. 源码的中的案例

  1. 找到 nacos-example 工程
  2. 查看 App.class 文件
public class App {
    public static void main(String[] args) throws NacosException {
        Properties properties = new Properties();
        properties.setProperty("serverAddr", "localhost:8848");
        properties.setProperty("namespace", "quickStart");
        NamingService naming = NamingFactory.createNamingService(properties);
        // 服务注册
        naming.registerInstance("nacos.test.3", "11.11.11.11", 8888, "TEST1");
        naming.registerInstance("nacos.test.3", "2.2.2.2", 9999, "DEFAULT");

        // 服务发现
        System.out.println("[Instances after register]  " + naming.getAllInstances("nacos.test.3", Lists.newArrayList("TEST1")));
        System.out.println("[Instances after register]  " + naming.getAllInstances("nacos.test.3", Lists.newArrayList("DEFAULT")));
        
    }
}

  1. 将 main 函数进行修改
    public static void main(String[] args) throws NacosException {
        Properties properties = new Properties();
        // nacos的ip地址
        properties.setProperty("serverAddr", "localhost:8848");
        // 命名空间
        properties.setProperty("namespace", "Test_study");
        // 用户名
        properties.setProperty("username", "nacos");
        // 密码
        properties.setProperty("password", "nacos");
        // 通过工厂模式根据传入的properties属性创建NamingService实例的方法
        NamingService naming = NamingFactory.createNamingService(properties);
        // 服务注册
        naming.registerInstance("nacos.test.3", "11.11.11.11", 8888);

    }
  1. 查看 NamingExample.class 文件
public class NamingExample {
    
    private static final String INSTANCE_SERVICE_NAME = "nacos.test.3";
    
    private static final String INSTANCE_IP = "11.11.11.11";
    
    private static final int INSTANCE_PORT = 8888;
    
    private static final String INSTANCE_CLUSTER_NAME = "TEST1";
    
    public static void main(String[] args) throws NacosException, InterruptedException {
        
        Properties properties = new Properties();
        properties.setProperty("serverAddr", System.getProperty("serverAddr", "localhost"));
        properties.setProperty("namespace", System.getProperty("namespace", "public"));
        
        NamingService naming = NamingFactory.createNamingService(properties);
        // 服务注册
        naming.registerInstance(INSTANCE_SERVICE_NAME, INSTANCE_IP, INSTANCE_PORT, INSTANCE_CLUSTER_NAME);
        // 服务订阅
        Executor executor = new ThreadPoolExecutor(1, 1, 0L, TimeUnit.MILLISECONDS, new LinkedBlockingQueue<Runnable>(),
                runnable -> {
                    Thread thread = new Thread(runnable);
                    thread.setName("test-thread");
                    return thread;
                });

        naming.subscribe(INSTANCE_SERVICE_NAME, new AbstractEventListener() {
        
            //EventListener onEvent is sync to handle, If process too low in onEvent, maybe block other onEvent callback.
            //So you can override getExecutor() to async handle event.
            @Override
            public Executor getExecutor() {
                return executor;
            }
        
            @Override
            public void onEvent(Event event) {
                System.out.println("[serviceName] " + ((NamingEvent) event).getServiceName());
                System.out.println("[instances from event] " + ((NamingEvent) event).getInstances());
            }
        });
        
        Thread.sleep(1000);
    
        System.out.println("[instances after register] " + naming.getAllInstances(INSTANCE_SERVICE_NAME));
        
        Thread.sleep(1000);
        // 服务下线
        naming.deregisterInstance(INSTANCE_SERVICE_NAME, INSTANCE_IP, INSTANCE_PORT, INSTANCE_CLUSTER_NAME);
    
        Thread.sleep(1000);
        
        System.out.println("[instances after deregister] " + naming.getAllInstances(INSTANCE_SERVICE_NAME));
        
        Thread.sleep(1000);
    }
}

3… 服务订阅流程梳理

3.1. 从 NamingFactory.createNamingService(properties);说起

  1. 打断点启动

  1. f7 进入NamingFactory.createNamingService(properties)方法

    public static NamingService createNamingService(Properties properties) throws NacosException {
        try {
            // 通过反射获取NacosNamingService类
            Class<?> driverImplClass = Class.forName("com.alibaba.nacos.client.naming.NacosNamingService");
            // 通过getConstructor()方法获取该类带有Properties参数的构造函数对象
            Constructor constructor = driverImplClass.getConstructor(Properties.class);
            // 通过newInstance()方法使用指定的参数实例化该类的对象
            return (NamingService) constructor.newInstance(properties);
        } catch (Throwable e) {
            throw new NacosException(NacosException.CLIENT_INVALID_PARAM, e);
        }
    }
  1. 进入导航 NacosNamingService 类,可以看到构造方法

  1. 构造方法打断点,重新启动

  1. f7 进入init(properties)方法

 private void init(Properties properties) throws NacosException {
        // 可忽略:异步加载 createEmptyJsonNode()和SpasAdapter.getAk() 组件
        PreInitUtils.asyncPreLoadCostComponent();
        // 可忽略:获取配置Nacos的相关参数如:地址、端口、命名空间等
        final NacosClientProperties nacosClientProperties = NacosClientProperties.PROTOTYPE.derive(properties);
        // 可忽略:打印日志
        NAMING_LOGGER.info(ParamUtil.getInputParameters(nacosClientProperties.asProperties()));
        // 可忽略:检查必要配置的属性
        ValidatorUtils.checkInitParam(nacosClientProperties);
        // 可忽略:获取命名空间
        this.namespace = InitUtils.initNamespaceForNaming(nacosClientProperties);
        // 可忽略:序列化
        InitUtils.initSerialization();
        // 可忽略:初始化上下文
        InitUtils.initWebRootContext(nacosClientProperties);
        // 可忽略:初始化日志
        initLogName(nacosClientProperties);
        // 可忽略:获取一个uuid
        this.notifierEventScope = UUID.randomUUID().toString();
        // 可忽略:给监听对象赋值
        this.changeNotifier = new InstancesChangeNotifier(this.notifierEventScope);
        // 注册一个事件监听器,并且指定该监听器的事件队列容量为16384
        NotifyCenter.registerToPublisher(InstancesChangeEvent.class, 16384);
        // 通知中心注册一个订阅者对象,以便接收通知。当有通知发布时,订阅者会收到通知并执行相应操作
        NotifyCenter.registerSubscriber(changeNotifier);
        this.serviceInfoHolder = new ServiceInfoHolder(namespace, this.notifierEventScope, nacosClientProperties);
        //  -----------------------------主要看这个-----------------------------------
        // Nacos服务器进行通信,实现服务的注册与发现等功能
        this.clientProxy = new NamingClientProxyDelegate(this.namespace, serviceInfoHolder, nacosClientProperties, changeNotifier);
    }
  1. f7 进入 NamingClientProxyDelegate(this.namespace, serviceInfoHolder, nacosClientProperties, changeNotifier);方法

public NamingClientProxyDelegate(String namespace, ServiceInfoHolder serviceInfoHolder,
        NacosClientProperties properties, InstancesChangeNotifier changeNotifier) throws NacosException {
    // 在需要时可以更新服务信息
    this.serviceInfoUpdateService = new ServiceInfoUpdateService(properties, serviceInfoHolder, this,
            changeNotifier);
    // 可以获取服务器列表、监控服务器状态、更新服务器列表等功能
    this.serverListManager = new ServerListManager(properties, namespace);
    this.serviceInfoHolder = serviceInfoHolder;
    // 用于处理安全性相关的操作。SecurityProxy类可能是用于验证用户身份、管理权限等安全功能
    this.securityProxy = new SecurityProxy(this.serverListManager.getServerList(),
            NamingHttpClientManager.getInstance().getNacosRestTemplate());
    // 该函数用于初始化安全代理
    initSecurityProxy(properties);
    // 初始化一个客户端代理对象,用于与命名服务进行交互
    this.httpClientProxy = new NamingHttpClientProxy(namespace, securityProxy, serverListManager, properties);
    this.grpcClientProxy = new NamingGrpcClientProxy(namespace, securityProxy, serverListManager, properties,
            serviceInfoHolder);
}
  1. f7 进入 new ServiceInfoUpdateService(properties, serviceInfoHolder, this, changeNotifier);方法
public ServiceInfoUpdateService(NacosClientProperties properties, ServiceInfoHolder serviceInfoHolder,
        NamingClientProxy namingClientProxy, InstancesChangeNotifier changeNotifier) {
    this.asyncQuerySubscribeService = isAsyncQueryForSubscribeService(properties);
    // 这创建一个定时任务的线程池
    this.executor = new ScheduledThreadPoolExecutor(initPollingThreadCount(properties),
            new NameThreadFactory("com.alibaba.nacos.client.naming.updater"));
    this.serviceInfoHolder = serviceInfoHolder;
    this.namingClientProxy = namingClientProxy;
    this.changeNotifier = changeNotifier;
}
  1. 既然是有线程池,那么当前类肯定有 run()方法,下下找

  1. 主要的代码解释
  • this.serviceKey = ServiceInfo.getKey(groupedServiceName, clusters); -----> 获取服务唯一的 key
  • ServiceInfo serviceObj = serviceInfoHolder.getServiceInfoMap().get(serviceKey); -----> 通过 key 获取服务的信息
  • serviceInfoHolder.processServiceInfo(serviceObj); -----> 反序列化操作
  1. 查看processServiceInfo(ServiceInfo serviceInfo)方法

public ServiceInfo processServiceInfo(ServiceInfo serviceInfo) {
    // 获取服务的key
    String serviceKey = serviceInfo.getKey();
    // 如果为空就返回null
    if (serviceKey == null) {
        NAMING_LOGGER.warn("process service info but serviceKey is null, service host: {}",
                JacksonUtils.toJson(serviceInfo.getHosts()));
        return null;
    }
    // 通过key获取jvm缓存的服务信息
    ServiceInfo oldService = serviceInfoMap.get(serviceInfo.getKey());
    // 判断serviceInfo是否为空或错误推送,错误就返回旧服务
    if (isEmptyOrErrorPush(serviceInfo)) {
        //empty or error push, just ignore
        NAMING_LOGGER.warn("process service info but found empty or error push, serviceKey: {}, "
                + "pushEmptyProtection: {}, hosts: {}", serviceKey, pushEmptyProtection, serviceInfo.getHosts());
        return oldService;
    }
    // 写入jvm缓存
    serviceInfoMap.put(serviceInfo.getKey(), serviceInfo);
    // 计算两个服务实例的差异
    InstancesDiff diff = getServiceInfoDiff(oldService, serviceInfo);
    if (StringUtils.isBlank(serviceInfo.getJsonFromServer())) {
        serviceInfo.setJsonFromServer(JacksonUtils.toJson(serviceInfo));
    }
    // 将本地缓存的大小设置给监控指标对象
    MetricsMonitor.getServiceInfoMapSizeMonitor().set(serviceInfoMap.size());
    if (diff.hasDifferent()) {
        NAMING_LOGGER.info("current ips:({}) service: {} -> {}", serviceInfo.ipCount(), serviceInfo.getKey(),
                JacksonUtils.toJson(serviceInfo.getHosts()));
        
        if (!failoverReactor.isFailoverSwitch(serviceKey)) {
            NotifyCenter.publishEvent(
                    new InstancesChangeEvent(notifierEventScope, serviceInfo.getName(), serviceInfo.getGroupName(),
                            serviceInfo.getClusters(), serviceInfo.getHosts(), diff));
        }
        // 将jvm缓存中的服务信息,写入到本地缓存中
        DiskCache.write(serviceInfo, cacheDir);
    }
    return serviceInfo;
}

3.2. 服务订阅总结

  1. 创建NacosNamingService 对象
    1. NamingFactory.createNamingService(); -----> 创建NacosNamingService
    2. 通过反射 NacosNamingService 实例,通过构造函数执行 init()方法
    3. init()方法
      1. NamingClientProxyDelegate() -----> 实现服务的注册与发现等功能
        1. 创建 ServiceInfoUpdateService()对象 -----> 定时拉取服务列表数据
          1. run()方法
            1. serviceInfoHolder.getServiceInfoMap().get(serviceKey); -----> 获取服务列表数据
            2. serviceInfoHolder.processServiceInfo(serviceObj); -----> 反序列化
              1. serviceInfoMap.put(serviceInfo.getKey(), serviceInfo); -----> 服务数据存储到serviceInfoMap 中服务的数据变更,会将数据拉取到serviceInfoMap

4. 服务注册流程梳理

4.1. 从 naming.registerInstance(“nacos.test.3”, “11.11.11.11”, 8888);说起

  1. 打断点启动

  1. f7 进入registerInstance()方法,再次 按 f7 直到进入到下面方法

/**
 * 服务注册
 * @param serviceName 服务名称
 * @param groupName   分组名称
 * @param ip          服务ip
 * @param port        服务端口
 * @param clusterName 集群名称
 * @throws NacosException
 */
public void registerInstance(String serviceName, String groupName, String ip, int port, String clusterName)
            throws NacosException {
    // nacos的配置信息对象
    Instance instance = new Instance();
    // ip
    instance.setIp(ip);
    // 端口
    instance.setPort(port);
    // 权重
    instance.setWeight(1.0);
    // 集群名称
    instance.setClusterName(clusterName);
    // 注册方法    -----  主要的方法
    registerInstance(serviceName, groupName, instance);
}
  1. 打开Instance 类查看配置类里有哪些属性
@JsonInclude(Include.NON_NULL)
public class Instance implements Serializable {
    
    /**
     * 此实例的唯一id
     */
    private String instanceId;
    
    /**
     * ip
     */
    private String ip;
    
    /**
     * 端口
     */
    private int port;
    
    /**
     * 权重.
     */
    private double weight = 1.0D;
    
    /**
     * 健康状态.
     */
    private boolean healthy = true;
    
    /**
     * 启用以接受请求
     */
    private boolean enabled = true;
    
    /**
     * 临时.
     *
     * @since 1.0.0
     */
    private boolean ephemeral = true;
    
    /**
     * 集群名字.
     */
    private String clusterName;
    
    /**
     * 服务名称.
     */
    private String serviceName;
    
    /**
     * user extended attributes.
     */
    private Map<String, String> metadata = new HashMap<>();
    
    ..... get/set
    
    
    
    ... toString
    
   .....
}

  1. f7 进入registerInstance(serviceName, groupName, instance);方法

/**
 * 注册
 * @param serviceName 服务名称
 * @param groupName   组名称
 * @param instance    配置信息
 * @throws NacosException
 */
@Override
public void registerInstance(String serviceName, String groupName, Instance instance) throws NacosException {
    // 校验服务的合法性
    NamingUtils.checkInstanceIsLegal(instance);
    // 检查并移除服务实例的分组名称前缀
    checkAndStripGroupNamePrefix(instance, groupName);
    // 将服务实例注册到Nacos注册中心    -----  主要的方法
    clientProxy.registerService(serviceName, groupName, instance);
}
  1. f7 进入clientProxy.registerService(serviceName, groupName, instance)方法

public void registerService(String serviceName, String groupName, Instance instance) throws NacosException {
    getExecuteClientProxy(instance).registerService(serviceName, groupName, instance);
}
  1. f7 进入registerService(serviceName, groupName, instance);方法
  • registerService 是一个接口,实现类有 4 个,nacos2.x 的默认实现类为 grpc 所以最终进入的是NamingGrpcClientProxy 类

/**
 * 校验是否为临时服务
 * @param serviceName 服务名称
 * @param groupName   服务组名
 * @param instance    配置信息
 * @throws NacosException
 */
@Override
public void registerService(String serviceName, String groupName, Instance instance) throws NacosException {
    NAMING_LOGGER.info("[REGISTER-SERVICE] {} registering service {} with instance {}", namespaceId, serviceName,
            instance);
    if (instance.isEphemeral()) {
        // 默认会走这里    原因:Instance类中  ephemeral属性为true
        registerServiceForEphemeral(serviceName, groupName, instance);
    } else {
        doRegisterServiceForPersistent(serviceName, groupName, instance);
    }
}
  1. f7 进入registerServiceForEphemeral(serviceName, groupName, instance);方法

private void registerServiceForEphemeral(String serviceName, String groupName, Instance instance)
        throws NacosException {
    // 缓存配置信息   就是一个map集合,还有一个定时任务的心跳机制
    redoService.cacheInstanceForRedo(serviceName, groupName, instance);
    // 服务注册
    doRegisterService(serviceName, groupName, instance);
}

redoService.cacheInstanceForRedo(serviceName, groupName, instance); 的 的心跳机制

    public NamingGrpcRedoService(NamingGrpcClientProxy clientProxy, NacosClientProperties properties) {
        setProperties(properties);
        // 定时任务的线程池
        this.redoExecutor = new ScheduledThreadPoolExecutor(redoThreadCount, new NameThreadFactory(REDO_THREAD_NAME));
        this.redoExecutor.scheduleWithFixedDelay(new RedoScheduledTask(clientProxy, this), redoDelayTime, redoDelayTime,
                TimeUnit.MILLISECONDS);
    }
  1. f7 进入doRegisterService(serviceName, groupName, instance)方法

/**
 * 将提供的服务实例注册到指定的服务和分组中
 *
 * @param serviceName 服务名
 * @param groupName   组名
 * @param instance    配置信息
 * @throws NacosException nacos exception
 */
public void doRegisterService(String serviceName, String groupName, Instance instance) throws NacosException {
    // 请求的一些属性
    InstanceRequest request = new InstanceRequest(namespaceId, serviceName, groupName,
            NamingRemoteConstants.REGISTER_INSTANCE, instance);
    // 将请求发送到服务器,并获取响应对象
    requestToServer(request, Response.class);
    // 通知系统服务实例已成功注册   // 就是改了一个标志
    redoService.instanceRegistered(serviceName, groupName);
}

redoService.instanceRegistered(serviceName, groupName);

/**
 * 如果{@code true}表示缓存的数据已经成功注册到服务器。
 */
private volatile boolean registered;

/**
 * 如果{@code true}表示缓存的数据正在从服务器注销。
 */
private volatile boolean unregistering;

public void registered() {
        this.registered = true;
        this.unregistering = false;
    }
  1. 进入requestToServer(request, Response.class);方法

    private <T extends Response> T requestToServer(AbstractNamingRequest request, Class<T> responseClass)
            throws NacosException {
        Response response = null;
        try {
            // 添加请求头
            request.putAllHeader(
                    getSecurityHeaders(request.getNamespace(), request.getGroupName(), request.getServiceName()));
            // 发送请求
            response = requestTimeout < 0 ? rpcClient.request(request) : rpcClient.request(request, requestTimeout);
            // .... 根据类型抛异常
            ...
        } catch (NacosException e) {
           // 抛异常
        }
    }
  1. f7 进入rpcClient.request(request)方法,再次按 f7 进入public Response request(Request request, long timeoutMills)

public Response request(Request request, long timeoutMills) throws NacosException {
    int retryTimes = 0;
    Response response;
    Throwable exceptionThrow = null;
    long start = System.currentTimeMillis();
    while (retryTimes <= rpcClientConfig.retryTimes() && (timeoutMills <= 0
            || System.currentTimeMillis() < timeoutMills + start)) {
        boolean waitReconnect = false;
        try {
            ... 异常校验
            response = this.currentConnection.request(request, timeoutMills);
            ... 异常校验
            // return response.
            lastActiveTimeStamp = System.currentTimeMillis();
            // 返回响应数据
            return response;
            
        } catch (Throwable e) {
            ... 处理异常
        }
        retryTimes++;
    }
    
    if (rpcClientStatus.compareAndSet(RpcClientStatus.RUNNING, RpcClientStatus.UNHEALTHY)) {
        // 用于在请求失败时切换服务器
        switchServerAsyncOnRequestFail();
    }
    
    ... 异常校验
}
  1. f7 进入this.currentConnection.request(request, timeoutMills);方法

@Override
public Response request(Request request, long timeouts) throws NacosException {
    // 转grpc类型
    Payload grpcRequest = GrpcUtils.convert(request);
    // 异步请求
    ListenableFuture<Payload> requestFuture = grpcFutureServiceStub.request(grpcRequest);
    Payload grpcResponse;
    try {
        // 获取返回的数据
        if (timeouts <= 0) {
            grpcResponse = requestFuture.get();
        } else {
            grpcResponse = requestFuture.get(timeouts, TimeUnit.MILLISECONDS);
        }
    } catch (Exception e) {
        throw new NacosException(NacosException.SERVER_ERROR, e);
    }
    // 再次转换回来,并且强制转为Response
    return (Response) GrpcUtils.parse(grpcResponse);
}

4.2. 服务注册流程总结

  1. 创建NacosNamingService 对象
    1. clientProxy.registerService(serviceName, groupName, instance); —> grpc 协议进行注册
      1. registerServiceForEphemeral(serviceName, groupName, instance); ----> 临时注册
        1. redoService.cacheInstanceForRedo(serviceName, groupName, instance); —> 缓存 + 心跳
        2. doRegisterService(serviceName, groupName, instance); ----> 远程调用
  2. doRegisterService(serviceName, groupName, instance); ----> 远程调用
    1. requestToServer(request, Response.class);
      1. rpcClient.request(request)
      2. while (retryTimes <= rpcClientConfig.retryTimes() && (timeoutMills <= 0 || System.currentTimeMillis() < timeoutMills + start)) 发送请求条件
      3. this.currentConnection.request(request, timeoutMills); —> 发送注册请求

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

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

相关文章

【文心智能体】通过工作流使用知识库来实现信息查询输出,一键查看旅游相关信息,让出行多一份信心

欢迎来到《小5讲堂》 这是《文心智能体平台》系列文章&#xff0c;每篇文章将以博主理解的角度展开讲解。 温馨提示&#xff1a;博主能力有限&#xff0c;理解水平有限&#xff0c;若有不对之处望指正&#xff01; 目录 创建灵感基本配置头像名称和简介人物设定角色与目标思考路…

一文读懂数据血缘分析原理与建设方法

什么是数据血缘分析?数据血缘为数据全生命周期过程中的数据关系&#xff0c;包括数据特征的变化&#xff0c;即数据的来龙去脉。主要内容包括数据的来源、数据的加工方式、映射关系以及数据的流出和消费。数据血缘分析就是针对数据分析中的血缘关系做分析&#xff0c;主要包含…

第二证券:10倍“妖股”一天暴跌98% 揭秘港股那些做局套路

尽管港股行情全体低迷&#xff0c;自上一年以来港股商场仍不乏短短时日暴升10倍以上的疑似“妖股”&#xff0c;这些公司普遍具有一起的特征&#xff1a;刚上市不久的新股、次新股&#xff0c;市值较小、股权高度会集。 证券时报记者通过查询发现&#xff0c;这类股票反面往往…

图片无损缩放PhotoZoom Pro 9.0.2绿色版 +免费赠送PhotoZoom激活优惠代码

PhotoZoom Pro 9.0.2 是一款专业的图片无损缩放软件&#xff0c;该软件采用了 benvista s-spline 独特技术&#xff0c;增强了对图像格式的支持&#xff0c;多处理器支持&#xff0c;GPU 加速&#xff0c;win10和 Photoshop CC 支持。带来一流的数字图形扩展与缩减技术。该软件…

torch torchvision 安装失败解决方法

按照安装教程在安装Torch torchvision的时候报错 报错内容&#xff1a; ERROR: Could not find a version that satisfies the requirement torch (from versions: none)ERROR: No matching distribution found for torch 出现这个情况&#xff0c;有说要换源什么的&#xff…

博客常见问题

hexo g 生成静态文件 hexo s 本地预览 hexo d 同步上传到git 1、输入hexo d &#xff0c;上传到git时&#xff0c;报错 看了下git的配置&#xff0c;没有问题&#xff0c;单机过去也能直接到我的git上 可能是传不过去&#xff0c;token的问题 最下面开发者设置&#xff0c;找到…

单片机毕业设计-基于单片机的运动手环

文章目录 前言资料获取设计介绍功能介绍程序代码部分参考 设计清单具体实现截图参考文献设计获取 前言 &#x1f497;博主介绍&#xff1a;✌全网粉丝10W,CSDN特邀作者、博客专家、CSDN新星计划导师&#xff0c;一名热衷于单片机技术探索与分享的博主、专注于 精通51/STM32/MSP…

2024年【浙江省安全员-C证】找解析及浙江省安全员-C证模拟考试题库

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 浙江省安全员-C证找解析是安全生产模拟考试一点通生成的&#xff0c;浙江省安全员-C证证模拟考试题库是根据浙江省安全员-C证最新版教材汇编出浙江省安全员-C证仿真模拟考试。2024年【浙江省安全员-C证】找解析及浙江…

如何在页面调用utility bar并传递参数至lwc组件

1.在app的utility item中添加lwc组件&#xff1a; 2.调用utility bar api的方式有两种&#xff1a; 方法一&#xff0c;通过lwc调用&#xff1a; import {LightningElement,api ,wire } from lwc; import { publish, MessageContext } from lightning/messageService; imp…

新手烤鸡测试——单烤没拉满

文章目录 电脑配置先来一波硬件信息硬盘检测cpu-Z跑分AIDA64单烤FPUDistInfoDiskMark查看硬盘读写次数 电脑配置先来一波 图吧下载地址 硬件信息 硬盘检测 cpu-Z跑分 AIDA64单烤FPU 压榨cpu性能结果 DistInfo 计算了一下到几天快两个月的使用时间也没差 DiskMark查看硬盘读…

批量复制指定文件夹——EXCEL VBA 实现

工作中往往需要复制特定文件夹&#xff0c;例如&#xff0c;一个文件夹中有100个文件夹&#xff0c;我只需要复制其中50个文件夹&#xff0c;这50个文件夹的名字放入excel表中第一列&#xff0c;从第二行开始&#xff08;注意&#xff1a;第一行的表头不能覆盖&#xff09;&…

三十三、Gin的中间件

目录 定义&#xff1a; 一、处理请求前执行 1、模拟请求前需要鉴权 2、使用use方法 3、实验 二、处理请求后执行 1、在具体方法中添加输出方便观察整个处理请求过程生命周期 2、在next方法后增加输出即可在处理完成请求后执行内容 3、最终执行结果 定义&#xff1a; 在…

大模型LLM算法工程师技术面试指南

大模型LLM算法工程师技术面试指南 AI大模型全套学习资料 “最先掌握AI的人&#xff0c;将会比较晚掌握AI的人有竞争优势”。 这句话&#xff0c;放在计算机、互联网、移动互联网的开局时期&#xff0c;都是一样的道理。 我在一线互联网企业工作十余年里&#xff0c;指导过不少…

防跌倒识别摄像机

防跌倒识别摄像机 是一种结合了人工智能技术和监控摄像技术的先进设备&#xff0c;旨在通过实时监测和分析监控画面中的行为动作&#xff0c;及时发现并预防跌倒事件的发生。这种摄像机在医疗、养老院、家庭等场所有着广泛的应用前景。 防跌倒识别摄像机在医疗领域具有重要意义…

社群空间站付费入群系统易支付版全套搭建教程

社群空间站9.9付费入群系统易支付版全套搭建教程 1.创建站点 2.搭建环境 php7.2 3.上传源码包 数据库批量修改sq9.dongge1.icu s10.dongge1.icu 改为你的域名 4.上传数据库 修改数据库文件/data/config/ 5.访问域名 6.账户密码 admin 123456 7.易支付修改地址是在/data…

SpringBoot2:请求处理原理分析-常用接口方法参数整理

文章目录 1、常用的原生API参数2、一些自带的复杂参数3、自定义Bean参数4、自定义参数转换服务5、总结 1、常用的原生API参数 作用说明&#xff1a;在接口方法参数放入一些原生API作为参数使用。 案例&#xff1a; 接口收参形式&#xff1a; RequestMapping("/test"…

16款facebook辅助工具,总有一款适合你!

Hey小伙伴们~&#x1f44b; 是不是想利用FB大展拳脚&#xff0c;却苦于不知道如何开始&#xff1f;别急&#xff0c;今天就给你们安利16个超实用的FB营销工具&#xff0c;涵盖了内容创建和发布的应用程序&#xff0c;以及数据追踪分析、商品销售等多个方面让你轻松get海外获客新…

浅谈:CDN下真实IP的暴露

免责声明:本文仅做分享! 目录 CDN简介&#xff1a; 国内常见CDN&#xff1a; 国外常见CDN&#xff1a; 判断CDN存在? 在线ping检测: nslookup: 寻找真实IP----> 1-DNS历史解析纪录 2-子域名查询 1.在线平台查询 2.工具爆破 3.搜索引擎 3-网站邮件头信息 4-S…

100个视频如何转换成1个二维码

使用场景描述&#xff1a;有50-100个视频&#xff0c;要实现扫一个二维码&#xff0c;就可以完整观看这50-100个视频的内容&#xff0c;这种情况下&#xff0c;可以使用列表专辑二维码功能来轻松实现。 使用步骤 STEP1 注册帐号 使用视频专辑列表二维码&#xff0c;您需要注册…

原生 iOS 引入 Flutter 报错 kernel_blob.bin 找不到

情况 在一次原生 iOS 项目中引入 Flutter 的过程中&#xff0c;在模拟器中运行出现报错&#xff1a; 未能打开文件“kernel_blob.bin”&#xff0c;因为它不存在。 如下图&#xff1a; 模拟器中一片黑 原因&解决方案 这个是因为 Flutter 的打包 iOS framework 命令中…