使用WinINet和WinHTTP实现Http访问
飘飘白云 l_zhaohui@163.com <chsdate year="2007" month="11" day="30" islunardate="False" isrocdate="False" w:st="on">2007-11-30</chsdate>
Http访问有两种方式,GET和POST,就编程来说GET方式相对简单点,它不用向服务器提交数据,在这个例程中我使用POST方式,提交数据value1与value2,并从服务器得到他们的和(value1 + value2)。
为实现Http访问,微软提供了二套API:WinINet, WinHTTP。WinHTTP比WinINet更加安全和健壮,可以这么认为WinHTTP是WinINet的升级版本。这两套API包含了很多相似的函数与宏定义,呵呵,详细对比请查阅msdn中的文章“Porting WinINet Applications to WinHTTP”,在线MSDN连接:http://msdn2.microsoft.com/en-us/library/aa384068.aspx。在这个例程中,通过一个宏的设置来决定是使用WinHttp还是WinINet。代码如下:
#define USE_WINHTTP //Comment this line to user wininet.
下面来说说实现Http访问的流程(两套API都一样的流程):
1, 首先我们打开一个Session获得一个HINTERNET session句柄;
2, 然后我们使用这个session句柄与服务器连接得到一个HINTERNET connect句柄;
3, 这样我们使用这个connect句柄来打开Http 请求得到一个HINTERNET request句柄;
4, 这样我们就可以使用这个request句柄来发送数据与读取从服务器返回的数据;
5, 最后依次关闭request,connect,session句柄。
在这个例程中以上各个流程都进行了简单封装,以便对比两套API函数的些许差异。下面让源代码说话,原工程是一个windows控制台工程,你可以很容易通过拷贝代码重建工程。
另:如果你从服务器得到的返回数据是utf8格式的文本数据,你将需要对返回的数据进行转换才能争取显示(中文,日文等)。仅供参考,转换为ATL CStringW的函数见下:
CStringWGetStringWFromUtf8(
const
std::string
&
str)
{
intlen=MultiByteToWideChar(CP_UTF8,0,str.c_str(),int(str.length()),0,0);
CStringWbuf;
WCHAR*dd=buf.GetBuffer(len);
len=MultiByteToWideChar(CP_UTF8,0,str.c_str(),int(str.length()),dd,len);
buf.ReleaseBuffer(len);
returnbuf;
}
完整代码如下:
//
HttpPost.cppwrittenbyl_zhaohui@163.com2
//
2007/11/30
3
#include
"
stdafx.h
"
4
#include
<
windows.h
>
5
#include
<
stdio.h
>
6
#include
<
stdlib.h
>
7

8
#define_ATL_CSTRING_EXPLICIT_CONSTRUCTORS9
#include
<
atlbase.h
>
10
#include
<
atlstr.h
>
11

12
#defineUSE_WINHTTP
//
Commentthislinetouserwininet.
13
#ifdefUSE_WINHTTP14
#include
<
winhttp.h
>
15
#pragmacomment(lib,
"
winhttp.lib
"
)16
#
else
17
#include
<
wininet.h
>
18
#pragmacomment(lib,
"
wininet.lib
"
)19
#endif20
#defineBUF_SIZE(
1024
)21

22
//
CrackedUrl
23

class
CrackedUrl
{24
intm_scheme;25
CStringWm_host;26
intm_port;27
CStringWm_path;28
public:29
CrackedUrl(LPCWSTRurl)30


{31

URL_COMPONENTSuc=
{0};32
uc.dwStructSize=sizeof(uc);33

34
constDWORDBUF_LEN=256;35

36
WCHARhost[BUF_LEN];37
uc.lpszHostName=host;38
uc.dwHostNameLength=BUF_LEN;39

40
WCHARpath[BUF_LEN];41
uc.lpszUrlPath=path;42
uc.dwUrlPathLength=BUF_LEN;43

44
WCHARextra[BUF_LEN];45
uc.lpszExtraInfo=extra;46
uc.dwExtraInfoLength=BUF_LEN;47

48
#ifdefUSE_WINHTTP49

if(!WinHttpCrackUrl(url,0,ICU_ESCAPE,&uc))
{50
printf("Error:WinHttpCrackUrlfailed!/n");51
}52

53
#else54

if(!InternetCrackUrl(url,0,ICU_ESCAPE,&uc))
{55
printf("Error:InternetCrackUrlfailed!/n");56
}57
#endif58
m_scheme=uc.nScheme;59
m_host=host;60
m_port=uc.nPort;61
m_path=path;62
}63

64
intGetScheme()const65


{66
returnm_scheme;67
}68

69
LPCWSTRGetHostName()const70


{71
returnm_host;72
}73

74
intGetPort()const75


{76
returnm_port;77
}78

79
LPCWSTRGetPath()const80


{81
returnm_path;82
}83

84
staticCStringAUrlEncode(constchar*p)85


{86

if(p==0)
{87
returnCStringA();88
}89

90
CStringAbuf;91

92

for(;;)
{93
intch=(BYTE)(*(p++));94

if(ch=='/0')
{95
break;96
}97

98

if(isalnum(ch)||ch=='_'||ch=='-'||ch=='.')
{99
buf+=(char)ch;100
}101

elseif(ch=='')
{102
buf+='+';103
}104

else
{105
charc[16];106
wsprintfA(c,"%%%02X",ch);107
buf+=c;108
}109
}110

111
returnbuf;112
}113
}
;114

