如果當(dāng)比較元素相同返回真時(shí),此時(shí)比較元素將會(huì)繼續(xù)向下遍歷,在極端情況下,例如程序中所有元素都是一樣的情況下,在這種情況下,就會(huì)出現(xiàn)訪問(wèn)越界,結(jié)果就是導(dǎo)致程序出現(xiàn)segment fault所以在寫(xiě)c++ stl中的比較函數(shù)是,bool返回真的時(shí)候,一定是“真的”大,或者小,等于的時(shí)候只能返回false。
推倒的過(guò)程就已經(jīng)學(xué)到了不少東西....
最近重寫(xiě)了以前的一個(gè)程序...原來(lái) 用了OO 代碼可以寫(xiě)的如此優(yōu)雅...
建議沒(méi)事了 推推以前的程序重寫(xiě) 學(xué)到的東西也是不少的 特別是對(duì)需求的理解 對(duì)架構(gòu)的設(shè)計(jì)
原來(lái)自己以前一直是把C++當(dāng)成C用 盡管里面有class 但是 如果思想是過(guò)程的 再好的OO語(yǔ)言也是無(wú)助的
改代碼... 好好學(xué)習(xí)OO
少了硬編碼 代碼確實(shí)好看了很多...
上個(gè)例子... 滾動(dòng)視圖部分的
修改前的代碼
void CTransportWnd::OnPaint()
{
CPaintDC dc(this);
m_tooltip.RemoveAllTools(); //清空所有標(biāo)記
CRect rect;
GetClientRect(rect);
DocToClient(rect);
HDC dcMem;
HBITMAP bm;
bm = CreateCompatibleBitmap(dc.m_hDC, rect.Width(), rect.Height());
dcMem = CreateCompatibleDC(dc);
SelectObject(dcMem, bm);
//////////////////////////
Graphics graphics(dcMem);
graphics.SetSmoothingMode(SmoothingModeHighQuality);
// background
graphics.FillRectangle(&SolidBrush(Color(255, 255, 255, 255)), rect.left, rect.top, rect.Width(), rect.Height());
// concrete flow
DrawAllPlaces(graphics, rect);
BitBlt(dc, 0, 0, rect.Width(), rect.Height(), dcMem, 0, 0, SRCCOPY);
DeleteObject(bm);
DeleteDC(dcMem);
//更改滾動(dòng)條
GetClientRect(&rect);
int iCount = m_vecPlaces.size();
int iTotalHeight = (iCount / 2) * (FLOW_HEIGHT+10);
SetScrollSizes( MM_LOENGLISH, CSize(rect.Width() - 20, iTotalHeight) );
}
修改過(guò)后的代碼
void CWorksiteWnd::OnDraw(CDC* pDC)
{
// TODO: 在此處為本機(jī)數(shù)據(jù)添加繪制代碼
m_tooTip.RemoveAllTools(); //清空所有標(biāo)記
CRect rect;
GetClientRect(rect);
//繪圖
Graphics graphics( pDC->m_hDC );
DrawAllPlaces(graphics, rect);
//更新滾動(dòng)條
GetClientRect(&rect);
int iCount = m_vecPlaces.size();
int iTotalHeight = iCount * (FLOW_HEIGHT+10) + 20;
SetScrollSizes( MM_TEXT, CSize(rect.Width() - 20, iTotalHeight) );
}
單單拋開(kāi)雙緩沖不說(shuō) 那個(gè)DocToClient這個(gè)函數(shù)簡(jiǎn)直是丑陋無(wú)比 而且是設(shè)備相關(guān)的(換了分辨率就變樣了...)
void CTransportWnd::DocToClient(CRect& rect)
{
CClientDC dc(this);
OnPrepareDC(&dc, NULL);
dc.LPtoDP(rect);
rect.NormalizeRect();
//rect.bottom += 5000;
//rect.top -= 198;
}
代碼的丑陋可能現(xiàn)在還看不出來(lái)...在上一段小小的代碼 后面畫(huà)圖部分的
void CTransportWnd::DrawAllPlaces(Graphics& graphics, CRect rect)
{
//rect.top += 188;
CRect rectTemp;
GetClientRect(&rectTemp);
rect.top += rectTemp.bottom;
//rect.top = 0;
vector<PLACE_STATION*>::iterator it = m_vecPlaces.begin();
簡(jiǎn)直就是war3里面的憎惡... 用肉縫起來(lái)的代碼 后面還有大量的
CPoint pointScorll = GetScrollPosition();
rectTemp.bottom += pointScorll.y;
rectTemp.top += pointScorll.y;
哎 今天花了一早上更改代碼 現(xiàn)在的代碼看著要舒服多了 還是聽(tīng)感謝侯捷的<深入淺出MFC> 昨晚無(wú)意間翻到的CScorllView這一節(jié) 一下就看明白代碼的設(shè)備相關(guān)性的問(wèn)題出在哪了... 看來(lái)還是要時(shí)不時(shí)翻翻書(shū)
前段時(shí)間自己寫(xiě)了一個(gè)日志類(lèi) 不過(guò)是MFC寫(xiě)的 今天閑著沒(méi)事 就用SDK重寫(xiě)了這個(gè)類(lèi) 算是增加一點(diǎn)可移植性吧...
類(lèi)的使用也很簡(jiǎn)單 初始化的時(shí)候傳遞一個(gè)日志名稱(chēng)
然后在需要寫(xiě)日志的地方 直接調(diào)用這個(gè)對(duì)象的WriteLog函數(shù)傳遞一個(gè)字符串進(jìn)去就行了
如果需要時(shí)間戳 則調(diào)用WriteLogWithTime函數(shù) 一樣的接口
支持設(shè)置日志大小 SetLogSize() //KB計(jì)算
代碼很簡(jiǎn)單 一百來(lái)行 不過(guò)用著還算挺方便的 下午就用這個(gè)日志類(lèi)找到了一個(gè)以前工程中的SQL相關(guān)聯(lián)的BUG 呵呵
.h文件
/*****************************************************
*文件名稱(chēng): Log_SDK.h
*功能說(shuō)明: 使用SDK + STL重寫(xiě)的一個(gè)日志類(lèi) 增加可移植性
*創(chuàng)建時(shí)間: 09.10.31
*文件作者: zip
*****************************************************/
#pragma once
#include <string>
using namespace std;
class LogSDK
{
public:
LogSDK(string strFileName);
~LogSDK(void);
BOOL InitLogSDK();
BOOL WriteLog(TCHAR* pContent);
BOOL WriteLogWithTime(TCHAR* pContent); //每次寫(xiě)一行 且?guī)r(shí)間
BOOL WriteLogAtFirstLine(TCHAR* pContent); //將最近的日志寫(xiě)在最前面
void SetLogSize(int iSize);
private:
HANDLE m_hFile;
string m_strFileName;
string m_strFormat;
CRITICAL_SECTION m_cs; //臨界區(qū)
BOOL m_bInit; //初始化標(biāo)志位
int m_iSize; //文件大小 -- KB計(jì)算
string GetTimeStr();
};
.cpp文件
/***************************************************
*文件名稱(chēng): Log_SDK.cpp
*功能說(shuō)明: LogSDK的實(shí)現(xiàn)文件
*創(chuàng)建時(shí)間: 09.10.31
*文件作者: zip
***************************************************/
#include "StdAfx.h"
#include ".\log_sdk.h"
LogSDK::LogSDK(string strFileName)
{
m_bInit = FALSE;
m_strFileName = strFileName;
m_iSize = 20; //默認(rèn)20KB
InitLogSDK();
}
LogSDK::~LogSDK(void)
{
CloseHandle(m_hFile);
}
BOOL LogSDK::InitLogSDK()
{
m_hFile = CreateFile(m_strFileName.c_str() , GENERIC_ALL ,
FILE_SHARE_WRITE , NULL , OPEN_ALWAYS , FILE_ATTRIBUTE_NORMAL , NULL);
if (m_hFile == INVALID_HANDLE_VALUE)
{
DWORD dwErrorCode = GetLastError();
TCHAR szErrMsg[255];
sprintf(szErrMsg , "打開(kāi)文件失敗 , 錯(cuò)誤代碼:%d" , dwErrorCode);
//AfxMessageBox(szErrMsg);
MessageBox(NULL , szErrMsg , _T("警告") ,IDOK);
return FALSE;
}
::InitializeCriticalSection(&m_cs); //初始化臨界區(qū)
m_bInit = TRUE;
return TRUE;
}
//普通的寫(xiě)日志
BOOL LogSDK::WriteLog(TCHAR* pContent)
{
ASSERT(m_bInit);
DWORD dwFileLenHigh;
DWORD dwFileLen = GetFileSize(m_hFile , &dwFileLenHigh);
if (dwFileLen > (DWORD)m_iSize * 1024) //文件過(guò)大 清空文件
{
SetFilePointer(m_hFile , 0 , NULL , FILE_BEGIN); //移動(dòng)文件指針
SetEndOfFile(m_hFile);//清空文件
}
string strContent(pContent);
strContent += "\r\n";
BOOL bRet = TRUE;
::EnterCriticalSection(&m_cs);
SetFilePointer(m_hFile , 0 , NULL , FILE_END); //移動(dòng)到文件末尾
DWORD dwWriteLen;
BOOL bSuccess = WriteFile(m_hFile , strContent.c_str() ,
(DWORD)strContent.length() , &dwWriteLen , NULL);
ASSERT(dwWriteLen == strContent.length());
if (!bSuccess)
{
DWORD dwErrorCode = GetLastError();
TCHAR szErrMsg[255];
sprintf(szErrMsg , "寫(xiě)入日志文件失敗 , 錯(cuò)誤代碼:%d" , dwErrorCode);
MessageBox(NULL , szErrMsg , _T("發(fā)生了嚴(yán)重的錯(cuò)誤") , IDOK);
bRet = FALSE; //此處不用返回 還要釋放臨界區(qū)
}
::LeaveCriticalSection(&m_cs);
return bRet;
}
//帶有時(shí)間戳的日志
BOOL LogSDK::WriteLogWithTime(TCHAR* pContent)
{
ASSERT(m_bInit);
DWORD dwFileLenHigh;
DWORD dwFileLen = GetFileSize(m_hFile , &dwFileLenHigh);
if (dwFileLen > (DWORD)m_iSize * 1024) //文件過(guò)大 清空文件
{
SetFilePointer(m_hFile , 0 , NULL , FILE_BEGIN);
SetEndOfFile(m_hFile);//清空文件
}
//strcat(pContent , "\r\n");
string strContent(pContent);
strContent += "\r\n";
strContent.insert(0 , " ");
strContent.insert(0 , GetTimeStr());
BOOL bRet = TRUE;
::EnterCriticalSection(&m_cs);
SetFilePointer(m_hFile , 0 , NULL , FILE_END); //移動(dòng)到文件末尾
DWORD dwWriteLen;
BOOL bSuccess = WriteFile(m_hFile , strContent.c_str() ,
(DWORD)strContent.length() , &dwWriteLen , NULL);
ASSERT(dwWriteLen == strContent.length());
if (!bSuccess)
{
DWORD dwErrorCode = GetLastError();
TCHAR szErrMsg[255];
sprintf(szErrMsg , "寫(xiě)入日志文件失敗 , 錯(cuò)誤代碼:%d" , dwErrorCode);
MessageBox(NULL , szErrMsg , _T("發(fā)生了嚴(yán)重的錯(cuò)誤") , IDOK);
bRet = FALSE; //此處不用返回 還要釋放臨界區(qū)
}
::LeaveCriticalSection(&m_cs);
return bRet;
}
//返回格式化的時(shí)間字符串
string LogSDK::GetTimeStr()
{
SYSTEMTIME sys;
::GetLocalTime(&sys); //獲取當(dāng)前系統(tǒng)時(shí)間
TCHAR szTime[64];
memset(szTime , 0 , 64);
sprintf(szTime , "%d-%02d-%02d %02d:%02d:%02d.%03d" ,
sys.wYear , sys.wMonth , sys.wDay , sys.wHour , sys.wMinute , sys.wSecond , sys.wMilliseconds);
return string(szTime);
}
void LogSDK::SetLogSize(int iSize)
{
m_iSize = iSize;
}
也可以算鍛煉一下API的操作吧...
今天準(zhǔn)備給一個(gè)視圖增加一個(gè)滾動(dòng)條 自然想到了從CScorllView中繼承 可是繼承了以后 發(fā)現(xiàn)滾動(dòng)的時(shí)候會(huì)出現(xiàn)殘影
如圖

滾動(dòng)條滾動(dòng)以后
在網(wǎng)上找到了一個(gè)例子 里面有一個(gè)函數(shù)引起了我的注意
void CPaintWithDoubleBufferView::DocToClient(CRect& rect)
{
CClientDC dc(this);
OnPrepareDC(&dc, NULL);
dc.LPtoDP(rect);
rect.NormalizeRect();
}
原來(lái)問(wèn)題出在這了...
坐標(biāo)沒(méi)有轉(zhuǎn)換 于是在自己的工程中增加了一個(gè)坐標(biāo)轉(zhuǎn)換的接口 再次調(diào)試 完成
說(shuō)實(shí)話 并不大懂為什么要這么做...
系統(tǒng)待機(jī)功能對(duì)于我這種筆記本用戶很常見(jiàn) 一直不知道怎么實(shí)現(xiàn)
今天搜索了一下 找到了一個(gè)命令行的 稍微一改 就能加到代碼中去了
system("rundll32.exe powrprof.dll SetSuspendState");
測(cè)試通過(guò)
這段時(shí)間上網(wǎng)算是找瘋了...界面真是相當(dāng)?shù)穆闊?br>下面推薦一個(gè)OutLook類(lèi) 可以很方便的給自己的程序增加一個(gè)OutLook樣式的視圖
文件上傳下來(lái)了 又需要的 自己研究研究吧
這段時(shí)間一直在用一個(gè)國(guó)內(nèi)的VC知識(shí)庫(kù)上的一個(gè)OutLook類(lèi) 今天再和這個(gè)國(guó)外的OutLook類(lèi)一比較.. 汗...
以后 還是應(yīng)該多多上上國(guó)外網(wǎng)站
效果圖就不上了 自己看吧 比較華麗
下載地址
http://www.shnenglu.com/Files/lijiakxl/OutBarDemo.rar
哎 新的工作就是做自己最薄弱的界面部分了...
一大堆東西都不知道 慢慢來(lái)吧
東西都記下來(lái)
首先就是OutLook樣式的分割窗口 + 多視圖切換 代碼先放著
//更換視圖的按鈕函數(shù)
LRESULT COutBarView::OnOutbarNotify(UINT lParam , LONG wParam)
{
switch(wParam)
{
case IDC_OUT_BAR:
XT_CONTENT_ITEM* pContentItems = m_WndOutLookBar.GetMenuItem((int)lParam);
ASSERT(pContentItems);
CMainFrame *pMainFrame = (CMainFrame *)AfxGetApp()->m_pMainWnd;
switch(lParam)
{
case 0:
pMainFrame->SwitchToView(RED);
break;
case 1:
pMainFrame->SwitchToView(BLACK);
break;
}
}
return 1;
}
//創(chuàng)建Outlook工具欄
int CMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
if (CFrameWnd::OnCreate(lpCreateStruct) == -1)
return -1;
if (!m_wndToolBar.CreateEx(this, TBSTYLE_FLAT, WS_CHILD | WS_VISIBLE | CBRS_TOP
| CBRS_GRIPPER | CBRS_TOOLTIPS | CBRS_FLYBY | CBRS_SIZE_DYNAMIC) ||
!m_wndToolBar.LoadToolBar(IDR_MAINFRAME))
{
TRACE0("未能創(chuàng)建工具欄\n");
return -1; // 未能創(chuàng)建
}
if (!m_wndStatusBar.Create(this) ||
!m_wndStatusBar.SetIndicators(indicators,
sizeof(indicators)/sizeof(UINT)))
{
TRACE0("未能創(chuàng)建狀態(tài)欄\n");
return -1; // 未能創(chuàng)建
}
// TODO: 如果不需要工具欄可停靠,則刪除這三行
m_wndToolBar.EnableDocking(CBRS_ALIGN_ANY);
EnableDocking(CBRS_ALIGN_ANY);
DockControlBar(&m_wndToolBar);
return 0;
}
//切換View 切換視圖
void CMainFrame::SwitchToView(eView nView)
{
CRect rect;
CCreateContext context;
GetClientRect(&rect);
CView* pOldView = (CView* )m_wndSplitter.GetPane(0 , 1);
CDocument* pDoc = (CDocument* )pOldView->GetDocument();
context.m_pCurrentDoc = pDoc;
context.m_pCurrentFrame = this;
context.m_pNewDocTemplate = pDoc->GetDocTemplate();
switch (nView)
{
case RED:
context.m_pNewViewClass = RUNTIME_CLASS(CTestView2);
context.m_pLastView = pOldView;
m_wndSplitter.DeleteView(0 , 1); //刪除舊的視圖
m_wndSplitter.CreateView(0 , 1 , //創(chuàng)建新視圖
RUNTIME_CLASS(CTestView2) , CSize(100, 100) , &context);
break;
case BLACK:
context.m_pNewViewClass = RUNTIME_CLASS(COutBarStyleView);
context.m_pLastView = pOldView;
m_wndSplitter.DeleteView(0 , 1); //刪除舊的視圖
m_wndSplitter.CreateView(0 , 1 , //創(chuàng)建新視圖
RUNTIME_CLASS(COutBarStyleView) , CSize(100, 100) , &context);
break;
}
m_wndSplitter.RecalcLayout();
m_wndSplitter.SetActivePane(0,1);
}
// CMainFrame 消息處理程序
BOOL CMainFrame::OnCreateClient(LPCREATESTRUCT lpcs, CCreateContext* pContext)
{
// 創(chuàng)建切分窗口
if (!m_wndSplitter.CreateStatic(this, 1, 2))
return FALSE;
if (!m_wndSplitter.CreateView(0, 0, RUNTIME_CLASS(COutBarView), CSize(100, 100), pContext) ||
!m_wndSplitter.CreateView(0, 1, RUNTIME_CLASS(COutBarStyleView), CSize(100, 100), pContext))
{
m_wndSplitter.DestroyWindow();
return FALSE;
}
return TRUE; //此處必須返回TRUE才能分割窗口
}
一
數(shù)據(jù)庫(kù)操作
m_pConn->Execute((_bstr_t)strSQL , &index , 1);
第二個(gè)參數(shù)的值是影響的行數(shù)
有了這個(gè) 就可以在Update數(shù)據(jù)庫(kù)的記錄的時(shí)候不用先Select查看是否存在數(shù)據(jù)而執(zhí)行兩條SQL語(yǔ)句影響服務(wù)器的效率了
二
捕獲ADO的數(shù)據(jù)庫(kù)操作的異常
catch(_com_error e)
{
::AfxMessageBox(e.Description() , MB_OK);
return FALSE;
}
三
將double類(lèi)型的時(shí)間轉(zhuǎn)換成時(shí)間字符串
CString strGPSTime;
COleDateTime GPStime(GPSDate.gpsTime);
strGPSTime = GPStime.Format("%Y-%m-%d %H:%M:%S"); //GPS時(shí)間
四
小技巧 在MFC的編輯框顯示信息的時(shí)候 我以前一般都是 直接
m_strMsg += "提示信息:";
這樣會(huì)出現(xiàn)往下拖 很煩人
今天剛想到一個(gè)小技巧
在插入信息的時(shí)候 我們可以插入到頭部去 這樣就不會(huì)滾動(dòng) 我們看到的就是最新的信息了
m_strMsg.Insert(0 , "信息提示:");
//10.22 新增
對(duì)話框的巧妙隱藏 不閃屏!
隱藏窗口
void CFlashThiefDlg::OnWindowPosChanging(WINDOWPOS* lpwndpos)
{
lpwndpos->flags &= ~SWP_SHOWWINDOW;
CDialog::OnWindowPosChanging(lpwndpos);
}
五
判斷數(shù)據(jù)庫(kù)中的一張表是否存在的函數(shù)
//判斷一張表是否存在
BOOL IsTableExsist(CString strTableName)
{
try
{
CStringArray arrTableNames;
_RecordsetPtr Recordset = m_pConn->OpenSchema(adSchemaTables);
_variant_t l_vDBTableName;
_bstr_t bstrTableType;
while(!Recordset->adoEOF)
{
l_vDBTableName = Recordset->GetCollect("TABLE_NAME");
bstrTableType = Recordset->GetCollect("TABLE_TYPE");
if ((bstrTableType == (_bstr_t)"TABLE")||(bstrTableType == (_bstr_t)"VIEWS"))
{
arrTableNames.Add((char *)_bstr_t(l_vDBTableName));
}
Recordset->MoveNext();
}
int iCount = arrTableNames.GetSize();
for (int i = 0 ; i < iCount ; i++)
{
CString& strName = arrTableNames[i];
if (strName == strTableName)
return TRUE;
}
}
catch(_com_error& e)
{
ASSERT(FALSE);
CString str;
str.Format("文件名稱(chēng): %s \n 所在代碼行 : %d 執(zhí)行SQL語(yǔ)句失敗 錯(cuò)誤原因 %s" , __FILE__ , __LINE__ , (LPCSTR)e.Description());
TRACE(str);
AfxMessageBox(str);
return FALSE;
}
return FALSE;
}
以后再繼續(xù)