Logger

相关链接

代码文件:

参考示例:

AimRT 中的独立日志组件

AimRT 提供了一个独立的通用日志组件,属于 aimrt::common::util 这个CMake Target,只需要#include "util/log_util.h"即可独立于 CPP 接口层使用。

其中提供了一些基础的日志宏,这些日志宏需要在调用时传入一个Logger对象,来定义日志打印行为的具体表现。日志句柄以模板 Concept 的形式定义,只要是类似于以下这个示例、包含GetLogLevelLog两个接口的 C++ 类的实例都可以作为日志句柄:

class YourLogger {
 public:
  uint32_t GetLogLevel() const {
    // ...
  }

  void Log(uint32_t lvl, uint32_t line, uint32_t column,
           const char* file_name, const char* function_name,
           const char* log_data, size_t log_data_size) const {
    // ...
  }

};

其中,日志等级分为以下 6 档:

  • Trace

  • Debug

  • Info

  • Warn

  • Error

  • Fatal

在有了日志句柄之后,开发者可以直接基于日志句柄提供的Log方法打印日志,也可以使用提供的日志宏来更方便的打印日志。注意,提供的日志宏基于 C++20 Format 语法,关于 C++20 Format 语法的详细使用方式请参考C++官方文档

AimRT 在util/log_util.h文件中,还提供了两种默认的Logger类型:

  • SimpleLogger :一个简单的同步日志句柄;

  • SimpleAsyncLogger : 一个简单的异步日志句柄;

这两种日志句柄一般用于单元测试等未启动 AimRT 实例时的场景。

AimRT 中的独立日志组件使用示例

以下是一些使用示例:

#include "util/log_util.h"

int Main() {
  // Use a simple log handle provided in 'util/log_util.h', which will synchronously print logs on the console
  auto lgr = aimrt::common::util::SimpleLogger();

  uint32_t n = 42;
  std::string s = "Hello world";

  // Normal log macro
  AIMRT_HANDLE_LOG(lgr, aimrt::common::util::kLogLevelInfo, "This is a test log, n = {}, s = {}", n, s);
  AIMRT_HL_TRACE(lgr, "This is a test trace log, n = {}, s = {}", n, s);
  AIMRT_HL_DEBUG(lgr, "This is a test debug log, n = {}, s = {}", n, s);
  AIMRT_HL_INFO(lgr, "This is a test info log, n = {}, s = {}", n, s);
  AIMRT_HL_WARN(lgr, "This is a test warn log, n = {}, s = {}", n, s);
  AIMRT_HL_ERROR(lgr, "This is a test error log, n = {}, s = {}", n, s);
  AIMRT_HL_FATAL(lgr, "This is a test fatal log, n = {}, s = {}", n, s);

  // Check the expression and print the log only when it is false
  AIMRT_HL_CHECK_ERROR(lgr, n == 41, "Expression is not right, n = {}", n);

  // Print logs and throw exceptions
  AIMRT_HL_ERROR_THROW(lgr, "This is a test error log, n = {}, s = {}", n, s);

  // Check the expression, print the log and throw an exception when it is false
  AIMRT_HL_CHECK_TRACE_THROW(lgr, n == 41, "Expression is not right, n = {}", n);

  // ...
}

此外,日志组件中还定义了一个默认的日志句柄获取接口GetLogger(),只要当前上下文中有GetLogger()这个方法,即可使用一些更简洁的日志宏,隐式的将GetLogger()方法返回的结果作为日志句柄,省略掉显式传递日志句柄这一步。示例如下:

#include "util/log_util.h"

auto GetLogger() {
  return aimrt::common::util::SimpleLogger();
}

int Main() {
  uint32_t n = 42;
  std::string s = "Hello world";

  // Normal log macro
  AIMRT_TRACE("This is a test trace log, n = {}, s = {}", n, s);
  AIMRT_DEBUG("This is a test debug log, n = {}, s = {}", n, s);
  AIMRT_INFO("This is a test info log, n = {}, s = {}", n, s);
  AIMRT_WARN("This is a test warn log, n = {}, s = {}", n, s);
  AIMRT_ERROR("This is a test error log, n = {}, s = {}", n, s);
  AIMRT_FATAL("This is a test fatal log, n = {}, s = {}", n, s);

  // Check the expression and print the log only when it is false
  AIMRT_CHECK_ERROR(n == 41, "Expression is not right, n = {}", n);

  // Print logs and throw exceptions
  AIMRT_ERROR_THROW("This is a test error log, n = {}, s = {}", n, s);

  // Check the expression, print the log and throw an exception when it is false
  AIMRT_CHECK_TRACE_THROW(n == 41, "Expression is not right, n = {}", n);

  // ...
}

AimRT 运行时日志句柄

在 AimRT 中,模块可以通过调用CoreRef句柄的GetLogger()接口,获取aimrt::logger::LoggerRef句柄,这是一个包含GetLogLevelLog接口的类,满足上一节中对日志句柄的要求,可以直接作为日志宏的参数。其核心接口如下:

namespace aimrt::logger {

class LoggerRef {
 public:
  // 获取日志等级
  uint32_t GetLogLevel() const;

  // 打印日志
  void Log(uint32_t lvl, uint32_t line, uint32_t column,
      const char* file_name, const char* function_name,
      const char* log_data, size_t log_data_size) const;
};

}  // namespace aimrt::logger

AimRT 运行时日志句柄使用示例

模块开发者可以直接参照以下示例的方式,使用分配给模块的日志句柄来打印日志:

#include "aimrt_module_cpp_interface/module_base.h"

class HelloWorldModule : public aimrt::ModuleBase {
 public:
  bool Initialize(aimrt::CoreRef core) override {
    logger_ = core_.GetLogger();

    uint32_t n = 42;
    std::string s = "Hello world";

    AIMRT_TRACE("This is a test trace log, n = {}, s = {}", n, s);
    AIMRT_DEBUG("This is a test debug log, n = {}, s = {}", n, s);
    AIMRT_INFO("This is a test info log, n = {}, s = {}", n, s);
    AIMRT_WARN("This is a test warn log, n = {}, s = {}", n, s);
    AIMRT_ERROR("This is a test error log, n = {}, s = {}", n, s);
    AIMRT_FATAL("This is a test fatal log, n = {}, s = {}", n, s);
  }

 private:
  auto GetLogger() { return logger_; }

 private:
  aimrt::logger::LoggerRef logger_;
};