115
//
CrackedUrl
116
HINTERNETOpenSession(LPCWSTRuserAgent
=
0
)117

{118
#ifdefUSE_WINHTTP119
returnWinHttpOpen(userAgent,NULL,NULL,NULL,NULL);;120
#else121
returnInternetOpen(userAgent,INTERNET_OPEN_TYPE_PRECONFIG,NULL,NULL,0);122
#endif123
}
124

125
HINTERNETConnect(HINTERNEThSession,LPCWSTRserverAddr,
int
portNo)126

{127
#ifdefUSE_WINHTTP128
returnWinHttpConnect(hSession,serverAddr,(INTERNET_PORT)portNo,0);129
#else130
returnInternetConnect(hSession,serverAddr,portNo,NULL,NULL,INTERNET_SERVICE_HTTP,0,0);131
#endif132
}
133

134
HINTERNETOpenRequest(HINTERNEThConnect,LPCWSTRverb,LPCWSTRobjectName,
int
scheme)135

{136
DWORDflags=0;137
#ifdefUSE_WINHTTP138

if(scheme==INTERNET_SCHEME_HTTPS)
{139
flags|=WINHTTP_FLAG_SECURE;140
}141

142
returnWinHttpOpenRequest(hConnect,verb,objectName,NULL,NULL,NULL,flags);143

144
#else145

if(scheme==INTERNET_SCHEME_HTTPS)
{146
flags|=INTERNET_FLAG_SECURE;147
}148

149
returnHttpOpenRequest(hConnect,verb,objectName,NULL,NULL,NULL,flags,0);150
#endif151
}
152

153
BOOLAddRequestHeaders(HINTERNEThRequest,LPCWSTRheader)154

{155
SIZE_Tlen=lstrlenW(header);156
#ifdefUSE_WINHTTP157
returnWinHttpAddRequestHeaders(hRequest,header,DWORD(len),WINHTTP_ADDREQ_FLAG_ADD);158
#else159
returnHttpAddRequestHeaders(hRequest,header,DWORD(len),HTTP_ADDREQ_FLAG_ADD);160
#endif161
}
162

163
BOOLSendRequest(HINTERNEThRequest,
const
void
*
body,DWORDsize)164

{165
#ifdefUSE_WINHTTP166
returnWinHttpSendRequest(hRequest,0,0,const_cast<void*>(body),size,size,0);167
#else168
returnHttpSendRequest(hRequest,0,0,const_cast<void*>(body),size);169
#endif170
}
171

172
BOOLEndRequest(HINTERNEThRequest)173

{174
#ifdefUSE_WINHTTP175
returnWinHttpReceiveResponse(hRequest,0);176
#else177
//ifyouuseHttpSendRequestExtosendrequestthenuseHttpEndRequestinhere!178
returnTRUE;179
#endif180
}
181

182
BOOLQueryInfo(HINTERNEThRequest,
int
queryId,
char
*
szBuf,DWORD
*
pdwSize)183

{184
#ifdefUSE_WINHTTP185
returnWinHttpQueryHeaders(hRequest,(DWORD)queryId,0,szBuf,pdwSize,0);186
#else187
returnHttpQueryInfo(hRequest,queryId,szBuf,pdwSize,0);188
#endif189
}
190

191
BOOLReadData(HINTERNEThRequest,
void
*
buffer,DWORDlength,DWORD
*
cbRead)192

