简介

Easylogging++ 是用于 C++ 应用程序的单头高效日志库。它非常强大,高度可扩展并且可以根据用户的要求进行配置。github链接:https://github.com/amrayn/easyloggingpp

Easylogging++ 在v9.89版只有一个头文件,之后改为一个头文件、一个源文件,目前最新版本是v9.97(本文使用的版本)。

使用

使用 Easylogging++只需要三个简单的步骤:

  • 下载最新版本
  • easylogging++.heasylogging++.cc包含到项目中
  • 使用单个宏进行初始化
  1. #include "easylogging++.h"
  2. INITIALIZE_EASYLOGGINGPP
  3. int main(int argc, char* argv[]) {
  4. LOG(INFO) << "My first info log using default logger";
  5. return 0;
  6. }

扩展

Easylogging++默认日志写在一个文件里面,而且没有按日期新建日志的功能,需要自己扩展一下。扩展功能如下:

  • 日志文件放在按年、月生成的文件夹内,每个日志级别单独一个日志文件,如“Log\2021\202108\20210818_INFO.log”
  • 每天生成新的日志文件,即日志文件按日期滚动
  • 根据日志文件的最后修改时间自动删除n天前的日志文件,仅支持Windows系统

我会尽量使用标准库和Easylogging++里面已有的功能来实现扩展功能,减少外部依赖项,也便于后面进行命名空间的合并。

配置日志路径

Easylogging++支持配置文件、程序代码两种方式配置日志路径,这里采用程序代码的方式配置日志路径,代码如下:

  1. static std::string LogRootPath = "D:\\Log";
  2. static el::base::SubsecondPrecision LogSsPrec(3);
  3. static std::string LoggerToday = el::base::utils::DateTime::getDateTime("%Y%M%d", &LogSsPrec);
  4. static void ConfigureLogger()
  5. {
  6. std::string datetimeY = el::base::utils::DateTime::getDateTime("%Y", &LogSsPrec);
  7. std::string datetimeYM = el::base::utils::DateTime::getDateTime("%Y%M", &LogSsPrec);
  8. std::string datetimeYMd = el::base::utils::DateTime::getDateTime("%Y%M%d", &LogSsPrec);
  9. std::string filePath = LogRootPath + "\\" + datetimeY + "\\" + datetimeYM + "\\";
  10. std::string filename;
  11. el::Configurations defaultConf;
  12. defaultConf.setToDefault();
  13. //建议使用setGlobally
  14. defaultConf.setGlobally(el::ConfigurationType::Format, "%datetime %msg");
  15. defaultConf.setGlobally(el::ConfigurationType::Enabled, "true");
  16. defaultConf.setGlobally(el::ConfigurationType::ToFile, "true");
  17. defaultConf.setGlobally(el::ConfigurationType::ToStandardOutput, "true");
  18. defaultConf.setGlobally(el::ConfigurationType::SubsecondPrecision, "6");
  19. defaultConf.setGlobally(el::ConfigurationType::PerformanceTracking, "true");
  20. defaultConf.setGlobally(el::ConfigurationType::LogFlushThreshold, "1");
  21. //限制文件大小时配置
  22. //defaultConf.setGlobally(el::ConfigurationType::MaxLogFileSize, "2097152");
  23. filename = datetimeYMd + "_" + el::LevelHelper::convertToString(el::Level::Global)+".log";
  24. defaultConf.set(el::Level::Global, el::ConfigurationType::Filename, filePath + filename);
  25. filename = datetimeYMd + "_" + el::LevelHelper::convertToString(el::Level::Debug) + ".log";
  26. defaultConf.set(el::Level::Debug, el::ConfigurationType::Filename, filePath + filename);
  27. filename = datetimeYMd + "_" + el::LevelHelper::convertToString(el::Level::Error) + ".log";
  28. defaultConf.set(el::Level::Error, el::ConfigurationType::Filename, filePath + filename);
  29. filename = datetimeYMd + "_" + el::LevelHelper::convertToString(el::Level::Fatal) + ".log";
  30. defaultConf.set(el::Level::Fatal, el::ConfigurationType::Filename, filePath + filename);
  31. filename = datetimeYMd + "_" + el::LevelHelper::convertToString(el::Level::Info) + ".log";
  32. defaultConf.set(el::Level::Info, el::ConfigurationType::Filename, filePath + filename);
  33. filename = datetimeYMd + "_" + el::LevelHelper::convertToString(el::Level::Trace) + ".log";
  34. defaultConf.set(el::Level::Trace, el::ConfigurationType::Filename, filePath + filename);
  35. filename = datetimeYMd + "_" + el::LevelHelper::convertToString(el::Level::Warning) + ".log";
  36. defaultConf.set(el::Level::Warning, el::ConfigurationType::Filename, filePath + filename);
  37. el::Loggers::reconfigureLogger("default", defaultConf);
  38. //限制文件大小时启用
  39. //el::Loggers::addFlag(el::LoggingFlag::StrictLogFileSizeCheck);
  40. }

