L表示long指針
這是為了兼容Windows 3.1等16位操作系統遺留下來(lái)的,在win32中以及其他的32為操作系統中, long指針和near指針及far修飾符都是為了兼容的作用。沒(méi)有實(shí)際意義。
P表示這是一個(gè)指針;C表示是一個(gè)常量;T表示在Win32環(huán)境中, 有一個(gè)_T宏,這個(gè)宏用來(lái)表示你的字符是否使用UNICODE, 如果你的程序定義了UNICODE或者其他相關(guān)的宏,那么這個(gè)字符或者字符串將被作為UNICODE字符串,否則就是標準的ANSI字符串;STR表示這 個(gè)變量是一個(gè)字符串。所以L(fǎng)PCTSTR就表示一個(gè)指向常固定地址的可以根據一些宏定義改變語(yǔ)義的字符串。同樣, LPCSTR就只能是一個(gè)ANSI字符串,在程序中我們大部分時(shí)間要使用帶T的類(lèi)型定義。 LPCTSTR == const TCHAR *,
CString 和 LPCTSTR 可以說(shuō)通用。 原因在于CString定義的自動(dòng)類(lèi)型轉換,沒(méi)什么奇特的,最簡(jiǎn)單的C++操作符重載而已。
常量字符串a(chǎn)nsi和unicode的區分是由宏_T來(lái)決定的。但是用_T("abcd")時(shí), 字符串"abcd"就會(huì )根據編譯時(shí)的是否定一_UNICODE來(lái)決定是char* 還是 w_char*。 同樣,TCHAR 也是相同目的字符宏。 看看定義就明白了。簡(jiǎn)單起見(jiàn),下面只介紹 ansi 的情況,unicode 可以類(lèi)推。
ansi情況下,LPCTSTR 就是 const char*, 是常量字符串(不能修改的)。
而LPTSTR 就是 char*, 即普通字符串(非常量,可修改的)。
這兩種都是基本類(lèi)型, 而CString 是 C++類(lèi), 兼容這兩種基本類(lèi)型是最起碼的任務(wù)了。
由于const char* 最簡(jiǎn)單(常量,不涉及內存變更,操作迅速), CString 直接定義了一個(gè)類(lèi)型轉換函數 operator LPCTSTR() {......}, 直接返回他所維護的字符串。
當你需要一個(gè)const char* 而傳入了CString時(shí), C++編譯器自動(dòng)調用 CString重載的操作符 LPCTSTR()來(lái)進(jìn)行隱式的類(lèi)型轉換。 當需要CString , 而傳入了 const char* 時(shí)(其實(shí) char* 也可以),C++編譯器則自動(dòng)調用CString的構造函數來(lái)構造臨時(shí)的 CString對象。
因此CString 和 LPCTSTR 基本可以通用。
但是 LPTSTR又不同了,他是 char*, 意味著(zhù)你隨時(shí)可能修改里面的數據,這就需要內存管理了(如字符串變長(cháng),原來(lái)的存貯空間就不夠了,則需要重新調整分配內存)。
所以 不能隨便的將 const char* 強制轉換成 char* 使用。
樓主舉的例子
LPSTR lpstr = (LPSTR)(LPCTSTR)string;
就是這種不安全的使用方法。
這個(gè)地方使用的是強制類(lèi)型轉換,你都強制轉換了,C++編譯器當然不會(huì )拒絕你,但同時(shí)他也認為你確實(shí)知道自己要做的是什么。因此是不會(huì )給出警告的。 強制的任意類(lèi)型轉換是C(++)的一項強大之處,但也是一大弊端。
這一問(wèn)題在 vc6 以后的版本(僅針對vc而言)中得到逐步的改進(jìn)(你需要更明確的類(lèi)型轉換聲明)。
其實(shí)在很多地方都可以看到類(lèi)似“LPSTR lpstr = (LPSTR)(LPCTSTR)string”地用法,
這 種情況一般是函數的約束定義不夠完善的原因, 比如一個(gè)函數接受一個(gè)字符串參數的輸入,里面對該字符串又沒(méi)有任何的修改,那么該參數就應該定義成 const char*, 但是很多初學(xué)者弄不清const地用法,或者是懶, 總之就是隨意寫(xiě)成了 char* 。 這樣子傳入CString時(shí)就需要強制的轉換一下。
這種做法是不安全的,也是不被建議的用法,你必須完全明白、確認該字符串沒(méi)有被修改。CString 轉換到 LPTSTR (char*), 預定的做法是調用CString的GetBuffer函數,使用完畢之后一般都要再調ReleaseBuffer函數來(lái)確認修改 (某些情況下也有不調用ReleaseBuffer的,同樣你需要非常明確為什么這么做時(shí)才能這樣子處理,一般應用環(huán)境可以不考慮這種情況)。 同時(shí)需要注意的是, 在GetBuffer 和 ReleaseBuffer之間,CString分配了內存交由你來(lái)處理,因此不能再調用其他的CString函數。
CString 轉LPCTSTR:
CString cStr;
const char *lpctStr=(LPCTSTR)cStr;
LPCTSTR轉CString:
LPCTSTR lpctStr;
CString cStr=lpctStr;