CTK插件框架学习-插件注册调用(03)

news2025/1/25 8:57:45

CTK插件框架学习-新建插件(02)icon-default.png?t=N7T8https://mp.csdn.net/mp_blog/creation/editor/136923735

一、CTK插件组成

  • 接口类:对外暴露的接口,供其他插件调用
  • 实现类:实现接口内的方法
  • 激活类:负责将插件注册到CTK框架中

二、接口、插件、服务三者关系

1、一对一

一个接口类由一个实现类实现,对应一个插件,注册一个服务

参见

CTK插件框架学习-新建插件(02)icon-default.png?t=N7T8https://mp.csdn.net/mp_blog/creation/editor/136923735

2、多对一(多态)

多个接口类由一个实现类实现,对应一个插件,注册多个服务
 

  • 接口类1
    #pragma once
    #include <QObject>
    
    class IService1
    {
    public:
    	virtual ~IService1() {}
    	virtual void printf1() = 0;
    };
    
    //此宏将当前这个接口类声明为接口,后面的字符串是这个接口的唯一标识。
    Q_DECLARE_INTERFACE(IService1, "zr.IService1")
    

  • 接口类2
    #pragma once
    #pragma once
    #include <QObject>
    
    class IService2
    {
    public:
    	virtual ~IService2() {}
    	virtual void printf2() = 0;
    };
    
    //此宏将当前这个接口类声明为接口,后面的字符串是这个接口的唯一标识。
    Q_DECLARE_INTERFACE(IService2, "zr.IService2")
    

  • 实现类
    
    ==============================TestPlugin2.ch==================================
    #pragma once
    
    #include "IService1.h"
    #include "IService2.h"
    
    class ctkPluginContext;
    class TestPlugin2 : public QObject, public IService1, public IService2
    {
    	Q_OBJECT
    	//当一个类继承这个接口类,表明需要实现这个接口类
    	Q_INTERFACES(IService1)
    	Q_INTERFACES(IService2)
    
    public:
        TestPlugin2(ctkPluginContext* contex);
    
    	virtual void printf1();
    	virtual void printf2();
    };
    
    ==============================TestPlugin2.cpp=================================
    
    #include "TestPlugin2.h"
    #include <qdebug.h>
    
    TestPlugin2::TestPlugin2(ctkPluginContext* contex)
    {
    }
    
    void TestPlugin2::printf1()
    {
    	qDebug() << "IService1 printf1";
    }
    
    void TestPlugin2::printf2()
    {
    	qDebug() << "IService2 printf2";
    }
    

  • 插件测试
#include "CTKPlugin.h"
#include <QtWidgets/QApplication>


