bison 编译sql的基本知识
一。bison是干什么的?
bison 是一个语法分析器,把用户输入的内容,根绝在.y文件中事先定义好的规则,构建一课语法分析树。(所谓的规则就是,匹配上对应字符之后,执行相应的动作。)
1.一个简单的语法例子和分析:
statement :NAME '=' expression
expression :NUMBER '+' NUMBER
|NUMBER '-' NUMBER
这个语法的意思是:
冒号(:)用来间隔一条规则的左边和右边。
statement 等价于 NAME '=' expression 。
由于在语法中规定大写字母 和引号的内容为终结符,所以NAME '=' 两个字符不再有含义,已经终结。
但是expression 是非终结符,根据第二行的规定,expression 又等价为 两个数的加法或者两个数的减法。其中竖线(|)表示一个语法符号有两种等价方式。 NUMBER '+' NUMBER NUMBER '-' NUMBER均是终结符,所以语法解析结束。
假如现在的输入是fred=12+13. 则语法解析树如下:(圆形都是非终结符,矩形都是终结符)
2.移进 规约分析
当bison处理一个语法分析树时,会创建一组状态,每个状态对应一个或者多个分析过的规则中的可能的位置。当读到的记号不足以结束一条规则的时候,就会把这个记号压入一个内部堆栈,然后切换到新状态,这个过程叫做移进。当压入栈内的所有的语法符号已经等价于一个规则的右部时,就把这些符号全部弹出,把规则的左部压入栈。这个过程叫做规约。
下面是一个例子:
fred=12+13
语法分析器一次移进一个记号。
堆栈:
fred
fred =
fred =12
fred=12 +
fred =12+13 把12+13 规约成expression,12+13弹出,expression压入
fred = expression 把fred = expression规约成statement fred = expression弹出,statement压入
statement
3.sql语言中的语法解析过程。(以select为例)
首先,需要一个词法分析器来识别SQL中的所有关键字。
程序的运行流程:(以select为例)
1.用户输入sql语句,调用sql.tab.c中的解析函数,得到select_statement的状态。(。y文件中为sql语句定义了好多个状态。)
2.select_statement作为参数,去匹配规则,并执行相应的动作,即构建一课语法树。这个匹配的过程由yyparse()库函数来完成!
下面这段代码是.y文件中,定义的关于select_statement部分的代码:
select_statement:
SELECT selection table_exp
{
struct stnode * p = create_non_terminal_node("select_section"); if(!p)
{
printf("error:create_non_terminal_node\n"); return ;
} if(!append_child(p, $))
{
printf("error:append_child\n"); return ;
} if(!append_child(p, $))
{
printf("error:append_child\n"); return ;
} if(!append_child(p, $))
{
printf("error:append_child\n"); return ;
} $$ = p;
}
selection:
scalar_exp_list
{
struct stnode * p = create_non_terminal_node("selection"); if(!p)
{
printf("error:create_non_terminal_node\n"); return ;
} if(!append_child(p, $))
{
printf("error:append_child\n"); return ;
} $$ = p;
}
| '*'
{
struct stnode * p = create_non_terminal_node("selection"); if(!p)
{
printf("error:create_non_terminal_node\n"); return ;
} if(!append_child(p, $))
{
printf("error:append_child\n"); return ;
} $$ = p;
}
;
scalar_exp_list:
scalar_exp
{
struct stnode * p = create_non_terminal_node("scalar_exp_list"); if(!p)
{
printf("error:create_non_terminal_node\n"); return ;
} if(!append_child(p, $))
{
printf("error:append_child\n"); return ;
} $$ = p;
}
| scalar_exp_list','scalar_exp
{
struct stnode * p = create_non_terminal_node("scalar_exp_list"); if(!p)
{
printf("error:create_non_terminal_node\n"); return ;
} if(!append_child(p, $))
{
printf("error:append_child\n"); return ;
} if(!append_child(p, $))
{
printf("error:append_child\n"); return ;
} if(!append_child(p, $))
{
printf("error:append_child\n"); return ;
} $$ = p;
}
;
table_exp:
from_clause
opt_order_by_clause
{
struct stnode * p = create_non_terminal_node("table_exp"); if(!p)
{
printf("error:create_non_terminal_node\n"); return ;
} if(!append_child(p, $))
{
printf("error:append_child\n"); return ;
} if(!append_child(p, $))
{
printf("error:append_child\n"); return ;
} $$ = p;
}
;
from_clause:
FROM table_ref_list opt_where_clause
{
struct stnode * p = create_non_terminal_node("from_clause"); if(!p)
{
printf("error:create_non_terminal_node\n"); return ;
} if(!append_child(p, $))
{
printf("error:append_child\n"); return ;
} if(!append_child(p, $))
{
printf("error:append_child\n"); return ;
} if(!append_child(p, $))
{
printf("error:append_child\n"); return ;
} $$ = p;
}
; table_ref_list:
table_ref
{
struct stnode * p = create_non_terminal_node("table_ref_list"); if(!p)
{
printf("error:create_non_terminal_node\n"); return ;
} if(!append_child(p, $))
{
printf("error:append_child\n"); return ;
} $$ = p;
}
| table_ref_list','table_ref
{
struct stnode * p = create_non_terminal_node("table_ref_list"); if(!p)
{
printf("error:create_non_terminal_node\n"); return ;
} if(!append_child(p, $))
{
printf("error:append_child\n"); return ;
} if(!append_child(p, $))
{
printf("error:append_child\n"); return ;
} if(!append_child(p, $))
{
printf("error:append_child\n"); return ;
} $$ = p;
}
;
table_ref:
table
{
struct stnode * p = create_non_terminal_node("table_ref"); if(!p)
{
printf("error:create_non_terminal_node\n"); return ;
} if(!append_child(p, $))
{
printf("error:append_child\n"); return ;
} $$ = p;
}
;
3.根据上面代码中的定义的规则,,,,展示下面的语法树的构建过程:
规则 动作: 语法树
select_statement : SELECT selection table_exp node("select_section") select_statement
SELECT selection table_exp
selection : scalar_exp_list node("selection") *
|*
table_exp : from_clause opt_order_by_clause node("table_exp") from_clause opt_order_by_clause
from_clause: FROM table_ref_list opt_where_clause node("from_clause") FROM table_ref_list opt_where_clause
table_ref_list :table_ref
| table_ref_list','table_ref node("table_ref_list ") table_ref
table_ref: table node("table ") table
bison 编译sql的基本知识的更多相关文章
- SQL server基础知识(表操作、数据约束、多表链接查询)
SQL server基础知识 一.基础知识 (1).存储结构:数据库->表->数据 (2).管理数据库 增加:create database 数据库名称 删除:drop database ...
- Sql Server 基础知识
Sql Server 基础知识: http://blog.csdn.net/t6786780/article/details/4525652 Sql Server 语句大全: http://www.c ...
- atitit.查看预编译sql问号 本质and原理and查看原生sql语句
atitit.查看预编译sql问号 本质and原理and查看原生sql语句 1. 预编译原理. 1 2. preparedStatement 有三大优点: 1 3. How to look gene ...
- SQL数据库基础知识-巩固篇<一>
SQL数据库基础知识-巩固篇<一>... =============== 首先展示两款我个人很喜欢的数据库-专用于平时个人SQL技术的练习<特点:体积小,好安装和好卸载,功能完全够用 ...
- mybatis中预编译sql与非预编译sql
预编译sql有缓存作用,非预编译没得 mybaits中带有#传参的有预编译左右,$没得 多用#传参 预编译语句的优势在于归纳为:一次编译.多次运行,省去了解析优化等过程:此外预编译语句能防止sql注入 ...
- APK反编译之一:基础知识—APK、Dalvik字节码和smali文件
refs: APK反编译之一:基础知识http://blog.csdn.net/lpohvbe/article/details/7981386 APK反编译之二:工具介绍http://blog.csd ...
- SQL语句之 知识补充
SQL语句之 知识补充 一.存储过程 运用SQL语句,写出一个像函数的模块,这就是存储过程. 需求: 编写存储过程,查询所有员工 -- 创建存储过程(必须要指定结束符号) -- 定义结束符号 DELI ...
- hibernate预编译SQL语句中的setParameter和setParameterList
使用预编译SQL语句和占位符參数(在jdbc中是?),可以避免由于使用字符串拼接sql语句带来的复杂性.我们先来简单的看下.使用预编译SQL语句的优点. 使用String sql = "se ...
- 转:sql server锁知识及锁应用
sql server锁(lock)知识及锁应用 提示:这里所摘抄的关于锁的知识有的是不同sql server版本的,对应于特定版本时会有问题. 一 关于锁的基础知识 (一). 为什么要引入锁 当多个用 ...
随机推荐
- android中实现毛笔效果(View 中画图)
近期有一个项目设计一个APP实现通过触摸屏实现毛笔写字效果.传统的绘画板程序直接通过Path的moveTo和LineTo便可实现简单的线条绘画程序.然而要达到毛笔的笔锋效果则须要更为具体点的设计.我的 ...
- github常见错误
如果输入$ Git remote add origin git@github.com:djqiang(github帐号名)/gitdemo(项目名).git 提示出错信息:fatal: remote ...
- sharding-jdbc从入门到出门(03)
经过端午节这2天对 sharding-jdbc一直怀揣成梦想的去学习,还是有一些没有解决的问题: 上一张图:
- On the importance of initialization and momentum in deep learning
Ilya Sutskever1 ilyasu@google.com James Martens jmartens@cs.toronto.edu George Dahl gdahl@cs.toronto ...
- Zabbix-Agent 客户端安装配置
1.安装Zabbix官方的yum源 [root@crazy-acong ~]# rpm -ivh http://repo.zabbix.com/zabbix/2.2/rhel/6/x86_64/zab ...
- Django之stark组件2
action批量处理功能 用户可以自定义批量处理功能,但是默认的有批量删除功能. ***思路*** 1,定义一个列表用来装action的类名并extend用户自定义列表 2.循环该列表获取到函数名(用 ...
- Android系统移植与调试之------->如何修改Android设备的默认休眠时间
1.找到~/mx0831-0525/frameworks/base/packages/SettingsProvider/res/values/ defaults.xml文件 2.修改默认休眠时间 3. ...
- ppm图像相关
PPM图像格式介绍 直接拿具体的数据来说明是最直接的,使用ue打开ppm文件,采用的都是十六进制asc码表示的,这里要注意地址00000000h中的最后一个字母是始终不变的,这原来没注意晕了我好久,第 ...
- swap 内存不足
参考:https://stackoverflow.com/questions/5682854/why-is-the-linker-terminating-on-me-when-i-build-clan ...
- C#线程使用学习
线程的入口函数可以不带输入参数,也可以带输入参数: form1.cs using System; using System.Collections.Generic; using System.Comp ...