专题:一个自制代码生成器(嵌入式脚本语言)之应用实例

初级代码游戏的专栏介绍与文章目录-CSDN博客

我的github:codetoys,所有代码都将会位于ctfc库中。已经放入库中我会指出在库中的位置。

这些代码大部分以Linux为目标但部分代码是纯C++的,可以在任何平台上使用。


专题:一个自制代码生成器(嵌入式脚本语言)之总述-CSDN博客

专题:一个自制代码生成器(嵌入式脚本语言)之对象模型-CSDN博客

专题:一个自制代码生成器(嵌入式脚本语言)之堆栈结构和总入口-CSDN博客

专题:一个自制代码生成器(嵌入式脚本语言)之核心逻辑-CSDN博客

专题:一个自制代码生成器(嵌入式脚本语言)之辅助逻辑-CSDN博客

专题:一个自制代码生成器(嵌入式脚本语言)之应用实例-CSDN博客(本篇)


目录

一、数据库模型

二、应用实例

2.1 创建模型

2.2 生成代码

2.3 生成的代码

三、如何创建模型


一、数据库模型

        数据库模型是根据数据库定义生成数据库操作相关代码的模型,生成的代码是静态代码,具有静态代码的编译时检查和运行效率。

        数据库模型支持表定义、主键、定制的DML操作。对于C++客户代码,所有操作都是基于函数的,函数名和参数清晰地表达了函数的功能。

        下面我们先了解一个应用实例,然后再研究如何编写模型。

二、应用实例

2.1 创建模型

bool Rooster_AddAllTable(CCTModel_UniversalDB* pCTM)
{CCTModel_UniversalDB::table* p;//资源类型,预定义的数值p = pCTM->AddNewTable("RESOURCE_TYPE_D", "资源类型");if (NULL == p)return false;p->AddMember("RESOURCE_TYPE_ID", "long", "资源类型");p->AddMember("COMMENT", "string", "备注");p->SetPK("RESOURCE_TYPE_ID");p->AddDML("InsertNewResourceType", "insert", "RESOURCE_TYPE_ID,COMMENT", "", "插入新资源类型");//资源使用类型,预定义的数值p = pCTM->AddNewTable("RESOURCE_USE_TYPE_D", "资源使用类型");if (NULL == p)return false;p->AddMember("RESOURCE_USE_TYPE_ID", "long", "资源使用类型ID");p->AddMember("COMMENT", "string", "备注");p->SetPK("RESOURCE_USE_TYPE_ID");p->AddDML("InsertNewResourceUseType", "insert", "RESOURCE_USE_TYPE_ID,COMMENT", "", "插入新资源使用类型");//资源p = pCTM->AddNewTable("RESOURCE_D", "资源");if (NULL == p)return false;p->AddMember("RESOURCE_ID", "long", "资源ID");p->AddMember("RESOURCE_NAME", "string", "资源名称");p->AddMember("RESOURCE_TYPE_ID", "long", "资源类型");p->AddMember("RESOURCE_WITH_SYS", "long", "资源是否区分sys/sys2");p->AddMember("COMMENT", "string", "备注");p->SetPK("RESOURCE_ID");p->AddDML("InsertNewResource", "insert", "*", "", "插入新资源");p->AddDML("GetAllResource", "select", "*", "", "获得全部");//任务定义p = pCTM->AddNewTable("TASK_D", "任务");if (NULL == p)return false;p->AddMember("TASK_ID", "long", "任务ID");p->AddMember("TASK_NAME", "string", "任务名称");p->AddMember("TASK_MULTIPLE", "long", "0非多开1多开");p->AddMember("TASK_WITH_SYS", "long", "是否区分sys/sys2");p->AddMember("TASK_PRIORITY", "long", "优先级,小的优先");p->AddMember("TASK_PLAN_TIME", "string", "执行时间,‘2,8-16’,时间点或时间段");p->AddMember("TASK_PLAN_INTERVAL", "long", "执行时间段的执行间隔(分钟0-59)");p->AddMember("TASK_START_CMD", "string", "启动命令");p->AddMember("COMMENT", "string", "备注");p->SetPK("TASK_ID");p->AddDML("InsertNewTaskDefine", "insert", "*", "", "插入新任务定义");p->AddDML("GetAllTask", "select", "*", "", "获得全部");//任务资源使用定义p = pCTM->AddNewTable("TASK_RESOURCE_USE_D", "任务资源使用");if (NULL == p)return false;p->AddMember("TASK_ID", "long", "任务ID");p->AddMember("RESOURCE_ID", "long", "资源ID");p->AddMember("RESOURCE_USE_TYPE_ID", "long", "资源使用类型ID");p->AddMember("COMMENT", "string", "备注");p->SetPK("TASK_ID,RESOURCE_ID");p->AddDML("InsertNewTaskResurceUseDefine", "insert", "TASK_ID,RESOURCE_ID,RESOURCE_USE_TYPE_ID,COMMENT", "", "插入新任务资源使用定义");p->AddDML("GetAllTaskResourceUse", "select", "*", "", "获得全部");//任务队列表p = pCTM->AddNewTable("TASK_QUEUE", "待执行的任务队列");if (NULL == p)return false;p->AddMember("TASK_ID", "long", "任务ID");p->AddMember("SYS", "long", "维度1");p->AddMember("SYS2", "long", "维度2");p->AddMember("TASK_NAME", "string", "任务名称(参考)");p->AddMember("TASK_PRIORITY", "long", "优先级,小的优先(参考)");p->AddMember("COMMENT", "string", "备注(参考)");p->AddMember("INSERT_TIME", "time", "开始时间", "", "TIME_LOG");p->SetPK("TASK_ID,SYS,SYS2");p->AddDML("InsertNewTaskQueue", "insert", "TASK_ID,SYS,SYS2,TASK_NAME,TASK_PRIORITY,COMMENT,INSERT_TIME", "", "插入新任务队列");p->AddDML("DeleteTaskQueue", "delete", "", "TASK_ID,SYS,SYS2", "删除任务队列");p->AddDML("GetAllTaskQueue", "select", "*", "", "获取全部任务队列");p->AddDML("GetTaskQueue", "select", "*", "TASK_ID,SYS,SYS2", "获取任务队列");//任务执行表p = pCTM->AddNewTable("TASK_LOG", "任务日志");if (NULL == p)return false;p->AddMember("TASK_LOG_ID", "long", "任务执行序列号SEQ_TASK_LOG");p->AddMember("TASK_ID", "long", "任务ID");p->AddMember("SYS", "long", "维度1");p->AddMember("SYS2", "long", "维度2");p->AddMember("PID", "long", "任务所在的进程,客户方写入");p->AddMember("START_TIME", "time", "开始时间", "", "TIME_LOG");p->AddMember("FINISH_TIME", "time", "结束时间", "", "TIME_LOG");p->AddMember("RETURN_VALUE", "long", "返回值 0成功 <0Rooster错误 >0任务错误");p->AddMember("UPDATE_TIME", "time", "记录更新时间", "", "TIME_LOG");p->SetPK("TASK_LOG_ID");p->AddDML("InsertNewTaskLog", "insert", "TASK_LOG_ID,TASK_ID,SYS,SYS2,START_TIME,UPDATE_TIME", "", "插入新任务日志");p->AddDML("UpdateTaskLog", "update", "PID,UPDATE_TIME", "TASK_LOG_ID", "更新任务日志", "finish_time=0");p->AddDML("FinishTaskLog", "update", "FINISH_TIME,RETURN_VALUE,UPDATE_TIME", "TASK_LOG_ID", "更新任务日志", "finish_time=0");p->AddDML("GetCurrentTask", "select", "*", "", "获取正在运行的任务", "finish_time=0");//序列if (NULL == pCTM->AddNewSequence("SEQ_TASK_LOG", "TASK日志序列", 0))return false;return true;
}

        这个代码创建了一组表和一个序列,每一行代码的功能都是显而易见的。

2.2 生成代码

int main(int argc, char** argv)
{if (!InitActiveApp("CCTModel_UniversalDB", 1024 * 1024, argc, argv))exit(1);if (sizeof(long) != 8){thelog << "非64位程序!" << ende;}G_IS_DEBUG = true;CCTModel_UniversalDB ctm;if (!Rooster_AddAllTable(&ctm)){thelog << "执行失败" << ende;return __LINE__;}if (!ctm.CreateCode("ns_rooster_struct", "RoosterStruct", ".")){thelog << "执行失败" << ende;return __LINE__;}thelog << "程序退出" << endi;return 0;
}

        有用的就两句:Rooster_AddAllTable创建模型,CreateCode生成代码。

2.3 生成的代码

        生成的一组文件:

        总入口头文件是_cc_CRoosterStruct.h:

//_cc_CRoosterStruct.h 自动生成的代码
//
// Copyright (c) ct  All rights reserved.
// 版权所有 ct 保留所有权利
////警告:本文件由CT系统自动生成,不可手工修改#ifndef CTFC_RoosterStruct_H
#define CTFC_RoosterStruct_H 1//如果需要使用内存数据库直连功能,请在包含本头文件之前定义 CTFC_RoosterStruct_H_SHM_DB
//如果需要使用MariaDB功能,请在包含本头文件之前定义 CTFC_RoosterStruct_H_MARIA_DB
#include "_cc_RoosterStruct_RESOURCE_TYPE_D.h"
#include "_cc_RoosterStruct_RESOURCE_USE_TYPE_D.h"
#include "_cc_RoosterStruct_RESOURCE_D.h"
#include "_cc_RoosterStruct_TASK_D.h"
#include "_cc_RoosterStruct_TASK_RESOURCE_USE_D.h"
#include "_cc_RoosterStruct_TASK_QUEUE.h"
#include "_cc_RoosterStruct_TASK_LOG.h"
#include "_cc_RoosterStruct_SEQ_TASK_LOG.h"using namespace ns_my_std;namespace ns_rooster_struct
{class CShmDB_RoosterStruct{public://表CRoosterStruct_RESOURCE_TYPE_D m_RESOURCE_TYPE_D;//资源类型CRoosterStruct_RESOURCE_USE_TYPE_D m_RESOURCE_USE_TYPE_D;//资源使用类型CRoosterStruct_RESOURCE_D m_RESOURCE_D;//资源CRoosterStruct_TASK_D m_TASK_D;//任务CRoosterStruct_TASK_RESOURCE_USE_D m_TASK_RESOURCE_USE_D;//任务资源使用CRoosterStruct_TASK_QUEUE m_TASK_QUEUE;//待执行的任务队列CRoosterStruct_TASK_LOG m_TASK_LOG;//任务日志//序列CRoosterStruct_SEQ_TASK_LOG m_SEQ_TASK_LOG;//TASK日志序列public:
#ifdef CTFC_RoosterStruct_H_SHM_DB
#define CTFC_RoosterStruct_H_WITH_DBtypedef CShmDB T_DB;
#endif#ifdef CTFC_RoosterStruct_H_MARIA_DB
#define CTFC_RoosterStruct_H_WITH_DBtypedef CMariaDB T_DB;
#endifprivate:
#ifdef CTFC_RoosterStruct_H_WITH_DBT_DB m_ShmDB;T_DB * pDB;
#endifpublic:
#ifdef CTFC_RoosterStruct_H_WITH_DB//返回内部数据库对象T_DB * getDB(){return pDB;}//直连初始化bool DBInit(char const * db = nullptr){
#ifdef CTFC_RoosterStruct_H_SHM_DBCShmActiveObjects * p;p = m_ShmDB.GetCShmActiveObjects();if (!p->isConnected() && !p->Attach(false))
#endif
#ifdef CTFC_RoosterStruct_H_MARIA_DBif(!m_ShmDB.Connect(db))
#endif{thelog << "CRoosterStruct连接失败" << ende;return false;}DEBUG_LOG << "CRoosterStruct连接成功" << endi;pDB = &m_ShmDB;m_RESOURCE_TYPE_D.Init(pDB);m_RESOURCE_USE_TYPE_D.Init(pDB);m_RESOURCE_D.Init(pDB);m_TASK_D.Init(pDB);m_TASK_RESOURCE_USE_D.Init(pDB);m_TASK_QUEUE.Init(pDB);m_TASK_LOG.Init(pDB);m_SEQ_TASK_LOG.Init(pDB);return true;}//直连断开bool DBUnInit(){
#ifdef CTFC_RoosterStruct_H_SHM_DBCShmActiveObjects * p;p = m_ShmDB.GetCShmActiveObjects();if (p->isConnected())p->Detach();
#endif
#ifdef CTFC_RoosterStruct_H_MARIA_DBpDB->Close();
#endifpDB = NULL;return true;}//直连创建所有数据库对象bool DBCreateAllDBObject(){//创建表if(!m_RESOURCE_TYPE_D.CreateTable())return false;if(!m_RESOURCE_USE_TYPE_D.CreateTable())return false;if(!m_RESOURCE_D.CreateTable())return false;if(!m_TASK_D.CreateTable())return false;if(!m_TASK_RESOURCE_USE_D.CreateTable())return false;if(!m_TASK_QUEUE.CreateTable())return false;if(!m_TASK_LOG.CreateTable())return false;//创建序列if(!m_SEQ_TASK_LOG.CreateSequence())return false;return true;}//直连显示所有数据库对象bool DBShowAllDBObject(){//显示表if(!m_RESOURCE_TYPE_D.Show())return false;if(!m_RESOURCE_USE_TYPE_D.Show())return false;if(!m_RESOURCE_D.Show())return false;if(!m_TASK_D.Show())return false;if(!m_TASK_RESOURCE_USE_D.Show())return false;if(!m_TASK_QUEUE.Show())return false;if(!m_TASK_LOG.Show())return false;//显示序列thelog << "sequence:" << endl << "-------------------------" << endl;theLog<< "SEQ_TASK_LOG " << m_SEQ_TASK_LOG.GetSequenceValue() << endl;theLog << "-------------------------" << endi;return true;}
#endif};
}
#endif

        这个文件看起来和普通手写的文件没有任何区别,除了它真的是用代码模板生成的。

        再来看看生成这个文件的模板_t_UniversalDB.h.ct是什么样的:

//_cc_C${SYS}.h 自动生成的代码
//
// Copyright (c) ct  All rights reserved.
// 版权所有 ct 保留所有权利
////警告:本文件由CT系统自动生成,不可手工修改#ifndef CTFC_${SYS}_H
#define CTFC_${SYS}_H 1//如果需要使用内存数据库直连功能,请在包含本头文件之前定义 CTFC_${SYS}_H_SHM_DB
//如果需要使用MariaDB功能,请在包含本头文件之前定义 CTFC_${SYS}_H_MARIA_DB
<%foreach table in ${tables}%>
#include "_cc_${SYS}_${table}.h"
<%endforeach%>
<%foreach sequence in sequences%>
#include "_cc_${SYS}_${sequence}.h"
<%endforeach%>using namespace ns_my_std;namespace ${NAMESPACE}
{class CShmDB_${SYS}{public://表<%foreach table in tables%>C${SYS}_${table} m_${table};//${table.comment}<%endforeach%>//序列<%foreach sequence in sequences%>C${SYS}_${sequence} m_${sequence};//${sequence.comment}<%endforeach%>public:
#ifdef CTFC_${SYS}_H_SHM_DB
#define CTFC_${SYS}_H_WITH_DBtypedef CShmDB T_DB;
#endif#ifdef CTFC_${SYS}_H_MARIA_DB
#define CTFC_${SYS}_H_WITH_DBtypedef CMariaDB T_DB;
#endifprivate:
#ifdef CTFC_${SYS}_H_WITH_DBT_DB m_ShmDB;T_DB * pDB;
#endifpublic:
#ifdef CTFC_${SYS}_H_WITH_DB//返回内部数据库对象T_DB * getDB(){return pDB;}//直连初始化bool DBInit(char const * db = nullptr){
#ifdef CTFC_${SYS}_H_SHM_DBCShmActiveObjects * p;p = m_ShmDB.GetCShmActiveObjects();if (!p->isConnected() && !p->Attach(false))
#endif
#ifdef CTFC_${SYS}_H_MARIA_DBif(!m_ShmDB.Connect(db))
#endif{thelog << "C${SYS}连接失败" << ende;return false;}DEBUG_LOG << "C${SYS}连接成功" << endi;pDB = &m_ShmDB;<%foreach table in tables%>m_${table}.Init(pDB);<%endforeach%><%foreach sequence in sequences%>m_${sequence}.Init(pDB);<%endforeach%>return true;}//直连断开bool DBUnInit(){
#ifdef CTFC_${SYS}_H_SHM_DBCShmActiveObjects * p;p = m_ShmDB.GetCShmActiveObjects();if (p->isConnected())p->Detach();
#endif
#ifdef CTFC_${SYS}_H_MARIA_DBpDB->Close();
#endifpDB = NULL;return true;}//直连创建所有数据库对象bool DBCreateAllDBObject(){//创建表<%foreach table in tables%>if(!m_${table}.CreateTable())return false;<%endforeach%>//创建序列<%foreach sequence in sequences%>if(!m_${sequence}.CreateSequence())return false;<%endforeach%>return true;}//直连显示所有数据库对象bool DBShowAllDBObject(){//显示表<%foreach table in tables%>if(!m_${table}.Show())return false;<%endforeach%>//显示序列thelog << "sequence:" << endl << "-------------------------" << endl;<%foreach sequence in sequences%>theLog<< "${sequence} " << m_${sequence}.GetSequenceValue() << endl;<%endforeach%>theLog << "-------------------------" << endi;return true;}
#endif};
}
#endif

        这就是一个代码模板编写的典型例子,用到了大量的循环处理和变量替换。看起来像asp?嗯,就是模仿asp啊。

三、如何创建模型

        下一篇将详细解释这个模型(原则上,除了最终生成的的是这个脚本生成器要求的对象结构外模型和脚本生成器并没有更多关联)。


(这里是结束,但不是整个系列的结束)

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

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

相关文章

信号处理--基于FBCSP滤波方法的运动想象分类

目录 理论 工具 方法 代码获取 理论 通用空间模式 (CSP) 算法可以用来有效构建最佳空间滤波器区分&#xff0c;然后实现运动想象的数据中的脑电信号的区分。然而&#xff0c;空间滤波器性能的好坏主要取决于其工作频带。如果脑电信号没有经过滤波或者滤波的频带范围不合适…

如何选择适合自己的软文推广平台

随着新媒体的兴起&#xff0c;越来越多的企业关注软文的推广&#xff0c;一篇好的软文离不开一个好的发布渠道。如何选择合适的发稿平台已经成为很多企业的痛点&#xff0c;所以我会根据自己的经验介绍一个常见的发稿平台。 1.门户网站 门户网站&#xff0c;这里就不解释哪些网…

【虹科分享】前Tableau工程师展示Domo如何与Tableau和Power BI高效结合

文章速览&#xff1a; Domo如何与Tableau协同工作如何将Domo数据集连接到Tableau工作簿如何从Domo连接到Tableau数据提取Domo如何与Power BI协同工作 现有的BI工具的不足该如何弥补&#xff0c;前Tableau工程师Tanner Brockbank的建议是&#xff0c;取长补短&#xff0c;结合…

Jenkins升级中的小问题

文章目录 使用固定版本安装根据jenkins页面下载war包升级jenkins重启jenkins报错问题解决 K8s部署过程中的一些小问题 ##### Jenkins版本小插曲 ​ 在Jenkins环境进行插件安装时全部清一色飘红&#xff0c;发现是因为Jenkins版本过低导致&#xff0c;报错的位置可以找到更新je…

【动手学深度学习】深入浅出深度学习之线性神经网络

目录 &#x1f31e;一、实验目的 &#x1f31e;二、实验准备 &#x1f31e;三、实验内容 &#x1f33c;1. 线性回归 &#x1f33b;1.1 矢量化加速 &#x1f33b;1.2 正态分布与平方损失 &#x1f33c;2. 线性回归的从零开始实现 &#x1f33b;2.1. 生成数据集 &#x…

网络安全:Kali Linux 进行SQL注入与XSS漏洞利用

目录 一、实验 1.环境 2.Kali Linux 进行SQL注入 3.Kali Linux 进行XSS漏洞利用 二、问题 1.XSS分类 2.如何修改beef-xss的密码 3.beef-xss 服务如何管理 4.运行beef报错 5.beef 命令的颜色有哪些区别 6.owasp-top-10 有哪些变化 一、实验 1.环境 &#xff08;1&a…

2024年腾讯云4核8g服务器并发数、优惠价格、支持多少人在线?

腾讯云4核8G服务器价格&#xff1a;轻量4核8G12M优惠价格646元15个月、CVM S5服务器4核8G配置1437元买1年送3个月。腾讯云4核8G服务器支持多少人同时在线&#xff1f;支持30个并发数&#xff0c;可容纳日均1万IP人数访问。腾讯云百科txybk.com整理4核8G服务器支持多少人同时在线…

Chrome之解决:浏览器插件不能使用问题(十三)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 优质专栏&#xff1a;多媒…

C程序编译、链接与项目构建

C程序编译、链接与项目构建 摘要C编译环境静、动态库介绍gcc与g和程序编译、链接Visual Studio创建和链接库动态库的显示调用Windows下显示动态库的加载/查找方式 Make介绍安装使用 CMake介绍安装使用构建方式内部构建外部构建构建使用静/动态库常用[系统]变量常用指令CMake模块…

YOLOv9改进策略:block优化 | Transformer架构ConvNeXt 网络在检测中大放异彩

&#x1f4a1;&#x1f4a1;&#x1f4a1;本文改进内容&#xff1a;Transformer架构 ConvNeXt 网络在图像分类和识别、分割领域大放异彩&#xff0c;同时对比 Swin-T 模型&#xff0c;在多种任务中其模型的大小和准确率均有一些提升&#xff0c;模型的 FLOPs 较大的减小且 Acc …

Tire树-不学面试后悔

先来一张图&#xff0c;看多少同学在面试中遇到这个题&#xff0c;然后被迫放弃&#xff0c;那就太可惜&#xff0c;因为这个题只要你往下看就会了 Trie&#xff08;发音类似 "try"&#xff09;或者说 前缀树 是一种树形数据结构&#xff0c;用于高效地存储和检索字…

Spring-ThreadLocal内存泄漏原因及解决办法

ThreadLocal原理回顾 ThreadLocal的原理&#xff1a;每个Thread内部维护着一个ThreadLocalMap&#xff0c;它是一个Map。这个映射表的Key是一个弱引用&#xff0c;其实就是ThreadLocal本身&#xff0c;Value是真正存的线程变量Object。 也就是说ThreadLocal本身并不真正存储线…

电源模块 YULIN俞霖科技DC/DC电源模块 直流升压 高压稳压

Features 最低工作电压&#xff1a;0.7V电压隔离&#xff1a;1000VDC /3000VDC 平均无故障时间&#xff1a; > 800,000 小时短路与电弧保护无最低负载要求&#xff1a;可空载工作输入电压&#xff1a;5、12、15、24VDCOutput 100,200、300、400、500 、600、800、 1000、1…

关于使用TCP-S7协议读写西门子PLC字符串的问题

我们可以使用TCP-S7协议读写西门子PLC&#xff0c; 比如PLC中定义一个String[50] 的地址DB300.20 地址DB300.20 DB块编号为300&#xff0c;偏移量【地址】是30 S7协议是西门子PLC自定义的协议&#xff0c;默认端口102&#xff0c;本质仍然是TCP协议的一种具体实现&#xff…

答题小程序功能细节揭秘:如何提升用户体验和满足用户需求?

答题小程序功能细节体现 随着移动互联网的快速发展&#xff0c;答题小程序成为了用户获取知识、娱乐休闲的重要平台。一款优秀的答题小程序不仅应该具备简洁易用的界面设计&#xff0c;更应该在功能细节上做到极致&#xff0c;以提升用户体验和满足用户需求。本文将从题库随机…

【YOLOv5改进系列(5)】高效涨点----添加密集小目标检测NWD方法

文章目录 &#x1f680;&#x1f680;&#x1f680;前言一、1️⃣ 修改loss.py文件1.1 &#x1f393; 修改11.2 ✨ 修改21.3 ⭐️相关代码的解释 二、2️⃣NWD实验2.1 &#x1f393; 实验一&#xff1a;基准模型2.2 ✨实验二&#xff1a;NWD权重设置0.52.3 ⭐️实验三&#xf…

蓝桥杯练习题——博弈论

1.必胜态后继至少存在一个必败态 2.必败态后继均为必胜态 Nim游戏 思路 2 3&#xff0c;先手必赢&#xff0c;先拿 1&#xff0c;然后变成 2 2&#xff0c;不管后手怎么拿&#xff0c;先手同样操作&#xff0c;后手一定先遇到 0 0 a1 ^ a2 ^ a3 … ^ an 0&#xff0c;先…

MySQL数据库----------探索高级SQL查询语句 (二)

目录 一、子查询 1.1多表查询 1.2多层嵌套 1.3 insert语句子查询 1.4update语句子查询 1.5delete语句子查询 1.6EXISTS 1.7子查询&#xff0c;别名as 二、mysql视图 2.1mysql视图介绍 2.2mysql作用场景[图]: 2.3视图功能&#xff1a; 2.4视图和表的区别和联系 区别…

保障校园网络安全用堡垒机的几个原因分析

校园&#xff0c;人人都熟悉的地方&#xff0c;梦想知识开始的地方。在互联网数字化快速发展的今天&#xff0c;网络安全的学习环境是非常必要的。所以采购保障校园网络安全工具是必要的。那为什么一定要用堡垒机呢&#xff1f;这里我们一起来简单分析一下原因。 保障校园网络…

前端如何判断元素是否到达可视区域

以图片显示为例&#xff1a; window.innerHeight 是浏览器可视区的高度&#xff1b;document.body.scrollTop || document.documentElement.scrollTop 是浏览器滚动的过的距离&#xff1b;imgs.offsetTop 是元素顶部距离文档顶部的高度&#xff08;包括滚动条的距离&#xff0…