#include <iostream>
#include <QStyleFactory>
#include <QDir>
#include <QDirIterator>
#include <QDebug>
#include "ctkPluginFrameworkFactory.h"
#include "ctkPluginFramework.h"
#include "ctkPluginException.h"
#include "ctkPluginContext.h"
#include "ctkPluginFrameworkLauncher.h"
#include "../TestPlugin2/IService1.h"
#include "../TestPlugin2/IService2.h"
/*
* 1、注意:Plugin-SymbolicName要满足这里的前缀是:TARGET/META-INF格式。TARGET的名字最好和工程名一致,不然可能出现device not open错误。
* 2、如果CTK初始化、插件安装启动等是在一个类中,则与CTK相关的变量应定义成类的属性,不能是成员变量,否则获取不到服务
* 3、CTK插件组成:
(1)每个插件有自己的注册器Activator,继承自QObject和ctkPluginActivator的一个类,并实现ctkPluginActivator的start、stop函数
(2)每个插件必须有一个资源文件,名称一般与插件名称一致,前缀必须为TARGET/META-INF,例:插件名称/META-INF
(3)每个插件必须添加一个元数据文件,名字必须为MANIFEST.MF,并添加到资源文件中
* 4、QSharedPointer framework这个对象既可以作为对象也可以作为对象指针,但要作为插件框架使用必须要用指针方法调用
* 5、生成的插件名(TARGET)不要有下划线,因为CTK会默认将插件名中的下划线替换成点号,最后导致找不到插件 
*/
int main(int argc, char *argv[])
{
	QApplication::setAttribute(Qt::AA_EnableHighDpiScaling);

    QApplication a(argc, argv);
	a.setApplicationName("ctktest");//Linux下没有名称报错

	QString path = QCoreApplication::applicationDirPath();

	// 启动插件工厂
	ctkPluginFrameworkFactory* ctkFrameWorkFactory = new ctkPluginFrameworkFactory;
	QSharedPointer<ctkPluginFramework> framework = ctkFrameWorkFactory->getFramework();
	try {
		framework->init();
		framework->start();
	}
	catch (const ctkPluginException& e)
	{
		std::cout << "framework init fail" << std::endl;
	}


	QSharedPointer<ctkPlugin> plugin;

	QDirIterator iter(path + "/plugins/", { "*.dll" }, QDir::NoFilter, QDirIterator::Subdirectories);
	while (iter.hasNext()) {
		//qDebug() << iter.next();
		QString dllPath = iter.next();
		QUrl url = QUrl::fromLocalFile(dllPath);
		try
		{
			plugin = framework->getPluginContext()->installPlugin(url);

			//获取MANIFEST.MF中的数据
			QHash<QString, QString> headers = plugin->getHeaders();
			ctkVersion version = ctkVersion::parseVersion(headers.value(ctkPluginConstants::PLUGIN_VERSION));
			QString name = headers.value(ctkPluginConstants::PLUGIN_NAME);
		}
		catch (ctkPluginException e) {
			std::cout << e.message().toStdString() << e.getType() << std::endl;
		}
	}

	try {
		plugin->start(ctkPlugin::START_TRANSIENT);//表示立即启用插件,不设置参数的话加载后也不会立即打印输出
	}
	catch (ctkPluginException e) {
		std::cout << e.message().toStdString() << e.getType() << std::endl;
	}


	//2、测试插件(多个接口一个实现一个服务一个插件)
	IService1* service1 = NULL;
	ctkServiceReference ref2 = framework->getPluginContext()->getServiceReference<IService1>();
	if (ref2)
	{
		service1 = framework->getPluginContext()->getService<IService1>(ref2);
	}
	if (service1)
	{
		service1->printf1();
	}
	IService2* service2 = NULL;
	ctkServiceReference ref3 = framework->getPluginContext()->getServiceReference<IService2>();
	if (ref3)
	{
		service2 = framework->getPluginContext()->getService<IService2>(ref3);
	}
	if (service2)
	{
		service2->printf2();
	}

	

	//ctkPlugin::State sta = plugin->getState();
	//ctkPluginFrameworkLauncher::stop();
	//plugin->stop(); 
	//plugin->uninstall();
	//sta = plugin->getState();

	

	CTKPlugin c;
	c.show();
    return a.exec();
}

3、一对多

一个接口类由多个实现类实现,对应多个插件,注册一个服务

