需求
能够满足旅客查询从任一城市到另一城市的最短里程 最低花费 最短时间 最少中转次数等问题
需要界面 mfc
部分代码如下:
// xmSearchDlg.cpp: 实现文件
//
#include "pch.h"
#include "framework.h"
#include "xmSearch.h"
#include "xmSearchDlg.h"
#include "afxdialogex.h"
#include <string>
#include <locale.h>
#ifdef _DEBUG
#define new DEBUG_NEW
#endif
int city_from;
int city_to;
int search;
#define INF 10000
using namespace std;
string station_name[7] = { "北京", "西安", "郑州", "徐州", "成都", "广州", "上海" };
double distan[7][7] = {
0, 2553, 695, 704, INF, INF, INF,
2553, 0, 511, INF, 812, INF, INF,
695, 511, 0, 349, INF, 1579, INF,
704, INF, 349, 0, INF, INF, 651,
INF, 812, INF, INF, 0, 2368, INF,
INF, INF, 1579, INF, 2368, 0, 1385,
INF, INF, INF, 651, INF, 1385, 0,
};
double hour[7][7] = {
0, 8, 2.3, 2.5, INF, INF, INF,
8, 0, 1.5, INF, 3, INF, INF,
2.3, 1.5, 0, 1.2, INF, 5, INF,
2.5, INF, 1.2, 0, INF, INF, 2,
INF, 3, INF, INF, 0, 7, INF,
INF, INF, 5, INF, 7, 0, 4,
INF, INF, INF, 2, INF, 4, 0,
};
double money[7][7] = {
0, 885, 202, 225, INF, INF, INF,
885, 0, 148, INF, 283, INF, INF,
202, 148, 0, 112, INF, 495, INF,
225, INF, 112, 0, INF, INF, 162,
INF, 283, INF, INF, 0, 684, INF,
INF, INF, 495, INF, 684, 0, 386,
INF, INF, INF, 162, INF, 386, 0,
};
double access[7][7] = {
0, 1, 1, 1, INF, INF, INF,
1, 0, 1, INF, 1, INF, INF,
1, 1, 0, 1, INF, 1, INF,
1, INF, 1, 0, INF, INF, 1,
INF, 1, INF, INF, 0, 1, INF,
INF, INF, 1, INF, 1, 0, 1,
INF, INF, INF, 1, INF, 1, 0,
};
double P[10], D[10];
void Dijkstra(double(*Edge)[7], int v0) {
//初始化
double Final[10] = { 0 };
for (int i = 0; i < 7; i++) {
P[i] = -1;
D[i] = Edge[v0][i];
}
//加入结点V0
D[v0] = 0;
Final[v0] = 1;
int k = v0;
for (int i = 0; i < 7; i++) {
double MIN = INF; //找到目前数组中权值最小的一条边
for (int j = 0; j < 7; j++) {
if (Final[j] == 0 && D[j] < MIN) {
MIN = D[j];
k = j;
}
}
Final[k] = 1; //标记该结点已经加入最短路径中
for (int w = 0; w < 7; w++) {
if (Final[w] == 0 && (MIN + Edge[k][w] < D[w])) {
D[w] = MIN + Edge[k][w]; //更新最小值
P[w] = k; //记录路径(说明最短路径中到w的要经过k)
}
}
}
}
// 用于应用程序“关于”菜单项的 CAboutDlg 对话框
class CAboutDlg : public CDialogEx
{
public:
CAboutDlg();
// 对话框数据
#ifdef AFX_DESIGN_TIME
enum { IDD = IDD_ABOUTBOX };
#endif
protected:
virtual void DoDataExchange(CDataExchange* pDX); // DDX/DDV 支持
// 实现
protected:
DECLARE_MESSAGE_MAP()
};
CAboutDlg::CAboutDlg() : CDialogEx(IDD_ABOUTBOX)
{
}
void CAboutDlg::DoDataExchange(CDataExchange* pDX)
{
CDialogEx::DoDataExchange(pDX);
}
BEGIN_MESSAGE_MAP(CAboutDlg, CDialogEx)
END_MESSAGE_MAP()
// CxmSearchDlg 对话框
CxmSearchDlg::CxmSearchDlg(CWnd* pParent /*=nullptr*/)
: CDialogEx(IDD_XMSEARCH_DIALOG, pParent)
{
m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
}
void CxmSearchDlg::DoDataExchange(CDataExchange* pDX)
{
CDialogEx::DoDataExchange(pDX);
DDX_Control(pDX, IDC_COMBO1, m_cbx1);
DDX_Control(pDX, IDC_COMBO2, m_cbx2);
DDX_Control(pDX, IDC_COMBO3, m_cbx3);
DDX_Control(pDX, IDC_EDIT1, c_edit);
}
BEGIN_MESSAGE_MAP(CxmSearchDlg, CDialogEx)
ON_WM_SYSCOMMAND()
ON_WM_PAINT()
ON_WM_QUERYDRAGICON()
ON_CBN_SELCHANGE(IDC_COMBO1, &CxmSearchDlg::OnCbnSelchangeCombo1)
ON_BN_CLICKED(IDC_BUTTON1, &CxmSearchDlg::OnBnClickedButton1)
ON_CBN_SELCHANGE(IDC_COMBO2, &CxmSearchDlg::OnCbnSelchangeCombo2)
ON_CBN_SELCHANGE(IDC_COMBO3, &CxmSearchDlg::OnCbnSelchangeCombo3)
ON_EN_CHANGE(IDC_EDIT1, &CxmSearchDlg::OnEnChangeEdit1)
END_MESSAGE_MAP()
// CxmSearchDlg 消息处理程序
BOOL CxmSearchDlg::OnInitDialog()
{
CDialogEx::OnInitDialog();
// 将“关于...”菜单项添加到系统菜单中。
// IDM_ABOUTBOX 必须在系统命令范围内。
ASSERT((IDM_ABOUTBOX & 0xFFF0) == IDM_ABOUTBOX);
ASSERT(IDM_ABOUTBOX < 0xF000);
CMenu* pSysMenu = GetSystemMenu(FALSE);
if (pSysMenu != nullptr)
{
BOOL bNameValid;
CString strAboutMenu;
bNameValid = strAboutMenu.LoadString(IDS_ABOUTBOX);
ASSERT(bNameValid);
if (!strAboutMenu.IsEmpty())
{
pSysMenu->AppendMenu(MF_SEPARATOR);
pSysMenu->AppendMenu(MF_STRING, IDM_ABOUTBOX, strAboutMenu);
}
}
// 设置此对话框的图标。 当应用程序主窗口不是对话框时,框架将自动
// 执行此操作
SetIcon(m_hIcon, TRUE); // 设置大图标
SetIcon(m_hIcon, FALSE); // 设置小图标
// TODO: 在此添加额外的初始化代码
//
// 下拉框添加
// 设置默认选项
m_cbx1.SetCurSel(0);
m_cbx2.SetCurSel(0);
m_cbx3.SetCurSel(0);
// 编辑框内容
return TRUE; // 除非将焦点设置到控件,否则返回 TRUE
}
void CxmSearchDlg::OnSysCommand(UINT nID, LPARAM lParam)
{
if ((nID & 0xFFF0) == IDM_ABOUTBOX)
{
CAboutDlg dlgAbout;
dlgAbout.DoModal();
}
else
{
CDialogEx::OnSysCommand(nID, lParam);
}
}
// 如果向对话框添加最小化按钮,则需要下面的代码
// 来绘制该图标。 对于使用文档/视图模型的 MFC 应用程序,
// 这将由框架自动完成。
void CxmSearchDlg::OnPaint()
{
if (IsIconic())
{
CPaintDC dc(this); // 用于绘制的设备上下文
SendMessage(WM_ICONERASEBKGND, reinterpret_cast<WPARAM>(dc.GetSafeHdc()), 0);
// 使图标在工作区矩形中居中
int cxIcon = GetSystemMetrics(SM_CXICON);
int cyIcon = GetSystemMetrics(SM_CYICON);
CRect rect;
GetClientRect(&rect);
int x = (rect.Width() - cxIcon + 1) / 2;
int y = (rect.Height() - cyIcon + 1) / 2;
// 绘制图标
dc.DrawIcon(x, y, m_hIcon);
}
else
{
CDialogEx::OnPaint();
}
}
//当用户拖动最小化窗口时系统调用此函数取得光标
//显示。
HCURSOR CxmSearchDlg::OnQueryDragIcon()
{
return static_cast<HCURSOR>(m_hIcon);
}
void CxmSearchDlg::OnCbnSelchangeCombo1()
{
// TODO: 在此添加控件通知处理程序代码
city_from = m_cbx1.GetCurSel();
}
void CxmSearchDlg::OnBnClickedButton1()
{
// TODO: 在此添加控件通知处理程序代码
if (search == 0) {
Dijkstra(distan, city_from);
}
else if (search == 1) {
Dijkstra(money, city_from);
}
else if (search == 2) {
Dijkstra(hour, city_from);
}
else {
Dijkstra(access, city_from);
}
CString str1;
string search_name[4] = { "最短里程", "最低花费", "最少时间","最少中转" };
string search_end[4] = { "公里", "元", "小时","次" };
for (int i = 0; i < 7; i++) {
if (i == city_to) {
str1.Format(TEXT("%s到%s, %s: \n"), CStringW(station_name[city_from].c_str()), CStringW(station_name[city_to].c_str()), CStringW(search_name[search].c_str()));
int j = i;
while (j != -1) {
str1.AppendFormat(TEXT("%s->"), CStringW(station_name[j].c_str()));
j = P[j];
}
str1.AppendFormat(TEXT("%s【%s为%.2f%s】\n"), CStringW(station_name[city_from].c_str()), CStringW(search_name[search].c_str()), D[i], CStringW(search_end[search].c_str()));
}
}
c_edit.SetWindowTextW(str1);
}
void CxmSearchDlg::OnCbnSelchangeCombo2()
{
// TODO: 在此添加控件通知处理程序代码
city_to = m_cbx2.GetCurSel();
}
void CxmSearchDlg::OnCbnSelchangeCombo3()
{
// TODO: 在此添加控件通知处理程序代码
search = m_cbx3.GetCurSel();
}
void CxmSearchDlg::OnEnChangeEdit1()
{
// TODO: 如果该控件是 RICHEDIT 控件,它将不
// 发送此通知,除非重写 CDialogEx::OnInitDialog()
// 函数并调用 CRichEditCtrl().SetEventMask(),
// 同时将 ENM_CHANGE 标志“或”运算到掩码中。
// TODO: 在此添加控件通知处理程序代码
}
运行截图如下: