首页 > 数据库 >MySQL和MongoDB如何JOIN查询?一个直接在本地运行的SQL执行引擎

MySQL和MongoDB如何JOIN查询?一个直接在本地运行的SQL执行引擎

时间:2023-08-07 18:05:13浏览次数:47  
标签:JOIN pip3 MongoDB 数据库 site 查询 install SQL

在微服务和云原生愈发流行的今天,数据的分布也愈发脱离单库单机而更加复杂,使用的数据库类型也会更多,但业务的复杂依然会带来了大量的数据查询和导出需求,而很多时候我们很难为数据量的大部分系统创建完整的BI数仓系统,这时候你是不是觉得为这些需求查询和导出数据就会是一个十分困难且耗时的工作?Syncany-SQL就是这样一个工具,来在不依赖数据库的情况下完成不同库表、不同机器和不同数据库类型间直接关联查询和聚合计算后直接导出到常用文件的工具。

直接在本地运行MySQL语法结构的SQL的执行引擎,只使用检查查询从常用MySQL、MongoDB、PostgreSQL、sqlserver、elasticsearch、influxdb、clickhouse、sqlite数据库加载数据及读取execl、csv、json和普通文本文件后,在内存中完成join和聚合运算,单条SQL的执行结果可以临时保存在内存中以便作为后续SQL的输入表继续处理,并且INSERT INTO语句执行指定”仅插入 I“、”存在更新否则插入 UI“、”存在更新否则插入其余删除 UDI“、”删除后插入 DI“四种合并数据模式,针对大数据量也可以分批执行。

整个程序共有两个项目组成,其中

https://github.com/snower/syncany-sql 负责解析SQL

https://github.com/snower/syncany 负责执行

特性

  • Join关联查询,支持在不同库表、不同机器和不同数据库类型之间Join查询
  • Having在内存中执行,支持复杂条件过滤,只包含聚合计算字段过滤时运行在聚合运算之前,否则在聚合计算之后执行
  • 完全在内存中完成Group By分组聚合计算和distinct去重聚合运算
  • Order By排序,如果排序完全是主表字段则使用数据库完成,否则内存中完成排序
  • 支持子查询
  • Where或join on条件支持简单子查询(仅支持and,比较操作符仅支持==、!=、>、>=、<、<=、in),其条件会被发送到数据库执行
  • Insert Into除支持正常写入数据库表外可直接写入到execl、json和csv中
  • Insert Into支持指定数据合并类型,共支持4中合并算法 (插入 I,更新否则插入 UI,更新否则插入其余删除 UDI,先删除再插入 DI)
  • 可以在设定每批数量后支持按批次执行
  • WHere、Join on和Having条件值都支持子查询返回
  • 支持set设置变量值,支持select info给变量赋值,执行访问变量
  • 支持使用常用MySQL函数

限制

  • Where和Join on条件仅可使用==、>、>=、<、<=、!=、in简单条件且仅可用and
  • Join仅支持Left Join模式,且关联条件中仅被关联表能添加常量条件,被关联表字段不可以是计算值,主表可是计算条件值
  • Join查询及子查询各表必须有别名
!!!特别注意:为何会有使用限制,来源于数据库加载数据的基本流程
  • 为简化数据库访问接口,每种数据库实现仅支持简单查询条件和排序来加载数据
  • JOIN关联查询使用IN查询从数据库加载数据后在内存中完成计算

安装

可使用pip直接安装

pip3 install syncanysql

# 执行成功后,执行
echo "select now(), 'hello world';" | syncany-sql
# 看到正确输出即为安装成功

因大多数情况使用数据库Driver很少,所以默认不安装数据库Driver,需要依据自己使用的数据库类型自行安装数据库Driver

pip3 install pymongo>=3.6.1
pip3 install PyMySQL>=0.8.1
pip3 install openpyxl>=2.5.0
pip3 install psycopg2>=2.8.6
pip3 install elasticsearch>=6.3.1
pip3 install influxdb>=5.3.1
pip3 install clickhouse_driver>=0.1.5
pip3 install redis>=3.5.3
pip3 install pymssql>=2.2.7

配置

配置文件支持json和yaml格式,默认加载当前目录的"config.[json|yaml]”和用户目录下"~/.syncany/config.[json|yaml]"文件,当前目录配置文件优先级高于用户目录,合并配置项后为最终加载配置。

详细介绍可查询示例配置文件信息 https://github.com/snower/syncany-sql/blob/main/docs/configure.md

配置完成后,命令行启动时会自动加载配置文件,并且添加的数据库信息将会在真正使用到的时候才会发起连接,并不会在程序启动就会尝试连接配置的数据库。

!!!特别注意的是,SQL中使用的数据库名称是配置文件中为每个数据库连接指定的名称,而不是数据库中真正的名称。如:
# 我们在配置文件中添加了以下数据库配置
databases:
  - name: test_db
    driver: mysql
    host: '127.0.0.1'
    port: 3306
    user: 'root'
    passwd: '123456'
    db: 'test'
    charset: 'utf8mb4'
    
 # 编写SQL是,假如我们需要查询127.0.0.1的test库,我们应该这样编写SQL
 
 select * from test_db;

示例

查询Nginx日志

-- 查询访问量最高的3个IP
SELECT seg0 AS ip, COUNT(*) AS cnt FROM `file://data/access.log?sep= ` GROUP BY seg0 ORDER BY cnt DESC LIMIT 3;

查询JSON文件

# 一个查询json文件的例子 https://github.com/snower/syncany-sql/tree/main/examples/demo
SELECT
    a.site_id,
    b.name AS site_name,
    IF(c.site_amount > 0, c.site_amount, 0) AS site_amount,
    MAX(a.timeout_at) AS timeout_at,
    MAX(a.vip_timeout_at) AS vip_timeout_at,
    now() as `created_at?`
