例子中的一句代碼,用來取得用戶從瀏覽器的輸入信息。
cgicc::form_iterator name = cgi.getElement("name");
之前已經創建好Cgicc類。想研究一下Cgicc類究竟做了什么。
可以猜測,在調用getElement之前,Cgicc已經存儲好了所有的輸入。
Cgicc的構造函數主體如下:
CGICCNS Cgicc::Cgicc(reader_function_t stream_reader)
: fEnvironment(stream_reader)
{
fFormData.reserve(40);
fFormFiles.reserve(5);
if(stringsAreEqual(getEnvironment().getRequestMethod(), "post"))
parseFormInput(getEnvironment().getPostData());
else
parseFormInput(getEnvironment().getQueryString());
}
在初始化參數表中,首先創建一個CgiEnvironment的類的實例,用來初始化系統信息,讀取從服務器傳來的各個參數。其中包括請求的方法(POST或者GET)。
然后為兩個vector分配存儲空間。然后通過查看請求的方法調用相應的函數。下面以post為例:
getEnvironment是一個內聯函數,用來返回Cgicc的私有變量fEnvironment.fEnvironment的getPostData()方法返回其私有變量fPostData(std::string);
而在CgiEnvirontment的構造函數中:
STDNS auto_ptr<char> temp(new char[getContentLength()]);
// use the appropriate reader function
if(stream_reader == NULL) {
STDNS cin.read(temp.get(), getContentLength());
if((unsigned long)STDNS cin.gcount() != getContentLength())
throw STDNS runtime_error("I/O error");
}
else {
// user specified a reader function
if((*stream_reader)
(temp.get(), getContentLength()) != getContentLength())
throw STDNS runtime_error("I/O error");
}
fPostData = STDNS string(temp.get(), getContentLength());
cin讀取用戶請求并存至temp中,并復制至fPostData中。這就是我們要處理的數據。(temp使用標準庫的auto_ptr,會自動銷毀)。
對fPostData的進一步處理則由Cgicc::parserFormInput來做。而parserFormInput則通過對整個字符串的解析將其分成鍵值對,形成一個FormEntry對象,并存入
fFormData(vector)中。所以,在cgicc類創建之后就可以直接通過查詢相應的輸入數據。getElement利用標準庫算法find_if在vector中查找相應的值。STL中一般
用迭代器查找時若無結果都會將迭代器置于容器后一位,所以判斷是否查找成功的代碼應該這樣寫:
if(name != cgi.getElements().end()) {
// iterator refers to a valid element
}