北京地铁出行线路规划系统项目总结(Java+Flask+Vue实现)
北京地铁出行线路规划系统项目总结
GitHub仓库地址:https://github.com/KeadinZhou/SE-Subway
Demo地址:http://10.66.2.161:8080/ (校内网)
项目需求
- 实现一个帮助进行地铁出行路线规划的命令行程序
- 地铁线路图数据需要与执行程序解耦
- 支持查询单条线路的所有站点
- 支持查询任意两站之间通过最少站数的路线
算法设计
项目中最主要的点在于:找出两个站点之间通过最少站数的路线。该点对应的经典模型是“在一个无向图中找出两点之间的最短路径”。各个站点即为无向图中的点,两站间的路径即为无向图中连接两个节点的边,在此需求下,所有边的长度均为1。
地铁最短路的换乘,最贴近实际的情况是“一次初始化,多次查询”,即在地图确定下来之后可能会有若干次不同点对的查询。而本需求中地铁线路图数据需要与执行程序解耦,每次查询对于地图数据都可能是不同的,所以初始化的时间应该均摊到每次查询中去。由于地铁地图属于稀疏图,且该图中每条边权恒定,故可以直接使用BFS算法解决。
细节上,为了解决两条线路覆盖的情况下(见下图),连续换乘的问题,需要更好的换乘算法。

解决方法:通过判断站点和前继站点的所在线路是否有重叠来判断是否需要换乘。
存储设计
数据存储上,可以采用存点/存边两种形式。在本项目中,存边会明显优于存点。
- 简洁易懂。数据中每个条目表示两个站点中的一条通路。
- 易于扩展。添加站点方便;数据不需要有序。
- 方便应用程度读取。数据中每个条目即为图中的一条边,方便建图。
具体到内容,整个数据文件可以被几条线路分成几个大的模块,每个模块存一条线路的数据,由于地铁站名不存在同名的情况,可以不存线路换乘站点的信息,通过同名同名判断即可。由于同一条边不可能同时属于两条线路,所以在边信息上不会存在冗余。每个模块由一个星号(*)开始,后面这个这条线的名称。接下来包含若干条边信息。每条边信息的格式为起点站名称 终点站名称。每条线的起点可以由建图后入度为 0 的点确定。
样例如下:
* 1号线
刘院 西横堤
果酒场 本溪路
...(省略)
咸水沽北 双桥河
* 9号线
天津站 大王庄
大王庄 十一经路
...(省略)
架构设计
系统整体的计算核心使用Java实现,通过命令行可以与之进行数据交互,通过指定地图数据和相关查询指令,可以实现所有的需求。同时,该核心可以通过外加另外的展示模块来实现人机交互的可视化。外加的数据显示模块采用B/S架构,后端使用Python/Flask实现,前端使用Node.js/Vue.js实现,前后端分离,两者使用接口进行数据交互。
实现细节
计算核心
数据初始化
private static void dataInit(String pathname) {
try(FileReader reader = new FileReader(pathname); BufferedReader br = new BufferedReader(reader) ) {
String line;
String lineName="";
boolean first=false;
while((line=br.readLine()) != null) {
String[] data = line.split(" ");
if(data[0].contains("*")){
lineName=data[1];
first=true;
continue;
}
int id1=getStationId(data[0]);
int id2=getStationId(data[1]);
if(first){
lineStart.put(lineName, id1);
first=false;
stations.get(id1).addLines(lineName);
}
stations.get(id2).addLines(lineName);
addEdge(id1, id2, lineName);
}
Station.count=stationCnt;
} catch (Exception e) {
e.printStackTrace();
}
}
通过约定的数据存储格式进行数据初始化,通过 * 星号来区别不同线路的数据。每条数据表示地铁图上的一条边,通过记录边数据和对应的线路名称来建立地铁图。
换乘检测
private static String checkSwitchLine(int id1, int id2, int id3){
String linea=edgeLines.get(getEdgeHash(id1,id2));
String lineb=edgeLines.get(getEdgeHash(id2,id3));
if(linea.equals(lineb)) return null;
return lineb;
}
通过检测相邻两条表是否同属于同一条线路来判断是否需要换乘。
路径查询
private static void getShortestPath(String start, String end) throws IOException {
int startID=getStationId(start);
if(startID>Station.count){
println("错误:没有 " + start + " 这个站");
return;
}
int endID=getStationId(end);
if(endID>Station.count){
println("错误:没有 " + end + " 这个站");
return;
}
Queue<Integer>q=new LinkedList<>();
q.offer(startID);
stations.get(startID).setVis(true);
while(!q.isEmpty()){
Station now=stations.get(q.poll());
for(int it:now.getEdges()){
Station tmp=stations.get(it);
if(tmp.isVis()) continue;
q.offer(it);
tmp.setVis(true);
tmp.setLastPoint(now.getId());
if(it==endID) break;
}
}
Stack<Integer> path=new Stack<>();
int tip=endID;
while(tip!=startID){
path.push(tip);
tip=stations.get(tip).getLastPoint();
}
ArrayList<Integer> res=new ArrayList<>();
res.add(startID);
while(!path.empty()){
int now=path.pop();
res.add(now);
}
println(res.size());
int tmpa=-1,tmpb=-1;
for(int it:res){
if(tmpa!=-1){
String switchMsg=checkSwitchLine(tmpa,tmpb,it);
if(switchMsg!=null){
println(switchMsg);
}
}
println(stations.get(it).getName());
tmpa=tmpb;
tmpb=it;
}
}
通过BFS算法进行最短路的寻路。找到路径之后检测路径上的所有站点是否需要换乘。
后端接口
后端接口由Python/Flask实现。Flask是一个轻量级的web框架,在接口类后端应用上有很大的优势。
全局数据接口
@app.route('/data', methods=['GET'])
def data():
lines = []
sites = set()
file = open('subway.txt', 'r')
for item in file:
ll = item.strip().split()
if ll[0].find('*') != -1:
lines.append(ll[1])
else:
sites.add(ll[0])
sites.add(ll[1])
return jsonify({
'lines': lines,
'sites': list(sites)
})
该接口用于查询地铁图中出现的所有线路和所有站点的列表,通过直接读取数据文件实现。
查询接口
GET_ALL_CMD = 'java subway -map subway.txt -a %s -o out.txt'
GET_PATH_CMD = 'java subway -map subway.txt -b %s %s -o out.txt'
@app.route('/query', methods=['POST'])
def query():
json_data = request.get_json()
all = json_data['isAll']
if all:
os.system(GET_ALL_CMD % json_data['query'])
else:
os.system(GET_PATH_CMD % (json_data['query'][0], json_data['query'][1]))
file = open('out.txt', 'r')
res = []
for item in file:
res.append(item.strip())
return jsonify({
'res': res
})
该接口通过前端提交的数据,选择查询线路和查询路径对应的计算核心命令,来使用计算核心来获取线路数据。
前端页面
前端页面是用户和程序交互的非常重要的手段。本项目的前端页面使用Vue.js配合ElementUI实现。
运行效果
计算核心

