Sqlite3 常见几种数据插入方式性能比较

2023-11-19

#include <QCoreApplication>
#include "mysqliteop.h"
#include <QDateTime>
#include <QDebug>
#include <QFile>
#include <QDataStream>
#include <string>
#include <vector>
using namespace std;

#define CHECKZERO(a) if((a)!= SQLITE_OK) throw("error.");

// 无优化
auto f1 = [](){
    sqlite3* db = nullptr;
    const char* path = "./perftest.db";
    CHECKZERO(sqlite3_open(path, &db));
    CHECKZERO(sqlite3_exec(db, "CREATE TABLE IF not exists Test(ID INTEGER,var0 INTEGER,var1 REAL,var2 TEXT);", 0, 0, 0));
    const int maxcount = 100;
    for (int i = 0; i < maxcount; i++) {
        CHECKZERO(sqlite3_exec(db, "INSERT INTO Test (ID,var0,var1,var2) VALUES (0,1,2.0,'\"hello sqlite3.\"');", 0, 0, 0));
    }
    CHECKZERO(sqlite3_close(db));
};

//关闭写同步
auto f2 = [](){
    sqlite3* db = nullptr;
    const char* path = "./perftest.db";
    CHECKZERO(sqlite3_open(path, &db));
    CHECKZERO(sqlite3_exec(db, "PRAGMA synchronous = OFF", 0, 0, 0));
    CHECKZERO(sqlite3_exec(db, "CREATE TABLE IF not exists Test(ID INTEGER,var0 INTEGER,var1 REAL,var2 TEXT);", 0, 0, 0));
    const int maxcount = 10000;
    for (int i = 0; i < maxcount; i++) {
        CHECKZERO(sqlite3_exec(db, "INSERT INTO Test (ID,var0,var1,var2) VALUES (0,1,2.0,'\"hello sqlite3.\"');", 0, 0, 0));
    }
    CHECKZERO(sqlite3_close(db));
};

//开启事务
auto f3 = [](){
    sqlite3* db = nullptr;
    const char* path = "./perftest.db";
    CHECKZERO(sqlite3_open(path, &db));
    CHECKZERO(sqlite3_exec(db, "PRAGMA synchronous = OFF", 0, 0, 0));
    CHECKZERO(sqlite3_exec(db, "CREATE TABLE IF not exists Test(ID INTEGER,var0 INTEGER,var1 REAL,var2 TEXT);", 0, 0, 0));
    CHECKZERO(sqlite3_exec(db, "BEGIN", 0, 0, 0));
    const int maxcount = 1000000;
    for (int i = 0; i < maxcount; i++) {
        CHECKZERO(sqlite3_exec(db, "INSERT INTO Test (ID,var0,var1,var2) VALUES (0,1,2.0,'\"hello sqlite3.\"');", 0, 0, 0));
        if (i % 10000 == 9999) {
            CHECKZERO(sqlite3_exec(db, "COMMIT", 0, 0, 0));
            CHECKZERO(sqlite3_exec(db, "BEGIN", 0, 0, 0));
        }
    }
    CHECKZERO(sqlite3_exec(db, "COMMIT", 0, 0, 0));
    CHECKZERO(sqlite3_close(db));
};


auto f4 = [](){
    //执行准备
    sqlite3* db = nullptr;
    const char* path = "./perftest.db";
    CHECKZERO(sqlite3_open(path, &db));
    CHECKZERO(sqlite3_exec(db, "PRAGMA synchronous = OFF", 0, 0, 0));
    CHECKZERO(sqlite3_exec(db, "CREATE TABLE IF not exists Test(ID INTEGER,var0 INTEGER,var1 REAL,var2 TEXT);", 0, 0, 0));
    // 执行准备
    sqlite3_stmt *pPrepare = nullptr;
    auto sql = "INSERT INTO Test (ID,var0,var1,var2) VALUES (?,?,?,?);";
    CHECKZERO(sqlite3_prepare_v2(db, sql, strlen(sql), &pPrepare, 0));
    CHECKZERO(sqlite3_exec(db, "BEGIN", 0, 0, 0));
    const int maxcount = 1000000;
    for (int i = 0; i < maxcount; i++) {
        CHECKZERO(sqlite3_reset(pPrepare));
        CHECKZERO(sqlite3_bind_int(pPrepare, 1, 0));
        CHECKZERO(sqlite3_bind_int(pPrepare, 2, 1));
        CHECKZERO(sqlite3_bind_double(pPrepare, 3, 2.0));
        const char* str = "hello sqlite3.";
        CHECKZERO(sqlite3_bind_text(pPrepare, 4, str, strlen(str), 0));
        int err = sqlite3_step(pPrepare);
        assert(SQLITE_DONE == err);
        if (i % 10000 == 9999) {
            CHECKZERO(sqlite3_exec(db, "COMMIT", 0, 0, 0));
            CHECKZERO(sqlite3_exec(db, "BEGIN", 0, 0, 0));
        }
    }
    CHECKZERO(sqlite3_exec(db, "COMMIT", 0, 0, 0));
    CHECKZERO(sqlite3_finalize(pPrepare)); // 释放
    CHECKZERO(sqlite3_close(db));
};


