2010年2月25日 星期四

LPTSTR,LPCSTR,LPCTSTR,LPSTR

 LPTSTR、LPCSTR、LPCTSTR、LPSTR的來源及意義


UNICODE:它是用兩個字節表示一個字符的方法。比如字符'A'在ASCII下面是一個字符,可'A'在UNICODE下面是兩個字符,高字符用0填充,而且漢字'程'在ASCII下面是兩個字節,而在UNICODE下仍舊是兩個字節
。UNICODE的用處就是定長表示世界文字,據統計,用兩個字節可以編碼現存的所有文字而沒有二義。

  
MBCS,它是多字節字符集,它是不定長表示世界文字的編碼。MBCS表示英文字母時就和ASCII一樣(這也是我們容易把MBCS和ASCII搞混的原因),但表示其他文字時就需要用多字節。  


WINDOWS 下面的程序設計可以支持MBCS和UNICODE兩種編碼的字符串,具體用那種就看你定義了MBCS宏還是UNICODE宏。MBCS宏對應的字符串指針 是char*也就是LPSTR,UNICODE對應的指針是unsigned   short*也就是LPWSTR,為了寫程序方便微軟定義了類型LPTSTR,在MBCS下他就是char*,   在UNICODE下它是unsigned   char*,這樣你就可以重定義一個宏進行不同字符集的轉換了。

LPTSTR、LPCSTR、LPCTSTR、LPSTR的意義:
LPSTR:32bit指針 指向一個字符串,每個字符佔1字節
LPCSTR:32-bit指針 指向一個常字符串,每個字符佔1字節
LPCTSTR:32-bit指針 指向一個常字符串,每字符可能佔1字節或2字節,取決於Unicode是否定義
LPTSTR:32-bit指針 每字符可能佔1字節或2字節,取決於Unicode是否定義

