上星期写完词法分析器的时候,曾遇上一个无关紧要却X疼的问题。毕竟是第一次完整地写整个语言的编译器(暂且这么叫着吧,解释器更靠谱),由于经验不足,在字符串解析这一块驻足了两天才解决掉,这里记录下来供以后参考。哦对了,之所以想自己手写词法分析器,并不是我不知道有自动工具可以自动生成,而是我不会用,嗯,果然高冷。

词法分析器的作用简而言之就是将语言分割成一个一个独立的词法单元(单词),并赋予一定的类型。(如果不了解其作用,建议参考词法分析)

例如:

a = 3 ;

我们就可以将其分课程一个个有意义的单元,并赋予类型:

<=,NE>

<3,NUM>

<;,SEMI>

和就是分割好的单词序列。在一般情况下,一门语言的词法分析器总会遇上要分析字符串成分的情况,比如表达式

val = "xxx" 或者 val = 'xxx'

我并未参考其他语言是如何分析,仅凭猜测,自认为应该解析成如下形式:

<=,NE>

<'/",SINGLEQUOTE/DOUBLEQUOTE>

当然也许这不是一个好的类型单词分配方法,但起码不会是一个错误方法吧,到现在,语法分析阶段工作的还挺好的,也许会有更好的办法,还需要多参考前辈们的。

这个时候问题就是,当遇上这样的句子时:

a = "xxxxxxx" b="ssssssss"

因为在一个SQL语句中不可能一辈子咱们就出现一个字符串单元,所以这个怎么解析当时很费了点脑筋。如果没注意,就会解析成

<=,NE>

<",DOUBLE_QUOTE>

<",DOUBLE_QUOTE>

<",DOUBLE_QUOTE>

<",DOUBLE_QUOTE>

如果不加以控制,这显然是错的,因为b=在这里显然是两个单词,而不是STRING。于是我开始寻觅各种办法解决这个问题:第一个想到的是用bool类型来控制,判断引号出现的单复数,如果是true则为复数,即收尾的符号,这样这个问题就解决了。但是,脑脑子里思考问题的解决方案永远是奔向理想目标中的其中一条道路,很多岔路是动手的时候出现的,于是真的出现问题了,参看下面伪代码,其中flag1与2分别代表单引号与双引号的判断flag。

get_next_token()
{
while (p != val.size())
{
if (flag)
{
std::string v=get_string();
continue;
}
switch (c)
{
case '\'':
if (flag==true)//the begin quote
{
flag = false;
}
else if (flag==false)
{
flag = true;
}
consume();
break;
default:
consume();
}
}
}

switch语句内的代码逻辑是没有任何问题的,问题出在,取字符串单词的判断上:当下一个单词是字符串时,取出之后便会执行continue,这时flag 是无法被改变状态的,所以当下一次取单词进入函数时又会进入开始的if逻辑,当时我在这种解决方法上进行了很多次的修改,均告失败,问题重重,于是只得另寻方案,每当这种时候都恨自己脑瓜不够机灵,想不到优雅的办法解决这种问题,当然了也许是条件限制,导致自己没法往优雅的解决方法上想,:p,我倾向于后者。后来也试过用用计数器的方式,也是失败了,掰着掰着就醉了,好一个,众人皆醒我独醉,醉完媳妇旁撅着睡。(诶?!我不是在寝室吗?)

好在把各种烂方法使了过后,想到了一个最终解决方法,使用了一个栈,当栈里保存着有引号的时候(当前符号落到引号上时在switch内的每个case压栈,如代码所示。),说明这一轮要取的单词属于字符串,当当前字符又落到引号上时,判断栈里是否有引号,如果有,则说明是收尾引号,这时清空栈。

//由于get_token函数过长,此处仅贴上部分片段
if (!quote_stack.empty())//string_identifier.first stores the quotes
{
if (quote_stack.top() == c)
{
consume();
char temp = quote_stack.top();
quote_stack.pop();
if (temp=='\'')
{
return token(tag::SINGLEQUOTE, "'");
}
return token(tag::DOUBLEQUOTE, "\"");
}
else
{
std::string id = STRINGS_WITH_TERMINATION(quote_stack.top()).c_str();
token tk(tag::STRING, id.c_str());
if (!id.empty())
{
return token(tk);
}
}
} //switch内部:
case '\'':
consume();
quote_stack.push('\'');
return token(tag::SINGLEQUOTE, "'");
case '"':
consume();
quote_stack.push('"');
return token(tag::DOUBLEQUOTE, "\"");