/*
* 1、通过ctkPluginConstants::SERVICE_RANKING和ctkPluginConstants::SERVICE_ID来调用不同的插件
* 2、插件不同但是在同一个dll内
* 3、插件获取策略:
*		插件容器中id最小的服务,id为插件注册时的SERVICE_RANKING属性
*		插件容器内id相同的情况,返回pid最小的服务
* 4、插件每次调用其他插件时只会生成一个实例,不会因多次调用产生多个服务实例
*/

  • 接口类
    #pragma once
    #include <QObject>
    
    /*
    * 1、通过ctkPluginConstants::SERVICE_RANKING和ctkPluginConstants::SERVICE_ID来调用不同的插件
    * 2、插件不同但是在同一个dll内
    * 3、插件获取策略:
    *		插件容器中id最小的服务,id为插件注册时的SERVICE_RANKING属性
    *		插件容器内id相同的情况,返回pid最小的服务
    * 4、插件每次调用其他插件时只会生成一个实例,不会因多次调用产生多个服务实例
    */
    class IService
    {
    public:
    	virtual ~IService() {}
    	virtual void printf() = 0;
    };
    
    //此宏将当前这个接口类声明为接口,后面的字符串是这个接口的唯一标识。
    Q_DECLARE_INTERFACE(IService, "zr.IService")
    

  • 实现类1
    #pragma once
    
    #include "IService.h"
    #include <QObject>
    
    class ctkPluginContext;
    class ServiceImp1 :
    	public QObject, public IService
    {
    	Q_OBJECT
    	//当一个类继承这个接口类,表明需要实现这个接口类
    	Q_INTERFACES(IService)
    	
    public:
    	ServiceImp1(ctkPluginContext* contex);
    	void printf();
    
    };
    
    

  • 实现类2
    #pragma once
    #include "IService.h"
    #include <QObject>
    
    
    class ctkPluginContext;
    class ServiceImp2 :
    	public QObject, public IService
    {
    	Q_OBJECT
    	//当一个类继承这个接口类,表明需要实现这个接口类
    	Q_INTERFACES(IService)
    
    public:
    	ServiceImp2(ctkPluginContext* contex);
    	void printf();
    };
    
    

  • 服务类
    =========================PluginActivator.h=============================
    #pragma once
    #include <qobject.h>
    #include "ctkPluginActivator.h"
    #include "ctkPluginContext.h"
    #include "ServiceImp1.h"
    #include "ServiceImp2.h"
    
    class PluginActivator :
    	public QObject, ctkPluginActivator
    {
    	Q_OBJECT
    	Q_INTERFACES(ctkPluginActivator)//向Qt的插件框架声明,希望将xxx插件放入到框架中。
    	Q_PLUGIN_METADATA(IID "TestPlugin3")//向qt框架申明插件(qt5版本)
    
    public:
    	PluginActivator();
    	void start(ctkPluginContext *context);
    	void stop(ctkPluginContext *context);
    private:
    	QScopedPointer<ServiceImp1> m_serviceImp1;//智能指针,自动析构回收
    	QScopedPointer<ServiceImp2> m_serviceImp2;//智能指针,自动析构回收
    };
    
    =========================PluginActivator.cpp=============================
    
    #include "PluginActivator.h"
    #include <QDebug>
    #include "ctkPluginContext.h"
    #include "ctkPluginFrameworkLauncher.h"
    
    PluginActivator::PluginActivator()
    {
    
    }
    void PluginActivator::start(ctkPluginContext *context)
    {
    	qDebug() << "my plugin3 start";
    	m_serviceImp1.reset(new ServiceImp1(context));
    	m_serviceImp2.reset(new ServiceImp2(context));
    
    	ctkDictionary dic1;
    	dic1.insert(ctkPluginConstants::SERVICE_RANKING, 1);
    	dic1.insert("name", "ServiceImp1");
    	context->registerService<IService>(m_serviceImp1.get(), dic1);
    
    	ctkDictionary dic2;
    	dic2.insert(ctkPluginConstants::SERVICE_RANKING, 2);
    	dic2.insert("name", "ServiceImp2");
    	context->registerService<IService>(m_serviceImp2.get(), dic2);
    
    }
    
    void PluginActivator::stop(ctkPluginContext *context)
    {
    	qDebug() << "my plugin3 stop";
    	Q_UNUSED(context)// Q_UNUSED,如果一个函数的有些参数没有用到、某些变量只声明不使用,但是又不想编译器、编辑器报警报,其他没有什么实际性作用
    
    	ctkServiceReference  ref1 = context->getServiceReference<IService>();
    	context->ungetService(ref1);
    
    	m_serviceImp1.reset(NULL);
    	m_serviceImp2.reset(NULL);
    
    	ctkPlugin::State sta = context->getPlugin()->getState();
    
    }

  • 插件测试
    #include "CTKPlugin.h"
    #include <QtWidgets/QApplication>
    
    
    #include <iostream>
    #include <QStyleFactory>
    #include <QDir>
    #include <QDirIterator>
    #include <QDebug>
    #include "ctkPluginFrameworkFactory.h"
    #include "ctkPluginFramework.h"
    #include "ctkPluginException.h"
    #include "ctkPluginContext.h"
    #include "ctkPluginFrameworkLauncher.h"
    #include "../TestPlugin3/IService.h"
    /*
    * 1、注意:Plugin-SymbolicName要满足这里的前缀是:TARGET/META-INF格式。TARGET的名字最好和工程名一致,不然可能出现device not open错误。
    * 2、如果CTK初始化、插件安装启动等是在一个类中,则与CTK相关的变量应定义成类的属性,不能是成员变量,否则获取不到服务
    * 53、CTK插件组成:
    (1)每个插件有自己的注册器Activator,继承自QObject和ctkPluginActivator的一个类,并实现ctkPluginActivator的start、stop函数
    (2)每个插件必须有一个资源文件,名称一般与插件名称一致,前缀必须为TARGET/META-INF,例:插件名称/META-INF
    (3)每个插件必须添加一个元数据文件,名字必须为MANIFEST.MF,并添加到资源文件中
    * 4、QSharedPointer framework这个对象既可以作为对象也可以作为对象指针,但要作为插件框架使用必须要用指针方法调用
    * 5、生成的插件名(TARGET)不要有下划线,因为CTK会默认将插件名中的下划线替换成点号,最后导致找不到插件 
    */
    int main(int argc, char *argv[])
    {
    	QApplication::setAttribute(Qt::AA_EnableHighDpiScaling);
    
        QApplication a(argc, argv);
    	a.setApplicationName("ctktest");//Linux下没有名称报错
    
    	QString path = QCoreApplication::applicationDirPath();
    
    	// 启动插件工厂
    	ctkPluginFrameworkFactory* ctkFrameWorkFactory = new ctkPluginFrameworkFactory;
    	QSharedPointer<ctkPluginFramework> framework = ctkFrameWorkFactory->getFramework();
    	try {
    		framework->init();
    		framework->start();
    	}
    	catch (const ctkPluginException& e)
    	{
    		std::cout << "framework init fail" << std::endl;
    	}
    
    	//QString dir = QCoreApplication::applicationDirPath();
    	//dir += "/plugins/TestPlugin.dll";
    	//QUrl url = QUrl::fromLocalFile(dir);
    	QSharedPointer<ctkPlugin> plugin;
    
    	QDirIterator iter(path + "/plugins/", { "*.dll" }, QDir::NoFilter, QDirIterator::Subdirectories);
    	while (iter.hasNext()) {
    		//qDebug() << iter.next();
    		QString dllPath = iter.next();
    		QUrl url = QUrl::fromLocalFile(dllPath);
    		try
    		{
    			plugin = framework->getPluginContext()->installPlugin(url);
    
    			//获取MANIFEST.MF中的数据
    			QHash<QString, QString> headers = plugin->getHeaders();
    			ctkVersion version = ctkVersion::parseVersion(headers.value(ctkPluginConstants::PLUGIN_VERSION));
    			QString name = headers.value(ctkPluginConstants::PLUGIN_NAME);
    		}
    		catch (ctkPluginException e) {
    			std::cout << e.message().toStdString() << e.getType() << std::endl;
    		}
    	}
    
    	try {
    		plugin->start(ctkPlugin::START_TRANSIENT);//表示立即启用插件,不设置参数的话加载后也不会立即打印输出
    	}
    	catch (ctkPluginException e) {
    		std::cout << e.message().toStdString() << e.getType() << std::endl;
    	}
    
    
    	//3、测试插件(一个接口多个实现一个服务多个插件<一个dll内>)
    	//3.1获取所有服务
    	
    	QList<ctkServiceReference> ref3List = framework->getPluginContext()->getServiceReferences<IService>();
    	foreach (ctkServiceReference var, ref3List)
    	{
    		if (var)
    		{
    			qDebug() << "service name:" << var.getProperty("name").toString();
    			qDebug() << "service ranking:" << var.getProperty(ctkPluginConstants::SERVICE_RANKING).toLongLong();
    			qDebug() << "service id:" << var.getProperty(ctkPluginConstants::SERVICE_ID).toLongLong();
    			IService* service3 = qobject_cast<IService*>(framework->getPluginContext()->getService(var));
    			if (service3)
    			{
    				service3->printf();
    			}
    		}
    	}
    
    	//3.2获取某些服务
    	IService* service3 = NULL;
    	ref3List = framework->getPluginContext()->getServiceReferences<IService>("(&(name=ServiceImp1))");
    	foreach(ctkServiceReference var, ref3List)
    	{
    		if (var)
    		{
    			qDebug() << "service name:" << var.getProperty("name").toString();
    			qDebug() << "service ranking:" << var.getProperty(ctkPluginConstants::SERVICE_RANKING).toLongLong();
    			qDebug() << "service id:" << var.getProperty(ctkPluginConstants::SERVICE_ID).toLongLong();
    			IService* service3 = qobject_cast<IService*>(framework->getPluginContext()->getService(var));
    			if (service3)
    			{
    				service3->printf();
    			}
    		}
    	}
    
    	//3.3获取一个服务,由service ranking 和service id决定
    	ctkServiceReference ref = framework->getPluginContext()->getServiceReference<IService>();
    	if (ref) {
    		IService* service3 = qobject_cast<IService*>(framework->getPluginContext()->getService(ref));
    		if (service3)
    		{
    			service3->printf();
    		}
    	}
    
    
    
    
    	//ctkPlugin::State sta = plugin->getState();
    	//ctkPluginFrameworkLauncher::stop();
    	//plugin->stop(); 
    	//plugin->uninstall();
    	//sta = plugin->getState();
    
    	
    
    	CTKPlugin c;
    	c.show();
        return a.exec();
    }
    

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

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

