OsmAnd/OsmAnd-java/src/main/java/net/osmand/router/TransportRoutePlanner.java

638 lines
20 KiB
Java
Raw Normal View History

package net.osmand.router;
import gnu.trove.iterator.TIntIterator;
import gnu.trove.list.array.TIntArrayList;
import gnu.trove.map.hash.TIntObjectHashMap;
import gnu.trove.map.hash.TLongObjectHashMap;
import java.io.IOException;
import java.util.ArrayList;
2018-08-04 13:03:06 +02:00
import java.util.Collections;
import java.util.Comparator;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.PriorityQueue;
import net.osmand.binary.BinaryMapIndexReader;
import net.osmand.binary.BinaryMapIndexReader.SearchRequest;
import net.osmand.data.LatLon;
import net.osmand.data.TransportRoute;
2018-08-15 14:43:47 +02:00
import net.osmand.data.TransportSchedule;
import net.osmand.data.TransportStop;
2018-08-05 22:35:28 +02:00
import net.osmand.osm.edit.Node;
import net.osmand.osm.edit.Way;
import net.osmand.util.MapUtils;
public class TransportRoutePlanner {
2018-08-05 10:55:08 +02:00
public List<TransportRouteResult> buildRoute(TransportRoutingContext ctx, LatLon start, LatLon end) throws IOException {
2018-08-04 13:03:06 +02:00
ctx.startCalcTime = System.currentTimeMillis();
List<TransportRouteSegment> startStops = ctx.getTransportStops(start);
List<TransportRouteSegment> endStops = ctx.getTransportStops(end);
TLongObjectHashMap<TransportRouteSegment> endSegments = new TLongObjectHashMap<TransportRouteSegment>();
for(TransportRouteSegment s : endStops) {
endSegments.put(s.getId(), s);
}
2018-08-04 13:03:06 +02:00
PriorityQueue<TransportRouteSegment> queue = new PriorityQueue<TransportRouteSegment>(new SegmentsComparator(ctx));
for(TransportRouteSegment r : startStops){
r.walkDist = (float) MapUtils.getDistance(r.getLocation(), start);
r.distFromStart = r.walkDist / ctx.cfg.walkSpeed;
queue.add(r);
}
double finishTime = ctx.cfg.maxRouteTime;
List<TransportRouteSegment> results = new ArrayList<TransportRouteSegment>();
while (!queue.isEmpty()) {
TransportRouteSegment segment = queue.poll();
2018-08-04 13:03:06 +02:00
TransportRouteSegment ex = ctx.visitedSegments.get(segment.getId());
if(ex != null) {
if(ex.distFromStart > segment.distFromStart) {
System.err.println(String.format("%.1f (%s) > %.1f (%s)", ex.distFromStart, ex, segment.distFromStart, segment));
}
continue;
}
2018-08-06 00:01:25 +02:00
ctx.visitedRoutesCount++;
2018-08-04 13:03:06 +02:00
ctx.visitedSegments.put(segment.getId(), segment);
if (segment.getDepth() > ctx.cfg.maxNumberOfChanges) {
continue;
}
if (segment.distFromStart > finishTime + ctx.cfg.finishTimeSeconds) {
break;
}
long segmentId = segment.getId();
TransportRouteSegment finish = null;
double minDist = 0;
double travelDist = 0;
double travelTime = 0;
TransportStop prevStop = segment.getStop(segment.segStart);
2018-08-05 23:46:01 +02:00
List<TransportRouteSegment> sgms = new ArrayList<TransportRouteSegment>();
2018-08-04 13:03:06 +02:00
for (int ind = 1 + segment.segStart; ind < segment.getLength(); ind++) {
segmentId ++;
ctx.visitedSegments.put(segmentId, segment);
TransportStop stop = segment.getStop(ind);
// could be geometry size
double segmentDist = MapUtils.getDistance(prevStop.getLocation(), stop.getLocation());
travelDist += segmentDist;
2018-08-15 14:43:47 +02:00
if(ctx.cfg.useSchedule) {
TransportSchedule sc = segment.road.getSchedule();
int interval = sc.avgStopIntervals.get(ind - 1);
travelTime += interval * 10;
} else {
travelTime += ctx.cfg.stopTime + segmentDist / ctx.cfg.travelSpeed;
}
2018-08-05 23:46:01 +02:00
sgms.clear();
sgms = ctx.getTransportStops(stop.x31, stop.y31, true, sgms);
2018-08-04 13:03:06 +02:00
for (TransportRouteSegment sgm : sgms) {
if (segment.wasVisited(sgm)) {
continue;
}
TransportRouteSegment rrs = new TransportRouteSegment(sgm);
rrs.parentRoute = segment;
rrs.parentStop = ind;
rrs.walkDist = MapUtils.getDistance(rrs.getLocation(), stop.getLocation());
rrs.parentTravelTime = travelTime;
rrs.parentTravelDist = travelDist;
double walkTime = rrs.walkDist / ctx.cfg.walkSpeed + ctx.cfg.changeTime;
rrs.distFromStart = segment.distFromStart + travelTime + walkTime;
queue.add(rrs);
}
TransportRouteSegment f = endSegments.get(segmentId);
double distToEnd = MapUtils.getDistance(stop.getLocation(), end);
if (f != null && distToEnd < ctx.cfg.walkRadius) {
if (finish == null || minDist > distToEnd) {
minDist = distToEnd;
finish = new TransportRouteSegment(f);
finish.parentRoute = segment;
finish.parentStop = ind;
finish.walkDist = distToEnd;
finish.parentTravelTime = travelTime;
finish.parentTravelDist = travelDist;
double walkTime = distToEnd / ctx.cfg.walkSpeed;
finish.distFromStart = segment.distFromStart + travelTime + walkTime;
}
}
prevStop = stop;
}
if (finish != null) {
if (finishTime > finish.distFromStart) {
finishTime = finish.distFromStart;
}
if(finish.distFromStart < finishTime + ctx.cfg.finishTimeSeconds) {
results.add(finish);
}
}
}
2018-08-04 13:03:06 +02:00
return prepareResults(ctx, results);
}
private List<TransportRouteResult> prepareResults(TransportRoutingContext ctx, List<TransportRouteSegment> results) {
Collections.sort(results, new SegmentsComparator(ctx));
List<TransportRouteResult> lst = new ArrayList<TransportRouteResult>();
2018-08-06 00:01:25 +02:00
System.out.println(String.format("Calculated %.1f seconds, found %d results, visited %d routes, loaded %d tiles (%d ms read, %d ms total),",
(System.currentTimeMillis() - ctx.startCalcTime) / 1000.0, results.size(), ctx.visitedRoutesCount,
ctx.quadTree.size(), ctx.readTime / (1000 * 1000), ctx.loadTime / (1000 * 1000)));
2018-08-04 13:03:06 +02:00
for(TransportRouteSegment res : results) {
TransportRouteResult route = new TransportRouteResult(ctx);
route.routeTime = res.distFromStart;
route.finishWalkDist = res.walkDist;
TransportRouteSegment p = res;
while (p != null) {
if (p.parentRoute != null) {
TransportRouteResultSegment sg = new TransportRouteResultSegment(p.parentRoute.road,
p.parentRoute.segStart, p.parentStop, p.parentRoute.walkDist);
route.segments.add(0, sg);
}
p = p.parentRoute;
}
// test if faster routes fully included
boolean include = false;
for(TransportRouteResult s : lst) {
if(includeRoute(s, route)) {
include = true;
break;
}
}
if(!include) {
lst.add(route);
System.out.println(route.toString());
} else {
// System.err.println(route.toString());
}
}
return lst;
}
private boolean includeRoute(TransportRouteResult fastRoute, TransportRouteResult testRoute) {
if(testRoute.segments.size() < fastRoute.segments.size()) {
return false;
}
int j = 0;
for(int i = 0; i < fastRoute.segments.size(); i++, j++) {
TransportRouteResultSegment fs = fastRoute.segments.get(i);
while(j < testRoute.segments.size()) {
TransportRouteResultSegment ts = testRoute.segments.get(j);
if(fs.route.getId().longValue() != ts.route.getId().longValue()) {
j++;
} else {
break;
}
}
if(j >= testRoute.segments.size()) {
return false;
}
}
return true;
}
private static class SegmentsComparator implements Comparator<TransportRouteSegment> {
public SegmentsComparator(TransportRoutingContext ctx) {
}
@Override
public int compare(TransportRouteSegment o1, TransportRouteSegment o2) {
2018-08-04 13:03:06 +02:00
return Double.compare(o1.distFromStart, o2.distFromStart);
}
}
2018-08-04 13:03:06 +02:00
public static class TransportRouteResultSegment {
public final TransportRoute route;
public final int start;
public final int end;
public final double walkDist ;
public TransportRouteResultSegment(TransportRoute route, int start, int end, double walkDist) {
this.route = route;
this.start = start;
this.end = end;
this.walkDist = walkDist;
}
2018-08-04 13:03:06 +02:00
public TransportStop getStart() {
return route.getForwardStops().get(start);
}
public TransportStop getEnd() {
return route.getForwardStops().get(end);
}
2018-08-05 22:35:28 +02:00
public List<Way> getGeometry() {
List<Way> list = new ArrayList<Way>();
route.mergeForwardWays();
List<Way> fw = route.getForwardWays();
double minStart = 150;
double minEnd = 150;
LatLon str = getStart().getLocation();
LatLon en = getEnd().getLocation();
int endInd = -1;
List<Node> res = new ArrayList<Node>();
for(int i = 0; i < fw.size() ; i++) {
List<Node> nodes = fw.get(i).getNodes();
for(int j = 0; j < nodes.size(); j++) {
Node n = nodes.get(j);
if(MapUtils.getDistance(str, n.getLatitude(), n.getLongitude()) < minStart) {
minStart = MapUtils.getDistance(str, n.getLatitude(), n.getLongitude());
res.clear();
}
res.add(n);
if(MapUtils.getDistance(en, n.getLatitude(), n.getLongitude()) < minEnd) {
endInd = res.size();
minEnd = MapUtils.getDistance(en, n.getLatitude(), n.getLongitude());
}
}
}
Way way = new Way(-1);
if (res.isEmpty()) {
for (int i = start; i <= end; i++) {
LatLon l = getStop(i).getLocation();
Node n = new Node(l.getLatitude(), l.getLongitude(), -1);
way.addNode(n);
}
list.add(way);
} else {
for(int k = 0; k < res.size() && k < endInd; k++) {
way.addNode(res.get(k));
}
}
list.add(way);
return list;
}
2018-08-04 13:03:06 +02:00
public double getTravelDist() {
double d = 0;
for (int k = start; k < end; k++) {
d += MapUtils.getDistance(route.getForwardStops().get(k).getLocation(),
route.getForwardStops().get(k + 1).getLocation());
}
return d;
}
2018-08-05 10:55:08 +02:00
public TransportStop getStop(int i) {
return route.getForwardStops().get(i);
}
}
2018-08-04 13:03:06 +02:00
public static class TransportRouteResult {
List<TransportRouteResultSegment> segments = new ArrayList<TransportRouteResultSegment>(4);
double finishWalkDist;
double routeTime;
2018-08-05 22:35:28 +02:00
private final TransportRoutingConfiguration cfg;
2018-08-04 13:03:06 +02:00
public TransportRouteResult(TransportRoutingContext ctx) {
2018-08-05 22:35:28 +02:00
cfg = ctx.cfg;
2018-08-04 13:03:06 +02:00
}
public List<TransportRouteResultSegment> getSegments() {
return segments;
}
public double getWalkDist() {
double d = finishWalkDist;
for (TransportRouteResultSegment s : segments) {
d += s.walkDist;
}
return d;
}
2018-08-05 22:35:28 +02:00
public double getRouteTime() {
return routeTime;
}
2018-08-04 13:03:06 +02:00
public int getStops() {
int stops = 0;
for(TransportRouteResultSegment s : segments) {
stops += (s.end - s.start);
}
return stops;
}
public double getTravelDist() {
double d = 0;
for (TransportRouteResultSegment s : segments) {
d += s.getTravelDist();
}
return d;
}
2018-08-05 22:35:28 +02:00
public double getTravelTime() {
2018-08-15 14:43:47 +02:00
if(cfg.useSchedule) {
int t = 0;
for(TransportRouteResultSegment s : segments) {
TransportSchedule sts = s.route.getSchedule();
for (int k = s.start; k < s.end; k++) {
t += sts.getAvgStopIntervals()[k] * 10;
}
}
return t;
}
2018-08-05 22:35:28 +02:00
return getTravelDist() / cfg.travelSpeed + cfg.stopTime * getStops() +
cfg.changeTime * getChanges();
}
public double getWalkTime() {
return getWalkDist() / cfg.walkSpeed;
}
2018-08-04 13:03:06 +02:00
public int getChanges() {
return segments.size() - 1;
}
@Override
public String toString() {
StringBuilder bld = new StringBuilder();
bld.append(String.format("Route %d stops, %d changes, %.2f min: %.2f m (%.1f min) to walk, %.2f m (%.1f min) to travel\n",
2018-08-05 22:35:28 +02:00
getStops(), getChanges(), routeTime / 60, getWalkDist(), getWalkTime() / 60.0,
getTravelDist(), getTravelTime() / 60.0));
2018-08-04 13:03:06 +02:00
for(int i = 0; i < segments.size(); i++) {
TransportRouteResultSegment s = segments.get(i);
bld.append(String.format(" %d. %s: walk %.1f m to '%s' and travel to '%s' by %s %d stops \n",
i + 1, s.route.getRef(), s.walkDist, s.getStart().getName(), s.getEnd().getName(), s.route.getName(), (s.end - s.start)));
}
bld.append(String.format(" F. Walk %.1f m to reach your destination", finishWalkDist));
return bld.toString();
}
}
public static class TransportRouteSegment {
2018-08-04 13:03:06 +02:00
final int segStart;
final TransportRoute road;
2018-08-15 14:43:47 +02:00
final int departureTime;
private static final int SHIFT = 10; // assume less than 1024 stops
TransportRouteSegment parentRoute = null;
2018-08-04 13:03:06 +02:00
int parentStop;
double parentTravelTime; // travel time
double parentTravelDist; // inaccurate
// walk distance to start route location (or finish in case last segment)
double walkDist = 0;
// main field accumulated all time spent from beginning of journey
double distFromStart = 0;
2018-08-15 14:43:47 +02:00
public TransportRouteSegment(TransportRoute road, int stopIndex) {
this.road = road;
this.segStart = (short) stopIndex;
2018-08-15 14:43:47 +02:00
this.departureTime = -1;
}
public TransportRouteSegment(TransportRoute road, int stopIndex, int depTime) {
this.road = road;
this.segStart = (short) stopIndex;
this.departureTime = depTime;
}
2018-08-04 13:03:06 +02:00
public TransportRouteSegment(TransportRouteSegment c) {
this.road = c.road;
this.segStart = c.segStart;
2018-08-15 14:43:47 +02:00
this.departureTime = c.departureTime;
2018-08-04 13:03:06 +02:00
}
2018-08-04 13:03:06 +02:00
public boolean wasVisited(TransportRouteSegment rrs) {
if (rrs.road.getId().longValue() == road.getId().longValue()) {
return true;
}
if(parentRoute != null) {
return parentRoute.wasVisited(rrs);
}
return false;
}
public TransportStop getStop(int i) {
return road.getForwardStops().get(i);
}
public LatLon getLocation() {
return road.getForwardStops().get(segStart).getLocation();
}
public int getLength() {
return road.getForwardStops().size();
}
public long getId() {
long l = road.getId() << SHIFT;
if(l < 0 ) {
throw new IllegalStateException("too long id " + road.getId());
}
if(segStart >= (1 << SHIFT)) {
throw new IllegalStateException("too many stops " + road.getId() + " " + segStart);
}
return l + segStart;
}
public int getDepth() {
if(parentRoute != null) {
return parentRoute.getDepth() + 1;
}
return 1;
}
@Override
public String toString() {
2018-08-04 13:03:06 +02:00
return String.format("Route: %s, stop: %s", road.getName(), road.getForwardStops().get(segStart).getName());
}
}
public static class TransportRoutingContext {
public RouteCalculationProgress calculationProgress;
2018-08-04 13:03:06 +02:00
public TLongObjectHashMap<TransportRouteSegment> visitedSegments = new TLongObjectHashMap<TransportRouteSegment>();
2018-08-05 23:46:01 +02:00
public TransportRoutingConfiguration cfg;
2018-08-04 13:03:06 +02:00
2018-08-05 23:46:01 +02:00
public TLongObjectHashMap<List<TransportRouteSegment>> quadTree;
2018-08-15 14:43:47 +02:00
public final Map<BinaryMapIndexReader, TIntObjectHashMap<TransportRoute>> routeMap =
new LinkedHashMap<BinaryMapIndexReader, TIntObjectHashMap<TransportRoute>>();
2018-08-06 00:01:25 +02:00
// stats
public long startCalcTime;
public int visitedRoutesCount;
public int wrongLoadedWays;
public int loadedWays;
public long loadTime;
public long readTime;
2018-08-04 13:03:06 +02:00
private final int walkRadiusIn31;
private final int walkChangeRadiusIn31;
2018-08-04 13:03:06 +02:00
public TransportRoutingContext(TransportRoutingConfiguration cfg, BinaryMapIndexReader... readers) {
this.cfg = cfg;
walkRadiusIn31 = (int) (cfg.walkRadius / MapUtils.getTileDistanceWidth(31));
walkChangeRadiusIn31 = (int) (cfg.walkChangeRadius / MapUtils.getTileDistanceWidth(31));
2018-08-05 23:46:01 +02:00
quadTree = new TLongObjectHashMap<List<TransportRouteSegment>>();
for (BinaryMapIndexReader r : readers) {
2018-08-15 14:43:47 +02:00
routeMap.put(r, new TIntObjectHashMap<TransportRoute>());
}
}
2018-08-04 13:03:06 +02:00
public List<TransportRouteSegment> getTransportStops(LatLon loc) throws IOException {
int y = MapUtils.get31TileNumberY(loc.getLatitude());
int x = MapUtils.get31TileNumberX(loc.getLongitude());
2018-08-05 23:46:01 +02:00
return getTransportStops(x, y, false, new ArrayList<TransportRouteSegment>());
}
2018-08-05 23:46:01 +02:00
public List<TransportRouteSegment> getTransportStops(int x, int y, boolean change, List<TransportRouteSegment> res) throws IOException {
return loadNativeTransportStops(x, y, change, res);
}
2018-08-05 23:46:01 +02:00
private List<TransportRouteSegment> loadNativeTransportStops(int sx, int sy, boolean change, List<TransportRouteSegment> res) throws IOException {
2018-08-04 13:03:06 +02:00
long nanoTime = System.nanoTime();
int d = change ? walkChangeRadiusIn31 : walkRadiusIn31;
int lx = (sx - d ) >> (31 - cfg.ZOOM_TO_LOAD_TILES);
int rx = (sx + d ) >> (31 - cfg.ZOOM_TO_LOAD_TILES);
int ty = (sy - d ) >> (31 - cfg.ZOOM_TO_LOAD_TILES);
int by = (sy + d ) >> (31 - cfg.ZOOM_TO_LOAD_TILES);
for(int x = lx; x <= rx; x++) {
for(int y = ty; y <= by; y++) {
2018-08-05 23:46:01 +02:00
int tileId = x << (cfg.ZOOM_TO_LOAD_TILES + 1) + y;
List<TransportRouteSegment> list = quadTree.get(tileId);
if(list == null) {
list = loadTile(x, y);
quadTree.put(tileId, list);
}
for(TransportRouteSegment r : list) {
TransportStop st = r.getStop(r.segStart);
if (Math.abs(st.x31 - sx) > walkRadiusIn31 || Math.abs(st.y31 - sy) > walkRadiusIn31) {
wrongLoadedWays++;
} else {
loadedWays++;
res.add(r);
}
}
2018-08-04 13:03:06 +02:00
}
}
loadTime += System.nanoTime() - nanoTime;
2018-08-05 23:46:01 +02:00
return res;
}
2018-08-05 23:46:01 +02:00
private List<TransportRouteSegment> loadTile(int x, int y) throws IOException {
2018-08-06 00:01:25 +02:00
long nanoTime = System.nanoTime();
2018-08-05 23:46:01 +02:00
List<TransportRouteSegment> lst = new ArrayList<TransportRouteSegment>();
2018-08-04 13:03:06 +02:00
int pz = (31 - cfg.ZOOM_TO_LOAD_TILES);
SearchRequest<TransportStop> sr = BinaryMapIndexReader.buildSearchTransportRequest(x << pz, (x + 1) << pz,
y << pz, (y + 1) << pz, -1, null);
TIntArrayList allPoints = new TIntArrayList();
2018-08-15 14:43:47 +02:00
TIntArrayList allPointsLoad = new TIntArrayList();
2018-08-05 23:46:01 +02:00
// should it be global?
TLongObjectHashMap<TransportStop> loadedTransportStops = new TLongObjectHashMap<TransportStop>();
2018-08-15 14:43:47 +02:00
for(BinaryMapIndexReader r : routeMap.keySet()) {
sr.clearSearchResults();
2018-08-15 14:43:47 +02:00
allPoints.clear();
allPointsLoad.clear();
List<TransportStop> stops = r.searchTransportIndex(sr);
for(TransportStop s : stops) {
2018-08-04 13:03:06 +02:00
if(!loadedTransportStops.contains(s.getId())) {
loadedTransportStops.put(s.getId(), s);
allPoints.addAll(s.getReferencesToRoutes());
}
}
2018-08-15 14:43:47 +02:00
if(allPoints.size() > 0) {
allPoints.sort();
TIntObjectHashMap<TransportRoute> loadedRoutes = routeMap.get(r);
TIntObjectHashMap<TransportRoute> routes = new TIntObjectHashMap<TransportRoute>();
TIntIterator it = allPoints.iterator();
int p = allPoints.get(0) + 1; // different
while(it.hasNext()) {
int nxt = it.next();
if (p != nxt) {
if (loadedRoutes.contains(nxt)) {
routes.put(nxt, loadedRoutes.get(nxt));
} else {
allPointsLoad.add(nxt);
}
}
}
r.loadTransportRoutes(allPointsLoad.toArray(), routes);
loadedRoutes.putAll(routes);
loadTransportSegments(routes, r, stops, lst);
}
}
2018-08-06 00:01:25 +02:00
readTime += System.nanoTime() - nanoTime;
2018-08-05 23:46:01 +02:00
return lst;
}
2018-08-15 14:43:47 +02:00
private void loadTransportSegments(TIntObjectHashMap<TransportRoute> routes, BinaryMapIndexReader r,
2018-08-05 23:46:01 +02:00
List<TransportStop> stops, List<TransportRouteSegment> lst) throws IOException {
for(TransportStop s : stops) {
2018-08-05 23:46:01 +02:00
for (int ref : s.getReferencesToRoutes()) {
TransportRoute route = routes.get(ref);
2018-08-05 23:46:01 +02:00
if (route != null) {
int stopIndex = -1;
2018-08-05 23:46:01 +02:00
double dist = TransportRoute.SAME_STOP;
for (int k = 0; k < route.getForwardStops().size(); k++) {
TransportStop st = route.getForwardStops().get(k);
2018-08-05 23:46:01 +02:00
double d = MapUtils.getDistance(st.getLocation(), s.getLocation());
if (d < dist) {
stopIndex = k;
2018-08-05 23:46:01 +02:00
dist = d;
}
}
2018-08-05 23:46:01 +02:00
if (stopIndex != -1) {
2018-08-15 14:43:47 +02:00
if(cfg.useSchedule) {
if(route.getSchedule() != null) {
TIntArrayList ti = route.getSchedule().tripIntervals;
int cnt = ti.size();
int t = 0;
for(int i = 0; i < cnt; i++) {
t += ti.getQuick(i);
if(t >= cfg.scheduleTimeOfDay) {
TransportRouteSegment segment = new TransportRouteSegment(route, stopIndex, t);
lst.add(segment);
break;
}
}
}
} else {
TransportRouteSegment segment = new TransportRouteSegment(route, stopIndex);
lst.add(segment);
}
2018-08-05 23:46:01 +02:00
} else {
2018-08-15 14:43:47 +02:00
System.err.println("Routing error: missing stop in route");
}
2018-08-05 23:46:01 +02:00
}
}
}
}
2018-08-15 14:43:47 +02:00
}
}