Skip to content

ltoddy/parser-tutorial

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

31 Commits
 
 
 
 
 
 
 
 
 
 

Repository files navigation

json parser

之前看到知乎上有人问,会写ParserTokenizer是什么水平,绝大情况下,屁用没有。小部分情况,就看你运气了。因为这东西,面试又不会加分,而且,如果你面试的小公司,可能面试官甚至都不懂你在说啥。

json这种数据格式,应该算是人人皆知的了,其语法规则不必赘述。

我想借助编写一份json parser来讲解语法解析,通过实践来学习。


简单来说,parser就是个转换器,输入是一个字符串,而输出是一个你自己定义一个数据结构。 对于字符串来说,他有各种各样的符号, 例如字符串r"{ "x": 10, "y": [20], "z": "some" }", 有左右花括号(一般来说,左括号叫开放括号,右括号叫做闭合括号),有逗号,有分号,有字符串,数字等等。

对于JSON,我们需要实现两个方法:

  • 用于解析JSON的 parse() 方法.
  • 以及将对象/值转换为JSON字符串的stringify()方法。

第一步,编写Tokenizer!

我们将一个字符串进行初次解析,将一个一个的符号,变成我们的数据结构(Token),每个Token会标识,“它”是什么, 例如:

一个字符串"some"可能会被转换成:

Token {
    type: string,
    content: "some"
}

对于Rust语言来说,提供了枚举,那么我们就可以借助枚举来定义我们的Token

src/token.rs

#[derive(Debug)]
pub enum Token {
    Comma,
    Colon,
    BracketOn,
    BracketOff,
    BraceOn,
    BraceOff,
    String(String),
    Number(f64),
    Boolean(bool),
    Null,
}

上述Token枚举就包含了JSON字符串里所有出现‘符号’的种类:逗号,分号,左方括号,右方括号,左花括号,右花括号,字符串,数字,布尔,和null。

对于将字符串解析成一系列Token的东西,我们称之为:Tokenizer

src/tokenizer.rs

pub struct Tokenizer<'a> {
    source: Peekable<Chars<'a>>,
}

对于Tokenizer,我们希望它能够有一个方法:next,每次调用这个方法,会返回 给我们一个Token,当没有Token返回的时候,则表示输入的字符串已经全部解析完。

很幸运,Rust的内置接口里面(trait我一般称作特征,这里写作了接口,这样子大众也更容易方便理解。),含有这么一个接口:

Iterator接口。

src/tokenizer.rs

impl<'a> Iterator for Tokenizer<'a> {
    type Item = Token;

    fn next(&mut self) -> Option<Self::Item> {
        'lex: while let Some(ch) = self.source.next() {
            return Some(match ch {
                ',' => Token::Comma,
                ':' => Token::Colon,
                '[' => Token::BracketOn,
                ']' => Token::BracketOff,
                '{' => Token::BraceOn,
                '}' => Token::BraceOff,
                '"' => Token::String(self.read_string(ch)),
                '0'..='9' => Token::Number(self.read_number(ch)),
                'a'..='z' => {
                    let label = self.read_symbol(ch);
                    match label.as_ref() {
                        "true" => Token::Boolean(true),
                        "false" => Token::Boolean(false),
                        "null" => Token::Null,
                        _ => panic!("Invalid label: {}", label),
                    }
                }
                _ => {
                    if ch.is_whitespace() {
                        continue 'lex;
                    } else {
                        panic!("Invalid character: {}", ch);
                    }
                }
            });
        }

        None
    }
}

我们一个一个读入字符串的字符,判断他归属于哪一个类型(token type),

从上面代码里看,对于那些符号的判断,最为简单,直接返回它对应的Token就可以了. 对于字符串,数字,符号(null, true, false),就稍微难一点判断了.

对于字符串,它的样子就像"this is a string",由一对双引号包围,更复杂一些的字符串,其含有转义字符: "This is a string\\n".

对于解析字符串,当我们首次遇到双引号字符时,我们判定,其随后的内容是一个字符串,当第二次遇到双引号的时候,我们判断,其字符串结束。

当遇到转移字符\的时候,我们所需要做的就是忽略第一个\,将之后的字符保存。

对于其他的字符,仅仅是遍历一遍保存便可。

src/tokenizer.rs

impl<'a> Tokenizer<'a> {
    fn read_string(&mut self, first: char) -> String {
        let mut value = String::new();
        let mut escape = false;
    
        while let Some(ch) = self.source.next() {
            if ch == first && escape == false {
                return value;
            }
            match ch {
                '\\' => {
                    if escape {
                        escape = false;
                        value.push(ch);
                    } else {
                        escape = true;
                    }
                }
                _ => {
                    value.push(ch);
                    escape = false;
                }
            }
        }
    
        value
    }
}

对于数字,其特征为数字开头,随后为数字,其中也可能包含一个小数点。所以,只要它是一数字开头, 我们便可判断它及其后面的字符串是一个完整的数字。并且,有且只可能有0个或1个小数点。

src/tokenizer.rs