相关文章

QT_day5:使用定时器实现闹钟

1、 程序代码&#xff1a; widget.h&#xff1a; #ifndef WIDGET_H #define WIDGET_H#include <QWidget> #include <QTime>//时间类 #include <QTimer>//时间事件类 #include <QTextToSpeech>//文本转语音类 QT_BEGIN_NAMESPACE namespace Ui { cla…

Nodejs 16与 gitbook搭建属于你自己的书本网站-第一篇

最近想重新搭建一个网站来存放自己的相关知识点&#xff0c;并向网络公开&#xff0c;有个hexo博客其实也不错的&#xff0c;但是总感觉hexo很多花里胡哨的玩意&#xff0c;导致挂载的博客异常卡&#xff0c;这样反而不利于我自己回顾博客了&#xff0c;于是我就开始钻研这个鬼…

IDEA的Scala环境搭建

目录 前言 Scala的概述 Scala环境的搭建 一、配置Windows的JAVA环境 二、配置Windows的Scala环境 编写一个Scala程序 前言 学习Scala最好先掌握Java基础及高级部分知识&#xff0c;文章正文中会提到Scala与Java的联系&#xff0c;简单来讲Scala好比是Java的加强版&#x…

商品数据化运营---初步整理

商品数据化运营是指利用数据分析技术来优化商品的销售、管理和推广过程。这种方法结合了数据科学、商业智能、市场分析和消费者行为研究等多个领域的技术和理论&#xff0c;旨在通过分析大量的商品和销售数据来提高商品的市场表现和企业的利润。 以下是商品数据化运营的一些关…