Windows使用兩種字符集ANSI和UNICODE,前者就是通常使用的單字節方式,但這種方式處理象中文這樣的雙字節字符不方便,容易出現半個漢字的情況。而後者是雙字節方式,方便處理雙字節字符。
WindowsNT 的所有與字符有關的函數都提供兩種方式的版本,而Windows9x只支持ANSI方式。_T一般同字常數相關,如_T("Hello"。如果你編譯一個 程序為ANSI方式,_T實際不起任何作用。而如果編譯一個程序為UNICODE方式,則編譯器會把"Hello"字符串以UNICODE方式保存。_T 和_L的區別在於,_L不管你是以什麼方式編譯,一律UNICODE方式保存.
Windows核心編程的第一章。
L是表示字符串資源為Unicode的。
比如
wchar_t Str[] = L"Hello World!";
這個就是雙子節存儲字符了。

_T是一個適配的宏~

#ifdef _UNICODE的時候
_T就是L
沒有#ifdef _UNICODE的時候
_T就是ANSI的。

比如
LPTSTR lpStr = new TCHAR[32];
TCHAR* szBuf = _T("Hello");
以上兩句使得無論是在UNICODE編譯條件下都是正確編譯的。

而且MS推薦你使用相匹配的字符串函數。
比如處理LPTSTR或者LPCTSTR 的時候,不要用strlen ,而是要用_tcslen

否則在UNICODE的編譯條件下,strlen不能處理 wchar_t*的字符串。
T是非常有意思的一個符號(TCHAR、LPCTSTR、LPTSTR、_T()、_TEXT()...),它表示使用一種中間類型,既不明確表示使用 MBCS,也不明確表示使用 UNICODE。那到底使用哪種字符集?編譯的時候才決定

在vc++中有著各種字符串的表示法,如您所說。        
首先char*   是指向ANSI字符數組的指針,其中每個字符佔據8位(有效數據是除掉最高位的其他7位),這裡保持了與傳統的C,C++的兼容。      
 LP的含義是長指針(long   pointer)。
LPSTR是一個指向以『\0』結尾的ANSI字符數組的指針,與char*可以互換使用,在win32中較多地使用 LPSTR。而LPCSTR中增加的『C'的含義是「CONSTANT」(常量),表明這種數據類型的實例不能被使用它的API函數改變,除此之外,它與 LPSTR是等同的。    
    為了滿足程序代碼國際化的需要,業界推出了Unicode標準,它提供了一種簡單和一致的表達字符串的方法,所有字符中的字節都是16位的值,其數量也可 以滿足差不多世界上所有書面語言字符的編碼需求,開發程序時使用Unicode(類型為wchar_t)是一種被鼓勵的做法。    
    LPWSTR與LPCWSTR由此產生,它們的含義類似於LPSTR與LPCSTR,只是字符數據是16位的wchar_t而不是char。        
 然後為了實現兩種編碼的通用,提出了TCHAR的定義:    
如果定義_UNICODE,聲明如下:     typedef   wchar_t   TCHAR;    
如果沒有定義_UNICODE,則聲明如下:     typedef   char   TCHAR;      
LPTSTR和LPCTSTR中的含義就是每個字符是這樣的TCHAR。        
CString類中的字符就是被聲明為TCHAR類型的,它提供了一個封裝好的類供用戶方便地使用。
如果您還需要進一步的信息,請參看http://msdn.microsoft.com/library/default.asp?url=/library/en-us/vclib/html/_crt_data_type_mappings.asp等其他有關信息。

LPTSTR、LPCSTR、LPCTSTR、LPSTR之間的轉換:

如何理解LPCTSTR類型?
2007-11-10 21:43

L表示long指針
這是為了兼容Windows 3.1等16位操作系統遺留下來的,在win32中以及其他的32為操作系統中, long指針和near指針及far修飾符都是為了兼容的作用。沒有實際意義。

P表示這是一個指針

C表示是一個常量
T表示在Win32環境中, 有一個_T宏
這個宏用來表示你的字符是否使用UNICODE, 如果你的程序定義了UNICODE或者其他相關的宏,那麼這個字符或者字符串將被作為UNICODE字符串,否則就是標準的ANSI字符串。

STR表示這個變量是一個字符串

所以LPCTSTR就表示一個指向常固定地址的可以根據一些宏定義改變語義的字符串。
同樣, LPCSTR就只能是一個ANSI字符串,在程序中我們大部分時間要使用帶T的類型定義。
LPCTSTR == const TCHAR *

CString 和 LPCTSTR 可以說通用。 原因在於CString定義的自動類型轉換,沒什麼奇特的,最簡單的C++操作符重載而已。

常 量字符串ansi和unicode的區分是由宏_T來決定的。但是用_T("abcd")時, 字符串"abcd"就會根據編譯時的是否定一_UNICODE來決定是char* 還是 w_char*。 同樣,TCHAR 也是相同目的字符宏。 看看定義就明白了。簡單起見,下面只介紹 ansi 的情況,unicode 可以類推。

ansi情況下,LPCTSTR 就是 const char*, 是常量字符串(不能修改的)。
而LPTSTR 就是 char*, 即普通字符串(非常量,可修改的)。
這兩種都是基本類型, 而CString 是 C++類, 兼容這兩種基本類型是最起碼的任務了。

由於const char* 最簡單(常量,不涉及內存變更,操作迅速), CString 直接定義了一個類型轉換函數
operator LPCTSTR() {......}, 直接返回他所維護的字符串。

當你需要一個const char* 而傳入了CString時, C++編譯器自動調用 CString重載的操作符 LPCTSTR()來進行隱式的類型轉換。
當需要CString , 而傳入了 const char* 時(其實 char* 也可以),C++編譯器則自動調用CString的構造函數來構造臨時的 CString對象。

因此CString 和 LPCTSTR 基本可以通用。


但是 LPTSTR又不同了,他是 char*, 意味著你隨時可能修改裡面的數據,這就需要內存管理了(如字符串變長,原來的存貯空間就不夠了,則需要重新調整分配內存)。
所以 不能隨便的將 const char* 強制轉換成 char* 使用。
樓主舉的例子
LPSTR lpstr = (LPSTR)(LPCTSTR)string;
就是這種不安全的使用方法。

這個地方使用的是強制類型轉換,你都強制轉換了,C++編譯器當然不會拒絕你,但同時他也認為你確實知道自己要做的是什麼。因此是不會給出警告的。
強制的任意類型轉換是C(++)的一項強大之處,但也是一大弊端。這一問題在 vc6 以後的版本(僅針對vc而言)中得到逐步的改進(你需要更明確的類型轉換聲明)。

其實在很多地方都可以看到類似
LPSTR lpstr = (LPSTR)(LPCTSTR)string;
地用法,這種情況一般是函數的約束定義不夠完善的原因
, 比如一個函數接受一個字符串參數的輸入,裡面對該字符串又沒有任何的修改,那麼該參數就應該定義成 const char*, 但是很多初學者弄不清const地用法,或者是懶, 總之就是隨意寫成了 char* 。 這樣子傳入CString時就需要強制的轉換一下。

這種做法是不安全的,也是不被建議的用法,你必須完全明白、確認該字符串沒有被修改

CString 轉換到 LPTSTR (char*), 預定的做法是調用CString的GetBuffer函數,使用完畢之後一般都要再調用ReleaseBuffer函數來確認修改 (某些情況下也有不調用ReleaseBuffer的,同樣你需要非常明確為什麼這麼做時才能這樣子處理,一般應用環境可以不考慮這種情況)。

同時需要注意的是, 在GetBuffer 和 ReleaseBuffer之間,CString分配了內存交由你來處理,因此不能再調用其他的CString函數。
CString 轉LPCTSTR:
CString cStr;
const char *lpctStr=(LPCTSTR)cStr;

LPCTSTR轉CString:
LPCTSTR lpctStr;
CString cStr=lpctStr;

combo box控件的使用

combo box控件的使用
1.你在編輯狀態下點那個控件的向下的三角形,就出冒出來一個可以調高度的東東。將高度調高,否則在執行時會不能顯示下拉選項。
2.為combo box添加選項,在編輯狀態下選combo box控件的屬性,選Data標籤,在編輯框中添加選項,按Ctrl-Enter來添加下一個選項。
3.為combo box添加變量
combo box有兩個變量,CComboBox類變量和CString變量。
CComboBox變量用來設置combo box的屬性,一般在cdialog類中的oninitdialog()函數中使用。添加方法是在classwizard中添加變量,添加時 Category中選Control。例如:CComboBox.SetCurSel(0)用來設置下拉菜單初始值,具體看MSDN。   
CString變量用來存儲下拉菜單所選的選項值。同樣,當選項發生變化時要使用UpdateData(TRUE)函數更新該變量的值。


-----------------------------------------------------------------------------------------------------------------------------------
一、初始化Combo Box控件
(1)在資源視圖中,選擇控件的"Properties",然後在Data選項卡里為控件添加初始化數據,換行是按Ctrl+Enter,然後,再修改Style選項卡里的Type屬性為Drop List。
(2)在程序初始化時動態添加
CString strTemp;
((CComboBox*)GetDlgItem(IDC_COMBO_CF))->ResetContent();//消除ID為IDC_COMBO_CF現有所有內容
for(int i=1;i<=100;i++)
{
   strTemp.Format("%d",i);
   ((CComboBox*)GetDlgItem(IDC_COMBO_CF))->AddString(strTemp); //為控件添加初始化數據
}
(3)刪除
DeleteString( UINT nIndex )//刪除指定行,
(4)插入
InsertString( int nIndex, LPCTSTR lpszItem )//將行插入到指定位置
(5)查找
FindString( int nStartAfter, LPCTSTR lpszItem )//可以在當前所有行中查找指定的字符傳的位置,nStartAfter指明從那一行開始進行查找。
int SelectString( int nStartAfter, LPCTSTR lpszItem )//可以選中包含指定字符串的行
二、如何控制Combo Box的下拉長度
1,首先要知道兩點:一、那就是在設計界面裡,點擊一下Combo Box的下拉箭頭,此時出現的調整框就是Combo Box的下拉調整框。
2,二、屬性裡有個 No integral height 鉤選項,表示最大長度為設計長度,如果實際內容比設計長度多,就出現滾動條,少就以實際長度顯示。
三、選擇其中的某行
(1)選中:
int iPos=((CComboBox*)GetDlgItem(IDC_COMBO_CF))->GetCurSel();//當前選中的行。
(2)設置
((CComboBox*)GetDlgItem(IDC_COMBO_CF))->SetCurSel(n)//設置第n行內容為顯示的內容。
四、取得Combo Box框內容
(1)取當前內容
((CComboBox*)GetDlgItem(IDC_COMBO_CF))->GetWindowText(strTemp);//將獲取的值存放到CString類型變量strTemp中。
如果定義了關聯的變量,例如m_combo,可以直接獲取:m_combo->GetWindowText(strTemp);

(2)取其他行內容
((CComboBox*)GetDlgItem(IDC_COMBO_CF))->GetLBText(n,strTemp); //其中n為從0開始的索引值
五、獲取當前選擇的行數
例如:((CComboBox*)GetDlgItem(IDC_COMBO_CF))->GetCurSel()返回的是當前選中值的行數,是整型。
或者:UINT m_row = m_combo->GetCurSel()
六、獲得焦點
通常要判斷控件是否獲得了焦點,可以用GetFocus()函數
例如:if(GetFocus()==GetDlgItem(IDC_EDIT_VALUE2))//判斷焦點是否在編輯框IDC_EDIT_VALUE2內。
但是combobox 的焦點不同,因為它是由edit和listbox兩部分組成的
所以獲得焦點要用GetParent():if ((GetFocus()->GetParent())==GetDlgItem(IDC_COMBO_CF))
-----------------------------------------------------------------------------------------------------------------------------------
一、如何添加/刪除Combo Box內容
1,在Combo Box控件屬性的Data標籤裡面添加,一行表示Combo Box下拉列表中的一行。換行用ctrl+回車。
2,在程序初始化時動態添加
如: //控件內容初始化
CString strTemp;
((CComboBox*)GetDlgItem(IDC_COMBO_CF))->ResetContent();//消除現有所有內容
for(int i=1;i<=100;i++)
{
   strTemp.Format("%d",i);
   ((CComboBox*)GetDlgItem(IDC_COMBO_CF))->AddString(strTemp);
}
3,下拉的時候添加
如: CString strTemp;
int iCount=((CComboBox*)GetDlgItem(IDC_COMBO_CF))->GetCount();//取得目前已經有的行數
if(iCount<1)//防止重複多次添加
{
   ((CComboBox*)GetDlgItem(IDC_COMBO_CF))->ResetContent();
   for(int i=1;i<=100;i++)
   {
    strTemp.Format("%d",i);
    ((CComboBox*)GetDlgItem(IDC_COMBO_CF))->AddString(strTemp);
   }
}
4,刪除
DeleteString( UINT nIndex )//刪除指定行,
5,插入
InsertString( int nIndex, LPCTSTR lpszItem )//將行插入到指定位置
6,查找
FindString( int nStartAfter, LPCTSTR lpszItem )//可以在當前所有行中查找指定的字符傳的位置,nStartAfter指明從那一行開始進行查找。
int SelectString( int nStartAfter, LPCTSTR lpszItem )//可以選中包含指定字符串的行
二、如何控制Combo Box的下拉長度
1,首先要知道兩點:一、那就是在設計界面裡,點擊一下Combo Box的下拉箭頭,此時出現的調整框就是Combo Box的下拉調整框。
2,二、屬性裡有個 No integral height 鉤選項,表示最大長度為設計長度,如果實際內容比設計長度多,就出現滾動條,少就以實際長度顯示。
三、選擇其中的某行
1,選中:
int iPos=((CComboBox*)GetDlgItem(IDC_COMBO_CF))->GetCurSel();//當前選中的行。
2,設置
((CComboBox*)GetDlgItem(IDC_COMBO_CF))->SetCurSel(n)//設置第n行內容為顯示的內容。
四、取得Combo Box框內容
1取當前內容
((CComboBox*)GetDlgItem(IDC_COMBO_CF))->GetWindowText(strTemp);

2取其他行內容
((CComboBox*)GetDlgItem(IDC_COMBO_CF))->GetLBText(n,strTemp);
-----------------------------------------------------------------------------------------------------------------------------------
兩個DropDownList 控件,選擇第一個後,第二個根據第一個搜索數據庫表內容填充信息。
問題是,第一個DropDownList 控件,我實現了SelectedIndexChanged事件,不過好像沒反應啊。
謝謝 
把Postback設成True
AutoPostBack="True"