impl<'a> Tokenizer<'a> {
    fn read_number(&mut self, first: char) -> f64 {
        let mut value = first.to_string();
        let mut point = false;
    
        while let Some(&ch) = self.source.peek() {
            match ch {
                '0'..='9' => {
                    value.push(ch);
                    self.source.next();
                }
                '.' => {
                    if !point {
                        point = true;
                        value.push(ch);
                        self.source.next();
                    } else {
                        return value.parse::<f64>().unwrap();
                    }
                }
                _ => return value.parse::<f64>().unwrap(),
            }
        }
    
        value.parse::<f64>().unwrap()
    }
}

对于符号null, true, false, { "is_symbol": true }比如这样子的json字符串。 它们不像字符串,由两个双引号包围,它们只是由单纯的英文小写字母组成。

src/tokenizer.rs

impl<'a> Tokenizer<'a> {
    fn read_symbol(&mut self, first: char) -> String {
        let mut symbol = first.to_string();
    
        while let Some(&ch) = self.source.peek() {
            match ch {
                'a'..='z' => {
                    symbol.push(ch);
                    self.source.next();
                }
                _ => break, // 遇到非英文小写字母,判定它结束
            }
        }
    
        symbol
    }
}

等到这里,如果实现了Tokenizer,让他能够不断地给我们解析Token,基本上第一个难点就算结束了。 因为,当我们把输入的字符串一个一个的解析成了一系列Token之后,剩下的很大一部分就是天高任鸟飞 的时候,为什么?很简单,Token也是我们自己定义的数据结构,而且它在内存中,我们想怎么用它就可以 怎么用它.

第二步,编写Parser!

下面,我们将我们一系列的Token解析成我们的JSON.

src/value.rs

pub enum Json {
    Null,
    String(String),
    Number(f64),
    Boolean(bool),
    Array(Vec<Json>),
    Object(HashMap<String, Json>),
}

如果你不清楚Json, 你可以看下: Introducing JSON

Parser接受字符串,借助我们刚才编写的Tokenizer, 然输出抽样语法树(一般来说,Parser接受字符串,然后输出抽象语法书,不过,管他呢,我们能实现我们想要实现的便可,管它具体的定义呢), 对于我们的Json Parser,输出的就是我们刚才定义的Json结构.

src/parser

pub struct Parser<'a> {
    tokenizer: Tokenizer<'a>,
}

这就是我们Parser的定义,它内含一个Tokenizer,要借助它生成的Toekn去变成Json

src/parser

impl<'a> Parser<'a> {
    pub fn parse(&mut self) -> Json {
        let token = self.step();

        self.parse_from(token)
    }

    fn step(&mut self) -> Token {
        self.tokenizer.next().expect("Unexpected end of JSON!!!")
    }

    fn parse_from(&mut self, token: Token) -> Json {
        match token {
            Token::Null => Json::Null,
            Token::String(s) => Json::String(s),
            Token::Number(n) => Json::Number(n),
            Token::Boolean(b) => Json::Boolean(b),
            Token::BracketOn => self.parse_array(),
            Token::BraceOn => self.parse_object(),
            _ => panic!("Unexpected token: {:?}", token),
        }
    }
}

以上代码就是Parser的核心了,其运行原理与Tokenizer相仿。

Json中的数据结构:booleanstringnull,以及array(以左方括号开头,右方括号结尾),object(以左花括号开头,右花括号结尾)。

借助于Tokenizer生成Token,进行模式匹配,当遇到Token::NullToken::String(s)Token::Number(n)Token::Boolean(b), 这些时,直接返回就可以了,毕竟我们已经在实现Tokenizer的时候处理过了。

当遇到Token::BracketOn,左括号!这是array开始的符号,那么我们交给self.parse_array()处理:

src/parser.rs

impl<'a> Parser<'a> {
    fn parse_array(&mut self) -> Json {
        let mut array = Vec::new();

        match self.step() {
            Token::BracketOff => return array.into(),
            token => array.push(self.parse_from(token)),
        }

        loop {
            match self.step() {
                Token::Comma => array.push(self.parse()),
                Token::BracketOff => break,
                token => panic!("Unexpected token {:?}", token),
            }
        }

        array.into()
    }
}

如上使我们如何处理array,当遇到右方括号的时候,表明array结束了,返回即可。 对于我们array类型,其每一个元素都可以为Json,并且,元素之间用逗号分割, 那么当遇到逗号Token::Comma的时候,就可以断定一个新的元素出现。

当遇到Token::BraceOn,左花括号!这是object开始的符号,那么我们交给self.parse_object()处理:

src/parser.rs

impl<'a> Parser<'a> {
    fn parse_object(&mut self) -> Json {
        let mut object = HashMap::new();

        match self.step() {
            Token::BraceOff => return object.into(),
            Token::String(key) => {
                match self.step() {
                    Token::Colon => do_nothing(),
                    token => panic!("Unexpected token {:?}", token),
                }
                let value = self.parse();
                object.insert(key, value);
            }
            token => panic!("Unexpected token {:?}", token),
        }

        loop {
            match self.step() {
                Token::Comma => {
                    let key = match self.step() {
                        Token::String(key) => key,
                        token => panic!("Unexpected token {:?}", token),
                    };
                    match self.step() {
                        Token::Colon => {}
                        token => panic!("Unexpected token {:?}", token),
                    }
                    let value = self.parse();
                    object.insert(key, value);
                }
                Token::BraceOff => break,
                token => panic!("Unexpected token {:?}", token),
            }
        }

        object.into()
    }
}

