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

初级代码游戏的专栏介绍与文章目录-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://xiahunao.cn/news/2904985.html

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

相关文章

Day29 集合的常用类

Day29 集合的常用类 文章目录 Day29 集合的常用类一、Collections二、 ConcurrentHashMap三、HashMap vs LinkedHashMap vs Hashtable vs ConcurrentHashMap四、LinkedHashMap五、Properties 一、Collections 1、概念&#xff1a; java.util.Collections是Java集合框架中的一…

使用C++ 20协程实现Raft共识算法

本文描述了如何在不使用任何额外库的情况下在c 20中实现Raft Server共识模块。文章分为三个主要部分: Raft算法的全面概述关于Raft服务器开发的详细说明对基于协程的自定义网络库的描述 该实现利用了C 20的强大功能&#xff0c;特别是协同程序&#xff0c;为构建分布式系统的…

RecyclerView 调用 notifyItemInserted 自动滚动到底部的问题

项目中发现一个奇怪的现象 RecyclerView 加载完数据以后&#xff0c;调用 notifyItemInserted 方法&#xff0c;RecyclerView 会滑动到底部。 简化后的效果图&#xff1a; 因为这个 RecyclerView 的适配器有一个 FootViewHolder&#xff0c;所以怀疑是 FootViewHolder 的问题…

车载以太网AVB交换机 gptp透明时钟 5口 全千兆 SW1500

全千兆车载以太网交换机 一、产品简要分析 5端口千兆车载以太网交换机&#xff0c;包含4个通道的1000BASE-T1接口使用罗森博格H-MTD和泰科MATEnet双接口&#xff0c;1个通道1000BASE-T标准以太网(RJ45接口)&#xff0c;可以实现车载以太网多通道交换&#xff0c;千兆和百兆车载…

GPT-1原理-Improving Language Understanding by Generative Pre-Training

文章目录 前言提出动机模型猜想模型提出模型结构模型参数 模型预训练训练的目标训练方式训练参数预训练数据集预训练疑问点 模型微调模型输入范式模型训练微调建议微调疑问点 实验结果分析 前言 首先想感慨一波 这是当下最流行的大模型的的开篇之作&#xff0c;由OpenAI提出。…

蓝桥杯-卡片换位

solution 有一个测试点没有空格&#xff0c;要特别处理&#xff0c;否则会有一个测试点运行错误&#xff01; 还有输入数据的规模在变&#xff0c;小心顺手敲错了边界条件 #include<iostream> #include<string> #include<queue> #include<map> #incl…

持续集成流程主要系统构成介绍(CI)

目录 一、概述 二、版本控制系统 2.1 概述 2.2 版本控制系统使用流程示意图 2.3 版本控制软件划分 2.3.1 集中式版本控制软件 2.3.2 分布式版本控制软件 2.3.3 总结 2.4 常用版本控制软件介绍 三、编译构建系统 3.1 概述 3.2 编译构建流程示意图 3.3 列举Java 源码…

Kafka重要配置参数全面解读(重要)

欢迎来到我的博客&#xff0c;代码的世界里&#xff0c;每一行都是一个故事 Kafka重要配置参数全面解读(重要 前言auto.create.topics.enableauto.leader.rebalance.enablelog.retention.{hour|minutes|ms}offsets.topic.num.partitions 和 offsets.topic.replication.factorlo…

回文数个数-第12届蓝桥杯选拔赛Python真题精选

[导读]&#xff1a;超平老师的Scratch蓝桥杯真题解读系列在推出之后&#xff0c;受到了广大老师和家长的好评&#xff0c;非常感谢各位的认可和厚爱。作为回馈&#xff0c;超平老师计划推出《Python蓝桥杯真题解析100讲》&#xff0c;这是解读系列的第43讲。 回文数个数&#…

macOS Sonoma如何查看隐藏文件

在使用Git进行项目版本控制时&#xff0c;我们可能会遇到一些隐藏文件&#xff0c;比如.gitkeep文件。它通常出现在Git项目的子目录中&#xff0c;主要作用是确保空目录也可以被跟踪。 终端命令 在尝试查看.gitkeep文件时&#xff0c;使用Terminal命令来显示隐藏文件 default…

