双十字路口交通仿真程序(VS2010+MFC)
这个程序是我上研二上学期时下一届师弟师妹们的面向对象课程大作业,当时我正好看过两三本 C++ 书籍,虽然忙着项目,但还是忙里偷闲检验了下自己。从设计到实现,耗时一周左右,完成于 2013 年年底。
虽是面向对象课程作业,但是只用到了“封装”,没有触及“继承”和“多态”,只是基于对象而已。
以前的做一个东西都会查阅不少资料,这次纯粹是靠想象了。
源码托管在 Github 上:点击进入链接,命名遵循“匈牙利命名法”尽量做到“self-documenting”,当然也有必要的注释。
可执行程序:点击进入链接
题目(完整pdf):



设计说明:
红绿灯类,保存灯的颜色、总持续时间、当前倒计时等,操作倒计时和颜色转换;
用一个枚举类型保存道路端点;
路径类,用一个 map 保存所有路径端点及其对应的折点,可通过路线端点获取其对应的路线折点;
车辆类,随机产生车辆时,就随机设置其颜色和行驶路线,类中保存车辆行驶路线端点、当前行驶区间、当前位置,可获取车辆状态或更新位置。
道路类,这是一个二维矩阵,安插了多个红绿灯,每一点都有一个标记,如果有车辆在此位置则激活。
仿真图示:定时扫描道路类中的所有点,获取对应的交通灯或车辆,然后绘制到界面上。
坐标假设:原点在左上角,向右为 x 轴正方向,向下为 y 轴正方向。
重难点:
1、红绿灯更替规则,封装到红绿灯类内从而得到简化;
2、复杂的转弯、让道规则,通过保存路线解决;
3、更新 1 秒:
// 车辆行进 1 秒
void CTrafficSimulationDlg::MoveOneSec(CMyPoint& ptCarPos)
{
for (auto it = m_vecCars.begin(); it != m_vecCars.end(); ++it) {
if (it->GetPos() == ptCarPos) { // 在车辆集合中找到该车辆
pair<CMyPoint, CMyPoint> pairCurInterval = it->GetCurInterval();
CMyPoint ptFrom = pairCurInterval.first;
CMyPoint ptTo = pairCurInterval.second; if (ptTo.GetX() == ptCarPos.GetX()
&& ptTo.GetY() == ptCarPos.GetY()) { // 行驶到一个区间的终点 vector<CMyPoint>& vecTurningPoints
= m_paths.GetTurningPoints(it->GetPath()); if (it->m_nPtIdx < vecTurningPoints.size()) { // 更新区间
ptFrom = ptTo;
ptTo = vecTurningPoints.at(it->m_nPtIdx++);
it->SetCurInterval(ptFrom, ptTo);
}
else { // 行驶到路线的终点,在车辆集合中删除该车
m_vecCars.erase(it);
m_roadMain.DeactivatePoint(ptCarPos);
return ;
}
} CMyPoint ptNewCarPos;
if (ptFrom.GetX() != ptTo.GetX()) { // 沿着 x 轴行驶
int n = ptTo.GetX() - ptFrom.GetX() > ? +: -;
ptNewCarPos = CMyPoint(ptCarPos.GetX()/gc_nScalar+n, ptCarPos.GetY()/gc_nScalar);
}
else { // 沿着 y 轴行驶
int n = ptTo.GetY() - ptFrom.GetY() > ? +: -;
ptNewCarPos = CMyPoint(ptCarPos.GetX()/gc_nScalar, ptCarPos.GetY()/gc_nScalar+n);
} bool bGreen = true;
if ( ptNewCarPos != ptTo // 如果不转弯,则查看红绿灯
&& m_setLightPos.find(ptCarPos) != m_setLightPos.end() ) // 遇到红绿灯
bGreen = m_roadMain.IsGreen(ptCarPos); // 车辆行进 1 秒
if ( bGreen && !m_roadMain.IsPointActivated(ptNewCarPos) ) {
m_roadMain.DeactivatePoint(ptCarPos);
m_roadMain.ActivatePoint(ptNewCarPos);
it->SetPos(ptNewCarPos);
} return ; // 更新完毕
}
}
}
是否要更换行驶区间(当前车辆位置是否与区间终点重合)、是否行驶到尽头(删除车辆);
判断当前区间是水平的(区间 x 轴坐标不相等)还是竖直的(区间 y 轴坐标不相等),然后决定沿着 x 或 y 的正方向还是反方向;
通过判断是否在路口转弯(新车辆位置与区间终点重合),从而决定是否忽略红绿灯。
运行截图:

