美文网首页
tutorial01

tutorial01

作者: 风之羁绊 | 来源:发表于2019-05-25 15:49 被阅读0次
  1. .h文件为了防止重复声明,使用

    ifndef LEPTJSON_H__

    define LEPTJSON_H__

    endif

  2. assert 断言的使用
    断言(assertion)是 C 语言中常用的防御式编程方式,减少编程错误。最常用的是在函数开始的地方,检测所有参数。有时候也可以在调用函数后,检查上下文是否正确。

C 语言的标准库含有 [assert()]这个宏(需 #include <assert.h>),提供断言功能。当程序以 release 配置编译时(定义了 NDEBUG 宏),assert() 不会做检测;而当在 debug 配置时(没定义 NDEBUG 宏),则会在运行时检测 assert(cond) 中的条件是否为真(非 0),断言失败会直接令程序崩溃。

另一个问题是,初学者可能会难于分辨何时使用断言,何时处理运行时错误(如返回错误值或在 C++ 中抛出异常)。简单的答案是,如果那个错误是由于程序员错误编码所造成的(例如传入不合法的参数),那么应用断言;如果那个错误是程序员无法避免,而是由运行时的环境所造成的,就要处理运行时错误(例如开启文件失败)。

  1. 宏的编写技巧
    有些同学可能不了解 EXPECT_EQ_BASE 宏的编写技巧,简单说明一下。反斜线代表该行未结束,会串接下一行。而如果宏里有多过一个语句(statement),就需要用 do { /.../ } while(0) 包裹成单个语句

第一次看带宏的代码,感觉有点恶心。。
.h文件

#ifndef LEPTJSON_H__
#define LEPTJSON_H__

typedef enum { LEPT_NULL, LEPT_FALSE, LEPT_TRUE, LEPT_NUMBER, LEPT_STRING, LEPT_ARRAY, LEPT_OBJECT } lept_type;

typedef struct {
    lept_type type;
}lept_value;

enum {
    LEPT_PARSE_OK = 0,
    LEPT_PARSE_EXPECT_VALUE,  
    LEPT_PARSE_INVALID_VALUE,
    LEPT_PARSE_ROOT_NOT_SINGULAR
};

int lept_parse(lept_value* v, const char* json);

lept_type lept_get_type(const lept_value* v);

#endif /* LEPTJSON_H__ */

根据enum的性质, LEPT_PARSE_EXPECT_VALUE, LEPT_PARSE_INVALID_VALU, LEPT_PARSE_ROOT_NOT_SINGULAR 这三个自然为1,2,3

LEPT_NULL, LEPT_FALSE, LEPT_TRUE, LEPT_NUMBER, LEPT_STRING, LEPT_ARRAY, LEPT_OBJECT为0到6
然后声明了两个接口,一个用来解析json串,第二个返回类型

然后先进入单元测试部分的代码
头文件引入

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "leptjson.h"

static int main_ret = 0;
static int test_count = 0;
static int test_pass = 0;

引入两个宏

#define EXPECT_EQ_BASE(equality, expect, actual, format) \
   do {\
       test_count++;\
       if (equality)\
           test_pass++;\
       else {\
           fprintf(stderr, "%s:%d: expect: " format " actual: " format "\n", __FILE__, __LINE__, expect, actual);\
           main_ret = 1;\
       }\
   } while(0)

#define EXPECT_EQ_INT(expect, actual) EXPECT_EQ_BASE((expect) == (actual), expect, actual, "%d")

用处就是根据返回值来输出测试的情况。

static void test_parse() {
    test_parse_null();
    test_parse_expect_value();
    test_parse_invalid_value();
    test_parse_root_not_singular();
}

int main() {
    test_parse();
    printf("%d/%d (%3.2f%%) passed\n", test_pass, test_count, test_pass * 100.0 / test_count);
    return main_ret;
}

主函数,主测试函数,测试这4种功能

static void test_parse_null() {
   lept_value v;
   v.type = LEPT_FALSE;
   EXPECT_EQ_INT(LEPT_PARSE_OK, lept_parse(&v, "null"));
   EXPECT_EQ_INT(LEPT_NULL, lept_get_type(&v));
}

然后回到.c文件中看

int lept_parse(lept_value* v, const char* json) {
    lept_context c;
    assert(v != NULL);
    c.json = json;
    v->type = LEPT_NULL;
    lept_parse_whitespace(&c);
    return lept_parse_value(&c, v);
}

lept_context 里面有一个指向常量的指针,指向的内容不可变
typedef struct {
const char* json;
}lept_context;
而常指针的写法为char* const json,指针不可另外指

把v的type定义为NULL,然后用把空白的去掉,指针前进,然后进入判断NULL,并返回返回值,并为type进行赋值

static void test_parse_expect_value() {
    lept_value v;

    v.type = LEPT_FALSE;
    EXPECT_EQ_INT(LEPT_PARSE_EXPECT_VALUE, lept_parse(&v, ""));
    EXPECT_EQ_INT(LEPT_NULL, lept_get_type(&v));

    v.type = LEPT_FALSE;
    EXPECT_EQ_INT(LEPT_PARSE_EXPECT_VALUE, lept_parse(&v, ""));
    EXPECT_EQ_INT(LEPT_NULL, lept_get_type(&v));
}

static void test_parse_invalid_value() {
    lept_value v;
    v.type = LEPT_FALSE;
    EXPECT_EQ_INT(LEPT_PARSE_INVALID_VALUE, lept_parse(&v, "nul"));
    EXPECT_EQ_INT(LEPT_NULL, lept_get_type(&v));

    v.type = LEPT_FALSE;
    EXPECT_EQ_INT(LEPT_PARSE_INVALID_VALUE, lept_parse(&v, "?"));
    EXPECT_EQ_INT(LEPT_NULL, lept_get_type(&v));
}

这两个类似,不讲了

static void test_parse_root_not_singular() {
    lept_value v;
    v.type = LEPT_FALSE;
    EXPECT_EQ_INT(LEPT_PARSE_ROOT_NOT_SINGULAR, lept_parse(&v, "null x"));
    EXPECT_EQ_INT(LEPT_NULL, lept_get_type(&v));
}

这个报错了

现在算看懂了代码了吧,但可学的还是不少,

然后要求改的内容为以下几点


要求改的点

代码改完,进入第二讲把

相关文章

网友评论

      本文标题:tutorial01

      本文链接:https://www.haomeiwen.com/subject/ujiazqtx.html