推动制药行业数字化转型:基于超融合架构的MES一体机解决方案

随着中国对信息化重视程度的不断加深&#xff0c;制药行业作为国民经济的重要支柱之一&#xff0c;也在积极寻求通过数字化手段提升产业效率与产品质量。自党的十六大提出“以信息化带动工业化”的战略以来&#xff0c;制药业的这一转型探索尤为迫切。 在现代制药生产中&#…

C# OpenCv Haar、LBP 人脸检测

目录 效果 代码 下载 效果 代码 using OpenCvSharp;namespace OPenCVDemo {class Program{static void Main(string[] args){// Load the cascadesvar haarCascade new CascadeClassifier("haarcascade_frontalface_default.xml");var lbpCascade new Casca…

CSS(六)

一、精灵图 1.1 为什么需要精灵图 一个网页中往往会应用很多小的背景图像作为修饰&#xff0c;当网页中的图像过多时&#xff0c;服务器就会频繁地接收和发送请求图片&#xff0c;造成服务器请求压力过大&#xff0c;这将大大降低页面的加载速度。 因此&#xff0c;为了有效…

国外的Java面试题和国内的相比谁更卷

前言 有很多朋友很好奇国外的Java面试题长啥样&#xff0c;今天我们就去找5道国外的面试来和国内的对比一下看看谁难一些&#xff01; 面试题分享 1. Is Java Platform Independent if then how?&#xff08; Java平台是独立的吗&#xff1f;&#xff09; Yes, Java is a…

【氮化镓】位错对氮化镓(GaN)电子能量损失谱(EEL)的影响

本文献《Influence of dislocations on electron energy-loss spectra in gallium nitride》由C. J. Fall等人撰写&#xff0c;发表于2002年。研究团队通过第一性原理计算&#xff0c;探讨了位错对氮化镓&#xff08;GaN&#xff09;电子能量损失谱&#xff08;EEL&#xff09;…

大话设计模式之迪米特法则

迪米特法则&#xff0c;也称为最少知识原则&#xff08;Law of Demeter&#xff09;&#xff0c;是面向对象设计中的一个重要原则&#xff0c;其核心思想是降低耦合度、减少对象之间的依赖关系&#xff0c;从而使系统更加灵活、易于维护和扩展。 根据迪米特法则&#xff0c;一…

librdkafka的简单使用

文章目录 摘要kafka是什么安装环境librdkafka的简单使用生产者消费者 摘要 本文是Getting Started with Apache Kafka and C/C的中文版&#xff0c; kafka的hello world程序。 本文完整代码见仓库&#xff0c;这里只列出producer/consumer的代码 kafka是什么 本节来源&#…

Python图像处理——计算机视觉中常用的图像预处理

概述 在计算机视觉项目中&#xff0c;使用样本时经常会遇到图像样本不统一的问题&#xff0c;比如图像质量&#xff0c;并非所有的图像都具有相同的质量水平。在开始训练模型或运行算法之前&#xff0c;通常需要对图像进行预处理&#xff0c;以确保获得最佳的结果。图像预处理…

StarRocks实战——多点大数据数仓构建

目录 前言 一、背景介绍 二、原有架构的痛点 2.1 技术成本 2.2 开发成本 2.2.1 离线 T1 更新的分析场景 2.2.2 实时更新分析场景 2.2.3 固定维度分析场景 2.2.4 运维成本 三、选择StarRocks的原因 3.1 引擎收敛 3.2 “大宽表”模型替换 3.3 简化Lambda架构 3.4 模…

jmeter总结之:Regular Expression Extractor元件

Regular Expression Extractor是一个后处理器元件&#xff0c;使用正则从服务器的响应中提取数据&#xff0c;并将这些数据保存到JMeter变量中&#xff0c;以便在后续的请求或断言中使用。在处理动态数据或验证响应中的特定信息时很有用。 添加Regular Expression Extractor元…