FROM
    (SELECT
        YIELD_DATA(sites) AS site_id,
            IF(vip_type = '2', GET_VALUE(rules, 0, 'timeout_time'), '') AS timeout_at,
            IF(vip_type = '1', GET_VALUE(rules, 0, 'timeout_time'), '') AS vip_timeout_at
    FROM
        `data/demo.json`
    WHERE
        start_date >= '2021-01-01') a
        JOIN
    `data/sites.json` b ON a.site_id = b.site_id
        JOIN
    (SELECT
        site_id, SUM(amount) AS site_amount
    FROM
        `data/orders.json`
    WHERE
        status <= 0
    GROUP BY site_id) c ON a.site_id = c.site_id
GROUP BY a.site_id;

更多常用例子可查看 https://github.com/snower/syncany-sql/tree/main/examples

Python调用接口

from syncanysql import ScriptEngine

with ScriptEngine() as engine:
    # 执行SQL
    engine.execute('''
        INSERT INTO `top_ips` SELECT
            ip, cnt
        FROM
            (SELECT
                seg0 AS ip, COUNT(*) AS cnt
            FROM
                `file:///var/log/nginx/access.log?sep= `
            GROUP BY seg0) a
        ORDER BY cnt DESC
        LIMIT 3;
    ''')
    # 获取执行结果
    print(engine.pop_memory_datas("top_ips"))

常用用途

数据导出

因为直接可以把查询结果通过insert into语句直接写入csv或excel文件,以及可以非常方便的直接在不同库表、不同机器和不同数据库类型间直接join和进行聚合运算,也可以方便的使用内存保存中间结果,所以可以大幅简化编写复杂SQL和数据导出的效率。

ETL工具

在指定insert into数据合并类型后可以非常方便的用于数据同步,在数据量级不高的系统中,配合superset之类的图表系统可以非常快的完成BI系统搭建,命令行的执行模式也可以使用Jenkins或者云函数之类的系统大幅提高稳定性或动态扩展能力。

本文由博客一文多发平台 OpenWrite 发布!

标签:JOIN,pip3,MongoDB,数据库,site,查询,install,SQL
From: https://blog.51cto.com/u_16191490/6995968

相关文章

  • PLSQL客户端刚安装好后无法访问服务器数据库的解决方法
    1、正常安装PLSQL2、安装客户端oracle客户端工具:InstantClientSetup64w-oracle简易客户端.exe3、配置环境变量,在path中添加上面安装客户端工具的地址。4、打开plsql,即可看到登录下面多了一个选项,即可以登录使用了。......
  • Airflow 2.2.6 + MySQL 8.0.27 + Redis 7.0.12 部署Airflow任务调度平台
    本docker-compose文件在centos7.9系统,docker版本为24.0.2上测试的如果你的docker版本低于24.xxx就需要手动安装docker-compose,高于24就不需要安装了,docker已经自带了官方文档,关于docker部署1.先执行mkdir-p./dags./logs./plugins./config./......
  • Navicat Premium的mysql备份和恢复
    备份:::::::恢复::::::......
  • Java 远程连接 SQLite 数据库
    Java可以使用JDBCAPI来连接SQLite数据库。但是,SQLite不支持远程连接,因为它是一种文件数据库,需要直接访问数据库文件。如果您需要从远程位置访问SQLite数据库,可以将SQLite数据库文件放在共享文件夹中,并将该文件夹映射到本地计算机上。然后,您可以在本地计算机上使用JDBC......
  • MySQL基础语句
    select*from表where字段in(select临时表名.字段from(另一个字段)临时表名);deletefrom表名where字段inselect临时表名.字段from(另一个字段)临时表名);update表名set字段=新值where字段in(select临时表名.字段from(另一个字段)临时表名));insertint......
  • mysql-多表查询
    小知识:1.为表取别名因为是对两张表进行查询了,那么每次写表名的话就有点麻烦,所以用一个简单的别名来代表表名格式:表名as别名2.为字段取别名给字段名取别名的原因是为了有些字段名显示得更加清楚,比如:selectf_priceas'价格'fromfruits; 3.sql语句执......
  • MySQL问题记录Can't connect to MySQL server on 'localhost' (10061)解决方法
    登录MySQL提示Can'tconnecttoMySQLserveron'localhost'(10061)进入安装目录bin目录,执行mysqld--install,启动MySQL点击查看代码cdD:\soft\MySQL\MySQLServer5.7\binmysqld--installnetstartmysql提示启动失败最后执行mysqld--initialize--user=root--......
  • PostgreSQL数据库版本升级
    PostgreSQL数据库版本升级Postgresql是一个非常活跃的社区开源项目,更新速度很快,每一次版本的更新都会积极的修复旧版本的BUG,性能上也会有不同幅度的提升。10之前的版本由三部分组成,10开始只有两部分数字组成。PostgreSQL版本发布规则,一年一个大版本,一个季度一个小版本;P......
  • mysql密码破解
    mysql破解密码1.设置mysql免密登陆#vim/etc/my.cnf.d/mysql-server.cnf#主配置文件[mysqld]下面添加以下内容[mysqld]skip-grant-tables#免密的登陆#重启mysql#systemctlrestartmysqld2.查看数据库中用户的密码,此时是加密状态mysql>selecthost,user,au......
  • MySQL事务与锁
    锁锁粒度数据库锁定机制简单来说,就是数据库为了保证数据的一致性,而使各种共享资源在被并发访问变得有序所设计的一种规则。MySQL数据库由于其自身架构的特点,存在多种数据存储引擎,每种存储引擎的锁定机制都是为各自所面对的特定场景而优化设计,所以各存储引擎的锁定机制也有较大区别......