posts - 23,  comments - 20,  trackbacks - 0
 
如果当比较元素相同返回真时,此时比较元素将会继续向下遍历,在极端情况下,例如程序中所有元素都是一样的情况下,在这种情况下,就会出现访问越界,结果就是导致程序出现segment fault

所以在写c++ stl中的比较函数是,bool返回真的时候,一定是“真的”大,或者小,等于的时候只能返回false。
posted @ 2016-10-08 11:19 李佳 阅读(1359) | 评论 (0)编辑 收藏
推倒的过程就已经学到了不少东西....
最近重写了以前的一个程序...原来 用了OO 代码可以写的如此优雅...
建议没事了  推推以前的程序重写  学到的东西也是不少的  特别是对需求的理解 对架构的设计

posted @ 2009-11-25 15:01 李佳 阅读(302) | 评论 (0)编辑 收藏
原来自己以前一直是把C++当成C用 尽管里面有class  但是  如果思想是过程的  再好的OO语言也是无助的

改代码... 好好学习OO

posted @ 2009-11-17 15:39 李佳 阅读(575) | 评论 (1)编辑 收藏
少了硬编码  代码确实好看了很多...
上个例子... 滚动视图部分的
修改前的代码
void CTransportWnd::OnPaint()
{    
    CPaintDC dc(
this);
    
    m_tooltip.RemoveAllTools();    
//清空所有标记

    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(255255255255)), rect.left, rect.top, rect.Width(), rect.Height());

    
// concrete flow
    DrawAllPlaces(graphics, rect);

    BitBlt(dc, 
00, rect.Width(), rect.Height(), dcMem, 00, SRCCOPY);

    DeleteObject(bm);
    DeleteDC(dcMem);

    
//更改滚动条
    GetClientRect(&rect);
    
int iCount = m_vecPlaces.size();
    
int iTotalHeight = (iCount / 2* (FLOW_HEIGHT+10);
    SetScrollSizes( MM_LOENGLISH, CSize(rect.Width() 
- 20, iTotalHeight) );
}