auto f5 = [](){
    //内存模式
    sqlite3* db = nullptr;
    const char* path = "./perftest.db";
    CHECKZERO(sqlite3_open(":memory:", &db));
    CHECKZERO(sqlite3_exec(db, "PRAGMA synchronous = OFF", 0, 0, 0));
    CHECKZERO(sqlite3_exec(db, "CREATE TABLE IF not exists Test(ID INTEGER,var0 INTEGER,var1 REAL,var2 TEXT);", 0, 0, 0));
    // 执行准备
    sqlite3_stmt *pPrepare = nullptr;
    auto sql = "INSERT INTO Test (ID,var0,var1,var2) VALUES (?,?,?,?);";
    CHECKZERO(sqlite3_prepare_v2(db, sql, strlen(sql), &pPrepare, 0));
    CHECKZERO(sqlite3_exec(db, "BEGIN", 0, 0, 0));
    const int maxcount = 10000000;
    for (int i = 0; i < maxcount; i++) {
        CHECKZERO(sqlite3_reset(pPrepare));
        CHECKZERO(sqlite3_bind_int(pPrepare, 1, 0));
        CHECKZERO(sqlite3_bind_int(pPrepare, 2, 1));
        CHECKZERO(sqlite3_bind_double(pPrepare, 3, 2.0));
        const char* str = "hello sqlite3.";
        CHECKZERO(sqlite3_bind_text(pPrepare, 4, str, strlen(str), 0));

        int err = sqlite3_step(pPrepare);
        assert(SQLITE_DONE == err);
        if (i % 10000 == 9999) {
            CHECKZERO(sqlite3_exec(db, "COMMIT", 0, 0, 0));
            CHECKZERO(sqlite3_exec(db, "BEGIN", 0, 0, 0));
        }
    }
    CHECKZERO(sqlite3_exec(db, "COMMIT", 0, 0, 0));
    CHECKZERO(sqlite3_finalize(pPrepare)); // 释放
    // 导出
    CHECKZERO(sqlite3_exec(db, "VACUUM INTO 'out.db3';", 0, 0, 0));
    CHECKZERO(sqlite3_close(db));
};


auto f6 = [](){
    //执行准备
    sqlite3* db = nullptr;
    const char* path = "./perftest.db";
    CHECKZERO(sqlite3_open(path, &db));
    CHECKZERO(sqlite3_exec(db, "PRAGMA synchronous = OFF", 0, 0, 0));
    CHECKZERO(sqlite3_exec(db, "CREATE TABLE IF not exists Test(ID INTEGER,var0 INTEGER,var1 REAL,var2 BLOB);", 0, 0, 0));

    // 执行准备
    sqlite3_stmt *pPrepare = nullptr;
    auto sql = "INSERT INTO Test (ID,var0,var1,var2) VALUES (?,?,?,?);";
    CHECKZERO(sqlite3_prepare_v2(db, sql, strlen(sql), &pPrepare, 0));
    CHECKZERO(sqlite3_exec(db, "BEGIN", 0, 0, 0));

    CHECKZERO(sqlite3_reset(pPrepare));
    CHECKZERO(sqlite3_bind_int(pPrepare, 1, 0));
    CHECKZERO(sqlite3_bind_int(pPrepare, 2, 1));
    CHECKZERO(sqlite3_bind_double(pPrepare, 3, 2.0));

    QFile file("./stitching_gray.bmp");
    if(!file.open(QIODevice::ReadOnly)){return -1;}
    QByteArray bytes = file.readAll();
    file.close();

    QDateTime tb = QDateTime::currentDateTime();
    CHECKZERO(sqlite3_bind_blob(pPrepare, 4, bytes, bytes.length(), SQLITE_STATIC));
    int err = sqlite3_step(pPrepare);
    assert(SQLITE_DONE == err);

    CHECKZERO(sqlite3_exec(db, "COMMIT", 0, 0, 0));
    CHECKZERO(sqlite3_finalize(pPrepare)); // 释放

    QDateTime te = QDateTime::currentDateTime();
    qDebug() << "Just insert :" << tb.toString() << " <==> " << te.toString() << (te.toMSecsSinceEpoch()  - tb.toMSecsSinceEpoch()) << "ms";

    CHECKZERO(sqlite3_close(db));
};



