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

C++ Programmer's Cookbook

{C++ 基礎} {C++ 高級} {C#界面,C++核心算法} {設計模式} {C#基礎}

Visual C++ .NET編程:托管C++概述

2000年6月,Microsoft推出了“Microsoft.NET下一代互聯網軟件和服務戰略”,引起IT行業的廣泛關注。2000年9月,Microsoft在舊金山發布了Enterprise 2000。同月,Microsoft原總裁兼首席執行官鮑爾默來到中國就“下一代互聯網”的主題進行演講,在中國掀起了一股“.NET旋風”。2000年11月,Microsoft在Comdex計算機大展上發布了Visual Studio.NET軟件,并展示了其.NET發展戰略的框架體系和開發工具的相關特性,全面加速了Microsoft以.NET技術進軍市場的步伐。

  Microsoft的.NET戰略意味著:Microsoft以及在Microsoft平臺上的開發者將會制造服務,而不是制造軟件。在未來幾年之內,Microsoft將陸續發布有關.NET的平臺和工具,用于在因特網上開發Web服務。那時,工作在.NET上的用戶、開發人員和IT工作人員都不再購買軟件、安裝軟件和維護軟件。取而代之的是,他們將定制服務,軟件會自動安裝,所有的維護和升級也會通過互聯網進行。“Microsoft.NET 代表了一個集合、一個環境、一個可以作為平臺支持下一代Internet的可編程結構。”這就是鮑爾默對.NET的描述。


  作為.NET的最新特性組成部分,Microsoft .NET Framework是一個用于構建,部署和運行Web服務及應用程序的平臺。它為將現有投資與下一代應用程序和服務的集成提供了高產的,基于標準的,多語言環境,同時它還用于解決Internet級應用程序的部署和操作問題。.NET框架包含三個主要部分:通用語言運行時,一組層次化的統一的類庫,及組件化版本的動態服務器主頁(稱為ASP.NET)。

  用于開發.NET Framework的語言有Visual C#、VB.NET和C++托管擴展(Managed Extensions for C++)。其中C#是開發.NET的元語言,而C++托管擴展是在C++基礎上建立起來的,用來為Visual C++程序員開發.NET框架應用程序而設計。為敘述方便,我們將C++托管擴展就稱之為“托管C++”。
為了幫助C/C++以及Visual C++程序員或愛好者快速使用托管C++開發.NET Framework程序,我們將陸續推出相關的一系列文章。

  本篇“托管C++概述”主要講述了什么是托管C++、開發.NET Framework(框架)的項目類型以及與標準C++之間的區別。

  1、什么是托管C++?

  在回答這個問題,首先要搞清楚什么是“托管”(Managed)。托管是.NET的一個專門概念,它是融于通用語言運行時(CLR)中的一種新的編程理念,因此我們完全可以把“托管”視為“.NET”。那么什么是“通用語言運行時”?通用語言運行時是.NET 框架應用程序的執行引摯。它提供了許多服務,其中包括:代碼管理(裝入和執行)、類型安全性驗證、元數據(高級類型信息)訪問、為管理對象管理內存、管理代碼,COM對象和預生成的DLLs(非管理代碼和數據)的交互操作性、對開發人員服務的支持等等。

  也就是說,使用托管C++意味著,我們的代碼可以被CLR所管理,并能開發出具有最新特性如垃圾自動收集、程序間相互訪問等的.NET框架應用程序。

  由托管概念所引發的C++應用程序包括托管代碼、托管數據和托管類三個組成部分。  

  (1) 托管代碼:.Net環境提供了許多核心的運行(RUNTIME)服務,比如異常處理和安全策略。為了能使用這些服務,必須要給運行環境提供一些信息代碼(元數據),這種代碼就是托管代碼。所有的C#、VB.NET、JScript.NET默認時都是托管的,但Visual C++默認時不是托管的,必須在編譯器中使用命令行選項(/CLR)才能產生托管代碼。

  (2) 托管數據:與托管代碼密切相關的是托管數據。托管數據是由公共語言運行的垃圾回收器進行分配和釋放的數據。默認情況下,C#、Visual Basic 和 JScript.NET 數據是托管數據。不過,通過使用特殊的關鍵字,C# 數據可以被標記為非托管數據。Visual C++數據在默認情況下是非托管數據,即使在使用 /CLR 開關時也不是托管的。

  (3) 托管類:盡管Visual C++數據在默認情況下是非托管數據,但是在使用C++的托管擴展時,可以使用“__gc”關鍵字將類標記為托管類。就像該名稱所顯示的那樣,它表示類實例的內存由垃圾回收器管理。另外,一個托管類也完全可以成為 .NET 框架的成員,由此可以帶來的好處是,它可以與其他語言編寫的類正確地進行相互操作,如托管的C++類可以從Visual Basic類繼承等。但同時也有一些限制,如托管類只能從一個基類繼承等。需要說明的是,在托管C++應用程序中既可使用托管類也可以使用非托管類。這里的非托管類不是指標準C++類,而是使用托管C++語言中的__nogc關鍵字的類。

2、用托管C++可以開發.NET框架的項目類型

  使用托管C++應該是C++程序員編寫.NET框架應用程序最好的一種選擇,通過集成在Visual Studio.NET開發環境的托管C++向導,我們可以創建以下幾種開發.NET框架的項目類型:

  (1) 托管C++應用程序:用來創建一個支持托管擴展的單獨C++應用程序,使用它還可創建任何類型的應用程序,包括.NET框架客戶應用程序。

  (2) 托管C++類庫:用來創建一個支持托管擴展的C++DLL,使用它可以生成一個能被.NET框架應用程序調用的托管類型的組件。

  (3) 托管C++空項目:用來創建一個空的托管項目,該項目只含有支持托管擴展的正確編譯和鏈接的開關選項。使用它能將一個已有的C++源文件進入到一個托管環境中。

  (4) 托管C++ Web服務:用于創建兩個項目,一個是C++托管擴展項目,另一個是部署項目。

  3、托管C++與標準C++的主要區別

  盡管托管C++是從標準C++建立而來的,但它與標準C++有著本質上的區別,這主要體現在以下幾個方面:

  (1) 廣泛采用“名稱空間”(namespace)

  名稱空間是類型的一種邏輯命名方案,.NET使用該命名方案用于將類型按相關功能的邏輯類別進行分組,利用名稱空間可以使開發人員更容易在代碼中瀏覽和引用類型。當然,我們也可將名稱空間理解成是一個“類庫名”。

  盡管很早Microsoft就在Visual C++中支持名稱空間的編程方式,但是很少引起Visual C++程序員的普遍關注。現在在托管C++程序中,我們必須使用這一方式,即使用#using和using關鍵字。例如下面的簡單程序代碼是在控制臺上輸出“Hello World”:

#using
using namespace System;
int main(void)
{
Console::WriteLine(S"Hello World");
return 0;
}

  代碼中,#using是用來將一個元數據文件輸入到托管C++程序中,這些文件可以是包含托管數據和結構的MSIL (Microsoft intermediate language,微軟中間語言)文件,如DLL、EXE、OBJ文件等。mscorlib.dll是.NET框架的一個核心類庫,包含主要的名稱空間System。程序的第二行代碼“using namespace System;”用來使用System名稱空間。System是.NET框架根名稱空間,包含最基本的類型,如用于數據流的輸入/輸出的System::IO等。

  在對托管C++程序開發的不斷深入,我們不久就會發現,許多類型的引用都要在程序的前面使用#using和using來進行。

  (2) 基本數據類型的變化

  我們知道,標準C++語言的數據類型是非常豐富的。而托管C++的數據類型更加豐富,不僅包含了標準C++中的數據類型,而且新增了__int64(64位整型)、Decimal(96位十進制數)、String*(字符串類型)和Object*(對象類型)等類型,表1-1列出它們各自數據類型。

類型描述標準C++類型名托管C++類型名長度(位)
布爾型boolbool8
字符型charsigned char8
無符號字符型unsigned charchar8
短整型short [int]short16
無符號短整型unsigned short [int]unsigned short16
整型intint 或 long32
無符號整型unsigned [int]unsigned int 或 long32
長整型long [int]long32
無符號長整型unsigned long [int]unsigned long32
單精度浮點型floatfloat32
雙精度浮點型doubledouble64
長雙精度浮點型long double--64
Unicode字符--wchar_t16
64位整型--__int6464
無符號64位整型--unsigned __int6464
96位十進制值--Decimal96
對象類型--Object*32
字符串類型--String*--


  需要注意的是,String和Object在定義一個變量時,注意要有星號(“*”),但這個變量不是指針變量,這與標準C++的含義是不一樣的。例如上面的代碼可以改為:

#using
using namespace System;
int main(void)
{
String* hello = S"Hello World";
Console::WriteLine(hello);
return 0;
}
 (3) 新增三個托管C++類型:__gc class、__value class和__gc interface

  一個__gc類或結構意味著該類或結構的生命周期是由.NET開發平臺自動管理及垃圾自動收集,用戶不必自已去調用delete來刪除。定義一個__gc類或結構和標準C++基本相似,所不同的是在class或struct前加上__gc,例如下面的代碼:

__gc class G {
public:
int k;
int sum(int);
};

G::sum(int i) {return i*(i + 1)/2;}
int main()
{
G * g = new G;
Console::WriteLine(g->sum(4)); // 結果輸出10
return 0;
}

  但要注意:

  A. 一個__gc類不能從一個非托管類中繼承,且不能包含從它派生的非托管類。但一個__gc類最多可以從一個托管類中繼承。

  B. 一個__gc類不能定義成一個友元類或包含一個友元成員函數。所謂友元函數,是用來讓外部函數訪問類中的私有和保護類型成員。

  C. 一個__gc類不能聲明或定義以及重載new或delete操作以及不能包含using等聲明。

  __value類是用來使用具有短生命期的小型數據項,它不同于__gc類。__gc類數據分配在CLR堆中,而__value類對象是在運行棧或稱為NDP(.NET Developer Platform,.NET開發者平臺)堆中創建的,從而避免了垃圾回收器不斷分配和釋放空間而帶來的開銷。一個__value類可以聲明成為一個局部變量、參數和返回值,也可嵌入到一個__gc類中或是作為一個靜態變量或在C++堆中分配的變量。例如下面的代碼:

#using
using namespace System;
__value struct V { int i; };
__gc struct G { V v; }; // 嵌入到__gc類中
V f(V v) { // 定義一個全局函數,其值存儲在運行棧中
v.i += 1; // 不影響原來形參v的值
return v; // 返回V結構類型的值
}
int main(void)
{
V v1 = {10}; // 在運行棧中聲明并初始化
V v2 = f(v1); // 調用f函數,此時v1中的i為10,而v2中的i為11
G *pG = new G; // 為G實例分配堆空間
pG->v = v1; // pG的v中的i為10
pG->v.i += v2.i; // pG的v中的i為10+11=21
Console::WriteLine(v1.i); // 輸出結果為10
Console::WriteLine(v2.i); // 輸出結果為11
Console::WriteLine(pG->v.i); // 輸出結果為21
return 0;
}

  除此之外,所有的__gc對象都是從類System::Object派生而來,因而能夠很容易使用作用在__gc類中的集合和映射功能。然而__value類型并沒有與這個基類所共享,因而不能直接將__value作為函數中的Object*實參。為了解決這個問題,.NET允許我們使用__box關鍵字將一個__value類型視為一個__gc對象。此時__value類型被封裝成一個__gc類樁子(Stub),并被復制到NDP堆中。由于在托管C++中,box不具備隱式轉換的功能,因此在轉換時必須指明轉換的類型。

  托管C++中的__gc接口最能體現COM接口的思想,它的定義和聲明是非常簡單的,它除了關鍵字不同外,與一個__gc類的聲明極為相似。例如下面的代碼定義了一個接口IMyBase,其中包含了一個f的方法:

__gc __interface Ibase {
void f();
};

  需要說明的是,接口中所有的方法默認時都是純虛的且都是公有的,我們不需要在方法之前使用virtual關鍵字或在方法之后加上“= 0”。其次,在一個__gc接口中不能包含數據成員以及靜態成員,也不能包含任何類的聲明。下面舉一個示例來說明__gc接口的使用:

#using
using namespace System;

__gc __interface Ibase1 {
int f(int);
};
__gc __interface Ibase2 {
int f(int);
};
__gc struct C: Ibase1, Ibase2 {
int f(int i) { // 接口方法的實現
return 2*i-1;
};
};

int main(void){
C* c = new C;
Console::WriteLine((c -> f(1)).ToString()); // 輸出結果為1
Console::WriteLine((__try_cast (c)->f(2)).ToString());
// 輸出結果為3

Console::WriteLine((__try_cast (c)->f(3)).ToString());
// 輸出結果為5

return 0;
}

  代碼中,__try_cast用來將某個對象轉換成一個指定類型,并當類型轉換失敗時自動處理由此產生的異常。ToString用來將對象描述成一個字符串。
(4) 簡化屬性操作

  在__gc類中可以使用.NET的屬性,這個屬性簡化了屬性函數的調用操作,這與標準C++中的屬性不一樣。在標準C++中分別通過get_和put_成員函數來設置或獲取相關屬性的值。現在,托管C++中的屬性操作就好比是對一個屬性變量進行操作,例如下列代碼:

#using
using namespace System;

__gc class G {
public:
__property int get_Size() {
Console::WriteLine(S"get_屬性");
return nSize;
};
__property void set_Size(int i) {
Console::WriteLine(S"set_屬性");
nSize = i;
};
private:
int nSize;
};

int main() {
G * pG = new G;
pG->Size = 10; // 調用set_Size
int i = pG->Size; // 調用get_Size
Console::WriteLine(i);
}

  程序結果為:

   set_屬性

   get_屬性

   10

  需要說明的是,托管C++使用__property關鍵字來定義一個屬性的成員函數。從代碼中可以看出設置和獲取屬性的成員函數名稱中分別使用了set_和get_,這樣編譯器會自動生成一個偽成員變量Size,這個變量名是set_和get_成員函數后面的名稱。注意不要再在get_成員函數代碼中使用這個偽成員變量Size,它會引起該函數的遞歸調用。

  (5) 托管C++的委派

  在C/C++中,一個函數的地址就是內存地址。這個地址不會帶有任何其它附加信息,如函數的參數個數、參數類型、函數的返回值類型以及這個函數的調用規范等。總之,C/C++的回調函數不具備類型安全性。而.NET框架在回調函數的基礎增加了提供類型安全的機制,稱為委派。

  托管C++的委派方法不像C#那么復雜,它簡化了委派絕大部分的內部機制,因而使得它的使用變成非常簡單容易。例如下面的代碼:

#using
using namespace System;

__delegate int GetDayOfWeek(); // 委派方法的聲明
__gc class MyCalendar
{
public:
MyCalendar() : m_nDayOfWeek(4) {}
int MyGetDayOfWeek() {
Console::WriteLine("非靜態方法");
return m_nDayOfWeek;
}
static int MyStaticGetDayOfWeek() {
Console::WriteLine("靜態方法");
return 6;
}
private:
int m_nDayOfWeek;
};

int main(void)
{
GetDayOfWeek * pGetDayOfWeek; // 聲明委派類型變量
int nDayOfWeek;

// 將類的靜態方法MyStaticGetDayOfWeek綁定成委派
pGetDayOfWeek = new GetDayOfWeek(0, &MyCalendar::MyStaticGetDayOfWeek);
nDayOfWeek = pGetDayOfWeek->Invoke(); // 委派的調用
Console::WriteLine(nDayOfWeek);

// 將一個類的實例綁定成委派
MyCalendar * pcal = new MyCalendar();
pGetDayOfWeek =
static_cast(Delegate::Combine(pGetDayOfWeek,
new GetDayOfWeek(pcal, &MyCalendar::MyGetDayOfWeek)));
nDayOfWeek = pGetDayOfWeek->Invoke();
Console::WriteLine(nDayOfWeek);

// 刪除綁定委派的類實例
pGetDayOfWeek =
static_cast(Delegate::Remove(pGetDayOfWeek,
new GetDayOfWeek(pcal, &MyCalendar::MyGetDayOfWeek)));

return 0;
}

  輸出結果是:

   靜態方法
  
   6

   靜態方法

   非靜態方法

   4

  4、結速語

  總之,使用托管C++是C++程序員編寫.NET框架應用程序最好的一種選擇,在充分理解.NET框架基礎上,避免了使用其他語言如C#、VB.NET所帶來的額外開銷。

posted on 2006-08-18 10:18 夢在天涯 閱讀(3465) 評論(1)  編輯 收藏 引用 所屬分類: CPlusPlusManage c++ /CLI

評論

# re: Visual C++ .NET編程:托管C++概述 2013-04-28 08:42 eryar

Mark,
good...  回復  更多評論   

公告

EMail:itech001#126.com

導航

統計

  • 隨筆 - 461
  • 文章 - 4
  • 評論 - 746
  • 引用 - 0

常用鏈接

隨筆分類

隨筆檔案

收藏夾

Blogs

c#(csharp)

C++(cpp)

Enlish

Forums(bbs)

My self

Often go

Useful Webs

Xml/Uml/html

搜索

  •  

積分與排名

  • 積分 - 1811980
  • 排名 - 5

最新評論

閱讀排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
              麻豆久久婷婷| 一区二区动漫| 亚欧成人在线| 亚洲资源在线观看| 国产日韩欧美日韩| 久久一区亚洲| 你懂的亚洲视频| 一区二区三区导航| 亚洲女性裸体视频| 1204国产成人精品视频| 亚洲二区免费| 欧美午夜一区二区| 欧美在线一二三四区| 久久久青草婷婷精品综合日韩| 亚洲国产成人高清精品| 亚洲美女性视频| 国产一区二区在线观看免费| 欧美激情片在线观看| 欧美日韩精品一区二区| 欧美一区二区免费| 免费成人激情视频| 91久久午夜| 欧美成人免费va影院高清| 亚洲精品乱码久久久久久日本蜜臀 | 亚洲巨乳在线| 国产亚洲精品久久久久久| 欧美国产激情| 国产精品有限公司| 欧美激情一区二区三区| 国产嫩草一区二区三区在线观看 | 亚洲国产一区二区三区a毛片| 国产精品久久久免费| 欧美成人免费播放| 国产欧美在线视频| 99国产精品久久| 亚洲国产中文字幕在线观看| 亚洲欧美成人在线| 日韩午夜中文字幕| 久久久天天操| 久久成人精品无人区| 欧美日韩国产综合网| 巨乳诱惑日韩免费av| 国产精品久久久一本精品| 亚洲国产精品国自产拍av秋霞| 韩国三级在线一区| 亚洲免费中文| 校园激情久久| 国产精品久久国产精麻豆99网站| 欧美电影电视剧在线观看| 国产一区二区三区av电影| 亚洲午夜久久久久久尤物 | 久久精品日韩| 久久精品一区二区国产| 国产精品久久久久久久久免费| 亚洲全黄一级网站| 亚洲日本中文字幕区| 久久久综合激的五月天| 久久久精品一区| 国产无遮挡一区二区三区毛片日本| 日韩小视频在线观看专区| 亚洲精品人人| 欧美精品一区三区在线观看| 欧美高清视频免费观看| 亚洲成人资源网| 久久亚洲春色中文字幕| 欧美成人在线免费视频| 亚洲精品美女久久7777777| 久久久亚洲国产美女国产盗摄| 久久女同精品一区二区| 韩曰欧美视频免费观看| 久久精品夜色噜噜亚洲aⅴ| 久久综合九色综合欧美狠狠| 韩国在线一区| 免费人成网站在线观看欧美高清| 欧美成人首页| 在线亚洲一区| 国产精品视频xxx| 欧美综合激情网| 欧美电影免费| 一道本一区二区| 国产精品久久久久秋霞鲁丝| 午夜久久久久久| 免费高清在线一区| 亚洲人妖在线| 国产精品久在线观看| 午夜综合激情| 亚洲第一成人在线| 宅男噜噜噜66一区二区| 国产精品xxx在线观看www| 欧美一区午夜精品| 亚洲国产精品福利| 欧美一区二区三区精品| 极品少妇一区二区三区精品视频| 欧美福利影院| 亚洲欧美日韩精品一区二区| 媚黑女一区二区| 亚洲一区二区视频在线观看| 国产一区二区三区高清| 欧美高清视频一二三区| 亚洲欧美日韩国产一区| 亚洲国产裸拍裸体视频在线观看乱了中文 | 亚洲女ⅴideoshd黑人| 久久天天躁狠狠躁夜夜爽蜜月| 亚洲伦理精品| 国产日韩一区在线| 欧美—级高清免费播放| 校园春色综合网| 亚洲六月丁香色婷婷综合久久| 久久精品水蜜桃av综合天堂| 夜夜嗨av一区二区三区四区 | 欧美日在线观看| 欧美大成色www永久网站婷| 亚洲午夜极品| 亚洲精品网站在线播放gif| 国产欧美韩日| 欧美午夜在线观看| 免费欧美电影| 久久国产精品久久久久久电车 | 亚洲精品孕妇| 国产精品网站在线观看| 欧美一区二区三区的| 99成人免费视频| 欧美大片18| 久久久亚洲影院你懂的| 亚洲永久在线| 99re6这里只有精品| 亚洲成人在线视频网站| 国产一区清纯| 国产欧美va欧美va香蕉在| 欧美视频一区在线| 欧美区高清在线| 嫩草影视亚洲| 免费日韩视频| 欧美成人高清| 嫩草国产精品入口| 快射av在线播放一区| 久久精品系列| 久久精品视频99| 久久久91精品国产一区二区三区| 亚欧成人在线| 欧美在线啊v一区| 久久国产视频网| 久久国产精品99精品国产| 性欧美xxxx视频在线观看| 亚洲免费在线视频| 性欧美18~19sex高清播放| 午夜精品区一区二区三| 亚洲欧美日韩一区二区在线 | 美女精品网站| 欧美激情第二页| 亚洲精品久久久久久久久| 亚洲国产另类久久精品| 亚洲精品免费在线观看| 日韩午夜视频在线观看| 亚洲深夜福利| 欧美一区二区三区啪啪| 久久久在线视频| 欧美人与性动交α欧美精品济南到| 欧美日韩精品一区二区三区四区| 欧美日韩综合视频| 国产日韩欧美一二三区| 精品动漫3d一区二区三区免费版 | 亚洲国产91| 洋洋av久久久久久久一区| 一本色道久久综合亚洲精品小说| 中日韩美女免费视频网站在线观看 | 另类av一区二区| 欧美日韩精品综合| 国产精品久久久久久久久免费桃花| 国产女主播在线一区二区| 在线观看欧美精品| 日韩亚洲视频| 久久狠狠久久综合桃花| 久久午夜av| 99精品国产高清一区二区| 亚洲欧美一区在线| 欧美成人一品| 国产精品一区二区黑丝| 永久免费精品影视网站| 99精品欧美| 久久综合九色综合网站| 亚洲第一天堂无码专区| 亚洲一级在线| 亚洲国产二区| 久久婷婷影院| 久久久久国产一区二区| 欧美日本韩国一区| 国产毛片精品视频| 99v久久综合狠狠综合久久| 久久久久99| 在线亚洲免费| 免费在线欧美黄色| 国产日本欧美视频| 99精品视频免费在线观看| 久久精品中文| av成人天堂| 欧美成人日韩| 在线免费不卡视频| 欧美一区二区高清| 亚洲国产裸拍裸体视频在线观看乱了 |