Godot 学习笔记(5):国际化多语言翻译,包含常用10种语言机翻!

文章目录 前言国际化翻译Api选择小牛测试 语言选择代码逻辑实体对象翻译帮助类导出模板读取文件翻译测试多语言测试 综合翻译文件准备测试代码测试结果 完整代码实体类翻译帮助类网络帮助类 最终效果翻译前翻译中翻译后 总结 前言 为了面向更大的市场&#xff0c;国际化是肯定…

【开发篇】六、查询大量数据导致内存溢出

文章目录 1、溢出场景2、快照文件分析3、本地环境复现4、结论5、解决思路 记录一个问题&#xff0c;工作中有个数据处理服务OOM&#xff0c;查了下镜像的dockerfile&#xff0c;发现JVM参数如下。很明显&#xff0c;一个数据服务&#xff0c;里面经手大量的数据对象&#xff0c…

探究分布式事务:深入ACID特性在分布式系统中的挑战与解决方案

✨✨谢谢大家捧场&#xff0c;祝屏幕前的小伙伴们每天都有好运相伴左右&#xff0c;一定要天天开心哦&#xff01;✨✨ &#x1f388;&#x1f388;作者主页&#xff1a; 喔的嘛呀&#x1f388;&#x1f388; ✨✨ 帅哥美女们&#xff0c;我们共同加油&#xff01;一起进步&am…

并发编程之CountDownLatch和CyclicBarrier的详细解析(带小案例)

CountDownLatch 倒计时锁存器 用来解决线程执行次序的问题 CountDownLatch主要有两个方法&#xff0c;当一个或多个线程调用await方法时&#xff0c;这些线程会阻塞。 其它线程调用countDown方法会将计数器减1(调用countDown方法的线程不会阻塞)&#xff0c; 当计数器的值变为…

B096-docker版jenkins环境搭建

目录 Jenkins持续集成工具的介绍Jenkins安装过程案例 tips&#xff1a;ssm项目需要放到tomcat中运行&#xff0c;springboot项目不需要&#xff0c;内置有tomcat&#xff0c;可直接命令行运行 Jenkins持续集成工具的介绍 Jenkins安装过程 docker版Jenkins需要先安装docker环境…

三个对象组练习.java

题目&#xff1a;定义数组存储3部汽车对象&#xff1b;汽车属性&#xff1a;品牌&#xff0c;价格&#xff0c;颜色&#xff1b;创造3个汽车对象&#xff0c;数据通过键盘录入而来&#xff0c;并把数据存储到数组当中 分析&#xff1a; 在main&#xff08;&#xff09;里面定义…

