1. 所谓第四代语言

SQL是一种典型的第四代语言,即4GL,这种语言的突出特点是编写者不需要关注怎么做,只需要告诉系统我要什么就可以。

虽然4GL是这样的一种语言,大大简化了编写者的编写难度,其实底层还是数据库的编写者帮我们隐藏了具体的实现细节。

举个例子,你妈妈叫你去做一碗西红柿炒鸡蛋,但是并没有告诉你如何做,这个时候你查资料,发现了从洗菜到炒菜的所有过程,然后炒了一碗西红柿炒鸡蛋出来。

对于你的妈妈来说,她很简单的发出了一个命令,得到了结果,不过她也不知道具体如何实现的,具体的实现细节被你在厨房掩盖了。

2. RadonDB的查询优化器

你在厨房关上门来做的事情,其实就是数据库查询优化器做的事情,把你的指令分解成具体的执行过程,然后将结果返回给终端客户。

现在来看看查询优化器的实现。其代码在optimizer包下面,只有两个文件:

  • optimizer.go 这个文件声明了一个接口
  • simple_optimizer.go 这个文件则是具体的实现。

先看看接口实现:

package optimizer

import (
"planner"
) // Optimizer interface.
type Optimizer interface {
BuildPlanTree() (*planner.PlanTree, error)
}

从这个接口上来看,查询优化器主要做的事情就是构建一颗查询计划树。接下来开始看看具体的实现过程,首先会看到一个结构体:

// SimpleOptimizer is a simple optimizer who dispatches the plans
type SimpleOptimizer struct {
log *xlog.Log
database string
query string
node sqlparser.Statement
router *router.Router
}

一般来说,执行一个SQL的时候总会遇到这样一个操作:

optimizer.NewSimpleOptimizer(log, database, query, node, router).BuildPlanTree()

都会新建一个Optimizer,然后新建一个计划树。其实就是新建了一个刚才的结构体,这是实现的代码:

// NewSimpleOptimizer creates the new simple optimizer.
func NewSimpleOptimizer(log *xlog.Log, database string, query string, node sqlparser.Statement, router *router.Router) *SimpleOptimizer {
return &SimpleOptimizer{
log: log,
database: database,
query: query,
node: node,
router: router,
}
}

注意这里的node,这是一个sqlparser.Statement,主要玩的就是这个东西。

好了,接下来就可以新建查询计划树了:

// BuildPlanTree used to build plan trees for the query.
func (so *SimpleOptimizer) BuildPlanTree() (*planner.PlanTree, error) {
log := so.log
database := so.database
query := so.query
node := so.node
router := so.router plans := planner.NewPlanTree()
switch node.(type) {
case *sqlparser.DDL:
node := planner.NewDDLPlan(log, database, query, node.(*sqlparser.DDL), router)
plans.Add(node)
case *sqlparser.Insert:
node := planner.NewInsertPlan(log, database, query, node.(*sqlparser.Insert), router)
plans.Add(node)
case *sqlparser.Delete:
node := planner.NewDeletePlan(log, database, query, node.(*sqlparser.Delete), router)
plans.Add(node)
case *sqlparser.Update:
node := planner.NewUpdatePlan(log, database, query, node.(*sqlparser.Update), router)
plans.Add(node)
case *sqlparser.Select:
nod := node.(*sqlparser.Select)
selectNode := planner.NewSelectPlan(log, database, query, nod, router)
plans.Add(selectNode)
case *sqlparser.Checksum:
node := planner.NewOthersPlan(log, database, query, node, router)
plans.Add(node)
default:
return nil, errors.Errorf("optimizer.unsupported.query.type[%+v]", node)
} // Build plantree.
if err := plans.Build(); err != nil {
return nil, err
}
return plans, nil
}

代码也不长,就全都贴出来了,其实很简单,就是对node的类型进行判断,根据不同的类型确定不同的plan。

今天这篇只是引导,所以大致看看就好,先看看其中一个分支的计划是怎么创建的:

// NewInsertPlan used to create InsertPlan
func NewInsertPlan(log *xlog.Log, database string, query string, node *sqlparser.Insert, router *router.Router) *InsertPlan {
return &InsertPlan{
log: log,
node: node,
router: router,
database: database,
RawQuery: query,
Typ: PlanTypeInsert,
Querys: make([]xcontext.QueryTuple, 0, 16),
}
}

这里只是返回一个结构体,没什么意思,有水平的地方在Build中,但是代码很长,所以今天就先不摊开来讲了。

留点悬念。

3. 小结

今天写的真轻松,因为要开始学习一个非常庞大的东西了。加油吧自己。