auto f_query = [](){
    // query using one bind parameter

    sqlite3_stmt * pStmt = nullptr;
    sqlite3* pDB = nullptr;
    const char* path = "./perftest.db";
    CHECKZERO(sqlite3_open(path, &pDB));

    // CHECKZERO(sqlite3_exec(pDB, "PRAGMA synchronous = OFF", 0, 0, 0));

    std::string str = "SELECT var2 FROM Test WHERE ID=?1";
    int iReturn = sqlite3_prepare_v2(pDB, str.c_str(), str.size() + 1, &pStmt, nullptr);
    if (iReturn != SQLITE_OK) {
        return 1;
    }

    printf("The statement %s has %d parameters(s).\n", str.c_str(), sqlite3_bind_parameter_count(pStmt));

    // fourth parameter is length = position of \0
    iReturn = sqlite3_bind_int(pStmt, 1, 0);
    if (iReturn != SQLITE_OK) {
        return 1;
    }

    std::vector<std::string> vecResults;
    char cBuffer[1024*100]={0};
    string strBuffer;
    while (sqlite3_step(pStmt) == SQLITE_ROW)
    {
        sprintf(cBuffer, "%s", sqlite3_column_blob(pStmt, 0));
        printf("%s\n",cBuffer);
        strBuffer = cBuffer;
        vecResults.push_back(strBuffer);
    }

    sqlite3_finalize(pStmt);
    CHECKZERO(sqlite3_close(pDB));

    printf("Found %d results.\n", vecResults.size());
    for (unsigned int i = 0, iEnd = vecResults.size(); i != iEnd; ++i)
    {
        printf("%d: %s\n", i, vecResults[i].c_str());
    }

} ;



/// test_insert
auto f_insert_test = [](){
    // 执行准备
    sqlite3* db = nullptr;
    const char* path = "./perftest.db";
    CHECKZERO(sqlite3_open(path, &db));
    CHECKZERO(sqlite3_exec(db, "PRAGMA synchronous = OFF", 0, 0, 0));
    CHECKZERO(sqlite3_exec(db, "CREATE TABLE IF not exists Test(ID INTEGER,var0 INTEGER,var1 REAL,var2 BLOB);", 0, 0, 0));

    // 执行准备
    sqlite3_stmt *pPrepare = nullptr;
    auto sql = "INSERT INTO Test (ID,var0,var1,var2) VALUES (?,?,?,?);";
    CHECKZERO(sqlite3_prepare_v2(db, sql, strlen(sql), &pPrepare, 0));
    CHECKZERO(sqlite3_exec(db, "BEGIN", 0, 0, 0));

    QFile file("./stitching_gray.bmp");
    if(!file.open(QIODevice::ReadOnly))return -1;
    QByteArray bytes = file.readAll();
    file.close();

    CHECKZERO(sqlite3_reset(pPrepare));
    CHECKZERO(sqlite3_bind_int(pPrepare, 1, 0));
    CHECKZERO(sqlite3_bind_int(pPrepare, 2, 1));
    CHECKZERO(sqlite3_bind_zeroblob(pPrepare, 3,  file.size() /*bytes.length()*/));

    int err = sqlite3_step(pPrepare);
    assert(SQLITE_DONE == err);
    CHECKZERO(sqlite3_exec(db, "COMMIT", 0, 0, 0));
    CHECKZERO(sqlite3_finalize(pPrepare)); // 释放

    QDateTime tb = QDateTime::currentDateTime();
    sqlite3_blob* blob = NULL;
    sqlite3_blob_open(db,path,"Test","var2",1,1,&blob);
    sqlite3_blob_write(blob,bytes,bytes.length(),0);

    QDateTime te = QDateTime::currentDateTime();
    qDebug() << "Just insert :" << tb.toString() << " <==> " << te.toString() << (te.toMSecsSinceEpoch()  - tb.toMSecsSinceEpoch()) << "ms";

    CHECKZERO(sqlite3_blob_close(blob));
    CHECKZERO(sqlite3_close(db));
};


