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

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();
    }
    

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

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

相关文章

CSS绘制三角形和梯形

以上效果对应的CSS依次如下&#xff0c;从左往右依次看就很直观了。 .border {width: 30px;height: 30px;margin: 10px;background-color: lightblue;&_1 {border: solid 1px #b160e7;}&_2 {border-top: solid 15px lightcoral;border-right: solid 15px lightgoldenr…

互联网、因特网、万维网的区别

互联网 internet&#xff1a;凡是能彼此通信的设备组成的网络就叫互联网&#xff0c;即使只有两台计算机&#xff0c;无论以何种技术使其彼此通信&#xff0c;都叫互联网。所以&#xff0c;根据互联网的覆盖规模可以分为&#xff1a; 局域网&#xff08;Local Area Network&am…

阿里云服务器经济型e实例特点、适用场景介绍和问题解答

阿里云服务器ECS经济型e系列是阿里云面向个人开发者、学生、小微企业&#xff0c;在中小型网站建设、开发测试、轻量级应用等场景推出的全新入门级云服务器&#xff0c;CPU处理器采用Intel Xeon Platinum架构处理器&#xff0c;支持1:1、1:2、1:4多种处理器内存配比&#xff0c…

腾讯云docker创建容器镜像及仓库

这里为了尽量简单&#xff0c;直接用腾讯云容器版本服务器 腾讯云有自己的镜像加速地址&#xff0c;速度还可以&#xff0c;单纯拉取容器还是够用的 但是当我push容器出现各种各样问题因为网络原因&#xff0c;国内访问docker官方镜像站非常麻烦&#xff0c;所以使用阿里的镜像…

储能系统--充电桩中国市场展望(四)

一、充电桩发展 充电桩产业十余年萌芽成长&#xff0c;迈入高速增长时代。2006-2015年为中国充电桩行业萌芽期&#xff0c;2006年&#xff0c;比亚迪在深圳总 部建立了第一座汽车充电站。2008年&#xff0c;北京市奥运会期间建设了国内第一个集中式充电站&#xff0c;在这个阶…

ctf.show_web

11.ctf.show_web11 解题步骤 密码为空&#xff0c;用 bp 抓包&#xff0c;去掉 session。 $password$_SESSION[password]&#xff1a;输入的password和session的结果一致 后端代码就是拿这个session的value值与我们输入的密码进行匹配, 由于这个value值我没解密出来, 所以这…

Unity中如何实现草的LOD

1&#xff09;Unity中如何实现草的LOD 2&#xff09;用Compute Shader处理图像数据后在安卓机上不能正常显示渲染纹理 3&#xff09;关于进游戏程序集加载的问题 4&#xff09;预制件编辑模式一直在触发自动保存 这是第379篇UWA技术知识分享的推送&#xff0c;精选了UWA社区的热…

Sakana 与 Jamba

这篇不是什么技术文章,入门没门槛,浅显易懂。 测试完了DBRX,还行吧,但是也没说给我带来多大惊喜,看的出来dataset选的挺好,比如中文语料的识别,也看得出来对推理做了很大的功夫,几乎所有的复杂逻辑全按COT by default呈现,这些是优点,要说缺点,没啥特点,现在说实话…

C语言:文件操作(2)

4.2 fputc的使用 这里写自定义目录标题 fputc的定义&#xff1a; 主要功能&#xff1a;一个字符一个字符的写进文件&#xff0c;将int类型的字符character写进文件流&#xff08;FILE* stream&#xff09;中&#xff0c;返回一个整形。如果成功fputc会返回写进文件的字符&…

C++STLmap,set

我最近开了几个专栏&#xff0c;诚信互三&#xff01; > |||《算法专栏》&#xff1a;&#xff1a;刷题教程来自网站《代码随想录》。||| > |||《C专栏》&#xff1a;&#xff1a;记录我学习C的经历&#xff0c;看完你一定会有收获。||| > |||《Linux专栏》&#xff1…