通过-a命令显示一条线上所有的站点。

通过-b命令查询两站之间的最短路径。

通过-o命令指定结果输出的文件。

计算核心的健壮性:不合法命令的检测。
WEB展示前端

查询主界面。(路径查询)

当查询时,输入关键字,系统将会自动匹配补全相关的站点供用户选择。

查询的结果。显示每一站的站名,并在需要换乘的站点后面标注换乘的线路。

线路站点查询,可以查询每条线路所包含的所有站点。

线路查询结果。
北京地铁出行线路规划系统项目总结(Java+Flask+Vue实现)的更多相关文章
- canves绘制北京地铁线路图,包括线路绘制,优先路线,单路径选择。
canves绘制北京地铁线路图,包括线路绘制,优先路线,单路径选择. 即将推出,后台涵盖各种语言,php,C#,java,nodejs等.
- Dijkstra算法_北京地铁换乘_android实现-附带源码.apk
Dijkstra算法_北京地铁换乘_android实现 android 2.2+ 源码下载 apk下载 直接上图片 如下: Dijkstra(迪杰斯特拉)算法是典型的最短路径路由算法,用于计 ...
- [转载]将别人的项目或JAVA文件导入到自己的Eclipse中时,常常会出现JAVA文件的中文注释变成乱码的情况,解决办法
eclipse 代码中文注释乱码 求解决 将别人的项目或JAVA文件导入到自己的Eclipse中时,常常会出现JAVA文件的中文注释变成乱码的情况,主要原因就是别人的IDE编码格式和自己的Eclips ...
- Java web项目引用java项目,类型找不到
Java web项目引用java项目,类型找不到 错误信息: java.lang.ClassNotFoundException: org.codehaus.jackson.map.ObjectMapp ...
- 北京地铁月度消费总金额计算(Python版)
最近业余时间在学习Python,这是那天坐地铁时突发奇想,想看看我这一个月的地铁费共多少钱,所以简单的构思了下思路,就直接开写了,没想到用Python来实现还挺简单的. 设计思路: 每次乘车正常消费7 ...
- WPF简易北京地铁效果图
这个是百度地图上北京地铁的地址http://map.baidu.com/?subwayShareId=beijing,131,我们先看下百度上面的效果图 我要实现的内容比较简单,就是绘制这些图,和在地 ...
- ubuntu下eclipse新建项目没有java project的解决办法
装好了eclipse之后却发现新建项目没有java project的选项,大致搜索了一下,并没有发现很好的解决方案(大都是让你重新安装什么的),于是开始瞎鼓捣,并且找到了一个方案: 在终端切换到roo ...
- Maven项目中java类报错-Cannot resolve symbol
电脑蓝屏了,强制重启之后再打开IDEA里面的项目,所有Java类文件都在报Cannot resolve symbo错误,可以确定所有依赖的包都有引用且jar包没有冲突. 经查询找到这个解决方法: 在I ...
- 【MyEcplise】导入项目报错:Errors running builder 'JavaScript Validator' on project '项目名'. java.lang.ClassCastException
导入项目报错:Errors running builder 'JavaScript Validator' on project '项目名'. java.lang.ClassCastException ...
随机推荐
- Saltstack_使用指南11_配置管理-状态之间依赖关系
1. 说明 下文的案例是根据上一篇文章进行的修改.因此请优先读取上一篇文章内容<Saltstack_10_配置管理-状态模块> 并且目录进行了变化,从 /srv/salt/lamp 变为了 ...
- [20190523]修改参数后一些细节注意.txt
[20190523]修改参数后一些细节注意.txt --//昨天远程给别人解决一个小问题,就是配置使用hugepage.一些细节必须注意,通过例子说明问题. 1.环境:# cat /proc/vers ...
- bayaim_当前国内外Hadoop的实际使用案例
当前国内外Hadoop的实际使用案例 说到Hadoop,很多朋友都想知道到底它是如何被应用的呢?通过下文,笔者就向大家介绍一下具体的情况. 1.Yahoo Yahoo是Hadoop的最大支持者,截至2 ...
- [日常] 前端资源测试机上忽略版本号的的nginx配置
利用nginx的rewrite的指令,可以实现url的重新跳转,rewrtie有四种不同的flag,分别是redirect(临时重定向).permanent(永久重定向).break和last.其中前 ...
- react界面跳转,滚动到顶部
在使用react-router-dom时,我们经常会遇到路由切换时滚动到浏览器顶部的问题. 滚动到顶部 Scroll to top 很多时候我们需要的是滚动到顶部“Scroll to top”,因为发 ...
- 7.Java基础_Java数据输入
import java.util.Scanner; public class Output { public static void main(String[] args){ Scanner sc=n ...
- GitHub密钥生成
前提电脑上需装有Git软件 这里提供百度云下载地址:https://pan.baidu.com/s/1r0y4XRyQCz7ZJBnZJhAtqw 提取码:88qf 1.登录GitHub账号 2.点 ...
- Requests 详解
什么是Requests Requests是用Python语言编写,基于urllib,他比urllib更加方便,可以节约我们的大量工作,完全满足HTTP测试需求
- 面向对象程序设计(Java) 第7周学习指导及要求
2019面向对象程序设计(Java)第7周学习指导及要求 (2019.10.11-2019.10.14) 学习目标 掌握四种访问权限修饰符的使用特点: 掌握Object类的用途及常用API: 掌握Ar ...
- Pwn-level3
题目地址 https://dn.jarvisoj.com/challengefiles/level3.rar.2047525b05c499c9dd189ba212bba1f8 借鉴 https://w ...