{193
#ifdefUSE_WINHTTP194
returnWinHttpReadData(hRequest,buffer,length,cbRead);195
#else196
returnInternetReadFile(hRequest,buffer,length,cbRead);197
#endif198
}
199

200
void
CloseInternetHandle(HINTERNEThInternet)201

{202
if(hInternet)203


{204
#ifdefUSE_WINHTTP205
WinHttpCloseHandle(hInternet);206
#else207
InternetCloseHandle(hInternet);208
#endif209
}210
}
211

212
int
_tmain(
int
argc,_TCHAR
*
argv[])213

{214
HINTERNEThSession=0;215
HINTERNEThConnect=0;216
HINTERNEThRequest=0;217
CStringWstrHeader(L"Content-type:application/x-www-form-urlencoded/r/n");218

219
//Testdata220
CrackedUrlcrackedUrl(L"http://www.easy-creator.net/test2/add.asp");221
CStringAstrPostData("value1=10&value2=14");222

223
//Opensession.224
hSession=OpenSession(L"HttpPostbyl_zhaohui@163.com");225

if(hSession==NULL)
{226
printf("Error:Opensession!/n");227
return-1;228
}229

230
//Connect.231
hConnect=Connect(hSession,crackedUrl.GetHostName(),crackedUrl.GetPort());232

if(hConnect==NULL)
{233
printf("Error:Connectfailed!/n");234
return-1;235
}236

237
//Openrequest.238
hRequest=OpenRequest(hConnect,L"POST",crackedUrl.GetPath(),crackedUrl.GetScheme());239

if(hRequest==NULL)
{240
printf("Error:OpenRequestfailed!/n");241
return-1;242
}243

244
//Addrequestheader.245

if(!AddRequestHeaders(hRequest,strHeader))
{246
printf("Error:AddRequestHeadersfailed!/n");247
return-1;248
}249

250
//Sendpostdata.251

if(!SendRequest(hRequest,(constchar*)strPostData,strPostData.GetLength()))
{252
printf("Error:SendRequestfailed!/n");253
return-1;254
}255

256
//Endrequest257

if(!EndRequest(hRequest))
{258
printf("Error:EndRequestfailed!/n");259
return-1;260
}261

262
charszBuf[BUF_SIZE];263
DWORDdwSize=0;264
szBuf[0]=0;265

266
//Queryheaderinfo.267
#ifdefUSE_WINHTTP268
intcontextLengthId=WINHTTP_QUERY_CONTENT_LENGTH;269
intstatusCodeId=WINHTTP_QUERY_STATUS_CODE;270
intstatusTextId=WINHTTP_QUERY_STATUS_TEXT;271
#else272
intcontextLengthId=HTTP_QUERY_CONTENT_LENGTH;273
intstatusCodeId=HTTP_QUERY_STATUS_CODE;274
intstatusTextId=HTTP_QUERY_STATUS_TEXT;275
#endif276
dwSize=BUF_SIZE;277

if(QueryInfo(hRequest,contextLengthId,szBuf,&dwSize))
{278
szBuf[dwSize]=0;279
printf("Contentlength:[%s]/n",szBuf);280
}281

282
dwSize=BUF_SIZE;283

if(QueryInfo(hRequest,statusCodeId,szBuf,&dwSize))
{284
szBuf[dwSize]=0;285
printf("Statuscode:[%s]/n",szBuf);286
}287

288
dwSize=BUF_SIZE;289

if(QueryInfo(hRequest,statusTextId,szBuf,&dwSize))
{290
szBuf[dwSize]=0;291
printf("Statustext:[%s]/n",szBuf);292
}293

294
//readdata.295

for(;;)
{296
dwSize=BUF_SIZE;297

if(ReadData(hRequest,szBuf,dwSize,&dwSize)==FALSE)
{298
break;299
}300

301

if(dwSize<=0)
{302
break;303
}304

305
szBuf[dwSize]=0;306
printf("%s/n",szBuf);//Outputvalue=value1+value2307
}308

309
CloseInternetHandle(hRequest);310
CloseInternetHandle(hConnect);311
CloseInternetHandle(hSession);312

313
return0;314
}
315
本文介绍如何使用WinINet和WinHTTP实现HTTP POST请求,包括解析URL、建立会话、连接服务器、发送及接收数据等步骤。通过示例代码展示了两种API之间的差异。
2579

被折叠的 条评论
为什么被折叠?



