- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个相当大的程序,但有一个小的内存问题。代码按预期运行,我得到了我想要的结果,但我想摆脱损坏。我运行了 valgrind,但我并不真正理解错误。
内容如下:
==11295==
==11295== HEAP SUMMARY:
==11295== in use at exit: 72,704 bytes in 2 blocks
==11295== total heap usage: 19,836 allocs, 19,834 frees, 1,247,711 bytes allocated
==11295==
==11295== 0 bytes in 1 blocks are definitely lost in loss record 1 of 2
==11295== at 0x4C2E80F: operator new[](unsigned long) (in /usr/lib/valgrind/vgpreload_memcheck-amd64-linux.so)
==11295== by 0x40197C: grid::setSize(int, int) (grid.cpp:75)
==11295== by 0x40B9AC: node::node() (node.cpp:17)
==11295== by 0x4032AF: main (main.cpp:15)
==11295==
==11295== 72,704 bytes in 1 blocks are still reachable in loss record 2 of 2
==11295== at 0x4C2DB8F: malloc (in /usr/lib/valgrind/vgpreload_memcheck-amd64-linux.so)
==11295== by 0x4EC3EFF: ??? (in /usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.21)
==11295== by 0x40104E9: call_init.part.0 (dl-init.c:72)
==11295== by 0x40105FA: call_init (dl-init.c:30)
==11295== by 0x40105FA: _dl_init (dl-init.c:120)
==11295== by 0x4000CF9: ??? (in /lib/x86_64-linux-gnu/ld-2.23.so)
==11295==
==11295== LEAK SUMMARY:
==11295== definitely lost: 0 bytes in 1 blocks
==11295== indirectly lost: 0 bytes in 0 blocks
==11295== possibly lost: 0 bytes in 0 blocks
==11295== still reachable: 72,704 bytes in 1 blocks
==11295== suppressed: 0 bytes in 0 blocks
==11295==
==11295== For counts of detected and suppressed errors, rerun with: -v
==11295== ERROR SUMMARY: 1 errors from 1 contexts (suppressed: 1 from 1)
我知道我创建的唯一指针是在 grid.setSize() 函数中,但这些都应该由析构函数处理。此外,第一个 valgrind block 表示丢失了 0 个字节。
我感到困惑的是第二个街区。我没有在任何地方使用过 malloc,我真的不知道这个 block 需要我做什么。
感谢任何帮助!
根据要求,这里是一些代码。
网格.cpp
#include <iostream>
#include <cstdlib>
#include <cstddef>
#include "grid.h"
using namespace std;
// initializes the double pointer
grid::grid()
{
m_grid = NULL;
}
// destroys the 2D game array
grid::~grid()
{
for (int i = 0; i < m_width; i++)
{
delete[] m_grid[i];
}
delete[] m_grid;
}
cell** grid::getGrid()
{
return m_grid;
}
// returns width
int grid::getWidth()
{
return m_width;
}
// returns height
int grid::getHeight()
{
return m_height;
}
// returns radiation level of specified cell
int grid::getCellRad(const int x, const int y) const
{
return m_grid[x][y].m_rad;
}
int grid::getXRad() const
{
return m_x_rad;
}
int grid::getYRad() const
{
return m_y_rad;
}
// clears grid pointer
void grid::setGridNull()
{
m_grid = NULL;
}
// creates a grid of size width * height and makes all cells unoccupied
void grid::setSize(const int grid_width, const int grid_height)
{
m_width = grid_width;
m_height = grid_height;
m_grid = new cell* [m_width];
// from left to right
for (int i = 0; i < m_width; i++)
{
m_grid[i] = new cell [m_height];
// from top to bottom
for (int j = 0; j < m_height; j++)
{
m_grid[i][j].m_occupied = 0;
}
}
}
// calculates radiation value for all cells in the grid
void grid::setRadiation(const int x_coord, const int y_coord,
const int rad_mag, const int decay_factor)
{
m_x_rad = x_coord;
m_y_rad = y_coord;
// from left to right
for (int i = 0; i < m_width; i++)
{
// from top to bottom
for (int j = 0; j < m_height; j++)
{
// radiation (equals) max radiation (minus)
// Manhattan distance between source and current (times) decay factor
m_grid[i][j].m_rad = rad_mag -
(abs(m_x_rad - i) + abs(m_y_rad - j)) * decay_factor;
// minimum radiation
if (m_grid[i][j].m_rad < 1)
m_grid[i][j].m_rad = 1;
}
}
return;
}
// makes a specified cell unavailable
void grid::setOccupied(const int x, const int y)
{
m_grid[x][y].m_occupied = 1;
return;
}
// makes a specified cell available
void grid::setUnoccupied(const int x, const int y)
{
m_grid[x][y].m_occupied = 0;
return;
}
grid& grid::operator=(const grid& g)
{
m_x_rad = g.m_x_rad;
m_y_rad = g.m_y_rad;
m_width = g.m_width;
m_height = g.m_height;
m_grid = new cell* [m_width];
// from left to right
for (int i = 0; i < m_width; i++)
{
m_grid[i] = new cell [m_height];
// from top to bottom
for (int j = 0; j < m_height; j++)
{
m_grid[i][j].m_occupied = g.m_grid[i][j].m_occupied;
m_grid[i][j].m_rad = g.m_grid[i][j].m_rad;
}
}
return *this;
}
// checks whether specified cell is available
bool grid::isAvailable(const int x, const int y) const
{
return !(m_grid[x][y].m_occupied);
}
// displays a map of available/unavailable cells
void grid::printOccupied() const
{
cout << "--------------------Occupied\n";
// from top to bottom
for (int i = 0; i < m_height; i++)
{
// from left to right
for (int j = 0; j < m_width; j++)
{
cout << (m_grid[j][i].m_occupied ? "1" : "0") << " ";
}
cout << "\n";
}
}
// displays a map of every cell's radiation level
void grid::printRadiation() const
{
cout << "--------------------Radiation\n";
// from top to bottom
for (int i = 0; i < m_height; i++)
{
// from left to right
for (int j = 0; j < m_width; j++)
{
cout << m_grid[j][i].m_rad
<< (m_grid[j][i].m_rad >= DOUBLE_DIGIT ? " " : " ");
}
cout << "\n";
}
}
主要.cpp
#include "main.h"
using namespace std;
int main()
{
clock_t start = clock();
double duration;
queue<node> frontier;
node root; // node that grabs initial puzzle state
unordered_map<int, basic_node> node_map;
int grid_width, grid_height, x_coord, y_coord, rad_mag, decay_factor,
num_alligators, num_turtles, num_trees;
string orientation;
root.setNumItems(VECTOR_SHIFT + num_alligators + num_turtles + num_trees);
if (!getInitialPuzzle(root.getGrid(), root.getItems(),
grid_width, grid_height, rad_mag, decay_factor,
num_alligators, num_turtles, num_trees,
x_coord, y_coord, orientation))
{
return -1;
}
frontier.push(node(root));
breadthFirstTreeSearch(frontier, node_map, num_alligators, num_trees);
// put the root in the hash table
basic_node r;
r.node_id = frontier.back().getID();
r.node_parent = -1;
r.node_cost = frontier.back().getItems()[1]. // only the boat has a cost
calcItemCost(frontier.back().getGrid());
// no action taken to get here
r.node_action_item = '\0';
r.node_action_number = 0;
r.node_action_move = '\0';
node_map.insert({r.node_id, r});
int j = 0;
while (!frontier.empty())
{
++j;
if (frontier.front().isGoal())
break;
// only need to check boat, alligators, and turtles
for (int i = 1; i < frontier.front().getItems().size() - num_trees; i++)
{
// only the boat can rotate
if (i == 1)
{
// if the boat can rotate ccw in its own grid
if (frontier.front().getItems()[i].
canRotateCCW(frontier.front().getGrid()))
{
// enqueue a copy of the current node and rotate that boat ccw
frontier.push(frontier.front());
frontier.back().getItems()[i].rotateCCW(frontier.back().getGrid());
basic_node b;
b.node_id = frontier.back().getID();
b.node_parent = frontier.back().getParent();
b.node_cost = frontier.back().getItems()[1].
calcItemCost(frontier.back().getGrid());
b.node_action_item = frontier.back().getItems()[1].getType();
b.node_action_number = 0; // there's only one boat
b.node_action_move = 'N'; // counterclockwise
node_map.insert({b.node_id, b});
}
// if the boat can rotate cw in its own grid
if (frontier.front().getItems()[i].
canRotateCW(frontier.front().getGrid()))
{
// enqueue a copy of the current node and rotate that boat cw
frontier.push(frontier.front());
frontier.back().getItems()[i].rotateCW(frontier.back().getGrid());
basic_node b;
b.node_id = frontier.back().getID();
b.node_parent = frontier.back().getParent();
b.node_cost = frontier.back().getItems()[1].
calcItemCost(frontier.back().getGrid());
b.node_action_item = frontier.back().getItems()[1].getType();
b.node_action_number = 0; // there's only one boat
b.node_action_move = 'C'; // clockwise
node_map.insert({b.node_id, b});
}
}
// if the boat, alligator, or turtle can move forward
if (frontier.front().getItems()[i].
canMoveForward(frontier.front().getGrid()))
{
// enqueue a copy of the current node and move the item forward
frontier.push(frontier.front());
frontier.back().getItems()[i].moveForward(frontier.back().getGrid());
basic_node b;
b.node_id = frontier.back().getID();
b.node_parent = frontier.back().getParent();
b.node_cost = frontier.back().getItems()[1].
calcItemCost(frontier.back().getGrid());
// gets B, A, or T for the item that moves
b.node_action_item = frontier.back().getItems()[i].getType();
// need to shift for boat and goal, and alligator if necessary
b.node_action_number = i -
(b.node_action_item == 'B' ? 1 : VECTOR_SHIFT) - // shift 1 if boat
(b.node_action_item == 'A' ? 0 : num_alligators);
// this will get the movement; one of {U, D, L, R}
b.node_action_move = (frontier.back().getItems()[i].getOrientation());
node_map.insert({b.node_id, b});
}
// if the alligator or turtle can move backward
if (frontier.front().getItems()[i].
canMoveBackward(frontier.front().getGrid()))
{
// enqueue a copy of the current node and move the item forward
frontier.push(frontier.front());
frontier.back().getItems()[i].moveBackward(frontier.back().getGrid());
basic_node b;
b.node_id = frontier.back().getID();
b.node_parent = frontier.back().getParent();
b.node_cost = frontier.back().getItems()[1].
calcItemCost(frontier.back().getGrid());
// gets A or T for the item that moves
b.node_action_item = frontier.back().getItems()[i].getType();
// need to shift for boat and goal, and alligator if necessary
b.node_action_number = i - VECTOR_SHIFT -
(b.node_action_item == 'A' ? 0 : num_alligators);
// this will get the movement; one of {U, D, L, R}
b.node_action_move = (frontier.back().getItems()[i].getOrientation());
// this will get the movement; one of {U, D, L, R}
switch (frontier.back().getItems()[i].getOrientation())
{
case 'L':
b.node_action_move = 'R';
break;
case 'R':
b.node_action_move = 'L';
break;
case 'U':
b.node_action_move = 'D';
break;
case 'D':
b.node_action_move = 'U';
break;
}
node_map.insert({b.node_id, b});
}
}
// all possible moves completed for the current state of the board
frontier.pop();
}
root.getGrid().printOccupied();
cout << endl;
frontier.front().getGrid().printRadiation();
duration = double(clock() - start) / CLOCKS_PER_MS;
writeSolutions(duration, frontier.front(), node_map, rad_mag, decay_factor,
num_alligators, num_turtles, num_trees);
return 0;
}
和 node.cpp
#include <cstddef>
#include <iostream>
#include "node.h"
int node::class_id = 0;
int node::num_items = 0;
node::node()
{
m_grid.setGridNull();
m_grid.setSize(0, 0);
m_grid.setRadiation(0, 0, 0, 0);
m_id = 0;
m_parent = 0;
}
node::node(const node& n)
{
m_grid = n.m_grid;
m_items = n.m_items;
m_id = ++class_id;
m_parent = n.m_id;
/*for (int i = 0; i < n.getNumItems(); i++)
{
m_items[i] = n.m_items[i];
}*/
}
int node::getNumItems() const
{
return num_items;
}
grid& node::getGrid()
{
return m_grid;
}
vector<item>& node::getItems()
{
return m_items;
}
int node::getID() const
{
return m_id;
}
int node::getParent() const
{
return m_parent;
}
void node::setNumItems(const int n)
{
num_items = n;
return;
}
void node::setID(const int i)
{
m_id = i;
return;
}
void node::setParent(const int p)
{
m_parent = p;
return;
}
node& node::operator=(node& n)
{
m_grid = n.m_grid;
m_parent = n.m_parent;
m_id = ++class_id;
for (int i = 0; i < n.getItems().size(); i++)
{
cout << "yo\n";
m_items[i] = n.m_items[i];
cout << "hey\n";
}
return *this;
}
// this will compare the boat and goal locations to determine goal node
bool node::isGoal() const
{
bool found = false;
// if we don't have 2+, we don't have a goal and a boat
if (m_items.size() >= 2)
{
// item 1 is the boat
// back of the boat is in the goal
if ((m_items[1].getXCoord() == m_items[0].getXCoord()) &&
m_items[1].getYCoord() == m_items[0].getYCoord())
{
found = true;
}
else
{
switch (m_items[1].getOrientation())
{
case 'R':
if ((m_items[1].getXCoord() == m_items[0].getXCoord() - 1) &&
m_items[1].getYCoord() == m_items[0].getYCoord())
{
found = true;
}
break;
case 'U':
if ((m_items[1].getXCoord() == m_items[0].getXCoord()) &&
m_items[1].getYCoord() == m_items[0].getYCoord() + 1)
{
found = true;
}
break;
case 'L':
if ((m_items[1].getXCoord() == m_items[0].getXCoord() + 1) &&
m_items[1].getYCoord() == m_items[0].getYCoord())
{
found = true;
}
break;
case 'D':
if ((m_items[1].getXCoord() == m_items[0].getXCoord()) &&
m_items[1].getYCoord() == m_items[0].getYCoord() - 1)
{
found = true;
}
break;
}
}
}
return found;
}
最佳答案
线索是栈帧中的_dl_init
。这是共享库的初始化函数。任何时候加载共享库时,它的初始化函数都会运行。有问题的内存块是在共享库的初始化函数中分配的。
共享库在其初始化函数中为其内部数据结构分配内存是很常见的。这是 C++ 运行时库,预计将在进程的生命周期内保持加载状态。因此,这并不是真正的内存泄漏,特别是因为“可能丢失”的指示表明 valgrind 设法在某处找到了指向内存块的指针。
但是,您自己的代码中确实存在内存泄漏。 “肯定输了”就是这个意思:肯定输了。你的类泄漏了内存,而 valgrind 没有找到任何指向它的悬挂指针,否则它会属于“可能丢失”类别。
关于c++ - Valgrind 说 malloc 内存损坏,还没有使用 malloc,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41916853/
我希望 valgrind 在发现第一个错误时停止并退出。 请勿推荐 --vgdb-error=1 :它不会退出 valgrind。您必须连接 gdb 并从那里终止。 --db-attach : 在最近
有人可以快速解释 Valgrind 的工作原理吗?一个例子:它如何知道内存何时被分配和释放? 最佳答案 Valgrind 基本上在“沙箱”中运行您的应用程序。在此沙箱中运行时,它能够插入自己的指令来进
我有一个因 SIGSEGV 而崩溃的应用程序。 --20183-- VALGRIND INTERNAL ERROR: Valgrind received a signal 11 (SIGSEGV) -
我有一个因 SIGSEGV 而崩溃的应用程序。 --20183-- VALGRIND INTERNAL ERROR: Valgrind received a signal 11 (SIGSEGV) -
我想使用 valgrind 检查长时间运行的进程是否存在内存泄漏。我怀疑我所追求的内存泄漏可能仅在执行几个小时后才会发生。我可以在 valgrind 下运行应用程序并获取 valgrind 日志,但这
我想用 valgrind 检查一个长时间运行的进程是否有内存泄漏。我怀疑我所追求的内存泄漏可能仅在执行数小时后才会发生。我可以在 valgrind 下运行应用程序并获得 valgrind 日志,但这样
如何在不通过 valgrind 命令选项启动它的情况下对每个 Process 实例执行 valgrind memcheck。 有没有办法将监控选项保存在进程中,而不是每次都使用 valgrind 命令
我使用了“--trace-children=yes”选项,我还使用了“--trace-children-skip=patt1,patt2,...”选项(过滤掉噪音过程)。但它对我来说仍然很慢,我的多进
我从 Valgrind 得到以下日志: MPK ==5263== 4 bytes in 1 blocks are still reachable in loss record 1 of 84 ==52
如何在 Valgrind 抑制文件中添加注释? 我需要为一个大型项目维护一个 Valgrind 抑制文件。我们从我们链接到的工具中过滤无法修复的错误。随着工具的新版本发布,此文件可能需要随着时间的推移
我有一个大程序要运行。使用 valgrind 需要几个小时才能运行。我听说有一些东西可以让我们为程序中的特定函数调用 valgrind。其余程序将正常执行(没有 valgrind env)。 任何人都
我可以用 valgrind 检测整数溢出缺陷吗?里面的哪个工具可以做到这一点? 最佳答案 Valgrind 没有可以检测整数溢出的工具。 您可能会使用 gcc 选项捕获这些错误: -ftrapv Th
我有一个简单的程序: int main(void) { const char sname[]="xxx"; sem_t *pSemaphor; if ((pSemaphor = sem_o
如何让 Valgrind 准确显示错误发生的位置?我编译了我的程序(通过 PuTTy 在 Windows 机器上通过 Linux 终端)添加了 -g 调试选项。 当我运行 Valgrind 时,我得到
或者最好是全部,而不仅仅是我的代码?我的程序使用 Gtk、Loudmouth 和其他一些东西,而这两个(以及它们背后的一些,libgcrypto、libssl)本身导致了如此多的错误,以至于我无法检测
我想尝试使用 valgrind 进行一些堆损坏检测。通过以下腐败“单元测试”: #include #include #include int main() { char * c = (ch
我看过类似的问题here ,但我的问题是我没有编辑 default.supp 文件的权限。例如,Valgrind 中是否有任何忽略所有抑制文件的命令行选项? 最佳答案 在 Valgrind 3.10.
我在一个运行无限循环的程序上使用 valgrind。 由于memcheck在程序结束后显示内存泄漏,但由于我的程序有无限循环,它永远不会结束。 那么有什么方法可以强制从 valgrind 时不时地转储
我一直在尝试使用 valgrind 查找一些可疑的内存错误。 在被分析的程序甚至到达我希望分析的点之前,它会因为对 mmap 的调用开始失败而退出。当它不在 valgrind 下时,这些调用会成功。
由于 OpenSSL 使用未初始化的内存,因此对使用 openldap2 的 libldap 的程序进行 Valgrind 是一件苦差事。存在一个 --ignore-fn选项,但仅适用于 Valgri
我是一名优秀的程序员,十分优秀!