- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
问题是关于 GRIB 解析器(链接到 GRIB 文件 https://github.com/Gifciak/GRIB),当我执行我的代码时(通过代码块或在 linux 上通过控制台 - g++ main.cpp -pedantic
)我收到一个错误,段错误,但它并不总是发生。
例如,当我编译 10 次时,有 8 次会出现错误,而 2 次一切正常,这将为我提供控制台输出和信息。
根据我的研究,问题出在 std::copy
上,因为它可能正在尝试复制一个不再存在的迭代器。
有人可以解释为什么会这样吗?为什么它不总是崩溃或成功?
#include <iostream>
#include <vector>
#include <fstream>
#include <iterator>
#include <algorithm>
using ByteVec = std::vector<uint8_t>;
template<typename T, size_t size = sizeof(T)>
auto getReverseEndianValue(const auto & iter) {
union {
T result;
char tmp[size];
} buffer;
auto reverseIter = std::make_reverse_iterator(std::next(iter, size));
std::copy(reverseIter, std::next(reverseIter, size), buffer.tmp);
return buffer.result;
}
enum Edition {
Edition_Unknown = -1,
Edition_GRIB1 = 1,
};
namespace section {
class IS {
public:
uint32_t magicFlag;
uint32_t size;
Edition edition;
static IS read(const auto & iter) {
IS result;
result.magicFlag = getReverseEndianValue<uint32_t>(iter);
result.size = getReverseEndianValue<uint32_t, 3>(iter + 4);
result.edition = (*(iter + 7) == 1 ? Edition_GRIB1 : Edition_Unknown);
return result;
}
};
class PDS {
public:
uint32_t size;
uint8_t tableVersion;
uint8_t indentificatorOfCenter;
uint8_t numProcessID;
uint8_t gridIndentification;
uint8_t flagForGDSorBMS;
uint8_t indParamAndUnit;
uint8_t indTypeOfLevelOrLayer;
uint16_t levelOrLayer;
uint8_t year;
uint8_t month;
uint8_t day;
uint8_t hour;
uint8_t minute;
uint8_t forecastTimeUnit;
uint8_t p1;
uint8_t p2;
uint8_t indTimeRange;
uint16_t averageOrAccumulate;
uint8_t missing;
uint8_t century;
uint8_t subcenterId;
uint16_t decimalScale;
ByteVec data;
static PDS read(const auto& iter) {
PDS result;
result.size = getReverseEndianValue<uint32_t, 3>(iter);
result.tableVersion = getReverseEndianValue<uint8_t>(iter + 3);
result.indentificatorOfCenter = getReverseEndianValue<uint8_t>(iter + 4);
result.numProcessID = getReverseEndianValue<uint8_t>(iter + 5);
result.gridIndentification = getReverseEndianValue<uint8_t>(iter + 6);
result.flagForGDSorBMS = getReverseEndianValue<uint8_t>(iter + 7);
result.indParamAndUnit = getReverseEndianValue<uint8_t>(iter + 8);
result.indTypeOfLevelOrLayer = getReverseEndianValue<uint8_t>(iter + 9);
result.levelOrLayer = getReverseEndianValue<uint16_t>(iter + 10);
result.year = getReverseEndianValue<uint8_t>(iter + 12);
result.month = getReverseEndianValue<uint8_t>(iter + 13);
result.day = getReverseEndianValue<uint8_t>(iter + 14);
result.hour = getReverseEndianValue<uint8_t>(iter + 15);
result.minute = getReverseEndianValue<uint8_t>(iter + 16);
result.forecastTimeUnit = getReverseEndianValue<uint8_t>(iter + 17);
result.p1 = getReverseEndianValue<uint8_t>(iter + 18);
result.p2 = getReverseEndianValue<uint8_t>(iter + 19);
result.indTimeRange = getReverseEndianValue<uint8_t>(iter + 20);
result.averageOrAccumulate = getReverseEndianValue<uint16_t>(iter + 21);
result.missing = getReverseEndianValue<uint8_t>(iter + 23);
result.century = getReverseEndianValue<uint8_t>(iter + 24);
result.subcenterId = getReverseEndianValue<uint8_t>(iter + 25);
result.decimalScale = getReverseEndianValue<uint16_t>(iter + 26);
return result;
}
};
}
class GribData {
private:
section::IS secIS;
section::PDS secPDS;
public:
void print() {
std::cout
<< "### Section IS ###\n"
<< "magicFlag: " << +secIS.magicFlag << "\n"
<< "size: " << +secIS.size << "\n"
<< "edition: " << +secIS.edition << "\n"
<< "\n### Section PDS ###\n"
<< "size: " << +secPDS.size << "\n"
<< "tableVersion: " << +secPDS.tableVersion << "\n"
<< "indentificatorOfCenter: " << +secPDS.indentificatorOfCenter << "\n"
<< "numProcessID: " << +secPDS.numProcessID << "\n"
<< "gridIndentification: " << +secPDS.gridIndentification << "\n"
<< "flagForGDSorBMS: " << +secPDS.flagForGDSorBMS << "\n"
<< "indParamAndUnit: " << +secPDS.indParamAndUnit << "\n"
<< "indTypeOfLevelOrLayer: " << +secPDS.indTypeOfLevelOrLayer << "\n"
<< "levelOrLayer: " << +secPDS.levelOrLayer << "\n"
<< "year: " << +secPDS.year << "\n"
<< "month: " << +secPDS.month << "\n"
<< "day: " << +secPDS.day << "\n"
<< "hour: " << +secPDS.hour << "\n"
<< "minute: " << +secPDS.minute << "\n"
<< "forecastTimeUnit: " << +secPDS.forecastTimeUnit << "\n"
<< "p1: " << +secPDS.p1 << "\n"
<< "p2: " << +secPDS.p2 << "\n"
<< "indTimeRange: " << +secPDS.indTimeRange << "\n"
<< "averageOrAccumulate: " << +secPDS.averageOrAccumulate << "\n"
<< "missing: " << +secPDS.missing << "\n"
<< "century: " << +secPDS.century << "\n"
<< "subcenterId: " << +secPDS.subcenterId << "\n"
<< "decimalScale: " << +secPDS.decimalScale << "\n";
}
static GribData loadData(const ByteVec& rawdata) {
GribData result;
constexpr char MAGIC_START[4] = { 'G', 'R', 'I', 'B' };
constexpr char MAGIC_END[4] = { '7', '7', '7', '7' };
auto start = std::search(rawdata.cbegin(),
rawdata.cend(),
std::begin(MAGIC_START),
std::end(MAGIC_START));
auto end = std::search(rawdata.cbegin(),
rawdata.cend(),
std::begin(MAGIC_END),
std::end(MAGIC_END));
ByteVec data(start, end + sizeof(MAGIC_END));
result.secIS = section::IS::read(data.cbegin());
result.secPDS = section::PDS::read(data.cbegin() + 8);
auto size = getReverseEndianValue<uint32_t, 3>(data.cbegin() + 4);
auto sec1 = getReverseEndianValue<uint32_t, 3>(data.cbegin() + 8);
auto sec2 = getReverseEndianValue<uint32_t, 3>(data.cbegin() + 8 + sec1);
auto sec3 = getReverseEndianValue<uint32_t, 3>(data.cbegin() + 8 + sec1 + sec2);
std::cout
<< "size: " << size << "\n"
<< "sec0: " << 8 << "\n"
<< "sec1: " << sec1 << "\n"
<< "sec2: " << sec2 << "\n"
<< "sec3: " << sec3 << "\n"
<< "end flag: " << sizeof(MAGIC_END) << "\n"
<< "sum: " << 8 + sec1 + sec2 + sec3 + sizeof(MAGIC_END) << "\n\n";
return result;
}
static GribData loadDataFromFile(const std::string& path) {
std::ifstream file(path, std::ios::binary);
ByteVec data;
std::copy(std::istreambuf_iterator<char>(file),
{},
std::back_inserter(data));
return loadData(data);
}
};
int main() {
auto grib = GribData::loadDataFromFile("message_2_G1.grib");
grib.print();
}
这是预期的结果,因为我从控制台复制了它
size: 4538sec0: 8sec1: 28sec2: 178sec3: 4320end flag: 4sum: 4538### Section IS ###magicFlag: 1196575042size: 1191186874edition: 1### Section PDS ###size: 28tableVersion: 2indentificatorOfCenter: 7numProcessID: 81gridIndentification: 37flagForGDSorBMS: 128indParamAndUnit: 33indTypeOfLevelOrLayer: 100levelOrLayer: 850year: 15month: 3day: 10hour: 0minute: 0forecastTimeUnit: 1p1: 0p2: 0indTimeRange: 10averageOrAccumulate: 0missing: 0century: 21subcenterId: 0decimalScale: 1
最佳答案
首先,使用 g++ main.cpp -pedantic
不是很有用,因为您没有启用任何警告。将 -Wall -Wextra
添加到您的编译器标志,以及 -g
以便您可以对其进行调试。
使用 -fsanitize=undefined
编译显示了一个运行时错误,该错误是由于在需要有效指针的地方使用空指针导致的:
/usr/include/c++/8/bits/stl_algobase.h:368:23: runtime error: null pointer passed as argument 2, which is declared to never be null
Segmentation fault (core dumped)
这意味着您的程序有错误。
使用 -D_GLIBCXX_DEBUG
编译会向 std::vector
添加额外的检查,这会告诉您问题所在:
/usr/include/c++/8/debug/safe_iterator.h:374:
Error: attempt to advance a past-the-end iterator 4 steps, which falls
outside its valid range.
Objects involved in the operation:
iterator @ 0x0x7fffb09ceb90 {
type = __gnu_debug::_Safe_iterator<__gnu_cxx::__normal_iterator<unsigned char const*, std::__cxx1998::vector<unsigned char, std::allocator<unsigned char> > >, std::__debug::vector<unsigned char, std::allocator<unsigned char> > > (constant iterator);
state = past-the-end;
references sequence with type 'std::__debug::vector<unsigned char, std::allocator<unsigned char> >' @ 0x0x7fffb09cf050
}
Aborted (core dumped)
您应该在调试器下运行该程序以查看此无效迭代器增量发生的位置。在 GDB 中运行该程序,然后使用其 up
命令向上移动堆栈显示错误来自此处,在 loadData
中:
constexpr char MAGIC_START[4] = { 'G', 'R', 'I', 'B' };
constexpr char MAGIC_END[4] = { '7', '7', '7', '7' };
auto start = std::search(rawdata.cbegin(),
rawdata.cend(),
std::begin(MAGIC_START),
std::end(MAGIC_START));
auto end = std::search(rawdata.cbegin(),
rawdata.cend(),
std::begin(MAGIC_END),
std::end(MAGIC_END));
ByteVec data(start, end + sizeof(MAGIC_END));
^^^^^^^^^^^^^^^^^^^^^^^
考虑当 rawdata
不包含 MAGIC_START
字符但包含 MAGIC_END
字符时会发生什么。 start
和 end
会形成有效的迭代器范围吗?
考虑当 rawdata
不包含 MAGIC_END
字符时会发生什么。 end + sizeof(MAGIC_END)
是否有效?
您不应假定对 std::search
的两次调用按预期工作。您应该通过测试 start == rawdata.end()
或 end == rawdata.end()
来添加一些错误检查。如果其中任何一个为真,则说明出了问题(可能是 rawdata
字符串中的错误输入)。
您还应该学习如何使用调试器,并了解编译器提供的用于检测错误的其他工具(例如 GCC 的 -fsanitize=undefined
和 -D_GLIBCXX_DEBUG
选项应该用来帮助确认错误的存在,并且应该使用 GDB 来查找这些错误发生的位置。
关于c++ - "Occasional"段错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56112751/
问题是关于 GRIB 解析器(链接到 GRIB 文件 https://github.com/Gifciak/GRIB),当我执行我的代码时(通过代码块或在 linux 上通过控制台 - g++ mai
不久前,智能客户端被吹捧为解决“偶尔连接”使用环境的解决方案,而诸如 Google Gears 之类的工具包也出于同样的原因而应运而生。在我看来,持续、可靠的互联网访问正变得越来越普遍(即使在商用飞机
我有以下代码使用 managedWifi api (http://managedwifi.codeplex.com/) 监听 wifi 连接/断开事件 public void wlanConnecti
我正在从文件中提取图标并在对话框中显示它们 const LPCWSTR path = L"c:\path\to\file"; const UINT nIconsCheck = ExtractIconE
我有一个 Node 应用程序,其服务器托管在heroku上。我的所有请求都会成功,直到我发送了大约 10 或 15 个请求。然后我开始收到 CORS 错误。知道为什么会发生这种情况吗? 尝试一下。 h
我正在开发一个 Chrome 扩展,下面的代码只是有时会导致错误,并非总是如此。大多数时候,这个扩展确实能正常工作,但有时它无法获取 getBackgroundPage() 并落入“else”部分。
有人知道为什么会这样吗?我的项目使用本地网络,所以只在一台设备上测试它没有意义,所以我想在 20 台设备上安装它。我这样做的方法是构建一次,然后使用组织者。但是,大约每 3 或 4 次传输,图标就无法
我尝试将 [String: Any] 数组保存为用户默认值,在某些情况下它可以工作,但在其他情况下则不行。我使用以下内容保存为默认值: static func savingQueueToDisk()
这只是偶尔发生,似乎是随机的,我无法重新创建此错误以对其进行追踪。 我知道这不是很多信息,可能有成千上万的问题可能会导致这种奇怪的行为。但也许你可以指出那些最明显的问题,或者给我提示如何解决我的问题,
我正在编写一个应用程序,每 40 毫秒(25Hz)记录一次手机的加速度。这个帧速率可以平均保持,但有时我会遇到 5'000ms - 50'000ms 时间帧之间的延迟。我想知道为什么会这样。 这里有一
我在R脚本中使用sendmailR发送通知。 有时通知失败并显示以下错误: Unknown SMTP code: 452 Error in if (code == lcode) { : argumen
我编写了一个与服务器通信的 TCP 客户端。在专用的“监听”线程中,我有如下代码。它应该只在有数据时才读取数据。 (if (stream.DataAvailable)) 奇怪的是,有时我的程序会崩溃,
我有一个浮点图,它通过 ajax 从 XML 文档中轮询数据。我使用模式:x 轴时间。它工作正常,除了间歇性地,xml Document 中的一些值被删除。这是通过 ajax 获取 xml 的代码。
When I upgrade webpack4.x to 5.88.1, I use the cache function of webpack to improve the build spe
当我尝试为预取的 Queryset 运行 objects.all() 时,我在 Django 中遇到了一个令人沮丧的间歇性错误。有一个问题,在查询集的实例化和通过它运行迭代之间,model._meta
我有一个单线程 FastCGI“Hello, World!”应用程序,有一些代码: error = FCGX_Init(); // error handling socket_descriptor =
我今天遇到了一个新问题。我发现了一些我正在使用的数据,看起来像这样(在 csv 文件中): Male,White,All Combined,1989,30-31,31,"59,546","18,141
我正在尝试使用 SURF 和 kNN 对对象进行分类。该代码运行良好,但偶尔会崩溃并显示“段错误”。我不确定我是否做错了什么,但我很确定它已得到纠正。如果您想重现问题,这里是输入文件。 Link to
我的一些用户遇到了这个崩溃(据他们说,它发生在使用应用程序 4-5 分钟后)但我自己无法重现: Application Specific Information: has active assert
通常当尝试通过 [[NSBundle mainBundle] pathForResource:@"iphone"ofType:@"splang"] 在我的应用程序中加载特定资源时,它工作正常,但每隔一
我是一名优秀的程序员,十分优秀!