npm ERR! errno CERT_HAS_EXPIRED

1 问题描述 使用npm命令安装相关依赖报错&#xff1a;npm ERR! code CERT_HAS_EXPIRED npm ERR! errno CERT_HAS_EXPIRED npm ERR! request to https://registry.npm.taobao.org/vue%2fcli failed, reason: certificate has expired报错示例图如下所示&#xff1a; 2原因分析…

实验室开放项目实验报告-01

实验室开放项目实验报告 实验名称&#xff1a;实验一输入输出格式 实验目的&#xff1a;熟练掌握程序设计竞赛中通常采用的输入输出格式和掌握不同格式输入输出数据的处理方法 实验内容&#xff1a; 在本地电脑中新建一个文件夹&#xff0c;用于存放C源程序&#xff0c;文件…

学习鸿蒙基础(9)

目录 一、鸿蒙国际化配置 二、鸿蒙常用组件介绍 三、鸿蒙像素单位介绍 四、鸿蒙布局介绍 1、Row与Column线性布局 2、层叠布局-Stack 3、弹性布局 4、栅格布局 5、网格布局 一、鸿蒙国际化配置 base目录下为默认的string。en_US对应美国的。zh_CN对应中国的。新增一个s…

[RAM] 图解 RAM 结构原理

主页&#xff1a; 元存储博客 文章目录 前言1. Channel2. Dimm3. Rank4. Bank5. Row6. Column7. Beat8. Burst Length总结 前言 从CPU至DRAM晶粒之间依据层级由大至小为channel>DIMM>rank>chip>bank>row/column。 图片来源&#xff1a; 电脑王 DRAM层级关系 DR…

应急响应靶机训练-Linux1题解

前言 接上文&#xff0c;应急响应靶机训练Linux1 靶机地址&#xff1a; 应急响应靶机-Linux(1) 最近感冒了&#xff0c;就没录视频版。 题解 目标&#xff1a;3个flag以及黑客的ip地址 登陆虚拟机 密码defend flag1: su history flag{thisismybaby} flag2&#xff1a;…

AI Agent智能应用从0到1定制开发 (包含全部教学视频和全部源码,附链接)

链接分享&#xff1a;https://pan.baidu.com/s/1Cjk-MNDY4DYQKvNI1L4Tzg?pwd3vcd 提取码:3vcd 全部教学视频和全部源码链接分享在下方。 第1章&#xff1a;多模型强应用&#xff1a;AI2.0时代应用开发者机会 在AI2.0时代&#xff0c;应用开发者面临着前所未有的机遇。这一章…

盘点最适合做剧场版的国漫,最后一部有望成为巅峰

最近《完美世界》动画官宣首部剧场版&#xff0c;主要讲述石昊和火灵儿的故事。这个消息一出&#xff0c;引发了很多漫迷的讨论&#xff0c;其实现在已经有好几部国漫做过剧场版&#xff0c;还有是观众一致希望未来会出剧场版的。那么究竟是哪些国漫呢&#xff0c;下面就一起来…

源支付 V7 版,150购买,需要的下载研究,拿走回复

源支付 V7 版&#xff0c;150购买&#xff0c;需要的下载研究,拿走回复 来自站长论坛搬砖&#xff0c;站长论坛搬砖&#xff0c;站长论坛搬砖&#xff0c;站长论坛搬砖&#xff0c;下载地址在最后 请按官方教程配置&#xff0c;专为个人站长打造的聚合免签系统&#xff0c;拥有…

视频监控平台和视频接入网关,支持HTTPS实现视频浏览和录像回放

目录 一、https协议介绍 &#xff08;一&#xff09;什么叫https &#xff08;二&#xff09;https的工作原理 &#xff08;三&#xff09;https和http的区别 二、视频监控平台为何要采用https &#xff08;一&#xff09;采用HTTPS浏览视频的好处体现在以下几个方面&…

手撕算法-跳跃游戏

描述 分析 如果某一个作为 起跳点 的格子可以跳跃的距离是 3&#xff0c;那么表示后面 3 个格子都可以作为 起跳点可以对每一个能作为 起跳点 的格子都尝试跳一次&#xff0c;把 能跳到最远的距离 不断更新如果可以一直跳到最后&#xff0c;就成功了 代码 class Solution {…