修改过后的代码
void CWorksiteWnd::OnDraw(CDC* pDC)
{
    
// TODO: 在此处为本机数据添加绘制代码
    m_tooTip.RemoveAllTools();    //清空所有标记

    CRect rect;    
    GetClientRect(rect);        

    
//绘图
    Graphics graphics( pDC->m_hDC );
    DrawAllPlaces(graphics, rect);

    
//更新滚动条
    GetClientRect(&rect);
    
int iCount = m_vecPlaces.size();
    
int iTotalHeight = iCount  * (FLOW_HEIGHT+10+ 20;
    SetScrollSizes( MM_TEXT, CSize(rect.Width() 
- 20, iTotalHeight) );    
}
单单抛开双缓冲不说   那个DocToClient这个函数简直是丑陋无比 而且是设备相关的(换了分辨率就变样了...)

void CTransportWnd::DocToClient(CRect& rect)
{    
    CClientDC dc(
this);
    OnPrepareDC(
&dc, NULL);
    dc.LPtoDP(rect);
    rect.NormalizeRect();
    
//rect.bottom += 5000;
    
//rect.top -= 198;
}
代码的丑陋可能现在还看不出来...在上一段小小的代码  后面画图部分的

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();
简直就是war3里面的憎恶...  用肉缝起来的代码  后面还有大量的
CPoint pointScorll = GetScrollPosition();
            rectTemp.bottom 
+= pointScorll.y;
            rectTemp.top 
+= pointScorll.y; 
哎  今天花了一早上更改代码   现在的代码看着要舒服多了  还是听感谢侯捷的<深入浅出MFC>  昨晚无意间翻到的CScorllView这一节  一下就看明白代码的设备相关性的问题出在哪了...   看来还是要时不时翻翻书

posted @ 2009-11-09 14:42 李佳 阅读(429) | 评论 (0)编辑 收藏
前段时间自己写了一个日志类  不过是MFC写的   今天闲着没事  就用SDK重写了这个类  算是增加一点可移植性吧...
类的使用也很简单  初始化的时候传递一个日志名称 
然后在需要写日志的地方 直接调用这个对象的WriteLog函数传递一个字符串进去就行了
如果需要时间戳  则调用WriteLogWithTime函数  一样的接口
支持设置日志大小  SetLogSize()   //KB计算
代码很简单  一百来行  不过用着还算挺方便的  下午就用这个日志类找到了一个以前工程中的SQL相关联的BUG 呵呵 

.h文件
/*****************************************************
*文件名称: Log_SDK.h
*功能说明: 使用SDK + STL重写的一个日志类 增加可移植性
*创建时间: 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);        //每次写一行 且带时间
    BOOL WriteLogAtFirstLine(TCHAR* pContent);    //将最近的日志写在最前面    
    void SetLogSize(int iSize);

private:
    HANDLE m_hFile;
    
string m_strFileName;
    
string m_strFormat; 
    CRITICAL_SECTION m_cs;                
//临界区
    BOOL m_bInit;                            //初始化标志位
    int m_iSize;                                //文件大小    -- KB计算

    
string GetTimeStr();
};

.cpp文件
/***************************************************
*文件名称: Log_SDK.cpp
*功能说明: LogSDK的实现文件
*创建时间: 09.10.31
*文件作者: zip
***************************************************
/
#include 
"StdAfx.h"
#include 
".\log_sdk.h"

LogSDK::LogSDK(
string strFileName)
{
    m_bInit 
= FALSE;
    m_strFileName 
= strFileName;
    m_iSize 
= 20;    //默认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 , 
"打开文件失败 , 错误代码:%d" , dwErrorCode);
        
//AfxMessageBox(szErrMsg);
        MessageBox(NULL , szErrMsg , _T("警告") ,IDOK);
        
return FALSE;
    }
    ::InitializeCriticalSection(
&m_cs);    //初始化临界区
    m_bInit = TRUE;
    
return TRUE;
}

//普通的写日志
BOOL LogSDK::WriteLog(TCHAR* pContent)
{
    ASSERT(m_bInit);
    DWORD dwFileLenHigh;
    DWORD dwFileLen 
= GetFileSize(m_hFile , &dwFileLenHigh);
    
if (dwFileLen > (DWORD)m_iSize * 1024)    //文件过大 清空文件
    {
        SetFilePointer(m_hFile , 
0 , NULL , FILE_BEGIN);    //移动文件指针
        SetEndOfFile(m_hFile);//清空文件
    }
    
string strContent(pContent);
    strContent 
+= "\r\n";

    BOOL bRet 
= TRUE;
    ::EnterCriticalSection(
&m_cs);
    SetFilePointer(m_hFile , 
0 , NULL , FILE_END);    //移动到文件末尾
    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 , 
"写入日志文件失败 , 错误代码:%d" , dwErrorCode);        
        MessageBox(NULL , szErrMsg , _T(
"发生了严重的错误") , IDOK);
        bRet 
= FALSE;    //此处不用返回  还要释放临界区
    }
    ::LeaveCriticalSection(
&m_cs);

    
return bRet;
}

//带有时间戳的日志
BOOL LogSDK::WriteLogWithTime(TCHAR* pContent)
{
    ASSERT(m_bInit);
    DWORD dwFileLenHigh;
    DWORD dwFileLen 
= GetFileSize(m_hFile , &dwFileLenHigh);
    
if (dwFileLen > (DWORD)m_iSize * 1024)    //文件过大 清空文件
    {
        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);    //移动到文件末尾
    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 , 
"写入日志文件失败 , 错误代码:%d" , dwErrorCode);        
        MessageBox(NULL , szErrMsg , _T(
"发生了严重的错误") , IDOK);
        bRet 
= FALSE;    //此处不用返回  还要释放临界区
    }
    ::LeaveCriticalSection(
&m_cs);

    
return bRet;
}

//返回格式化的时间字符串
string LogSDK::GetTimeStr()
{
    SYSTEMTIME sys;
    ::GetLocalTime(
&sys);    //获取当前系统时间
    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的操作吧...

posted @ 2009-10-31 17:58 李佳 阅读(1106) | 评论 (1)编辑 收藏
今天准备给一个视图增加一个滚动条   自然想到了从CScorllView中继承 可是继承了以后  发现滚动的时候会出现残影
如图

滚动条滚动以后

在网上找到了一个例子  里面有一个函数引起了我的注意
void CPaintWithDoubleBufferView::DocToClient(CRect& rect)
{
    CClientDC dc(
this);
    OnPrepareDC(
&dc, NULL);
    dc.LPtoDP(rect);
    rect.NormalizeRect();
}
原来问题出在这了...
坐标没有转换  于是在自己的工程中增加了一个坐标转换的接口  再次调试   完成

说实话  并不大懂为什么要这么做...

posted @ 2009-10-28 15:44 李佳 阅读(351) | 评论 (0)编辑 收藏
系统待机功能对于我这种笔记本用户很常见   一直不知道怎么实现
今天搜索了一下  找到了一个命令行的   稍微一改  就能加到代码中去了  

system("rundll32.exe powrprof.dll SetSuspendState");

测试通过

posted @ 2009-10-22 12:02 李佳 阅读(443) | 评论 (2)编辑 收藏
这段时间上网算是找疯了...界面真是相当的麻烦
下面推荐一个OutLook类   可以很方便的给自己的程序增加一个OutLook样式的视图
文件上传下来了   又需要的  自己研究研究吧
这段时间一直在用一个国内的VC知识库上的一个OutLook类   今天再和这个国外的OutLook类一比较.. 汗...
以后   还是应该多多上上国外网站    

效果图就不上了   自己看吧   比较华丽

下载地址
http://www.cppblog.com/Files/lijiakxl/OutBarDemo.rar
posted @ 2009-09-21 16:02 李佳 阅读(490) | 评论 (1)编辑 收藏
哎  新的工作就是做自己最薄弱的界面部分了...
一大堆东西都不知道  慢慢来吧 
东西都记下来 
首先就是OutLook样式的分割窗口  +   多视图切换  代码先放着

//更换视图的按钮函数
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;
}

//创建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(
"未能创建工具栏\n");
        
return -1;      // 未能创建
    }

    
if (!m_wndStatusBar.Create(this||
        
!m_wndStatusBar.SetIndicators(indicators,
          
sizeof(indicators)/sizeof(UINT)))
    {
        TRACE0(
"未能创建状态栏\n");
        
return -1;      // 未能创建
    }
    
// 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 ,    //创建新视图
            RUNTIME_CLASS(CTestView2) , CSize(100100) , &context);
        
break;
        
    
case BLACK:
        context.m_pNewViewClass 
= RUNTIME_CLASS(COutBarStyleView);
        context.m_pLastView 
= pOldView;
        m_wndSplitter.DeleteView(
0 , 1);    //删除旧的视图
        m_wndSplitter.CreateView(0 , 1 ,    //创建新视图
            RUNTIME_CLASS(COutBarStyleView) , CSize(100100) , &context);
        
break;
    }
    m_wndSplitter.RecalcLayout();  
    m_wndSplitter.SetActivePane(
0,1); 
}

// CMainFrame 消息处理程序
BOOL CMainFrame::OnCreateClient(LPCREATESTRUCT lpcs, CCreateContext* pContext)
{
    
// 创建切分窗口
    if (!m_wndSplitter.CreateStatic(this12))
        
return FALSE;

    
if (!m_wndSplitter.CreateView(00, RUNTIME_CLASS(COutBarView), CSize(100100), pContext) ||
        
!m_wndSplitter.CreateView(01, RUNTIME_CLASS(COutBarStyleView), CSize(100100), pContext))
    {
        m_wndSplitter.DestroyWindow();
        
return FALSE;
    }
    
return TRUE;    //此处必须返回TRUE才能分割窗口
}

posted @ 2009-09-16 10:22 李佳 阅读(623) | 评论 (0)编辑 收藏

数据库操作 
m_pConn->Execute((_bstr_t)strSQL , &index , 1);
第二个参数的值是影响的行数
有了这个  就可以在Update数据库的记录的时候不用先Select查看是否存在数据而执行两条SQL语句影响服务器的效率了


捕获ADO的数据库操作的异常
catch(_com_error e)
    {
        ::AfxMessageBox(e.Description() , MB_OK);
        
return FALSE;
    }
三 
将double类型的时间转换成时间字符串
CString strGPSTime;
    COleDateTime GPStime(GPSDate.gpsTime);
    strGPSTime 
= GPStime.Format("%Y-%m-%d %H:%M:%S");    //GPS时间

小技巧 在MFC的编辑框显示信息的时候  我以前一般都是 直接
m_strMsg += "提示信息:";
这样会出现往下拖  很烦人  
今天刚想到一个小技巧
在插入信息的时候 我们可以插入到头部去  这样就不会滚动 我们看到的就是最新的信息了
m_strMsg.Insert(0 , "信息提示:");

//10.22 新增
对话框的巧妙隐藏   不闪屏!
隐藏窗口
void CFlashThiefDlg::OnWindowPosChanging(WINDOWPOS* lpwndpos)
{
    lpwndpos
->flags &= ~SWP_SHOWWINDOW;
    CDialog::OnWindowPosChanging(lpwndpos);    
}


判断数据库中的一张表是否存在的函数
//判断一张表是否存在
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(
"文件名称: %s \n 所在代码行 : %d 执行SQL语句失败 错误原因 %s" , __FILE__ , __LINE__ , (LPCSTR)e.Description());
        TRACE(str);        
        AfxMessageBox(str);    
        
return FALSE;
    }

    
return FALSE;
}

以后再继续

posted @ 2009-09-11 16:06 李佳 阅读(693) | 评论 (0)编辑 收藏
仅列出标题
共3页: 1 2 3 
<2024年4月>
31123456
78910111213
14151617181920
21222324252627
2829301234
567891011

常用链接

留言簿

随笔分类

随笔档案

文章档案

相册

搜索

  •  

最新评论

阅读排行榜

评论排行榜