auto f_preftest = [](std::function<void()> f){
    QDateTime tb = QDateTime::currentDateTime();
    f();
    QDateTime te = QDateTime::currentDateTime();

    qDebug() << tb.toString() << " <==> " << te.toString() << (te.toMSecsSinceEpoch()  - tb.toMSecsSinceEpoch()) << "ms";
};


int main(int argc,char* argv[])
{

    /// 性能測試
    /// 無優化
    //f_preftest(f1);  // "周三 12月 7 09:59:14 2022"  <==>  "周三 12月 7 09:59:26 2022" 11442 ms(百)

    /// 關閉讀寫同步
    //f_preftest(f2);  // "周三 12月 7 10:00:32 2022"  <==>    "周三 12月 7 10:00:43 2022" 11262 ms(萬)

    /// 開啟事務
    // f_preftest(f3); //"周三 12月 7 10:03:08 2022"  <==>  "周三 12月 7 10:03:12 2022" 3209 ms(百萬)

    /// 關閉讀寫同步 + 開啟事務
    // f_preftest(f4); // "周三 12月 7 10:08:32 2022"  <==>  "周三 12月 7 10:08:40 2022" 8260 ms (千萬) (734 ms 百萬)

    /// 內存模式
     //f_preftest(f5);  // 7110ms (千万)

    /// 大數據寫入
    f_preftest(f6); // 105 ms

    /// 大數據讀
    // f_preftest(f_query); // 10ms

    //f_preftest(f_insert_test);
}

总结: 内存模式最快,千万条在7110ms,每条不到7ms;其次是开始事务并关闭读写同步模式;
对于秒级百万条数据插入来说毫无压力。后面我会对sqlite3的源码进行由浅到深的剖析。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Sqlite3 常见几种数据插入方式性能比较 的相关文章

  • 在 SQL 中按键组对行进行顺序编号?

    SQL中有没有办法按顺序添加行号按关键组 假设一个表包含任意 CODE NAME 元组 示例表 CODE NAME A Apple A Angel A Arizona B Bravo C Charlie C Cat D Dog D Dopp
  • Qt:如何连接到 SQLite?

    我安装了 SQLite3 解压到 c sqlite 创建了一个数据库 c sqlite mzsales 现在我试图在 QTableView 中显示其内容 QSqlDatabase db QSqlDatabase addDatabase QS
  • Django 中从 sqlite 迁移到 postgresql

    我想迁移自sqlite to PostgreSQL db 我安装了 postgresql 并在其 shell 上创建数据库 然后配置我的 django 设置如下 default ENGINE django db backends postg
  • Android中不同线程的数据库访问

    我有一个在 AsyncTasks 中从互联网下载数据的服务 它解析数据并将其存储在数据库中 该服务持续运行 当服务写入数据库时 活动会尝试从数据库中读取更改 我有一个数据库助手 有多种写入和读取方法 这会导致问题吗 可能尝试从两个不同的线程
  • hive sql查找最新记录

    该表是 create table test id string name string age string modified string 像这样的数据 id name age modifed 1 a 10 2011 11 11 11 1
  • java库维护数据库结构

    我的应用程序一直在开发 所以偶尔 当版本升级时 需要创建 更改 删除一些表 修改一些数据等 通常需要执行一些sql代码 是否有一个 Java 库可用于使我的数据库结构保持最新 通过分析类似 db structure version 信息并执
  • 分组和切换列和行

    我不知道这是否会被正式称为枢轴 但我想要的结果是这样的 Alex Charley Liza 213 345 1 23 111 5 42 52 2 323 5 23 1 324 5 我的输入数据采用这种形式 Apt Name
  • 一个表可以有多个主键吗?

    我现在很困惑 也许你可以帮助我更好地理解这个问题 即一个表可以有两个主键 如果是 那么如何 如果没有 那为什么 您询问是否可以有多个主键field你当然可以 您只能有一个主键 但它可以包含唯一标识行所需的任意数量的列 创建表时使用类似这样的
  • SQLite 返回错误的 2013 年周数?

    我有一个简单的 SQL 来计算 SQLite 报告中的周数 SELECT STRFTIME W date column 2009 2012 年的数据是正确的 2013 年我总是得到错误的周数 例如 SELECT STRFTIME W 201
  • 使用联接更新表?

    我正在尝试使用表 B 中的数据更新表 A 我以为我可以做这样的事情 update A set A DISCOUNT 3 from INVOICE ITEMS A join ITEM PRICE QUNTITY B on A ITEM PRI
  • SQLite CreateDatabase 不支持错误

    我将 Entity Framework 4 2 CF 与 SQLite 一起使用 但是当我尝试启动该应用程序时 出现 提供商不支持 CreateDatabase 错误 这是我的模型映射 protected override void OnM
  • 将两个表合并为一个输出

    假设我有两张表 已知营业时间 ChargeNum CategoryID Month Hours 111111 1 2 1 09 10 111111 1 3 1 09 30 111111 1 4 1 09 50 222222 1 3 1 09
  • 获取在任何日期创建的表的列表?

    我遇到了这样的情况 我想查找我在 2012 年 9 月 14 日 2012 年 9 月 14 日 在 sql server 上创建的表 是否有任何查询会列出在此日期创建的这些表 SELECT FROM sys tables WHERE cr
  • SQL Server:为什么 ISO-8601 格式的日期依赖于语言?

    我需要一些帮助来理解 SQL Server 中的日期格式处理 如果您尝试以下操作 它将返回正确的结果 SET LANGUAGE English SELECT CAST 2013 08 15 AS DATETIME 2013 08 15 00
  • SQL 使用另一列的键和最大值设置列

    我需要根据同一 ID 的 duration 列的最大值更新 max register 列 将值设置为 1 其他值设置为 0 初始表 Id duration max register 1 0 0 1 7 0 1 3 0 2 10 0 2 5
  • IIF(...) 不是公认的内置函数

    我正在尝试在 Microsoft SQL Server 2008 R2 中使用它 SET SomeVar SomeOtherVar IIF SomeBool value when true value when false 但我收到一个错误
  • 如何使用原始 SQL 查询实现搜索功能

    我正在创建一个由 CS50 的网络系列指导的应用程序 这要求我仅使用原始 SQL 查询而不是 ORM 我正在尝试创建一个搜索功能 用户可以在其中查找存储在数据库中的书籍列表 我希望他们能够查询 书籍 表中的 ISBN 标题 作者列 目前 它
  • 在android中创建SQLite数据库

    我想在我的应用程序中创建一个 SQLite 数据库 其中包含三个表 我将向表中添加数据并稍后使用它们 但我喜欢保留数据库 就好像第一次安装应用程序时它会检查数据库是否存在 如果存在则更新它 否则如果不存在则创建一个新数据库 此外 我正在制作
  • 获取mysql中逗号分隔行中不同值的计数

    一个表 Jobs 有 2 列 JobId 城市 当我们保存工作时 工作位置可能是多个城市 如下所示 JobId City 1 New York 2 New York Ohio Virginia 3 New York Virginia 我如何
  • Oracle:按月分区表

    我的解决方案 德语几个月 PARTITION BY LIST to char GEBURTSDATUM Month PARTITION p1 VALUES JANUAR PARTITION p2 VALUES Februar PARTITI