双十字路口交通仿真程序(VS2010+MFC)的更多相关文章
- VS2010/MFC编程入门之五(MFC消息映射机制概述)
VS2010/MFC编程入门之五(MFC消息映射机制概述)-软件开发-鸡啄米 http://www.jizhuomi.com/software/147.html 上一讲鸡啄米为大家简单分析了MFC应用 ...
- VS2010/MFC编程入门之四(MFC应用程序框架分析)
VS2010/MFC编程入门之四(MFC应用程序框架分析)-软件开发-鸡啄米 http://www.jizhuomi.com/software/145.html 上一讲鸡啄米讲的是VS2010应用 ...
- VS2010/MFC编程入门之三(VS2010应用程序工程中文件的组成结构)
VS2010/MFC编程入门之三(VS2010应用程序工程中文件的组成结构)-软件开发-鸡啄米 http://www.jizhuomi.com/software/143.html 鸡啄米在上一讲中 ...
- VS2010/MFC编程入门之二(利用MFC向导生成单文档应用程序框架)
VS2010/MFC编程入门之二(利用MFC向导生成单文档应用程序框架)-软件开发-鸡啄米 http://www.jizhuomi.com/software/141.html 上一讲中讲了VS20 ...
- VS2010/MFC编程入门之一(VS2010与MSDN安装过程图解)
原文地址: VS2010/MFC编程入门之一(VS2010与MSDN安装过程图解)-软件开发-鸡啄米 http://www.jizhuomi.com/software/139.html 上一讲中鸡 ...
- VS2010/MFC编程入门教程之目录和总结
鸡啄米的这套VS2010/MFC编程入门教程到此就全部完成了,虽然有些内容还未涉及到,但帮助大家进行VS2010/MFC的入门学习业已足够.以此教程的知识为基础,学习VS2010/MFC较为深入的内容 ...
- 《VS2010/MFC编程入门教程》——读书笔记
推荐两个比较好的学习网站:http://v.dxsbb.com/jisuanji/555/ http://www.jizhuomi.com/software/257.html MFC全称Microso ...
- VS2010/MFC编程入门之五十二(Ribbon界面开发:创建Ribbon样式的应用程序框架)
上一节中鸡啄米讲了GDI对象之画刷CBrush,至此图形图像的入门知识就讲完了.从本节开始鸡啄米将为大家带来Ribbon界面开发的有关内容.本文先来说说如何创建Ribbon样式的应用程序框架. Rib ...
- VS2010/MFC编程入门之五十(图形图像:GDI对象之画笔CPen)
上一节中鸡啄米讲了CDC类及其屏幕绘图函数,本节的主要内容是GDI对象之画笔CPen. GDI对象 在MFC中,CGdiObject类是GDI对象的基类,通过查阅MSDN我们可以看到,CGdiObje ...
随机推荐
- 使用Sentry集中化日志管理
在调试程序中,通过日志分期来排查BUG是一个重要手段,它可以说是程序调试的利器. 关于日志管理 随着应用组件变多,那么各coder对输出日志五花八门,有写入stdout,有写stderr, 有写到sy ...
- Asp.Net MVC anti-forgery token的问题:nameidentifier or identityprovider not present
当使用ClaimsIdentity的时候,Asp.Net MVC在生成AntiForgeryToken的时候会默认使用User.Identity中两种ClaimsType的值:NameIdentifi ...
- 如何修改本地hosts文件?
1.window7修改本地hosts文件 # window7系统hosts文件位置 C:\Windows\System32\drivers\etc 2.linux # linux系统hosts文件位置 ...
- MySQL 第四天
回顾 列属性: 主键, 自增长, 唯一键 关系: 一对一,一对多和多对多 范式: 三层范式 1NF: 字段设计必须符合原子性 2NF: 不存在部分依赖(没有复合主键) 3NF: 不存在传递依赖 ...
- PAT 1067. 试密码(20)
当你试图登录某个系统却忘了密码时,系统一般只会允许你尝试有限多次,当超出允许次数时,账号就会被锁死.本题就请你实现这个小功能. 输入格式: 输入在第一行给出一个密码(长度不超过20的.不包含空格.Ta ...
- go语言之并发编程同步一
前面介绍了采用go语法的并行操作以及channel.既然是并行操作,那么就涉及到数据原子性以及同步的问题.所以在Go里面也需要采用同步的机制. 互斥锁: 由标准库代码包sync中的Mutex结构体类型 ...
- (4.14)存储:RAID在数据库存储上的应用
关键词:(4.14)存储:RAID在数据库存储上的应用 转自:http://blog.51cto.com/qianzhang/1251260 随着单块磁盘在数据安全.性能.容量上呈现出的局限,磁盘阵列 ...
- tomcat8.5.11的manager页面总是提示403的问题
修改conf/tomcat-users.xml加入: <role rolename="manager"/> <role rolename="manage ...
- Java技术相关
1.System.getProperty("user.dir") References:API
- Linux:Ubuntu下部署Web运行环境
Linux:Ubuntu下部署Web运行环境 本次博客将会从三部分内容详述Ubuntu系统下Web运行环境的配置: 依次是:FTP服务器的搭建.MYSQL数据库的搭建.JDK的安装等. 参考文章如下: ...