gpt4 book ai didi

C错误处理异常

转载 作者:太空宇宙 更新时间:2023-11-04 08:04:17 25 4
gpt4 key购买 nike

我想在 C 中做一些错误处理。我喜欢 C# 异常,所以决定做类似的事情。目标是:

  • 调用函数中的一行错误处理,如 C# 中的抛出异常
  • 中断发生错误的调用函数中的代码序列
  • 知道发生错误的文件/行
  • 关于错误的口头描述

您认为这种做法有什么问题吗?

调用函数:

EError activate_add(uint32_t key)
{
if (activate_bufferSize+1>=activate_bufferMax) THROW_ERROR("activate list full");
activate_buffer[activate_bufferSize].Id = activate_bufferSize+1;
activate_buffer[activate_bufferSize].StoredKey = key;
activate_bufferSize++;
TRY(activate_saveToEEProm());
NO_ERROR();
}

我的解决方案如下:

#ifndef ERROR_H_
#define ERROR_H_


typedef enum
{
Message_None,
Message_Error,
Message_Warning,
Message_Info
}EMessage_type;

typedef struct
{
EMessage_type message_type;
char* message;
const char* file;
uint32_t line;
}EErrorStruct,*EError;

#define THROW_ERROR(message, ...) { EError __err=GET_MESSAGE(Message_Error,(char*)__FILE__,__LINE__,message,##__VA_ARGS__); SEND_MESSAGE(__err); return __err;}
#define TRY(__x) { EError __err = __x; if (__err->message_type==Message_Error) { return __err;}}

#define TRYHAL(__x) { HAL_StatusTypeDef __res = __x; if (__res != HAL_OK) { THROW_ERROR("HAL problem: %u",__res);}}
#define TRYFAT(__x) { FRESULT __res = __x; if (__res != FR_OK) { THROW_ERROR("FAT problem: %u",__res);}}

#define NO_ERROR() { return GET_MESSAGE(Message_None,NULL,0,NULL,0);}
#define SEND_ERROR(message , ...) { SEND_MESSAGE(GET_MESSAGE(Message_Error,(char*)__FILE__,__LINE__,message,##__VA_ARGS__)); }
#define SEND_WARN(message , ...) { SEND_MESSAGE(GET_MESSAGE(Message_Warning,(char*)__FILE__,__LINE__,message,##__VA_ARGS__)); }
#define SEND_INFO(message , ...) { SEND_MESSAGE(GET_MESSAGE(Message_Info,(char*)__FILE__,__LINE__,message,##__VA_ARGS__)); }

EError GET_MESSAGE(EMessage_type messagetype,const char* file, uint32_t line,const char *format, ... );
void SEND_MESSAGE(EError err);
uint8_t ISFAILED(EError err);


#endif /* ERROR_H_ */

源文件

#include <error.h>
#include "stdio.h"
#include "stdarg.h"

static EErrorStruct error;
static const EError m = &error;
static char buffer[300];

EError GET_MESSAGE(EMessage_type messagetype,const char* file, uint32_t line,const char *format, ... )
{

va_list args;
va_start(args,format);
vsprintf(buffer, format, args);
va_end(args);

m->message = buffer;
m->message_type = messagetype;
m->file = file;
m->line = line;

return m;
}



void SEND_MESSAGE(EError err)
{
switch (err->message_type) {
case Message_Error:
printf("ERRO: %s \r\n\t\t\t\t\t\t\t\t\t\t at %s line: %u\r\n",err->message,err->file,(unsigned int)err->line);
break;
case Message_Warning:
printf("WARN: %s \r\n\t\t\t\t\t\t\t\t\t\t at %s line: %u\r\n",err->message,err->file,(unsigned int)err->line);
break;
case Message_Info:
printf("INFO: %s \r\n\t\t\t\t\t\t\t\t\t\t at %s line: %u\r\n",err->message,err->file,(unsigned int)err->line);
break;
default:
break;
}

}
uint8_t ISFAILED(EError err)
{
if (err->message_type == Message_Error) return 1;


return 0;
}

最佳答案

以下都被广泛认为是不好的做法:

  • 类函数宏。
  • 可变参数宏。
  • 可变函数。
  • 通过宏发明您自己的语言功能。
  • 使用非标准语言功能,例如 ##__VA_ARGS__

这些都是不好的做法有很多原因:不存在的类型安全、很难阅读、难以调试/维护、可移植性等等。

因此,将上述所有内容组合到一个程序中是一个非常糟糕的想法。


C 中的错误处理是通过事实上的行业标准处理的,基本上是这样的:

  • 错误代码通过枚举定义。
  • 任何API函数的返回值都是为错误代码保留的。
  • 出错时该函数返回一个错误代码。
  • 编译器设置为警告级别,如果您忽略函数的结果,您会收到警告。
  • 源代码文档说明了在成功和错误时其他参数会发生什么/不会发生什么。

这是其他 C 程序员所期望并立即理解的。他们不理解或期望一些自制的异常处理系统。

此外,在支持异常处理的语言中也存在很多批评。参见 Why is exception handling bad?

关于C错误处理异常,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43936894/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com