- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
这是我第一次使用 map 编写 Web 应用程序。
我正在尝试为来自给定 OSM map 的每个节点创建带有邻接表的无向图。
当我在小 map 上测试时一切正常。
我解码 OSM 映射(等于 XML 文件),然后从 OSM 对象创建我收到的无向图。
当我尝试从较大的 map 创建图表时,问题就出现了。
以6MB大小的 map 为例:
节点数:24828
路数:4535
每种方式平均有5个节点。
所有这些加起来就是:24828 * 4535 * 5 = 562,974,900 次迭代!
直觉上,为了找到每个节点的邻居,我需要从节点列表中的每个节点 2 中以各种方式遍历每个节点 1。
如果 node1 等于 node2 我需要将下一个和上一个节点作为它的邻居。
我花了大约 1:30 分钟来完成:
我正在构建将在智能手机上运行并计算随机运行路径的网络应用程序。
如果用户只需要等待 1:30 分钟来创建图形,它将无法使用。
我熟悉 BFS\DFS,但在这种情况下它们不会帮助我,因为我需要构建图形。
也许还有其他有效的方法来为节点创建邻接表?
我如何构建邻接表:
public static List<Node> GetNodesFromXML(Osm i_Osm) {
List<Node> o_Nodes = new ArrayList<Node>();
long id;
double latitude;
double longtitude;
Map<Node, List<Node>> o_AdjList = new HashMap<Node, List<Node>>();
for (Osm.Node nodeChild : i_Osm.getNode()) {
id = nodeChild.getId();
latitude = nodeChild.getLat();
longtitude = nodeChild.getLon();
Node node = new Node(id, latitude, longtitude);
for (Osm.Way way : i_Osm.getWay()) // go over the node list of the specific way objects
{
for (Osm.Way.Nd nd : way.getNd()) {
// some manipulation to create the adjacency list
}
}
//List<Long> nodeAdjacenciesByRef = getNodeAdjacenciesByRef(node, i_Osm.getWay(), i_Osm.getNode());
// List<Edge> nodeAdjacencies = getNodeAdjacencies1(node, i_Osm.getWay(), i_Osm.getNode());
// List<Edge> nodeAdjacencies = getAdjacenciesListFromRefList(node, nodeAdjacenciesByRef, i_Osm.getNode());
// node.SetAdjacencies(nodeAdjacencies);
o_Nodes.add(node);
}
for(Node node : o_Nodes)
{
}
o_Nodes = updateAdjacenciesToAllNodes(o_Nodes);
return o_Nodes;
}
我用于图形的类:
// Node.java
public class Node implements Comparable<Node>
{
private long m_Id;
private List<Edge> m_Adjacencies = new ArrayList<Edge>();
private double m_Longtitude;
private double m_Latitude;
private Node m_Prev;
private double m_MinDistance = Double.POSITIVE_INFINITY; // this is for Dijkstra Algorithm
//used to reconstruct the track when we found the approproate length of the user request
//from the current level to the destination
public Node(long i_Id, double i_Latitude, double i_Longtitude)
{
m_Id = i_Id;
m_Latitude = i_Latitude;
m_Longtitude = i_Longtitude;
}
...
}
// Graph.java
private List<Node> m_Nodes = new ArrayList<>();
private List<Way> m_Ways = new ArrayList<>();
private List<Relation> m_Relations = new ArrayList<>();
private Bounds m_Bounds;
public Graph(List<Node> i_Nodes, List<Way> i_Ways, List<Relation> i_Relations, Bounds i_Bounds) {
m_Nodes = i_Nodes;
m_Ways = i_Ways;
m_Relations = i_Relations;
m_Bounds = i_Bounds;
}
...
}
// Edge.java
public class Edge {
Node m_Source;
Node m_Destination;
double m_Weight;
public Edge(Node i_Source, Node i_Destination, double i_Weight) {
m_Source = i_Source;
m_Destination = i_Destination;
m_Weight = i_Weight;
}
...
}
编辑:已解决:
我使用了哈希表。这样我将能够在 O(1) 中获取每个节点。
所以我在所有节点上运行一次(1 秒或更短时间)并创建这张 map 。
在我有了这个映射之后,我可以在没有外部循环的情况下以各种方式传递每个节点。
重新设计后,整个过程大约需要 3 秒。
解决方法如下:
public static List<Node> GetNodesFromXML(Osm i_Osm) {
List<Node> o_Nodes = new ArrayList<Node>();
long id;
double latitude;
double longtitude;
Map<Long, Node> o_NodesByRef = new HashMap<Long, Node>();
for (Osm.Node nodeChild : i_Osm.getNode()) {
id = nodeChild.getId();
latitude = nodeChild.getLat();
longtitude = nodeChild.getLon();
Node node = new Node(id, latitude, longtitude);
//o_Nodes.add(node);
o_NodesByRef.put(id, node);
}
o_Nodes = addAdjacencies(o_NodesByRef, i_Osm.getWay());
//o_Nodes = updateAdjacenciesToAllNodes(o_Nodes);
return o_Nodes;
}
private static List<Node> addAdjacencies(Map<Long, Node> i_NodesByRef , List<Osm.Way> i_Ways) {
List<Node> o_Nodes = new ArrayList<Node>();
long ndId;
int nodeIndex;
int lastNodeIndex;
Node previousNode;
Node nextNode;
double weight;
//System.out.println(i_SourceNode.getNodeId());
for (Osm.Way way : i_Ways) // go over the node list of the specific way objects
{
if (way.getNd().size() > 1) {
for (Osm.Way.Nd nd : way.getNd()) {
if(i_NodesByRef.containsKey(nd.getRef()))// found node in way
{
Node node = i_NodesByRef.get(nd.getRef());
nodeIndex = way.getNd().indexOf(nd);
Edge edge1;
Edge edge2;
Osm.Way.Nd temp_nd;
lastNodeIndex = way.getNd().size() - 1;
if (nodeIndex == 0) // node is the first in the way
{
temp_nd = way.getNd().get(nodeIndex + 1);
nextNode = i_NodesByRef.get(temp_nd.getRef());
weight = CoordinateMath.getDistanceBetweenTwoNodes(node, nextNode);
edge1 = new Edge(node, nextNode, weight);
i_NodesByRef.get(node.getNodeId()).getAdjacencies().add(edge1);
} else if (lastNodeIndex == nodeIndex) // node is the last
{
temp_nd = way.getNd().get(nodeIndex - 1);
previousNode = i_NodesByRef.get(temp_nd.getRef());
weight = CoordinateMath.getDistanceBetweenTwoNodes(node, previousNode);
edge1 = new Edge(node, previousNode, weight);
i_NodesByRef.get(node.getNodeId()).getAdjacencies().add(edge1);
} else // node is in the middle
{
temp_nd = way.getNd().get(nodeIndex - 1);
previousNode = i_NodesByRef.get(temp_nd.getRef());
weight = CoordinateMath.getDistanceBetweenTwoNodes(node, previousNode);
// node -> previousNode
edge1 = new Edge(node, previousNode, weight);
i_NodesByRef.get(node.getNodeId()).getAdjacencies().add(edge1);
temp_nd = way.getNd().get(nodeIndex + 1);
nextNode = i_NodesByRef.get(temp_nd.getRef());
weight = CoordinateMath.getDistanceBetweenTwoNodes(node, nextNode);
// node -> nextNode
edge2 = new Edge(node, nextNode, weight);
i_NodesByRef.get(node.getNodeId()).getAdjacencies().add(edge2);
}
}
}
}
}
for(Map.Entry<Long, Node> entry : i_NodesByRef.entrySet())
{
o_Nodes.add(entry.getValue());
}
return o_Nodes;
}
最佳答案
问题是您试图存储一个邻接表,其中的元素是整个节点。
Map<Node, List<Node>> o_AdjList = new HashMap<Node, List<Node>>();
而且效率不高,因为该图存储了所有节点信息并占用了大量内存。
相反,我将只使用节点 ID 作为整数来制作图表:
Map<Integer, TreeSet<Integer>> o_AdjList = new HashMap<Integer, TreeSet<Integer>>();
并将剩余的节点信息保存在一个单独的更高效的结构中,例如 HashSet。
这意味着您将拥有一个仅使用整数 id 的图形表示,该对象比第一个图形小很多倍。现在处理器可以缓存更多并运行 SSSP更快。
如果你真的想要一个有实际节点的图,你可以在之后构建它。这就是我从方式
构建图表的方法:
for (Osm.Way way : i_Osm.getWay()) {
//I will asume the getNd() returns some sort of array or list an you can access the next or previous element
for (Osm.Way.Nd nd : way.getNd()) {
if(o_AdjList contains key nd){
o.AdjList.get(nd).add(nextNd);
} else {
o.AdjList.put(nd,nextNd);
}
}
当然,您必须实现一些 walk 方法,但在那之后……您已经设置好节点。
关于java - 如何在线性时间内从大型 OSM map 构建具有邻接表的无向图,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32286855/
我有一台 MySQL 服务器和一台 PostgreSQL 服务器。 需要从多个表中复制或重新插入一组数据 MySQL 流式传输/同步到 PostgreSQL 表。 这种复制可以基于时间(Sync)或事
如果两个表的 id 彼此相等,我尝试从一个表中获取数据。这是我使用的代码: SELECT id_to , email_to , name_to , status_to
我有一个 Excel 工作表。顶行对应于列名称,而连续的行每行代表一个条目。 如何将此 Excel 工作表转换为 SQL 表? 我使用的是 SQL Server 2005。 最佳答案 这取决于您使用哪
我想合并两个 Django 模型并创建一个模型。让我们假设我有第一个表表 A,其中包含一些列和数据。 Table A -------------- col1 col2 col3 col
我有两个表:table1,table2,如下所示 table1: id name 1 tamil 2 english 3 maths 4 science table2: p
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 1 年前。 Improve th
下面两个语句有什么区别? newTable = orginalTable 或 newTable.data(originalTable) 我怀疑 .data() 方法具有性能优势,因为它在标准 AX 中
我有一个表,我没有在其中显式定义主键,它并不是真正需要的功能......但是一位同事建议我添加一个列作为唯一主键以随着数据库的增长提高性能...... 谁能解释一下这是如何提高性能的? 没有使用索引(
如何将表“产品”中的产品记录与其不同表“图像”中的图像相关联? 我正在对产品 ID 使用自动增量。 我觉得不可能进行关联,因为产品 ID 是自动递增的,因此在插入期间不可用! 如何插入新产品,获取产品
我有一个 sql 表,其中包含关键字和出现次数,如下所示(尽管出现次数并不重要): ____________ dog | 3 | ____________ rat | 7 | ____
是否可以使用目标表中的LAST_INSERT_ID更新源表? INSERT INTO `target` SELECT `a`, `b` FROM `source` 目标表有一个自动增量键id,我想将其
我正在重建一个搜索查询,因为它在“我看到的”中变得多余,我想知道什么 (albums_artists, artists) ( ) does in join? is it for boosting pe
以下是我使用 mysqldump 备份数据库的开关: /usr/bin/mysqldump -u **** --password=**** --single-transaction --databas
我试图获取 MySQL 表中的所有行并将它们放入 HTML 表中: Exam ID Status Assigned Examiner
如何查询名为 photos 的表中的所有记录,并知道当前用户使用单个查询将哪些结果照片添加为书签? 这是我的表格: -- -- Table structure for table `photos` -
我的网站都在 InnoDB 表上运行,目前为止运行良好。现在我想知道在我的网站上实时发生了什么,所以我将每个页面浏览量(页面、引荐来源网址、IP、主机名等)存储在 InnoDB 表中。每秒大约有 10
我在想我会为 mysql 准备两个表。一个用于存储登录信息,另一个用于存储送货地址。这是传统方式还是所有内容都存储在一张表中? 对于两个表...有没有办法自动将表 A 的列复制到表 B,以便我可以引用
我不是程序员,我从这个表格中阅读了很多关于如何解决我的问题的内容,但我的搜索效果不好 我有两张 table 表 1:成员 id*| name | surname -------------------
我知道如何在 ASP.NET 中显示真实表,例如 public ActionResult Index() { var s = db.StaffInfoDBSet.ToList(); r
我正在尝试运行以下查询: "insert into visits set source = 'http://google.com' and country = 'en' and ref = '1234
我是一名优秀的程序员,十分优秀!