QT mysql 数据库线程池 与数据库操作封装

最近事情比较多很久没有写学习笔记了,数据库线程池,+ 数据库封装,虽说数据库操作有很多不需要写sql 的,ORM 封装的方式去操作数据库。但是从业这些年一直是自己动手写sql ,还是改不了这个习惯。不说了直接上代码。

数据库线程池

ConnectionPool.h 文件#ifndef CONNECTIONPOOL_H
#define CONNECTIONPOOL_H#include <QtSql>
#include <QQueue>
#include <QString>
#include <QMutex>
#include <QMutexLocker>#include "ConnectionPool.h"class ConnectionPool {
public:static void release(); // 关闭所有的数据库连接static QSqlDatabase createConnection();                 // 获取数据库连接static void closeConnection(QSqlDatabase connection); // 释放数据库连接回连接池~ConnectionPool();private:static ConnectionPool& getInstance();ConnectionPool();ConnectionPool(const ConnectionPool &other);ConnectionPool& operator=(const ConnectionPool &other);QSqlDatabase createConnection(const QString &connectionName); // 创建数据库连接void initialize();void loadConfigFile();public:QQueue<QString> usedConnectionNames;   // 已使用的数据库连接名QQueue<QString> unusedConnectionNames; // 未使用的数据库连接名QJsonObject mJsonObject;// 数据库信息QString hostName;QString databaseName;QString username;QString password;QString databaseType;int     port;bool    testOnBorrow;    // 取得连接的时候验证连接是否有效QString testOnBorrowSql; // 测试访问数据库的 SQLint maxWaitTime;  // 获取连接最大等待时间int waitInterval; // 尝试获取连接时等待间隔时间int maxConnectionCount; // 最大连接数static QMutex mutex;static QWaitCondition waitConnection;static ConnectionPool *instance;
};#endif // CONNECTIONPOOL_H
ConnectionPool.cpp#include "ConnectionPool.h"
#include <QDebug>QMutex ConnectionPool::mutex;
QWaitCondition ConnectionPool::waitConnection;
ConnectionPool* ConnectionPool::instance = NULL;ConnectionPool::ConnectionPool()
{initialize();
}ConnectionPool::~ConnectionPool()
{// 销毁连接池的时候删除所有的连接foreach(QString connectionName, usedConnectionNames){QSqlDatabase::removeDatabase(connectionName);}foreach(QString connectionName, unusedConnectionNames){QSqlDatabase::removeDatabase(connectionName);}
}void ConnectionPool::loadConfigFile()
{QString path = qApp->applicationDirPath();QString strFile;strFile = path + "/config/DBConfig.json";QFile file(strFile);if(!file.open(QIODevice::ReadOnly | QIODevice::Text)){qDebug() << "could't open projects json";return;}QString value = file.readAll();file.close();QJsonParseError parseJsonErr;QJsonDocument document = QJsonDocument::fromJson(value.toUtf8(), &parseJsonErr);if(!(parseJsonErr.error == QJsonParseError::NoError)){qDebug() << parseJsonErr.errorString();return;}mJsonObject=document.object();
}void ConnectionPool::initialize()
{//loadConfigFile();hostName     = "127.0.0.1";databaseName = "test";username     = "root";password     = "admin";databaseType = "QMYSQL";port         = 3306;testOnBorrow = true;testOnBorrowSql = "SELECT 1";waitInterval = 200;maxWaitTime     = 5000;maxConnectionCount = 10;}ConnectionPool& ConnectionPool::getInstance()
{if (NULL == instance){QMutexLocker locker(&mutex);if (NULL == instance){instance = new ConnectionPool();}}return *instance;
}void ConnectionPool::release()
{QMutexLocker locker(&mutex);delete instance;instance = NULL;
}QSqlDatabase ConnectionPool::createConnection()
{ConnectionPool& pool = ConnectionPool::getInstance();QString connectionName;QMutexLocker locker(&mutex);// 已创建连接数int connectionCount = pool.unusedConnectionNames.size() + pool.usedConnectionNames.size();// 如果连接已经用完,等待 waitInterval 毫秒看看是否有可用连接,最长等待 maxWaitTime 毫秒for (int i = 0;i < pool.maxWaitTime&& pool.unusedConnectionNames.size() == 0 && connectionCount == pool.maxConnectionCount;i += pool.waitInterval){waitConnection.wait(&mutex, pool.waitInterval);// 重新计算已创建连接数connectionCount = pool.unusedConnectionNames.size() + pool.usedConnectionNames.size();}qDebug() << "connectionCount:"<<connectionCount;qDebug() << "pool.maxConnectionCount:"<<pool.maxConnectionCount;if (pool.unusedConnectionNames.size() > 0){// 有已经回收的连接,复用它们connectionName = pool.unusedConnectionNames.dequeue();}else if (connectionCount < pool.maxConnectionCount){// 没有已经回收的连接,但是没有达到最大连接数,则创建新的连接connectionName = QString("Connection-%1").arg(connectionCount + 1);}else{// 已经达到最大连接数qDebug() << "Cannot create more connections.";return QSqlDatabase();}// 创建连接QSqlDatabase db = pool.createConnection(connectionName);// 有效的连接才放入 usedConnectionNamesif (db.isOpen()){pool.usedConnectionNames.enqueue(connectionName);}return db;
}void ConnectionPool::closeConnection(QSqlDatabase connection)
{ConnectionPool& pool = ConnectionPool::getInstance();QString connectionName = connection.connectionName();// 如果是我们创建的连接,从 used 里删除,放入 unused 里if (pool.usedConnectionNames.contains(connectionName)){QMutexLocker locker(&mutex);pool.usedConnectionNames.removeOne(connectionName);pool.unusedConnectionNames.enqueue(connectionName);waitConnection.wakeOne();}
}QSqlDatabase ConnectionPool::createConnection(const QString &connectionName)
{// 连接已经创建过了,复用它,而不是重新创建if (QSqlDatabase::contains(connectionName)){QSqlDatabase db1 = QSqlDatabase::database(connectionName);if (testOnBorrow){// 返回连接前访问数据库,如果连接断开,重新建立连接qDebug() << "Test connection on borrow, execute:" << testOnBorrowSql << ", for" << connectionName;QSqlQuery query(testOnBorrowSql, db1);if (query.lastError().type() != QSqlError::NoError && !db1.open()){qDebug() << "Open datatabase error:" << db1.lastError().text();return QSqlDatabase();}}return db1;}// 创建一个新的连接QSqlDatabase db = QSqlDatabase::addDatabase(databaseType, connectionName);db.setHostName(hostName);db.setDatabaseName(databaseName);db.setUserName(username);db.setPassword(password);db.setPort(port);if (!db.open()){qDebug() << "Open datatabase error:" << db.lastError().text();return QSqlDatabase();}return db;
}

数据库封装:

SqlDatabase.h#ifndef SQLDATABASE_H
#define SQLDATABASE_H#include <QJsonArray>
#include <QJsonValue>
#include <QJsonObject>class  SqlDatabase
{
public:SqlDatabase();~SqlDatabase();public:static void InsertDB(QString strTableName,QJsonArray jsonValue);static void UpdateDB(QString strTableName,QJsonArray jsonValue,QString strColumnCondition);static int  InsertDB(QString strTableName,QJsonObject jsonValue);static void UpdateDB(QString strTableName,QJsonObject jsonValue,QString strCondition);static void QueryRecord(QString strSql,QJsonArray &nJsonValue);static int  QueryCount(QString strSql);static bool Delete(QString strSql);static bool Delete(QString strTableName,QString strCondition);
};#endif // SQLDATABASE_H

SqlDatabase.cpp#include "SqlDatabase.h"
#include "ConnectionPool.h"
#include <QDebug>#pragma execution_character_set("utf-8")SqlDatabase::SqlDatabase()
{}SqlDatabase::~SqlDatabase()
{ConnectionPool::Destroy();
}void SqlDatabase::InsertDB(QString strTableName,QJsonArray jsonValues)
{QSqlDatabase db =ConnectionPool::CreateConnection();QString strValues="";QString strNames="";bool iskeyName=false;for(int j=0;j<jsonValues.size();j++){QJsonObject::iterator it;QString strKeyValue;QJsonObject tmpObj =jsonValues.at(j).toObject();for(it=tmpObj.begin();it!=tmpObj.end();it++){if(strKeyValue.isEmpty()){if(it.value().isDouble()){it.value()=QString::number(it.value().toDouble(),'f',12);}strKeyValue=QString("'%1'").arg(it.value().toString());if(!iskeyName){strNames=QString("%1").arg(it.key());}}else{if(it.value().isDouble()){it.value()=QString::number(it.value().toDouble(),'f',12);}strKeyValue+=QString(",'%1'").arg(it.value().toString());if(!iskeyName){strNames+=QString(",%1").arg(it.key());}}}iskeyName =true;if(strValues.isEmpty()){strValues +="("+strKeyValue+")";}else{strValues +=",("+strKeyValue+")";}}QString strSql=QString("INSERT INTO %1 (%2)  VALUES %3 ").arg(strTableName).arg(strNames).arg(strValues);QSqlQuery query(db);if(!query.exec(strSql)){qDebug()<<"Failed to INSERT:"<<strSql;}ConnectionPool::CloseConnection(db);
}void SqlDatabase::UpdateDB(QString strTableName,QJsonArray jsonValue,QString strColumnCondition)
{QString mHeadSql= QString(" UPDATE  %1 m,( ").arg(strTableName);QString mEndSql=" ) n ";QString mSetConditionSql="";QString mValueSql="";QString mCondition="";QString strSql="";for(int i=0;i<jsonValue.size();i++){QJsonObject jsonObject=jsonValue.at(i).toObject();QJsonObject::iterator it;QString strValue="";if(!mValueSql.isEmpty()){mValueSql +=  " UNION ";}for(it=jsonObject.begin();it!=jsonObject.end();it++){if(it.value().isDouble()){it.value()=QString::number(it.value().toDouble(),'f',3);}if(strValue ==""){strValue =QString(" SELECT '%0' as `%1`").arg(it.value().toString()).arg(it.key());mSetConditionSql =QString(" SET m.%0 = n.%1").arg(it.key()).arg(it.key());}else{strValue +=QString(",'%0' as `%1`").arg(it.value().toString()).arg(it.key());mSetConditionSql += QString(" ,m.%0 = n.%1").arg(it.key()).arg(it.key());}	}mValueSql += strValue;}mCondition += QString(" WHERE m.%0 = n.%1").arg(strColumnCondition).arg(strColumnCondition);strSql =mHeadSql +mValueSql + mEndSql + mSetConditionSql+ mCondition;qDebug()<<strSql;
}int SqlDatabase::InsertDB(QString strTableName,QJsonObject jsonValue)
{QSqlDatabase db =ConnectionPool::CreateConnection();QString strValues="";QString strNames="";int nLastNum=0;QJsonObject::iterator it;for(it=jsonValue.begin();it!=jsonValue.end();it++){if(strValues.isEmpty()){if(it.value().isDouble()){it.value()=QString::number(it.value().toDouble(),'f',12);}strValues=QString("'%1'").arg(it.value().toString());strNames=QString("%1").arg(it.key());}else{if(it.value().isDouble()){it.value()=QString::number(it.value().toDouble(),'f',12);}strValues+=QString(",'%1'").arg(it.value().toString());strNames+=QString(",%1").arg(it.key());}}QString strSql=QString("INSERT INTO %1 (%2)  VALUES(%3) ").arg(strTableName).arg(strNames).arg(strValues);QSqlQuery query(db);if(!query.exec(strSql)){qDebug()<<"Failed to InsertDB:"<<query.lastError().text();}else{nLastNum=query.lastInsertId().toInt();}ConnectionPool::CloseConnection(db);return  nLastNum;
}void SqlDatabase::UpdateDB(QString strTableName,QJsonObject jsonValue,QString strCondition)
{QSqlDatabase db =ConnectionPool::CreateConnection();QString strValues="";QString strNames="";QJsonObject::iterator it;for(it=jsonValue.begin();it!=jsonValue.end();it++){if(strValues.isEmpty()){if(it.value().isDouble()){it.value()=QString::number(it.value().toDouble(),'f',12);}strValues=QString("%1='%2'").arg(it.key()).arg(it.value().toString());}else{if(it.value().isDouble()){it.value()=QString::number(it.value().toDouble(),'f',12);}strValues+=QString(",%1='%2'").arg(it.key()).arg(it.value().toString());}}QString strSql=QString("UPDATE  %1  SET %2  %3 ").arg(strTableName).arg(strValues).arg(strCondition);QSqlQuery query(db);if(!query.exec(strSql)){qDebug()<<"Failed to UpdateDB:"<<query.lastError().text();}ConnectionPool::CloseConnection(db);
}void SqlDatabase::QueryRecord(QString strSql,QJsonArray &nJsonValue)
{QSqlDatabase db =ConnectionPool::CreateConnection();QSqlQuery query(db);if(!query.exec(strSql)){qDebug()<<"Failed to QueryRecord:"<<query.lastError().text();}while(query.next()){QSqlRecord qResultRecord=query.record();QJsonObject jsonObject;for(int fileIndex =0; fileIndex<qResultRecord.count();fileIndex++){if(query.value(fileIndex).isNull()){jsonObject.insert(qResultRecord.fieldName(fileIndex),QJsonValue::Null);}else if(query.value(fileIndex).type() ==QVariant::Int){jsonObject.insert(qResultRecord.fieldName(fileIndex),query.value(fileIndex).toInt());}else if(query.value(fileIndex).type() == QVariant::Double){jsonObject.insert(qResultRecord.fieldName(fileIndex),query.value(fileIndex).toDouble());}else if(query.value(fileIndex).type() == QVariant::LongLong){jsonObject.insert(qResultRecord.fieldName(fileIndex),query.value(fileIndex).toLongLong());}else{jsonObject.insert(qResultRecord.fieldName(fileIndex),query.value(fileIndex).toString());}}nJsonValue.append(jsonObject);}ConnectionPool::CloseConnection(db);
}int SqlDatabase::QueryCount(QString strSql)
{QSqlDatabase db =ConnectionPool::CreateConnection();QSqlQuery query(db);int totalCnt=0;QString strSqlCount=QString(" SELECT COUNT(*) AS Cnt FROM  (%1) t ").arg(strSql);if(!query.exec(strSqlCount)){qDebug()<<"Failed to QueryCount:"<<query.lastError().text();}while(query.next()){totalCnt=query.value("Cnt").toInt();}ConnectionPool::CloseConnection(db);return totalCnt;
}bool SqlDatabase::Delete(QString strSql)
{bool bRet =true;QSqlDatabase db =ConnectionPool::CreateConnection();QSqlQuery query(db);if(!query.exec(strSql)){qDebug()<<"Failed to Delete:"<<query.lastError().text();bRet = false;}ConnectionPool::CloseConnection(db);return bRet;
}bool SqlDatabase::Delete(QString strTableName,QString strCondition)
{bool bRet =true;QSqlDatabase db =ConnectionPool::CreateConnection();QSqlQuery query(db);QString strSql=QString(" DELETE FROM %1 %2 ").arg(strTableName).arg(strCondition);if(!query.exec(strSql)){qDebug()<<"Failed to Delete:"<<query.lastError().text();bRet = false;}ConnectionPool::CloseConnection(db);return bRet;
}

此处说一下批量更新

源数据是这样需要更新

id:1      shorName  :aaaaa       score1:56.9   ranking  :22

id:2      shorName  :bbbbb       score1:89.9  score3:59.9  ranking  :27

id:3      shorName  :ccccc       score1:76.9   score2:81.9  ranking  :29

sql 语句 批量更新

UPDATE test m,(SELECT'1.000' AS `id`,'a' AS `name`,'22.000' AS `ranking`,'56.900' AS `score1`,'39.500' AS `score2`,'56.700' AS `score3`,'aaaa' AS `shorName` UNIONSELECT'2.000' AS `id`,'b' AS `name`,'27.000' AS `ranking`,'89.900' AS `score1`,'39.500' AS `score2`,'59.900' AS `score3`,'bbbbb' AS `shorName` UNIONSELECT'3.000' AS `id`,'c' AS `name`,'29.000' AS `ranking`,'76.900' AS `score1`,'72.900' AS `score2`,'81.900' AS `score3`,'ccccc' AS `shorName` ) n SET m.id = n.id,m.NAME = n.NAME,m.ranking = n.ranking,m.score1 = n.score1,m.score2 = n.score2,m.score3 = n.score3,m.shorName = n.shorName 
WHEREm.id = n.id

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

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

相关文章

【C语言】数据结构——栈和队列实例探究

&#x1f497;个人主页&#x1f497; ⭐个人专栏——数据结构学习⭐ &#x1f4ab;点击关注&#x1f929;一起学习C语言&#x1f4af;&#x1f4ab; 目录 导读&#xff1a;一、 栈1. 栈的概念及结构2. 栈的实现3. 实现代码3.1 定义结构体3.2 初始化栈3.3 销毁栈3.4 入栈3.5 出栈…

Rust生态系统:探索常用的库和框架

大家好&#xff01;我是lincyang。 今天我们来探索Rust的生态系统&#xff0c;特别是其中的一些常用库和框架。 Rust生态系统虽然相比于一些更成熟的语言还在成长阶段&#xff0c;但已经有很多强大的工具和库支持各种应用的开发。 常用的Rust库和框架 Serde&#xff1a;一个…

python pymodbus库使用教程(以Modbus RTU为例)

文档&#xff1a; https://pymodbus.readthedocs.io/en/latest/ 源码&#xff1a; https://github.com/riptideio/pymodbus/ 文章目录 Python PyModbus库使用教程&#xff1a;以Modbus RTU为例介绍安装PyModbus配置串行连接导入必要的模块创建Modbus客户端实例 建立连接连接…

CQ 社区版 V2.6.0 发布 | SQL闪回、权限看板、新增数据源人大金仓等

前言 HELLO&#xff0c;大家好&#xff0c;又到了 CloudQuery 社区版发版时间&#xff01;本次更新版本为 v2.6.0&#xff0c;亮点多多&#xff0c;我们直入主题一起来看&#xff01; 一、本期亮点 新增 3 种数据源支持 V2.6.0&#xff0c;新增三种国产数据源支持&#xff…

【广州华锐互动】VR溺水预防教育:在虚拟世界中学会自救!

在现代社会中&#xff0c;水上安全和救援行动的重要性不言而喻。尤其在自然灾害、游泳事故或航海事故中&#xff0c;有效的救援行动可以挽救许多生命。然而&#xff0c;传统的救援训练往往存在成本高、风险大、效率低等问题。在这样的背景下&#xff0c;虚拟现实&#xff08;VR…

为什么要写测试用例,测试用例写给谁看?

&#x1f4e2;专注于分享软件测试干货内容&#xff0c;欢迎点赞 &#x1f44d; 收藏 ⭐留言 &#x1f4dd; 如有错误敬请指正&#xff01;&#x1f4e2;交流讨论&#xff1a;欢迎加入我们一起学习&#xff01;&#x1f4e2;资源分享&#xff1a;耗时200小时精选的「软件测试」资…

buildAdmin 后端控制器的代码分析

buildAdmin的代码生成&#xff0c;很像是 fastadmin 的生成模式&#xff0c;当我们利用数据库生成了一个控制器的时候&#xff0c;我们可以看到&#xff0c; 它的生成代码很简洁 <?phpnamespace app\admin\controller\askanswer;use app\common\controller\Backend;/*** 回…

数据中台之用户画像

用户画像应用领域较为广泛,适合于各个产品周期,从新用户的引流到潜在用户的挖掘、 从老用户 的培养到流失用户的回流等。通过挖掘用户兴趣、偏好、人口统计特征,可以 直接 作用于提升营销精准 度、推荐匹配度,最终提升产品服务和企业利润。还包括广告投放、产品布局和行业报…

webshell之无扩展免杀

1.php加密 这里是利用phpjiami网站进行加密&#xff0c;进而达到加密效果 加密前&#xff1a; 查杀效果 可以看到这里D某和某狗都查杀 里用php加密后效果 查杀效果 可以看到这里只有D某会显示加密脚本&#xff0c;而某狗直接绕过 2.dezend加密 可以看到dezend加密的特征还是…

SA实战 ·《SpringCloud Alibaba实战》第14章-服务网关加餐:SpringCloud Gateway核心技术

大家好,我是冰河~~ 一不小心《SpringCloud Alibaba实战》专栏都更新到第14章了,再不上车就跟不上了,小伙伴们快跟上啊! 在《SpringCloud Alibaba实战》专栏前面的文章中,我们实现了用户微服务、商品微服务和订单微服务之间的远程调用,并且实现了服务调用的负载均衡。也基…

联想拯救者Lenovo Legion R9000K 2021H(82N6)原装出厂Windows10/Win11系统ISO镜像

链接&#xff1a;https://pan.baidu.com/s/13NkeCXNdV0Ib5eeRnZUeAQ?pwdnlr7 提取码&#xff1a;nlr7 拯救者笔记本电脑原厂WIN系统自带所有驱动、出厂主题壁纸、系统属性专属LOGO标志、Office办公软件、联想电脑管家等预装程序 所需要工具&#xff1a;16G或以上的U盘 文…

第15届蓝桥杯Scratch选拔赛中级(STEMA)真题2023年10月

一、单选题 1.运行以下哪个程序后&#xff0c;巨嘴鸟会向下移动&#xff1f;&#xff08; &#xff09; A. B. C. D. 2.运行以下程序后&#xff0c; 能看到几只河豚鱼&#xff08; &#xff09;&#xff1f; A.3 B.4 C.6 D.7 3.以下运算结果为“False”的是&#xff08…

财报解读:电商GMV增长30%后,快手将坚守本地生活?

快手逐渐讲好了其高质量成长的故事。 根据财报&#xff0c;快手三季度业绩超出预期&#xff0c;其中&#xff0c;营收279.5亿元&#xff0c;同比增长20.8%&#xff1b;调整后净利润31.7亿元&#xff0c;同比扭亏为盈。 而联系市场环境来看&#xff0c;三季度广告、电商市场较…

webpack环境变量的设置

现在虽然vite比较流行&#xff0c;但对于用node写后端来说&#xff0c;webpack倒是成了一个很好的打包工具&#xff0c;可以很好的保护后端的代码。所以这块的学习还是不能停下来&#xff0c;接下来我们来针对不同的环境做不同的设置写好笔记。 引用场景主要是针对服务器的各种…

设计师不能忽视的几个宝藏图标设计工具

在这个快速变化的时代&#xff0c;设计师对创新和实用工具的需求越来越大。这就要求我们及时跟上潮流&#xff0c;不断探索和尝试最新、最有价值的图标设计工具。只有这样&#xff0c;我们才能在竞争激烈的设计市场中脱颖而出。以下是我们精心挑选的2024年值得一试的图标设计工…

MySQL数据库:开源且强大的关系型数据库管理系统

大家好&#xff0c;我是咕噜-凯撒&#xff0c;数据在当今信息化时代的重要性不可忽视。作为企业和组织的重要资产&#xff0c;数据的管理和存储变得至关重要&#xff0c;MySQL作为一种关系型数据库管理系统&#xff0c;具有非常多的优势&#xff0c;下面简单的探讨一下MySQL数据…

labview 安捷伦 34970A 采集温度等

本文详细描述了怎么用安捷伦34970A采集温度&#xff0c;并列出了labview的下载链接&#xff0c;具有一定的参考价值。 1.必要条件&#xff1a; RS-232电缆一根 IO Libraries Suite 软件 BenchLink Data Logger 软件 软件可以在http://www.keysight.com.cn下载 检查RS-232…

Hive安装配置 - 本地模式

文章目录 一、Hive运行模式二、安装配置本地模式Hive&#xff08;一&#xff09;安装配置MySQL1、删除系统自带的MariaDB2、上传MySQL组件到虚拟机3、在主节点上安装MySQL组件4、在主节点上配置MySQL&#xff08;1&#xff09;查看MySQL服务状态&#xff08;2&#xff09;查看M…

笔记本外接显示器的一些基本操作

1>&#xff0c;安装问题直接问客服&#xff0c;正常情况是将显示屏接上电源&#xff0c;然后用先将显示屏和笔记本的HDMI接口连接即可。 按下组合键 win p ,选择 “复制”。 2>&#xff0c;接上显示屏后&#xff0c;原笔记本无声音&#xff1f; 1、找到笔记本电脑右下…

2023“亚太杯”大学生数学建模竞赛

2023亚太杯数学建模C题 中国新能源电动汽车的发展趋势 解题思路、数据 该题并没有提供数据集&#xff0c;对所需数据进行收集整理是对题目进行求解的基础。在本题中&#xff0c;主要需要以下数据&#xff1a;新能源汽车历史销售量、新能汽车相关专利的历史数量、充电桩历史数…