1. Log机制

做一些大型项目的时候,对代码的调试最有效的办法往往是最直接、最简单的log机制

即对可以出设置打印店,对应打印信息进行调试(当然是有gdb也许你会觉得很高大上,但是实际项目中,gdb比较繁琐,且比较适合老司机使用)。

如果log机制打印的好,则会事半功倍,大大提升调试的效率。博主今天就工作当中的使用到的log机制分享给大家,希望对大家有所帮助。

先看几张图:

1). Trace:

格式: 日期+时间+文件名+ <函数名 所在行> + : + 打印的信息

2). Error Log:

格式: Time + 设备信息 + 系统信息 + 系统事件编号 + 错误信息

2. Trace code

/* 功能描述: 获取当前系统日期,并转化成字符串输出 */
int get_date_string(char *str)
{
time_t now;
struct tm *tm_now; time(&now);
tm_now = localtime(&now); sprintf(str,"%4d-%02d-%02d %02d:%02d:%02d", \
tm_now->tm_year+1900, tm_now->tm_mon+1, tm_now->tm_mday,\
tm_now->tm_hour, tm_now->tm_min, tm_now->tm_sec); return 0;
} /* 功能描述: 获取当前系统时间,返回值单位为微妙 */
uint64_t get_sys_time_usec()
{
timeval tv;
gettimeofday(&tv,NULL);
return (tv.tv_usec);
}
/******************************************************************************
*宏定义: Trace_Log
*功能描述: 将中间过程信息打印到文件中
*全局影响:
*说明:必须使用宏,否则打印的函数名、行号为本函数的
* 作者: 修改时间 内容
* Jimmy 2018-03-22 创建
*
*
******************************************************************************/
#define Trace_Log(fd, fmt, arg...) \
{ \
\
int err_code = 0; \
char time_str[128]; /* 存储本地时间信息<年月日时分秒> */ \
char code_info[128]; /* 存储调用的<文件-函数-行号> */ \
char str[1024]; /* 存储传入的参数*/ \
char usec_str[32]; /*存放微秒*/ \
/* 1. 检查输入参数是否合法 */ \
if(fd < 0) \
{ \
printf("Input arguments [fd=%d] error! ", fd); \
err_code = -1; \
} \
\
if(0 == err_code) \
{ \
memset(time_str, '\0', sizeof(time_str)); \
memset(code_info, '\0', sizeof(code_info)); \
memset(str, '\0', sizeof(str)); \
memset(usec_str, '\0', sizeof(usec_str)); \
/* 2. 获取系统时间,获取调用者的文件/函数/行号 */ \
get_date_string(time_str); \
\
sprintf(usec_str, "+%06dus", get_sys_time_usec()); \
sprintf(code_info, ": %-10s <%-10s %u> : ", __FILE__, __FUNCTION__, __LINE__); \
\
strcat(time_str, usec_str); \
strcat(time_str, code_info); /*将两者结合*/ \
\
sprintf(str, fmt, ##arg); /*将传入的参数按格式写入到buf*/ \
\
/* 3. 将本地时间 - 文件/函数/行信息 - 字符串信息 写入到相应的文件中*/\
write(fd, (void *)time_str, strlen(time_str)); \
usleep(1000); \
write(fd, (void *)str, strlen(str)); \
usleep(1000); \
write(fd, "\n", 1); \
} \
}

3. Error Log code

/******************************************************************************
*宏定义: Error_Log
*功能描述: 将错误信息打印到文件中 *说明:必须使用宏,否则打印的函数名、行号为本函数的
* 作者: 修改时间 内容
* Jimmy 2018-03-28 创建
*
*
******************************************************************************/ #define Error_Log(fmt, arg...) \
{ \
char tmp[64]; \
char time_str[128]; /* 存储本地时间信息<年月日时分秒> */ \
char code_info[64]; /* 存储调用的<文件-函数-行号> */ \
char usec_str[12]; /*存储微妙时间*/ \
char error_info[512]; /*存储错误信息*/ \
char dev_id[15]; /*存储设备的ID信息*/ \
char systemVer[8]; /*存储系统的版本号*/ \
char process[8]; /*存储运行软件的版本号*/ \
char stm32Ver[8]; /*存储底层硬件的版本号*/ \
\
if(g_err_fd != -1) \
{ \
memset(time_str, '\0', sizeof(time_str)); \
memset(code_info, '\0', sizeof(code_info)); \
memset(usec_str, '\0', sizeof(usec_str)); \
\
memset(dev_id, '\0', sizeof(dev_id)); \
memset(systemVer, '\0', sizeof(systemVer)); \
memset(process, '\0', sizeof(process)); \
memset(stm32Ver, '\0', sizeof(stm32Ver)); \
\
memset(error_info, '\0', sizeof(error_info)); \
memset(tmp, '\0', sizeof(tmp)); \
\
/* 1. 获取时间值 */ \
get_date_string(time_str); \
sprintf(usec_str, "+%06dus", (get_sys_time_usec()%1000000LLU)); \
strcat(time_str, usec_str); \
write(g_err_fd, "\nTIME\t\t\t\t: ", strlen("\nTIME\t\t\t\t: ")); \
write(g_err_fd, (void *)time_str, strlen(time_str)); \
write(g_err_fd, "\n", 1); \
\
/* 2. 获取设备id和软件版本信息 (这些根据实际项目修改*/ \
GetSysCmdInfo(dev_id, "read_misc mid | tr -d '\n'"); \
GetSysCmdInfo(systemVer, "read_misc version | tr -d '\n'"); \
GetSysCmdInfo(process, "read_misc software_version | tr -d '\n'"); \
GetSysCmdInfo(stm32Ver, "read_misc stm32_version | tr -d '\n'"); \
\
write(g_err_fd, "DEVICE_ID\t\t\t: ", strlen("DEVICE_ID\t\t\t: ")); \
write(g_err_fd, (void *)dev_id, strlen(dev_id)); \
write(g_err_fd, "\n", 1); \
\
write(g_err_fd, "SYSTEM_VERSION\t\t: ", strlen("SYSTEM_VERSION\t\t: "));\
write(g_err_fd, (void *)systemVer, strlen(systemVer)); \
write(g_err_fd, "\n", 1); \
\
write(g_err_fd, "PROCESS\t\t\t: ", strlen("PROCESS\t\t: "));\
write(g_err_fd, (void *)process, strlen(process)); \
write(g_err_fd, "\n", 1); \
\
write(g_err_fd, "SYSTEM_EVENT\t\t: ", strlen("SYSTEM_EVENT\t\t: ")); \
write(g_err_fd, (void *)stm32Ver, strlen(stm32Ver)); \
write(g_err_fd, "\n", 1); \
\
/* 3. 获取文件的行列信息 */ \
sprintf(code_info, "FILE\t\t\t\t: %-10s <%-10s %u> : \n", __FILE__, __FUNCTION__, __LINE__); \
write(g_err_fd, (void *)code_info, strlen(code_info)); \
\
/* 4. 获取出错信息*/ \
sprintf(error_info, fmt, ##arg); \
write(g_err_fd, "ERROR_DETAIL\t\t: " , strlen("ERROR_DETAIL\t\t: " )); \
write(g_err_fd, (void *)error_info, strlen(error_info)); \
write(g_err_fd, "\n", 1); \
\
/* 5. 写入结束信息 */ \
for(int i=0; i < sizeof(tmp)-1; i++) \
{tmp[i]='-';} \
tmp[sizeof(tmp)-1] = '\0'; \
write(g_err_fd, (void *)tmp, strlen(tmp)); \
write(g_err_fd, "\n", 1); \
bzero(tmp, sizeof(tmp)); \
} \
\
else \
{ \
printf("The g_err_fd was not inited !!!"); \
} \
}

Trace & Error log in file的更多相关文章

  1. MYSQL 5.7 无法启动(Could not open error log file errno 2)

    前两天电脑中毒, 病毒好像把mysql的 log.err 文件给删掉了.然后服务一直启动不了:Could not open error log file errno 2. 然后疯狂百度,搜索的结果大多 ...

  2. mariadb:InnoDB: Error: log file ./ib_logfile0 is of different size 0 5242880 bytes

    mariadb 启动中 InnoDB: Error: log file ./ib_logfile0 is of different size 0 起因:线上正在运行的系统,因为需要调整性能,变更了my ...

  3. Nginx启动报错: could not open error log file: open() &q

    启动nginx报如下错误: nginx: [alert] could not open error log file: open() "/usr/local/nginx/logs/error ...

  4. 报错问题:InnoDB: Error: log file ./ib_logfile0 is of different size

    InnoDB: Error: log file ./ib_logfile0 is of different size bytesInnoDB: than specified in the .cnf f ...

  5. E/Trace: error opening trace file: No such file or directory

    E/Trace: error opening trace file: No such file or directory (2) 有这一个错误,想了一下,然后发现是 AdroidManifest.xm ...

  6. The server instance Witness rejected configure request; read its error log file for more information. The reason 1427, and state 31, can be of use for

    数据库服务器做了镜像之后,发现有错误信息 The server instance Witness rejected configure request; read its error log file ...

  7. Got fatal error 1236 from master when reading data from binary log: 'Could not find first log file name in binary log index file'系列一:

    从库报这个错误:Got fatal error 1236 from master when reading data from binary log: 'Could not find first lo ...

  8. mysql从库Last_IO_Error: Got fatal error 1236 from master when reading data from binary log: 'Could not find first log file name in binary log index file'报错处理

    年后回来查看mysql运行状况与备份情况,登录mysql从库查看主从同步状态 mysql> show slave status\G; *************************** . ...

  9. Got fatal error 1236 from master when reading data from binary log: 'Could not find first log file name in binary log index file'

    setup slave from backup i got error Got fatal error 1236 from master when reading data from binary l ...

随机推荐

  1. 通过mybatis向数据库中插入日期数据

    遇到的问题: 通过mybatis向数据库中插入日期格式数据,发现只有年月日, 没有小时分钟和秒 当你想在实体类中使用java.util.Date类型,而且还想在数据库中保存时分秒时, 解决办法: 你可 ...

  2. 撸.NET Core的正确姿势

    特点 案例基于刚发布的.NET Core 2.1 只需一台Linux服务器搞定一切, 全程无需自己配置dotnet环境, 需要熟悉git docker基础知识可有可无, 过了下面几个步骤,你就已经入门 ...

  3. ASP.NET MVC 中 Autofac依赖注入DI 控制反转IOC 了解一下

    先简单了解一这个几个 名词的意思. 控制反转(IOC) 依赖注入(DI) 并不是某种技术. 而是一种思想.一种面向对象编程法则 什么是控制反转(IOC)?  什么是依赖注入(DI) 可以点击下面链接 ...

  4. 「PKUSC2018」主斗地(暴搜)

    这道斗地主比 \(PKUWC\) 那道可做多了... 我们用 \(NOIP\) 那道斗地主的思路:暴搜出三代和四代,贪心出散牌. 还有jry为什么要出xx网友而不出他的另一个老婆 我们发现两个人的每回 ...

  5. ROC,AUC,Precision,Recall,F1的介绍与计算

    1. 基本概念 1.1 ROC与AUC ROC曲线和AUC常被用来评价一个二值分类器(binary classifier)的优劣,ROC曲线称为受试者工作特征曲线 (receiver operatin ...

  6. cad2014卸载/安装失败/如何彻底卸载清除干净cad2014注册表和文件的方法

    cad2014提示安装未完成,某些产品无法安装该怎样解决呢?一些朋友在win7或者win10系统下安装cad2014失败提示cad2014安装未完成,某些产品无法安装,也有时候想重新安装cad2014 ...

  7. I/O类型

    同步和异步 synchronous  asyncronous 关注的是消息通知机制 同步:调用发出之后不会立即返回,但一旦返回,则返回即是最终结果. 异步:调用发出之后,被调用方立即返回消息,但返回的 ...

  8. Postgres 的 Range 类型

    mysql 不支持 Range 类型 零.介绍 1. 适用场景: a.可以用于实现 是否满足薪资需求 的功能 b.可以用于实现 是否符合上线时间 的功能 一.定义 1.类型范围 Postgres Se ...

  9. redis集成相关工具类

    package cn.yiyuan.util; import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; i ...

  10. odoo开发笔记-自定义发送邮件模板

    1. 首先激活开发者模式 2. 点击设置 - Email - 模板 - “选择你需要修改的模板” 我们以销售模块-报价单 邮件模板为例 来说明. quote order 原先默认模板,发出的邮件显示效 ...