如果想软件每个功能模块生成自己的日志,可以参考上面的代码自己实现,实现时注意以下两点:

  • 使用“%Y%M”配置文件路径时,Easylogging++只会识别第一个格式符,如“\%datetime{%Y%M}\%datetime{%Y%M}”生成的路径是“\202108\%datetime{%Y%M}”。
  • Easylogging++目前不支持文件名中加入日志级别,需要自己实现,如“\%datetime{%Y%M}%level.log”生成的路径是“\202108%level.log”。

这些问题可以按我上面的方法避开,或者修改源代码进行修复,源代码的修改部分会放在文章最后。

时间滚动日志

Easylogging++没有按时间滚动日志的功能,该功能需要检查当前的时间并决定是否生成新日志文件(文件名必须包含时间信息),关键问题只有两个:

  • 检查时间的时机:选择在每条日志写之前检查一次,因此需要监控每条日志的写入。
  • 生成新日志文件:直接调用上面的“ConfigureLogger()”方法覆盖日志的配置即可。

注:如果使用定时器来检查当前时间,修改系统时间时日志文件无法及时更新。

监控每条日志的写入需要实现一个继承LogDispatchCallback的类,代码如下:

  1. class LogDispatcher : public el::LogDispatchCallback
  2. {
  3. protected:
  4. void handle(const el::LogDispatchData* data) noexcept override {
  5. m_data = data;
  6. // 使用记录器的默认日志生成器进行调度
  7. dispatch(m_data->logMessage()->logger()->logBuilder()->build(m_data->logMessage(),
  8. m_data->dispatchAction() == el::base::DispatchAction::NormalLog));
  9. //此处也可以写入数据库
  10. }
  11. private:
  12. const el::LogDispatchData* m_data;
  13. void dispatch(el::base::type::string_t&& logLine) noexcept
  14. {
  15. el::base::SubsecondPrecision ssPrec(3);
  16. static std::string now = el::base::utils::DateTime::getDateTime("%Y%M%d", &ssPrec);
  17. if (now != LoggerToday)
  18. {
  19. LoggerToday= now;
  20. ConfigureLogger();
  21. }
  22. }
  23. };

LogDispatcher的使用方法如下:

  1. el::Helpers::installLogDispatchCallback<LogDispatcher>("LogDispatcher");
  2. LogDispatcher* dispatcher = el::Helpers::logDispatchCallback<LogDispatcher>("LogDispatcher");
  3. dispatcher->setEnabled(true);

自动删除日志

自动删除日志文件夹下最后修改时间在n天前的日志,代码如下:

  1. //删除文件路径下n天前的日志文件,由于删除日志文件导致的空文件夹会在下一次删除
  2. //isRoot为true时,只会清理空的子文件夹
  3. void DeleteOldFiles(std::string path, int oldDays, bool isRoot)
  4. {
  5. // 基于当前系统的当前日期/时间
  6. time_t nowTime = time(0);
  7. //文件句柄
  8. intptr_t hFile = 0;
  9. //文件信息
  10. struct _finddata_t fileinfo;
  11. //文件扩展名
  12. std::string extName = ".log";
  13. std::string str;
  14. //是否是空文件夹
  15. bool isEmptyFolder = true;
  16. if ((hFile = _findfirst(str.assign(path).append("\\*").c_str(), &fileinfo)) != -1)
  17. {
  18. do
  19. {
  20. //如果是目录,迭代之
  21. //如果不是,检查文件
  22. if ((fileinfo.attrib & _A_SUBDIR))
  23. {
  24. if (strcmp(fileinfo.name, ".") != 0 && strcmp(fileinfo.name, "..") != 0)
  25. {
  26. isEmptyFolder = false;
  27. DeleteOldFiles(str.assign(path).append("\\").append(fileinfo.name), oldDays, false);
  28. }
  29. }
  30. else
  31. {
  32. isEmptyFolder = false;
  33. str.assign(fileinfo.name);
  34. if ((str.size() >= extName.size()) && (str.substr(str.size() - extName.size()) == extName))
  35. {
  36. //是日志文件
  37. if ((nowTime - fileinfo.time_write) / (24 * 3600) > oldDays)
  38. {
  39. str.assign(path).append("\\").append(fileinfo.name);
  40. system(("attrib -H -R " + str).c_str());
  41. system(("del/q " + str).c_str());
  42. }
  43. }
  44. }
  45. } while (_findnext(hFile, &fileinfo) == 0);
  46. _findclose(hFile);
  47. if (isEmptyFolder && (!isRoot))
  48. {
  49. system(("attrib -H -R " + path).c_str());
  50. system(("rd/q " + path).c_str());
  51. }
  52. }
  53. }

里面的删除操作是通过调用批处理命令实现,网上有一个自动删除过期文件的完整批处理命令,不过我从来没成功过。

可以在每天新建日志文件时调用删除方法,删除文件可能会耗费一些时间,最好重新开一个线程,代码如下:

  1. static int LogCleanDays = 30;
  2. std::thread task(el::DeleteOldFiles, LogRootPath, LogCleanDays, true);

封装到一个头文件

上面的代码比较分散,实际使用时可以全部放到“easylogginghelper.h”头文件中,然后在项目中引用。头文件提供一个初始化函数“InitEasylogging()”来初始化所有配置,头文件代码如下:

  1. #pragma once
  2. #ifndef EASYLOGGINGHELPER_H
  3. #define EASYLOGGINGHELPER_H
  4. #include "easylogging++.h"
  5. #include <io.h>
  6. #include <thread>
  7. INITIALIZE_EASYLOGGINGPP
  8. namespace el
  9. {
  10. static int LogCleanDays = 30;
  11. static std::string LogRootPath = "D:\\Log";
  12. static el::base::SubsecondPrecision LogSsPrec(3);
  13. static std::string LoggerToday = el::base::utils::DateTime::getDateTime("%Y%M%d", &LogSsPrec);
  14. //删除文件路径下n天前的日志文件,由于删除日志文件导致的空文件夹会在下一次删除
  15. //isRoot为true时,只会清理空的子文件夹
  16. void DeleteOldFiles(std::string path, int oldDays, bool isRoot)
  17. {
  18. // 基于当前系统的当前日期/时间
  19. time_t nowTime = time(0);
  20. //文件句柄
  21. intptr_t hFile = 0;
  22. //文件信息
  23. struct _finddata_t fileinfo;
  24. //文件扩展名
  25. std::string extName = ".log";
  26. std::string str;
  27. //是否是空文件夹
  28. bool isEmptyFolder = true;
  29. if ((hFile = _findfirst(str.assign(path).append("\\*").c_str(), &fileinfo)) != -1)
  30. {
  31. do
  32. {
  33. //如果是目录,迭代之
  34. //如果不是,检查文件
  35. if ((fileinfo.attrib & _A_SUBDIR))
  36. {
  37. if (strcmp(fileinfo.name, ".") != 0 && strcmp(fileinfo.name, "..") != 0)
  38. {
  39. isEmptyFolder = false;
  40. DeleteOldFiles(str.assign(path).append("\\").append(fileinfo.name), oldDays, false);
  41. }
  42. }
  43. else
  44. {
  45. isEmptyFolder = false;
  46. str.assign(fileinfo.name);
  47. if ((str.size() > extName.size()) && (str.substr(str.size() - extName.size()) == extName))
  48. {
  49. //是日志文件
  50. if ((nowTime - fileinfo.time_write) / (24 * 3600) > oldDays)
  51. {
  52. str.assign(path).append("\\").append(fileinfo.name);
  53. system(("attrib -H -R " + str).c_str());
  54. system(("del/q " + str).c_str());
  55. }
  56. }
  57. }
  58. } while (_findnext(hFile, &fileinfo) == 0);
  59. _findclose(hFile);
  60. if (isEmptyFolder && (!isRoot))
  61. {
  62. system(("attrib -H -R " + path).c_str());
  63. system(("rd/q " + path).c_str());
  64. }
  65. }
  66. }
  67. static void ConfigureLogger()
  68. {
  69. std::string datetimeY = el::base::utils::DateTime::getDateTime("%Y", &LogSsPrec);
  70. std::string datetimeYM = el::base::utils::DateTime::getDateTime("%Y%M", &LogSsPrec);
  71. std::string datetimeYMd = el::base::utils::DateTime::getDateTime("%Y%M%d", &LogSsPrec);
  72. std::string filePath = LogRootPath + "\\" + datetimeY + "\\" + datetimeYM + "\\";
  73. std::string filename;
  74. el::Configurations defaultConf;
  75. defaultConf.setToDefault();
  76. //建议使用setGlobally
  77. defaultConf.setGlobally(el::ConfigurationType::Format, "%datetime %msg");
  78. defaultConf.setGlobally(el::ConfigurationType::Enabled, "true");
  79. defaultConf.setGlobally(el::ConfigurationType::ToFile, "true");
  80. defaultConf.setGlobally(el::ConfigurationType::ToStandardOutput, "true");
  81. defaultConf.setGlobally(el::ConfigurationType::SubsecondPrecision, "6");
  82. defaultConf.setGlobally(el::ConfigurationType::PerformanceTracking, "true");
  83. defaultConf.setGlobally(el::ConfigurationType::LogFlushThreshold, "1");
  84. //限制文件大小时配置
  85. //defaultConf.setGlobally(el::ConfigurationType::MaxLogFileSize, "2097152");
  86. filename = datetimeYMd + "_" + el::LevelHelper::convertToString(el::Level::Global)+".log";
  87. defaultConf.set(el::Level::Global, el::ConfigurationType::Filename, filePath + filename);
  88. filename = datetimeYMd + "_" + el::LevelHelper::convertToString(el::Level::Debug) + ".log";
  89. defaultConf.set(el::Level::Debug, el::ConfigurationType::Filename, filePath + filename);
  90. filename = datetimeYMd + "_" + el::LevelHelper::convertToString(el::Level::Error) + ".log";
  91. defaultConf.set(el::Level::Error, el::ConfigurationType::Filename, filePath + filename);
  92. filename = datetimeYMd + "_" + el::LevelHelper::convertToString(el::Level::Fatal) + ".log";
  93. defaultConf.set(el::Level::Fatal, el::ConfigurationType::Filename, filePath + filename);
  94. filename = datetimeYMd + "_" + el::LevelHelper::convertToString(el::Level::Info) + ".log";
  95. defaultConf.set(el::Level::Info, el::ConfigurationType::Filename, filePath + filename);
  96. filename = datetimeYMd + "_" + el::LevelHelper::convertToString(el::Level::Trace) + ".log";
  97. defaultConf.set(el::Level::Trace, el::ConfigurationType::Filename, filePath + filename);
  98. filename = datetimeYMd + "_" + el::LevelHelper::convertToString(el::Level::Warning) + ".log";
  99. defaultConf.set(el::Level::Warning, el::ConfigurationType::Filename, filePath + filename);
  100. el::Loggers::reconfigureLogger("default", defaultConf);
  101. //限制文件大小时启用
  102. //el::Loggers::addFlag(el::LoggingFlag::StrictLogFileSizeCheck);
  103. }
  104. class LogDispatcher : public el::LogDispatchCallback
  105. {
  106. protected:
  107. void handle(const el::LogDispatchData* data) noexcept override {
  108. m_data = data;
  109. // 使用记录器的默认日志生成器进行调度
  110. dispatch(m_data->logMessage()->logger()->logBuilder()->build(m_data->logMessage(),
  111. m_data->dispatchAction() == el::base::DispatchAction::NormalLog));
  112. //此处也可以写入数据库
  113. }
  114. private:
  115. const el::LogDispatchData* m_data;
  116. void dispatch(el::base::type::string_t&& logLine) noexcept
  117. {
  118. el::base::SubsecondPrecision ssPrec(3);
  119. static std::string now = el::base::utils::DateTime::getDateTime("%Y%M%d", &ssPrec);
  120. if (now != LoggerToday)
  121. {
  122. LoggerToday = now;
  123. ConfigureLogger();
  124. std::thread task(el::DeleteOldFiles, LogRootPath, LogCleanDays, true);
  125. }
  126. }
  127. };
  128. static void InitEasylogging()
  129. {
  130. ConfigureLogger();
  131. el::Helpers::installLogDispatchCallback<LogDispatcher>("LogDispatcher");
  132. LogDispatcher* dispatcher = el::Helpers::logDispatchCallback<LogDispatcher>("LogDispatcher");
  133. dispatcher->setEnabled(true);
  134. }
  135. }
  136. #endif

使用时只需要调用一次“el::InitEasylogging();”即可,代码如下:

  1. #include "easylogging++.h"
  2. #include "easylogginghelper.h"
  3. int main()
  4. {
  5. el::InitEasylogging();
  6. for (size_t i = 0; i < 10000; i++)
  7. {
  8. LOG(TRACE) << "***** trace log *****" << i;
  9. LOG(DEBUG) << "***** debug log *****" << i;
  10. LOG(ERROR) << "***** error log *****" << i;
  11. LOG(WARNING) << "***** warning log *****" << i;
  12. LOG(INFO) << "***** info log *****" << i;
  13. //不要轻易使用,程序会退出
  14. //LOG(FATAL) << "***** fatal log *****" << i;
  15. Sleep(100);
  16. }
  17. }

源代码优化(不推荐)

上面说到Easylogging++只会识别第一个时间格式符且不识别等级格式符,只需要修改TypedConfigurations::resolveFilename函数的实现即可,代码如下:

  1. std::string TypedConfigurations::resolveFilename(Level level,const std::string& filename)
  2. {
  3. std::string resultingFilename = filename;
  4. std::size_t dateIndex = std::string::npos;
  5. std::string dateTimeFormatSpecifierStr = std::string(base::consts::kDateTimeFormatSpecifierForFilename);
  6. //if改为while
  7. while ((dateIndex = resultingFilename.find(dateTimeFormatSpecifierStr.c_str())) != std::string::npos) {
  8. while (dateIndex > 0 && resultingFilename[dateIndex - 1] == base::consts::kFormatSpecifierChar) {
  9. dateIndex = resultingFilename.find(dateTimeFormatSpecifierStr.c_str(), dateIndex + 1);
  10. }
  11. if (dateIndex != std::string::npos) {
  12. const char* ptr = resultingFilename.c_str() + dateIndex;
  13. // Goto end of specifier
  14. ptr += dateTimeFormatSpecifierStr.size();
  15. std::string fmt;
  16. if ((resultingFilename.size() > dateIndex) && (ptr[0] == '{')) {
  17. // User has provided format for date/time
  18. ++ptr;
  19. int count = 1; // Start by 1 in order to remove starting brace
  20. std::stringstream ss;
  21. for (; *ptr; ++ptr, ++count) {
  22. if (*ptr == '}') {
  23. ++count; // In order to remove ending brace
  24. break;
  25. }
  26. ss << *ptr;
  27. }
  28. //注释掉此语句
  29. //resultingFilename.erase(dateIndex + dateTimeFormatSpecifierStr.size(), count);
  30. fmt = ss.str();
  31. } else {
  32. fmt = std::string(base::consts::kDefaultDateTimeFormatInFilename);
  33. }
  34. base::SubsecondPrecision ssPrec(3);
  35. std::string now = base::utils::DateTime::getDateTime(fmt.c_str(), &ssPrec);
  36. base::utils::Str::replaceAll(now, '/', '-'); // Replace path element since we are dealing with filename
  37. base::utils::Str::replaceAll(resultingFilename, dateTimeFormatSpecifierStr + "{"+ fmt+"}", now);
  38. }
  39. }
  40. //替换等级
  41. base::utils::Str::replaceAll(resultingFilename, base::consts::kSeverityLevelFormatSpecifier, LevelHelper::convertToString(level));
  42. base::utils::Str::replaceAll(resultingFilename, base::consts::kSeverityLevelShortFormatSpecifier, LevelHelper::convertToShortString(level));
  43. return resultingFilename;
  44. }

修改TypedConfigurations::resolveFilename函数的实现时,记得修改头文件里面的定义和所有该函数的调用。不推荐直接修改源代码,修改源代码不利于后期的版本更新。

附件

Easylogging++的使用及扩展的更多相关文章

  1. Asp.net Boilerplate之AbpSession扩展

    当前Abp版本1.2,项目类型为MVC5. 以属性的形式扩展AbpSession,并在"记住我"后,下次自动登录也能获取到扩展属性的值,版权归"角落的白板报"所 ...

  2. 恢复SQL Server被误删除的数据(再扩展)

    恢复SQL Server被误删除的数据(再扩展) 大家对本人之前的文章<恢复SQL Server被误删除的数据> 反应非常热烈,但是文章里的存储过程不能实现对备份出来的日志备份里所删数据的 ...

  3. .NET Core中间件的注册和管道的构建(3) ---- 使用Map/MapWhen扩展方法

    .NET Core中间件的注册和管道的构建(3) ---- 使用Map/MapWhen扩展方法 0x00 为什么需要Map(MapWhen)扩展 如果业务逻辑比较简单的话,一条主管道就够了,确实用不到 ...

  4. .NET Core中间件的注册和管道的构建(2)---- 用UseMiddleware扩展方法注册中间件类

    .NET Core中间件的注册和管道的构建(2)---- 用UseMiddleware扩展方法注册中间件类 0x00 为什么要引入扩展方法 有的中间件功能比较简单,有的则比较复杂,并且依赖其它组件.除 ...

  5. 采用EntityFramework.Extended 对EF进行扩展(Entity Framework 延伸系列2)

    前言 Entity Framework 延伸系列目录 今天我们来讲讲EntityFramework.Extended 首先科普一下这个EntityFramework.Extended是什么,如下: 这 ...

  6. Dapper扩展之~~~Dapper.Contrib

    平台之大势何人能挡? 带着你的Net飞奔吧!http://www.cnblogs.com/dunitian/p/4822808.html#skill 上一篇文章:Dapper逆天入门~强类型,动态类型 ...

  7. ExtJS 4.2 Date组件扩展:添加清除按钮

    ExtJS中除了提供丰富的组件外,我们还可以扩展他的组件. 在这里,我们将在Date日期组件上添加一个[清除]按钮,用于此组件已选中值的清除. 目录 1. Date组件介绍 2. 主要代码说明 3. ...

  8. .NET Core的文件系统[5]:扩展文件系统构建一个简易版“云盘”

    FileProvider构建了一个抽象文件系统,作为它的两个具体实现,PhysicalFileProvider和EmbeddedFileProvider则分别为我们构建了一个物理文件系统和程序集内嵌文 ...

  9. Hawk 6. 编译和扩展开发

    Hawk是开源项目,因此任何人都可以为其贡献代码.作者也非常欢迎使用者能够扩展出更有用的插件. 编译 编译需要Visual Stuido,版本建议使用2015, 2010及以上没有经过测试,但应该可以 ...