52 vue 中 image 资源直接使用 路径 和 使用require 的差异

前言 这也是 最近碰到的一个比较有趣的问题 是在 http 请求较多的场景下触发的情况 一般 我们的 Vue 中使用图片的地方, 一般会使用 require(“$imgPath”) 或者 “/$imgPath” 来配置图片的资源 然后 这个在目标页面 http 请求比较多的情况下, 两者 会有一些 差异, 我们…

嵌入式Qt 布局管理器QBoxLayout

一.存在问题 二.布局管理器 三.布局接口函数的使用 TestBtn1.setText("Test Button 1"); TestBtn1.setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); TestBtn1.setMinimumSize(160, 30); 使用setSizePolicy&#xff0c;那么 TestBtn1按钮 就会随着…

网页布局案例 浮动

这里主要讲浮动 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>Title</title><style>*{padding: 0;margin: 0;}.header{height: 40px;background-color: #333;}.nav{width: 1226px;heig…

计算机网络-TCP/IP 网络模型

TCP/IP网络模型各层的详细描述&#xff1a; 应用层&#xff1a;应用层为应用程序提供数据传输的服务&#xff0c;负责各种不同应用之间的协议。主要协议包括&#xff1a; HTTP&#xff1a;超文本传输协议&#xff0c;用于从web服务器传输超文本到本地浏览器的传送协议。FTP&…

二维数组定义 求和,最值,求平均值 JS

定义二维数组 二维数组的求和&#xff0c;最值&#xff0c;求平均值 Eg1 // 二维数组 const matrix [[1, 2, 3],[4, 5, 6],[7, 8, 9] ];// 初始化求和、最大值和最小值 let sum 0; let max Number.MIN_VALUE; let min Number.MAX_VALUE;// 遍历二维数组 for (let i 0; i…

yolov5+关键点检测实现溺水检测与警报提示(代码+原理)

往期热门博客项目回顾&#xff1a; 计算机视觉项目大集合 改进的yolo目标检测-测距测速 路径规划算法 图像去雨去雾目标检测测距项目 交通标志识别项目 yolo系列-重磅yolov9界面-最新的yolo 姿态识别-3d姿态识别 深度学习小白学习路线 //正文开始&#xff01; 人…

数据库mysql--------------脚本增量备份大全

目录 一、数据库上云迁移的方案&#xff1f; 1.1 方案一&#xff1a;使用脱机冷备份 1.2 方案二&#xff1a; 二、脚本增量备份 三、总结 一、数据库上云迁移的方案&#xff1f; 1.1 方案一&#xff1a;使用脱机冷备份 冷迁移----物理冷备 首先需要关闭数据库服务&#xff…

2024免费且保证100%的恢复成功率的数据恢复软件EasyRecovery

EasyRecovery是一款在市场上广受欢迎的数据恢复软件&#xff0c;具备许多强大而实用的功能。首先&#xff0c;它支持多种媒体类型的数据恢复&#xff0c;包括硬盘驱动器、存储设备、光学媒体、多媒体/移动设备以及RAID系统等。这意味着&#xff0c;无论数据是从哪种类型的设备中…

统计XML文件内标签的种类和其数量及将xml格式转换为yolov5所需的txt格式

1、统计XML文件内标签的种类和其数量 对于自己标注的数据集&#xff0c;需在标注完成后需要对标注好的XML文件校验&#xff0c;下面是代码&#xff0c;只需将SrcDir换成需要统计的xml的文件夹即可。 import os from tqdm import tqdm import xml.dom.minidomdef ReadXml(File…

【科技素养题】少儿编程 蓝桥杯青少组科技素养题 信息素养真题及解析第26套

少儿编程 科技素养 信息素养真题第26套 1、本次考试名称STEMA是STEM Assessment 的缩写。在保持第一个和最后一个字母不变的情况下,将 STEMA 的字母排列组合,一共可以组成()个与原先不同的组合。 A、5 B、6 C、12 D、20 答案:A 考点分析:主要考查小朋友们的逻辑思维…