gpt4 book ai didi

C++ 使这个 Djikstra 实现更快

转载 作者:塔克拉玛干 更新时间:2023-11-03 04:15:00 26 4
gpt4 key购买 nike

我正在为一个大图(40k 节点,100k 弧)实现 Djikstra 算法。对于较短的路径,对于较大的路径(从一端到另一端),搜索时间不到一秒,需要花费几分钟的时间。我也在搜索后绘制路径,所以我使用了一些 Qt 对象。我怎样才能让它更快?由于 map 结构,我在搜索邻居时感觉我在浪费时间。

这是类(class)

class PathFinder {
public:
void findPath2(const Node & start, Node & finish);

static PathFinder& getInstance();
PathFinder();
PathFinder(Gps gps);
~PathFinder();

unsigned int* getPrev() const;
void setPrev(unsigned int* prev);
QVector<unsigned int> makePath(int target);

GraphicNode* getTo();
GraphicNode* getFrom();

void setTo(GraphicNode* node);
void setFrom(GraphicNode* node);

class Compare
{
public:
bool operator() (std::pair<Node*, int> a, std::pair<Node*, int> b)
{
return a.second > b.second;
}
};


private:
static PathFinder* _pathfinder;
Gps _gps;
GraphicNode* _from;
GraphicNode* _to;

unsigned int* _prev;
unsigned int* _dist;
unsigned int _notVisited;

bool selectedNode = false;


Node* getMinNode();
bool hasNotVisited();
};

这是搜索功能

void PathFinder::findPath2(const Node& start, Node& finish)
{
QVector<Node> nodes=_gps.graph().nodes();

std::priority_queue<std::pair<Node*,int>,std::vector<std::pair<Node*, int>>,Compare> q;

_dist[start.id()] = 0;
for (int i = 0; i < nodes.size(); i++) {
std::pair<Node*, int> p = std::make_pair(const_cast<Node*>(&nodes.at(i)), _dist[i]);
q.push(p);
}

while (!q.empty()) {
std::pair<Node*, int> top = q.top();
q.pop();
Node* minNode = top.first;
QMap<Node*, unsigned short> nextNodes = minNode->nextNodes();

if (*minNode == finish) {
return;
}
int minNodeId = minNode->id();
for (QMap<Node*, unsigned short>::iterator iterator=nextNodes.begin(); iterator != nextNodes.end(); iterator++) {
Node* nextNode = iterator.key();
int altDist = _dist[minNodeId] + nextNodes.value(nextNode);
int nextNodeId = nextNode->id();
if (altDist < _dist[nextNodeId]) {
_dist[nextNodeId] = altDist;
_prev[nextNodeId] = minNodeId;
std::pair<Node*, int> p = std::make_pair(nextNode, _dist[nextNodeId]);
q.push(p);
}
}
}
}

这是节点的结构,它包含一个以权重为值的到它的邻居的映射,x和y是稍后绘制它的坐标,不要介意

class Node {
private:
unsigned short _id;
double _y;
double _x;
QMap<Node*, unsigned short> _nextNodes;
bool _visited = false;


public:
Node();
Node(unsigned short id, double longitude, double latitude);

unsigned short id() const;
double y() const;
void setY(double y);
double x() const;
void setX(double x);

bool operator==(const Node& other);
void addNextNode(Node* node, unsigned short length);

QMap<Node*, unsigned short> nextNodes() const;
};

最佳答案

如果您的图表永远不会改变,解决方案是将其切割成更小的图表。

  1. 计算每个小图的边之间的最短距离并存储结果(edge1,edge2,距离,内部最短路径)。
  2. 计算两点之间的距离时,在到达“小图”边时使用存储的结果。

我相信这就是 GoogleMaps 和其他路径查找器所使用的技巧。

缺点是初始成本(如果你的图表真的永远不会改变,你可以将这些“迷你”最短路径存储在一个文件或数据库中,一劳永逸)。您必须仔细选择小图的大小,因为访问存储的结果也会有(时间)成本(无论是在大内存映射还是数据库中)。必须找到一个平衡点。

如果相同的路径搜索经常返回,另一个想法是存储搜索次数最多的结果。

关于C++ 使这个 Djikstra 实现更快,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49839363/

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