NQueenDlg.cpp
上传用户:hncsjd
上传日期:2022-07-08
资源大小:3772k
文件大小:7k
源码类别:

其他智力游戏

开发平台:

Visual C++

  1. // NQueenDlg.cpp : implementation file
  2. //
  3. #include "stdafx.h"
  4. #include "NQueen.h"
  5. #include "NQueenDlg.h"
  6. #ifdef _DEBUG
  7. #define new DEBUG_NEW
  8. #undef THIS_FILE
  9. static char THIS_FILE[] = __FILE__;
  10. #endif
  11. /////////////////////////////////////////////////////////////////////////////
  12. // CAboutDlg dialog used for App About
  13. class CAboutDlg : public CDialog
  14. {
  15. public:
  16. CAboutDlg();
  17. // Dialog Data
  18. //{{AFX_DATA(CAboutDlg)
  19. enum { IDD = IDD_ABOUTBOX };
  20. //}}AFX_DATA
  21. // ClassWizard generated virtual function overrides
  22. //{{AFX_VIRTUAL(CAboutDlg)
  23. protected:
  24. virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV support
  25. //}}AFX_VIRTUAL
  26. // Implementation
  27. protected:
  28. //{{AFX_MSG(CAboutDlg)
  29. //}}AFX_MSG
  30. DECLARE_MESSAGE_MAP()
  31. };
  32. CAboutDlg::CAboutDlg() : CDialog(CAboutDlg::IDD)
  33. {
  34. //{{AFX_DATA_INIT(CAboutDlg)
  35. //}}AFX_DATA_INIT
  36. }
  37. void CAboutDlg::DoDataExchange(CDataExchange* pDX)
  38. {
  39. CDialog::DoDataExchange(pDX);
  40. //{{AFX_DATA_MAP(CAboutDlg)
  41. //}}AFX_DATA_MAP
  42. }
  43. BEGIN_MESSAGE_MAP(CAboutDlg, CDialog)
  44. //{{AFX_MSG_MAP(CAboutDlg)
  45. // No message handlers
  46. //}}AFX_MSG_MAP
  47. END_MESSAGE_MAP()
  48. /////////////////////////////////////////////////////////////////////////////
  49. // CNQueenDlg dialog
  50. CNQueenDlg::CNQueenDlg(CWnd* pParent /*=NULL*/)
  51. : CDialog(CNQueenDlg::IDD, pParent)
  52. {
  53. //{{AFX_DATA_INIT(CNQueenDlg)
  54. m_N = 8;
  55. //}}AFX_DATA_INIT
  56. // Note that LoadIcon does not require a subsequent DestroyIcon in Win32
  57. m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
  58. }
  59. void CNQueenDlg::DoDataExchange(CDataExchange* pDX)
  60. {
  61. CDialog::DoDataExchange(pDX);
  62. //{{AFX_DATA_MAP(CNQueenDlg)
  63. DDX_Control(pDX, IDC_QUEEN_PANEL, m_panel);
  64. DDX_Text(pDX, IDC_BOARD_SIZE, m_N);
  65. DDV_MinMaxLong(pDX, m_N, 1, 10);
  66. //}}AFX_DATA_MAP
  67. }
  68. BEGIN_MESSAGE_MAP(CNQueenDlg, CDialog)
  69. //{{AFX_MSG_MAP(CNQueenDlg)
  70. ON_WM_SYSCOMMAND()
  71. ON_WM_PAINT()
  72. ON_WM_QUERYDRAGICON()
  73. ON_BN_CLICKED(IDC_ABOUT, OnAbout)
  74. ON_BN_CLICKED(IDC_CHANGEN, OnChangeN)
  75. ON_BN_CLICKED(IDC_START, OnStart)
  76. ON_BN_CLICKED(IDC_NEXT, OnNext)
  77. ON_BN_CLICKED(IDC_STOP, OnStop)
  78. //}}AFX_MSG_MAP
  79. END_MESSAGE_MAP()
  80. /////////////////////////////////////////////////////////////////////////////
  81. // CNQueenDlg message handlers
  82. BOOL CNQueenDlg::OnInitDialog()
  83. {
  84. CDialog::OnInitDialog();
  85. // Add "About..." menu item to system menu.
  86. // IDM_ABOUTBOX must be in the system command range.
  87. ASSERT((IDM_ABOUTBOX & 0xFFF0) == IDM_ABOUTBOX);
  88. ASSERT(IDM_ABOUTBOX < 0xF000);
  89. CMenu* pSysMenu = GetSystemMenu(FALSE);
  90. if (pSysMenu != NULL)
  91. {
  92. CString strAboutMenu;
  93. strAboutMenu.LoadString(IDS_ABOUTBOX);
  94. if (!strAboutMenu.IsEmpty())
  95. {
  96. pSysMenu->AppendMenu(MF_SEPARATOR);
  97. pSysMenu->AppendMenu(MF_STRING, IDM_ABOUTBOX, strAboutMenu);
  98. }
  99. }
  100. // Set the icon for this dialog.  The framework does this automatically
  101. //  when the application's main window is not a dialog
  102. SetIcon(m_hIcon, TRUE); // Set big icon
  103. SetIcon(m_hIcon, FALSE); // Set small icon
  104. // TODO: Add extra initialization here
  105. m_N = 8;
  106. m_panel.SetSize(m_N);
  107. running = false;
  108. GetDlgItem(IDC_NEXT)->EnableWindow(false);
  109. GetDlgItem(IDC_STOP)->EnableWindow(false);
  110. return TRUE;  // return TRUE  unless you set the focus to a control
  111. }
  112. void CNQueenDlg::OnSysCommand(UINT nID, LPARAM lParam)
  113. {
  114. if ((nID & 0xFFF0) == IDM_ABOUTBOX)
  115. {
  116. CAboutDlg dlgAbout;
  117. dlgAbout.DoModal();
  118. }
  119. else
  120. {
  121. CDialog::OnSysCommand(nID, lParam);
  122. }
  123. }
  124. // If you add a minimize button to your dialog, you will need the code below
  125. //  to draw the icon.  For MFC applications using the document/view model,
  126. //  this is automatically done for you by the framework.
  127. void CNQueenDlg::OnPaint() 
  128. {
  129. if (IsIconic())
  130. {
  131. CPaintDC dc(this); // device context for painting
  132. SendMessage(WM_ICONERASEBKGND, (WPARAM) dc.GetSafeHdc(), 0);
  133. // Center icon in client rectangle
  134. int cxIcon = GetSystemMetrics(SM_CXICON);
  135. int cyIcon = GetSystemMetrics(SM_CYICON);
  136. CRect rect;
  137. GetClientRect(&rect);
  138. int x = (rect.Width() - cxIcon + 1) / 2;
  139. int y = (rect.Height() - cyIcon + 1) / 2;
  140. // Draw the icon
  141. dc.DrawIcon(x, y, m_hIcon);
  142. }
  143. else
  144. {
  145. CDialog::OnPaint();
  146. }
  147. }
  148. // The system calls this to obtain the cursor to display while the user drags
  149. //  the minimized window.
  150. HCURSOR CNQueenDlg::OnQueryDragIcon()
  151. {
  152. return (HCURSOR) m_hIcon;
  153. }
  154. void CNQueenDlg::OnAbout() //"关于"按钮单击事件响应函数
  155. {
  156. CAboutDlg dlgAbout;
  157. dlgAbout.DoModal();
  158. }
  159. void CNQueenDlg::OnChangeN() //"更改规模N"按钮单击事件响应函数
  160. {
  161. int N1 = m_N; //保存更新前的数值
  162. UpdateData(TRUE); //编辑框中读数据
  163. if( m_N<1 || m_N >10 )  m_N = N1; //新的数值超出范围,强制改回为原来的值
  164. UpdateData(FALSE); //将成员变量m_N的值写入编辑框中
  165. if( N1!=m_N )
  166. {
  167. int queen2[20] = {-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1};
  168. m_panel.SetQueen(queen2);
  169. m_panel.SetSize(m_N); //如果m_N的值更新了,则改变棋盘的规模
  170. }
  171. }
  172. int queen1[21][21]; //queen1[i][j]表示棋盘(i,j)位置上的状态
  173. int tmpq[730][21]; //存储每个解(N<=10的N皇后问题,最多有724个解(N=10时))
  174. //如果在第r行、第c列放置一个皇后,更前面已经放置好的皇后没有冲突,则返回1,否则返回0
  175. int judge( int r, int c, int N )
  176. {
  177. int i;
  178. for( i=0; i<r; i++ ) //检查前面r-1行各行,是否在第c列有皇后
  179. {
  180. if(queen1[i][c]==1)  break;
  181. }
  182. if(i<r)  return 0;
  183. //检查前面r-1行各行,与[r][c]位置位于同一条主对角线上是否有皇后
  184. for( i=1; r-i>=0&&c-i>=0; i++ )
  185. {
  186. if(queen1[r-i][c-i]==1)  break;
  187. }
  188. if(r-i>=0&&c-i>=0)  return 0;
  189. //检查前面r-1行各行,与[r][c]位置位于同一条主对角线上是否有皇后
  190. for( i=1; r-i>=0&&c+i<N; i++ )
  191. {
  192. if(queen1[r-i][c+i]==1)  break;
  193. }
  194. if(r-i>=0&&c+i<N)  return 0;
  195. return 1;
  196. }
  197. void CNQueenDlg::search( int step, int N )
  198. {
  199. int i, j, k;
  200. for(k=0; k<N; k++ )
  201. {
  202. if( judge(step,k,N) ) //如果在第step行、第k列可以放置一个皇后
  203. {
  204. queen1[step][k]=1;
  205. if( step==N-1 )
  206. {
  207. for( i=0; i<N; i++ )
  208. {
  209. for( j=0; j<N; j++ )
  210. if( queen1[i][j]==1 )  tmpq[count][i] = j;
  211. }
  212. count++;
  213. }
  214. search(step+1,N);
  215. queen1[step][k]=0;
  216. }
  217. }
  218. }
  219. void CNQueenDlg::OnStart() //"开始搜索"按钮单击事件响应函数
  220. {
  221. GetDlgItem(IDC_NEXT)->EnableWindow(false);
  222. GetDlgItem(IDC_STOP)->EnableWindow(false);
  223. int queen2[20] = {-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1};
  224. m_panel.SetQueen(queen2);
  225. count = 0;
  226. memset(queen1,0,sizeof(queen1));
  227. memset(tmpq,0,sizeof(tmpq));
  228. search(0,m_N);
  229. char str[40];
  230. sprintf( str, "求解完毕!共%d个解!", count );
  231. MessageBox( "可以点击“下一个解”来演示!",str, MB_OK );
  232. GetDlgItem(IDC_NEXT)->EnableWindow(true);
  233. GetDlgItem(IDC_STOP)->EnableWindow(true);
  234. index = 0;
  235. running = true;
  236. }
  237. void CNQueenDlg::OnNext() //"下一个解"按钮单击事件响应函数
  238. {
  239. m_panel.SetQueen(tmpq[index]);
  240. index++;
  241. if(index>=count)
  242. {
  243. MessageBox( "演示完毕!",NULL, MB_OK );
  244. GetDlgItem(IDC_NEXT)->EnableWindow(false);
  245. GetDlgItem(IDC_STOP)->EnableWindow(false);
  246. running = false;
  247. }
  248. }
  249. void CNQueenDlg::OnStop() //"停止"按钮单击事件响应函数
  250. {
  251. if( !running )  return;
  252. GetDlgItem(IDC_NEXT)->EnableWindow(false);
  253. GetDlgItem(IDC_STOP)->EnableWindow(false);
  254. running = false;
  255. }