这个方法目前运行良好,由于任务的特殊性,栈内最多会容纳两个字符,由于stack内部由deque实现(C++ STL),空间上多少浪费了一点,不过这个方法将任务简化,并且也挺好理解,同时相比flag的方法,flag更容易有在其他函数中无意赋值导致全局变量污染问题的风险。当然了,您可以将其替换为一个两个字节的数组,抽象成一个类来解决,我这里暂时先不做优化。

其实这也只是一个权宜之计,我相信一定有优雅且更加高效的设计或者方法,期待可以学到。

jcSQL词法分析器对字符串token的解析的更多相关文章

  1. QT json字符串生成和解析

    1         QT json字符串生成和解析 1.1  QT Json解析流程 (1)  字符串转化为QJsonDocument QJsonParseError json_error; QJso ...

  2. redis之字符串命令源代码解析(二)

    形象化设计模式实战             HELLO!架构                     redis命令源代码解析 在redis之字符串命令源代码解析(一)中讲了get的简单实现,并没有对 ...

  3. Java字符串常量池及字符串判等解析

    一.理解"=="的含义 "=="常用于两个对象的判等操作,在Java中,"=="主要有以下两种用法: 1.基础数据类型:比较的是他们的值是否 ...

  4. jwt认证生成后的token后端解析

    一.首先前端发送token token所在的位置headers {'authorization':token的值',Content-Type':application/json} 在ajax写 //只 ...

  5. Python 生成 JWT(json web token) 及 解析方式

    一.关于 jwt 的原理及概念可以自行在网络上搜索了解一下,这里推荐一篇写的比较好的博客 深入了解Json Web Token之概念篇 另附 JWT 的官方文档: https://jwt.io/int ...

  6. C语言之字符串典型例题解析

    今天又遇见几个好题,和以前的一些凑一块写一篇文章,作为我延迟去自习室的一个借口吧. 首先是第一题 int fun(char* s){ char* t = s; while(*t++); return ...

  7. 各自平台token获取解析及用户信息的获取

    1.auth根据手机号码获取auth平台session_token记统一认证的user_id与pass_id [dwliuchao1@GD-QHD-CNG152TFKX-12.55 logs]$ cd ...

  8. JSON字符串 拼接与解析

    常用方式: json字符串拼接(目前使用过两种方式): 1.运用StringBuilder拼接 StringBuilder json = new StringBuilder(); json.appen ...

  9. 后台给前台传JSON字符串,前台解析并拼接显示

    后台传JSON public class CourseType : IHttpHandler { Epoint.PeiXun.Bizlogic.BLL.CourseLibrary.PX_CourseT ...

随机推荐

  1. OpenCV学习笔记(一)安装及运行第一个OpenCV程序

    1.下载及安装 OpenCV是一套开源免费的图形库,主要有C/C++语言编写,官网: http://opencv.org/ .在 http://opencv.org/downloads.html 可以 ...

  2. JavaMail简单接收邮件

    一个简单的例子,收取所有邮件并在控制台输出. package cn.jmail.test; import java.io.*; import java.util.*; import javax.mai ...

  3. 理解ThreadLocal(一)

    早在JDK 1.2的版本中就提供java.lang.ThreadLocal,ThreadLocal为解决多线程程序的并发问题提供了一种新的思路.使用这个工具类可以很简洁地编写出优美的多线程程序. Th ...

  4. mac osx 10.9 ftp server端口

    开启 FTP Serversudo -s launchctl load -w /System/Library/LaunchDaemons/ftp.plist 关闭 FTP Serversudo -s ...

  5. View的工作原理(二)——layout

    1.当View的measure被确定后,会调用ViewGroup的layout方法,之后使用onLayout方法(同样也是系统未自动重写,要我们自己完成)遍历子View(根Measure的形式是一样得 ...

  6. 1003 Crashing Balloon

    考察DFS的应用,判断两个数的因子. #include <stdio.h> int f1,f2; void DFS(int m,int n,int k){ ){ f2=; ) f1=; } ...

  7. Mongo客户端

    http://www.linuxidc.com/Linux/2012-07/64233.htm http://www.oschina.net/p/rockmongo http://www.cnblog ...

  8. Poj 1269 Intersecting Lines_几何模板

    #include <iostream> #include <math.h> #include <iomanip> #define eps 1e-8 #define ...

  9. 751D·PARK北京时尚设计广场_百度百科

    751D·PARK北京时尚设计广场_百度百科 751D·PARK北京时尚设计广场

  10. linux修改系统时间date命令加clock -w

    http://m.jb51.net/LINUXjishu/117784.html 修改linux系统时间的方法(date命令) 11-18 23:22:27作者:脚本之家 命令格式为: date -s ...