随机推荐

  • AntV 柱状图

    AntV 柱状图图表 Step 1 npm install antv g2 Step 2 创建柱状图容器 div div 代码截图 代码生成效果 源码 const chartData 0 date Jan num 4 1 date Feb
  • 一篇文章实习心得

    1 爬虫实习 2月 如果公司已经搭建好了爬虫框架比如scrapy那么爬的方向可能也是固定的 代码复用率应该很高 只需要分析页面的逻辑 以及想要爬的字段 自己按照前辈写的代码修改就好了 如果公司没有搭建好框架 你是公司的第一个爬虫工程师 你要
  • mysql的主键规则_MySQL主键(PRIMARY KEY)

    主键 PRIMARY KEY 的完整称呼是 主键约束 MySQL 主键约束是一个列或者列的组合 其值能唯一地标识表中的每一行 这样的一列或多列称为表的主键 通过它可以强制表的实体完整性 选取设置主键约束的字段 主键约束即在表中定义一个主键来
  • Python如何执行shell脚本

    Python如何执行shell脚本 自从出了Pyhon3 5之后 os模块下的system os popen 基本被废弃了 因此如下只介绍2种方式 一 使用commands模块 有三个方法可以使用 1 commands getstatuso
  • wireshark 实用过滤表达式

    wireshark 实用过滤表达式 针对ip 协议 端口 长度和内容 1 关键字 与 eq 和 等同 可以使用 and 表示并且 或 or 表示或者 非 和 not 都表示取反 多组条件联合过滤数据包的命令 就是通过每个单个的条件命令与关键
  • python教程29-继承的基本使用、继承的注意事项、类方法和静态方法回顾、私有属性的继承特点、新式类和经典类

    python教程 小白入门2021 4 5 学习目标 文章目录 python教程 小白入门2021 4 5 P 168 继承的基本使用 P169 继承的注意事项 P170 类方法和静态方法回顾 P171 私有属性的继承特点 P172 新式类
  • 干货:Java正确获取客户端真实IP方法整理

    在JSP里 获取客户端的IP地址的方法是 request getRemoteAddr 这种方法在大部分情况下都是有效的 但是在通过了Apache Squid等反向代理软件就不能获取到客户端的真实IP地址了 如果使用了反向代理软件 将http
  • Leetcode.406 经典算法题:根据身高重建队列

    假设有打乱顺序的一群人站成一个队列 数组 people 表示队列中一些人的属性 不一定按顺序 每个 people i hi ki 表示第 i 个人的身高为 hi 前面 正好 有 ki 个身高大于或等于 hi 的人 请你重新构造并返回输入数组
  • 使用Idea更新数据库表的数据

    tags IDEA 我们在做案例的时候 经常需要改变数据表中的数据来进行简单测试 那我们在Idea下是如何修改数据表的数据的呢 我们可以看下面的图片 只要选择updata就行了 后面再按自动提交的标志 如果您觉得这篇文章帮助到了您 可以给作
  • JavaWeb05(删除&增加&修改功能实现&连接数据库)

    目录 一 实现删除功能 1 1 url如何传参 xx do 参数 参数值 参数名 参数值 1 2 servlet如何拿对应值 根据参数名拿到对应的参数值 String str req getParameter 参数名 1 3 如何询问 nc
  • sql: paging in SQL Server

    sql server 2012 及以上 涂聚文 Geovin Du geovindu 注 SELECT FROM BookKindList ORDER BY SELECT NULL OFFSET 0 ROWS FETCH NEXT 3 RO
  • 数学建模4 论文写作排版和技巧

    文字 标题一 四号黑体 标题二 三 小四号黑体 正文 宋体小四 行距1 5 标题前后空0 5行 英文和数字使用Times New Roman 小四 包括表格中的内 表头在表格上方 需写成 表1 什么什么表 黑体小五加粗 居中 图名在图下 需
  • Saas-Export项目之部门数据回显(下拉框和单选框回显)

    Saas Export项目之部门数据回显 之前的经验 做数据回显一般就是在数据修改时需要查询出这条信息 然后再将数据拆分每一项逐一赋值在表单上 通常赋值就行 而且都是input type text这种类型的 所有只要在input里增加val
  • 推荐 4 个本月火火火的开源项目

    本期推荐开源项目目录 1 开源的 AI 生成图像 APP 2 属于你的微信聊天机器人 3 好玩的 Windows 桌面应用 4 Windows 12 网页版 01 开源的 AI 生成图像 APP 该开源项目技术栈基于 Next js Cha
  • 与Miriam Suzanne一起进行Sass,Susy,单元测试和寻找声音

    In this episode of the Versioning Show Tim and David are joined by Miriam Suzanne best known for Susy a responsive layou
  • jmockit使用(一) —— mock 系统时间

    一 mock工具被广泛的应用于单测中 尤其是当测试环境系统依赖的外部项较多 而且不受控制时 jmockit一般有2种方式 一种是基于行为的方式 使用者在单测类中写 Expectations 里面放置 预期 会被执行的代码段和返回 这就是一种
  • 12款最常使用的网络爬虫工具推荐

    网络爬虫在当今的许多领域得到广泛应用 它的作用是从任何网站获取特定的或更新的数据并存储下来 网络爬虫工具越来越为人所熟知 因为网络爬虫简化并自动化了整个爬取过程 使每个人都可以轻松访问网站数据资源 使用网络爬虫工具可以让人们免于重复打字或复
  • VTK7.1.1+VS2017+QT的安装与配置

    本文转载自 https blog csdn net hebbely article details 81067505 简述 为了实现RealSense的PCL点云显示 需要VTK支持 由于整个平台在Qt环境实现 VTK编译为Qt插件 QVT
  • YOLO v5 引入解耦头部

    YOLO v5 引入解耦头部 最近想开个 深度学习模型搭建 opencv方面的训练营 有兴趣可以私聊我 文章目录 YOLO v5 引入解耦头部 前言 一 解耦头部示意图 二 在YOLO v5 中引入解耦头部 1 修改common py文件
  • Sqlite3 常见几种数据插入方式性能比较

    include