随机推荐

  1. Java实验项目三——编程实现Person类,学生类的设计及其继承关系

    Program: 编程实现Person类,学生类的设计及其继承关系 代码如下: 定义抽象类Person 1 /* 2 * Description:建立抽象类 3 * 4 * Written By:Ca ...

  2. 使用微服务Blog.Core开源框架的一些坑

    1.使用SqlSuger组件时同一API无法自动切库 1.1 在生成Model时在类上加上特性 1.2 一个接口如果使用了多个数据库实例,会出现库找不到,需要使用ChangeDataBase切库 2. ...

  3. 干掉 Postman?测试接口直接生成API文档,这个工具贼好用

    大家好,我是小富~ 前几天粉丝群有小伙伴问,有啥好用的API文档工具推荐,无意间发现了一款工具,这里马不停蹄的来给大家分享一下. ShowDoc一个非常适合团队的在线API文档工具,也支持用docke ...

  4. C语言:case详解

    C语言虽然没有限制 if else 能够处理的分支数量,但当分支过多时,用 if else 处理会不太方便,而且容易出现 if else 配对出错的情况.例如,输入一个整数,输出该整数对应的星期几的英 ...

  5. ADB 关闭指定应用 并打开

    import subprocess,time sjh="192.168.1.102:5555" aa1="adb -s {0} shell pm clear com.ku ...

  6. python爬取北京政府信件信息01

    python爬取,找到目标地址,开始研究网页代码格式,于是就开始根据之前学的知识进行爬取,出师不利啊,一开始爬取就出现了个问题,这是之前是没有遇到过的,明明地址没问题,就是显示网页不存在,于是就在百度 ...

  7. kubespray-2.14.2安装kubernetes-1.18.10(ubuntu-20.04.1)

    欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...

  8. C# BS方向 该如何规划学习?【学习路线指南】

    C#学习路线指南 花费了几天时间整理了C#学习路线图,可以说是纯野生C#程序员成长路线规划,在这里分享下,希望可以帮助到想从事C#开发的同学们.本人阅历尚浅,有些知识点可能分享不正确,希望广大网友评论 ...

  9. mysql - 按条件统计

    在表中分别统计mt =0 和 mt>0 的个数 方法一:select count(if(mt=0,1,null)) as a,count(if(mt>0,1,null)) as b fro ...

  10. Charles抓包工具永久破解+https抓包需要安装安全证书+防止请求乱码

    1.charles4.5.6版本安装+永久破解 链接:https://pan.baidu.com/s/1Z49AE6TG2IXUY-7qoyGU4g 提取码:3i97 安装好charles之后,把下载 ...