学习RadonDB源码(三)的更多相关文章

  1. 学习RadonDB源码(一)

    1. 可能是开始也可能是结束 RadonDB是国内知名云服务提供商青云开源的一款产品,下面是一段来自官方的介绍: QingCloud RadonDB 是基于 MySQL 研发的新一代分布式关系型数据库 ...

  2. 学习RadonDB源码(二)

    1. 为我新的一天没有放弃而喝彩 学习是一件很容易放弃的事情,因为就算是不学,我也能在现在的岗位上发光发热.可是人不就是一个热爱折腾的种群吗? 今天没有放弃不代表明天没有放弃,也许放弃的可能性大于坚持 ...

  3. 【菜鸟学习jquery源码】数据缓存与data()

    前言 最近比较烦,深圳的工作还没着落,论文不想弄,烦.....今天看了下jquery的数据缓存的代码,参考着Aaron的源码分析,自己有点理解了,和大家分享下.以后也打算把自己的jquery的学习心得 ...

  4. DotNetty网络通信框架学习之源码分析

    DotNetty网络通信框架学习之源码分析 有关DotNetty框架,网上的详细资料不是很多,有不多的几个博友做了简单的介绍,也没有做深入的探究,我也根据源码中提供的demo做一下记录,方便后期查阅. ...

  5. 学习 vue 源码 -- 响应式原理

    概述 由于刚开始学习 vue 源码,而且水平有限,有理解或表述的不对的地方,还请不吝指教. vue 主要通过 Watcher.Dep 和 Observer 三个类来实现响应式视图.另外还有一个 sch ...

  6. 我该如何学习spring源码以及解析bean定义的注册

    如何学习spring源码 前言 本文属于spring源码解析的系列文章之一,文章主要是介绍如何学习spring的源码,希望能够最大限度的帮助到有需要的人.文章总体难度不大,但比较繁重,学习时一定要耐住 ...

  7. 一起学习vue源码 - Object的变化侦测

    作者:小土豆biubiubiu 博客园:www.cnblogs.com/HouJiao/ 掘金:https://juejin.im/user/58c61b4361ff4b005d9e894d 简书:h ...

  8. 手牵手,从零学习Vue源码 系列一(前言-目录篇)

    系列文章: 手牵手,从零学习Vue源码 系列一(前言-目录篇) 手牵手,从零学习Vue源码 系列二(变化侦测篇) 手牵手,从零学习Vue源码 系列三(虚拟DOM篇) 陆续更新中... 预计八月中旬更新 ...

  9. 手牵手,从零学习Vue源码 系列二(变化侦测篇)

    系列文章: 手牵手,从零学习Vue源码 系列一(前言-目录篇) 手牵手,从零学习Vue源码 系列二(变化侦测篇) 陆续更新中... 预计八月中旬更新完毕. 1 概述 Vue最大的特点之一就是数据驱动视 ...

随机推荐

  1. 雪花ID实现新增数据同步

    雪花ID实现新增数据同步 GUID生成的ID,可以确保是唯一ID,但却是无序的,不适合用于数据同步. 雪花算法生成的ID(INT64)能够按照时间有序(升序)生成.只要电脑上的时间是正确的,新增的记录 ...

  2. 在 delphi (Object Pascal 语言)中,使用 array 关键字进行数组定义。

    如果需要定义二维数组可以采取以下定义形式: 一.静态数组定义 静态数组定义,通常用于数组元素的数目确定的情况.定义形式如下: 示例: 1 2 3 4 5 6 7 8 9 10 11 type   // ...

  3. linux内核在哪里处理设备树中compatible为"syscon"的节点?

    答: linux内核源码drivers/mfd/syscon.c中的of_syscon_register()接口对regmap_config进行初始化 注: linux内核源码版本为5.1.0

  4. tensorflow 如何看shape

    https://blog.csdn.net/yinxingtianxia/article/details/78121941 输入: x= tf.truncated_normal([, , ], dty ...

  5. python获取公网ip的几种方式

    python获取公网ip的几种方式 转 https://blog.csdn.net/conquerwave/article/details/77666226 from urllib2 import u ...

  6. shell编程系列8--文本处理三剑客之grep和egrep

    grep和egrep 第一种形式:grep [option] [pattern] [file1,file2...] 第二种形式:command | grep [option] [pattern] gr ...

  7. 002-05-RestTemplate 使用常见问题-非200请求捕获

    一.概述 1.1.RestTemplate报错时时获取不到错误信息 捕获异常后 try { result = restTemplate.postForEntity(faceConfig.getApiU ...

  8. layui时间控件闪退的问题

    项目上线,发现后台管理系统layui的子页面出现时间控件闪退的问题,根本选取不到时间. 其原因是:如果出现页面找到多个节点,只有第一个节点能正常使用后面的节点都会闪退,可以理解为目前laydate不支 ...

  9. Spring Cloud微服务安全实战- 2-1 环境安装

    下面这些.后续随着讲课逐步再去安装. 2019年1月开始 jdk是收费的 找jdk最后一个免费版本 8u192这是jdk1.8最后的一个免费版本 STS spring提供的ide可以方便的开发spri ...

  10. jsp、freemarker、velocity、thymeleaf

    1.概述在java领域,表现层技术主要有三种, (1)jsp; (2)freemarker; (3)velocity; (4)thymeleaf; 2.jsp优点: 1.功能强大,可以写java代码 ...