parse_object同理。

做到这里,目标之一的parse函数就能够实现了:

src/lib.rs

pub fn parse(s: &str) -> Json {
    let mut parser = Parser::new(s);
    parser.parse()
}

第三步, stringify

当我们实现从一个字符串变成Json结构后,也要实现Json结构变回原来的字符串。

src/code_generator.rs

pub struct CodeGenerator {
    value: String,
}

impl CodeGenerator {
    pub fn new() -> Self {
        Self {
            value: String::new(),
        }
    }

    pub fn gather(&mut self, json: &Json) {
        self.write_json(json)
    }

    pub fn product(self) -> String {
        self.value
    }

    fn write(&mut self, slice: &str) {
        self.value.push_str(slice);
    }

    fn write_char(&mut self, ch: char) {
        self.value.push(ch);
    }

    fn write_json(&mut self, json: &Json) {
        match *json {
            Json::Null => self.write("null"),
            Json::Boolean(ref b) => self.write(if *b { "true" } else { "false" }),
            Json::Number(ref n) => self.write(&n.to_string()),
            Json::String(ref s) => self.write(&format!("{:?}", s)),
            Json::Array(ref a) => self.write_array(a),
            Json::Object(ref o) => self.write_object(o),
        }
    }
}

定义一个CodeGenerator结构体,接受一个Json,然后产生一个String

对于将Json转化成String这个功能,相对来说就简单多了。如果你过去学过比如Java,Python Ruby等等一些面向对象的语言的话,都会知道一个对象的方法:toString(Python是__str__,Ruby是to_s)。

换句话说,我们就是给Json增添一个toString方法。而且,Json是我们自己定义的有规则的数据结构,实现它变成 String的操作就简单了许多。

那么,如上述代码,还是利用模式匹配,去实现JsonString的转换。

唯一有一点点复杂的也就是ArrayObject的转换。

src/code_generator.rs

impl CodeGenerator {
    fn write_array(&mut self, array: &[Json]) {
        self.write_char('[');

        for (i, elem) in array.iter().enumerate() {
            self.write_json(elem);
            if i != (array.len() - 1) {
                self.write_char(',');
            }
        }

        self.write_char(']');
    }
}

对于一个Array,他形如[element1, element2, element3, ...],左右两边各有一个方括号。 里面的元素之间由逗号相隔(除了最后一个元素外,其他元素后尾随一个逗号)。

src/code_generator.rs

impl CodeGenerator {
    fn write_object(&mut self, object: &HashMap<String, Json>) {
        self.write_char('{');

        for (i, (key, value)) in object.iter().enumerate() {
            self.write(&format!("{:?}", key));
            self.write_char(':');
            self.write_json(value);
            if i != (object.len() - 1) {
                self.write_char(',');
            }
        }

        self.write_char('}');
    }
}

对于Object,它形如"{ "key1": value1, "key2": value2, ... }",左右两边各有一个花括号。 里面元素为key-value形式,keyvalue之间有一个冒号,并且,keyString类型,valueJson类型。 同时在key-valuekey-value也有逗号分隔。

最后,实现我们的stingify方法:

pub fn stringify<T>(o: T) -> String
where
    T: Into<Json>,
{
    let mut gen = CodeGenerator::new();
    gen.gather(&o.into());
    gen.product()
}

第四步,实现泛型

src/implement.rs

macro_rules! impl_from_num_for_json {
    ($($t:ident)*) => {
        $(
            impl From<$t> for Json {
                fn from(n: $t) -> Json {
                    Json::Number(n as f64)
                }
            }
        )*
    };
}

impl_from_num_for_json!(u8 i8 u16 i16 u32 i32 u64 i64 usize isize f32 f64);

impl From<bool> for Json {
    fn from(b: bool) -> Json {
        Json::Boolean(b)
    }
}

impl From<String> for Json {
    fn from(s: String) -> Json {
        Json::String(s)
    }
}

impl<'a> From<&'a str> for Json {
    fn from(s: &'a str) -> Json {
        Json::String(s.to_string())
    }
}

impl From<Vec<Json>> for Json {
    fn from(v: Vec<Json>) -> Self {
        Json::Array(v)
    }
}

impl From<HashMap<String, Json>> for Json {
    fn from(mut map: HashMap<String, Json>) -> Self {
        let mut object = HashMap::new();

        for (key, value) in map.drain() {
            object.insert(key, Json::from(value));
        }

        Json::Object(object)
    }
}

下一步?

  • 错误处理, rust提供了Result枚举,以及语法糖来做错误处理。(尽可能的在Rust中避免使用panic!
  • 过程宏,实现jsonify过程宏,使得用户定义的数据结构能够反序列化Json和序列化成Json
  • 实现json formatter

更多的学习

请移步至 wiki

About

write your own json parser.

Topics

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages