MFC向指定网站post数据

22 篇文章 1 订阅

最近有个需求需向指定网站Get数据,用于注册账号。例如:http://baidu.com  Get时需要携带账号和密码信息 如:http://baidu.com?code=abc&password=123456

当然还有经过md5加密过得签名。记得以前有做过类似的需求,网上搜也是一大堆,看的我有些眼花了。以下是我实现用到的两个接口:

CString Get_Http(CString url)
{
	CString strLogMsg, strRes;
	try
	{
		CInternetSession session;  
		CHttpFile *pFile = (CHttpFile *)session.OpenURL(url);  
		if ( pFile )
		{   
			DWORD dwStateCode = 0;
			pFile->QueryInfoStatusCode(dwStateCode);
			strLogMsg.Format(_T("Http QueryInfoStatusCode:%ld"), dwStateCode);			
			if (dwStateCode >= 400)
			{
				pFile->Close();
				delete pFile ;
				return _T("");
			}

			char buf[1024] = {'\0'};
			CString str;
			CString s ;

			UINT uSize = 0;

			UINT nTotal = 0;

			while (true)
			{
				memset(buf,0,1024);
				uSize = pFile->Read(buf,1024);
				if (uSize>0)
				{
					nTotal += uSize;
				}
				if (uSize < 1024 || uSize <= 0)
				{
					break;
				}

			}
			buf[nTotal] = '\0';
			strRes = buf;

			pFile->Close();
			delete pFile ;

		}
	}
	catch( CException *e )
	{
		TCHAR   szCause[255];
		e->GetErrorMessage(szCause, 255);		
		e->Delete();    
	} 
	return strRes;
}
BOOL GetMd5(CString strInput, CString &strOutput)
{  
	HCRYPTPROV hProv=NULL;
	if(CryptAcquireContext(&hProv,NULL,NULL,PROV_RSA_FULL,CRYPT_VERIFYCONTEXT)==FALSE)       //获得CSP中一个密钥容器的句柄
	{
		return FALSE;
	}
	HCRYPTPROV hHash=NULL;
	//初始化对数据流的hash,创建并返回一个与CSP的hash对象相关的句柄。这个句柄接下来将被    CryptHashData调用。
	if(CryptCreateHash(hProv,CALG_MD5,0,0,&hHash)==FALSE)
	{
		return FALSE;
	}

	DWORD lenOfBytes;	
	USES_CONVERSION;
	char* input = W2A(strInput);
	lenOfBytes = strlen(input);
	if(CryptHashData(hHash,(BYTE*)input,lenOfBytes,0)==FALSE)      //hash文件  
	{
		return FALSE;
	}		

	BYTE *pbHash = NULL;
	DWORD dwHashLen=sizeof(DWORD);
	if (!CryptGetHashParam(hHash,HP_HASHVAL,NULL,&dwHashLen,0))    
	{
		return FALSE;
	}
	pbHash=(byte*)malloc(dwHashLen);
	if(CryptGetHashParam(hHash,HP_HASHVAL,pbHash,&dwHashLen,0))//获得md5值 
	{
		for(DWORD i=0;i<dwHashLen;i++)         //输出md5值 
		{
			TCHAR str[1024]={0};
			CString strFilePartM=_T("");
			_stprintf(str,_T("%02x"),pbHash[i]);
			strOutput+=str;
		}
	} 
	free(pbHash);

	//善后工作
	if(CryptDestroyHash(hHash)==FALSE)          //销毁hash对象  
	{
		return FALSE;			
	}
	if(CryptReleaseContext(hProv,0)==FALSE)
	{
		return FALSE;		
	}
	return TRUE;
} 

最近在网上还找到了一个用c++封装的get和post的类,顺便贴出来下:

// CHttpClient 命令目标
#include <afxinet.h>
#include <string>
using namespace std ;
 
#define  IE_AGENT  _T("Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1; SV1; .NET CLR 2.0.50727)")
 
// 操作成功
#define SUCCESS        0
// 操作失败
#define FAILURE        1
// 操作超时
#define OUTTIME        2
 
class CHttpClient : public CObject
{
public :
    CHttpClient (LPCTSTR strAgent = IE_AGENT);
    virtual ~CHttpClient ( void);
 
    int HttpGet ( LPCTSTR strUrl , LPCTSTR strPostData , string & strResponse );
    int HttpPost ( LPCTSTR strUrl , LPCTSTR strPostData , string & strResponse );
	//,该函数用来对url进行编码。将非ASCII字符转化成"%XX%XX%XX"格式,isWide和isChangeEqualAndSlash分别用来决定是否将获取到的内容转成宽字符和是否转换'='和'/'转换出的url按照UTF-8编码
    static CString CHttpClient :: UrlEncode (CString strUnicode ,LPCTSTR url= NULL ,BOOL isWide =FALSE , BOOL isChangeEqualAndSlash= FALSE );
private :
    int ExecuteRequest ( LPCTSTR strMethod , LPCTSTR strUrl , LPCTSTR strPostData, string&strResponse );
    void Clear ();
 
private :
    CInternetSession * m_pSession ;
    CHttpConnection * m_pConnection ;
    CHttpFile *m_pFile ;
};

// HttpClient.cpp : 实现文件
//
#include "stdafx.h"
#include "MyHttp.h"
 
#define  BUFFER_SIZE  4*1024
 
#define  NORMAL_CONNECT             INTERNET_FLAG_KEEP_CONNECTION
#define  SECURE_CONNECT             NORMAL_CONNECT | INTERNET_FLAG_SECURE
#define  NORMAL_REQUEST             INTERNET_FLAG_RELOAD | INTERNET_FLAG_DONT_CACHE
#define  SECURE_REQUEST             NORMAL_REQUEST | INTERNET_FLAG_SECURE | INTERNET_FLAG_IGNORE_CERT_CN_INVALID
 
CString CHttpClient ::UrlEncode ( CString strUnicode ,LPCTSTR url ,BOOL isWide, BOOL isChangeEqualAndSlash )
{
    LPCWSTR unicode = T2CW( strUnicode );
     int len = WideCharToMultiByte ( CP_UTF8 , 0, unicode , - 1, 0 , 0 , 0 , 0 );
     if (! len)
         return strUnicode ;
     char *utf8 = new char [ len + 1 ];
     char *utf8temp = utf8;
    WideCharToMultiByte ( CP_UTF8 , 0, unicode , - 1, utf8 , len + 1 , 0 , 0 );
    utf8 [len ] = 0 ;   
    CString strTemp , strEncodeData ;
     if (url != NULL)
        strEncodeData = url;
    
     while ((* utf8) != 0)
     {
         if ((*utf8 )<= 127&&(* utf8 )>0 &&(* utf8)!= '+' &&(*utf8 )!= ' ' &&(*utf8 )!= '?'
             &&(* utf8)!= '%' &&(*utf8 )!= '#'&&(! isChangeEqualAndSlash ||((*utf8 )!= '='&&(* utf8 )!='/' )))
            strEncodeData +=( char)(* utf8 );
         else {
            strTemp . Format( _T ("%%%2x" ), ( BYTE )*utf8 );
            strEncodeData += strTemp ;
         }
         ++ utf8;
     }
 
     delete [] utf8temp ;
     if (isWide ==FALSE )
     {
        CStringA stra ( strEncodeData .GetBuffer (0 ));
         return ( LPCWSTR )stra . GetBuffer (0 );
     } else return strEncodeData ;
 
}
 
// CHttpClient 成员函数
CHttpClient ::CHttpClient ( LPCTSTR strAgent )
{
    m_pSession = new CInternetSession (strAgent );
    m_pConnection = NULL ;
    m_pFile = NULL ;
}
 
CHttpClient ::~CHttpClient ( void)
{
    Clear ();
     if (NULL != m_pSession )
     {
        m_pSession -> Close();
         delete m_pSession ;
        m_pSession = NULL ;
     }
}
 
void CHttpClient ::Clear ()
{
     if (NULL != m_pFile )
     {
        m_pFile -> Close();
         delete m_pFile ;
        m_pFile = NULL ;
     }
 
     if (NULL != m_pConnection )
     {
        m_pConnection -> Close();
         delete m_pConnection ;
        m_pConnection = NULL ;
     }
}
 
int CHttpClient ::ExecuteRequest (LPCTSTR strMethod , LPCTSTR strUrl , LPCTSTR strPostData, string&strResponse )
{
    CString strServer ;
    CString strObject ;
    DWORD dwServiceType ;
    INTERNET_PORT nPort ;
    strResponse = "" ;
 
    AfxParseURL ( strUrl, dwServiceType , strServer , strObject , nPort );
 
     if (AFX_INET_SERVICE_HTTP != dwServiceType && AFX_INET_SERVICE_HTTPS != dwServiceType )
     {
         return FAILURE ;
     }
 
     try
     {
        m_pConnection = m_pSession ->GetHttpConnection (strServer ,
            dwServiceType == AFX_INET_SERVICE_HTTP ? NORMAL_CONNECT : SECURE_CONNECT,
            nPort );
        m_pFile = m_pConnection ->OpenRequest (strMethod , strObject ,
             NULL , 1 , NULL , NULL ,
             ( dwServiceType == AFX_INET_SERVICE_HTTP ? NORMAL_REQUEST : SECURE_REQUEST));
 
         //DWORD dwFlags;
         //m_pFile->QueryOption(INTERNET_OPTION_SECURITY_FLAGS, dwFlags);
         //dwFlags |= SECURITY_FLAG_IGNORE_UNKNOWN_CA;
         set web server option
         //m_pFile->SetOption(INTERNET_OPTION_SECURITY_FLAGS, dwFlags);
 
        m_pFile -> AddRequestHeaders (_T ("Accept: *,*/*" ));
        m_pFile -> AddRequestHeaders (_T ("Accept-Language: zh-cn" ));
        m_pFile -> AddRequestHeaders (_T ("Content-Type: application/x-www-form-urlencoded" ));
        m_pFile -> AddRequestHeaders (_T ("Accept-Encoding: deflate")); //gzip,
 
        m_pFile -> SendRequest (NULL , 0 , ( LPVOID)( LPCTSTR )strPostData , strPostData == NULL ? 0: _tcslen( strPostData ));
        char szChars [BUFFER_SIZE + 1 ] = { 0 };
        string strRawResponse = "" ;
        UINT nReaded = 0 ;
         while (( nReaded = m_pFile -> Read(( void *)szChars , BUFFER_SIZE )) > 0 )
         {
			 szChars [ nReaded ] = '\0' ;
            strRawResponse += szChars ;
            memset ( szChars , 0, BUFFER_SIZE + 1 );
         }
 
         int unicodeLen = MultiByteToWideChar (CP_UTF8 , 0 , strRawResponse. c_str(), -1 , NULL , 0 );
        WCHAR * pUnicode = new WCHAR [ unicodeLen + 1];
        memset ( pUnicode ,0 ,(unicodeLen + 1)* sizeof (wchar_t ));
 
        MultiByteToWideChar ( CP_UTF8 ,0 ,strRawResponse . c_str(),- 1 , pUnicode, unicodeLen );
        CString cs ( pUnicode );
         //AfxMessageBox(cs);
        CStringA stra ( cs. GetBuffer (0 ));
        strResponse = stra. GetBuffer (0 ); //(LPCSTR)(LPCWSTR)cs;
        stra . ReleaseBuffer ();
         delete [] pUnicode ;
        pUnicode = NULL ;
        Clear ();
     }
     catch ( CInternetException * e )
     {
        Clear ();
        DWORD dwErrorCode = e-> m_dwError ;
        e -> Delete();
 
        DWORD dwError = GetLastError ();
 
         //PRINT_LOG("dwError = %d", dwError, 0);
 
         if ( ERROR_INTERNET_TIMEOUT == dwErrorCode )
         {
             return OUTTIME ;
         }
         else
         {
             return FAILURE ;
         }
     }
     return SUCCESS ;
}
 
int CHttpClient ::HttpGet (LPCTSTR strUrl , LPCTSTR strPostData , string & strResponse )
{
     return ExecuteRequest ( _T( "GET" ), strUrl , strPostData , strResponse );
}
 
int CHttpClient ::HttpPost (LPCTSTR strUrl , LPCTSTR strPostData , string & strResponse )
{
     return ExecuteRequest ( _T( "POST" ), strUrl , strPostData , strResponse );
}

 
 





  • 1
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: MFC发送POST请求JSON数据,可以通过以下步骤实现: 1. 设置请求头 首先需要设置请求头,包括Content-Type和Content-Length。Content-Type设置为application/json,Content-Length设置为JSON数据的长度。 2. 创建JSON字符串 创建一个JSON字符串,可以使用CJsonObject等库来构造JSON字符串。 3. 创建连接对象 创建连接对象,使用CInternetSession::GetHttpConnection方法建立连接。 4. 发送请求 使用CHttpFile::SendRequest方法发送POST请求,并将JSON字符串作为请求内容发送出去。 5. 接收响应 使用CHttpFile::QueryInfoStatusCode获取HTTP状态码,判断请求是否成功。如果成功,使用CHttpFile::ReadString方法读取响应内容。 6. 关闭连接 使用CHttpFile::Close方法关闭连接。 以上就是MFC发送POST请求JSON数据的步骤。需要注意的是,在使用MFC发送POST请求时,需要在请求字符串中添加转义字符,以确保JSON数据能够正确传输。在发送请求前,可以通过调试工具查看请求内容,以帮助排查请求发送过程中的问题。 ### 回答2: MFC是微软的一套资源库,包含了许多封装好的类和模块,能够方便地开发Windows应用程序。而json是一种轻量级的数据交换格式,经常用于Web服务之间的数据传输。在MFC中使用Post请求发送json数据也是很常见的一种操作。 发送Post请求的方式有很多种,MFC中可以使用CInternetSession和CHttpConnection等类进行实现。首先,需要创建一个Internet会话对象,并通过该对象建立一个HTTP连接。在建立HTTP连接时,需要设置服务地址和端口号等参数。然后,可以构造一个HTTP请求,指定请求类型为POST,并将json数据填充到请求正文中。最后,通过连接对象的SendRequest方法,将HTTP请求发送给Web服务端,等待服务端响应。 具体实现可以按照以下步骤进行: 1.引入相关头文件: #include <afxinet.h> #include <afxdisp.h> #include <afxwin.h> #include <afx.h> 2.创建Internet会话对象: CInternetSession* pSession = new CInternetSession(); 3.连接Web服务: CHttpConnection * pConnect = pSession->GetHttpConnection(L"www.example.com",INTERNET_DEFAULT_HTTP_PORT,NULL,NULL); 4.构造HTTP请求: CString strUrl = L"/api/data"; CHttpFile* pFile = pConnect->OpenRequest(CHttpConnection::HTTP_VERB_POST,strUrl); 5.设置请求头信息: pFile->AddRequestHeaders(L"Content-Type: application/json"); 6.构造json数据: CString strJsonData = L"{\"key\":\"value\"}"; 7.设置请求正文: pFile->SendRequestEx(strJsonData,len,strPostData.GetLength()); 8.等待服务端响应: CString strResponse; pFile->ReadString(strResponse); 9.关闭连接: pFile->Close(); delete pConnect; delete pFile; 10.释放会话对象: delete pSession; 以上就是在MFC中发送Post请求json数据的方法。在实际开发中,还需要根据具体情况进行调整和优化。 ### 回答3: MFC(Microsoft Foundation Class)是微软公司开发的一组类库,用于在Windows操作系统上开发图形用户界面程序。MFC常用于开发Windows桌面应用程序,它提供了一套面向对象的框架,让开发过程更加高效、方便。 在MFC中发送POST请求并传输JSON数据可以通过以下步骤实现: 1. 首先,需要使用CInternetSession对象建立一个Internet会话。 ``` CInternetSession session(_T("MySession")); ``` 2. 接着,需要通过CInternetSession对象创建一个HttpConnect连接并指定服务器信息和端口号。 ``` CHttpConnection* pHttpConnect = session.GetHttpConnection(_T("localhost"), INTERNET_DEFAULT_HTTP_PORT); ``` 3. 创建一个HttpPost请求并设置请求头和请求体,请求体可以使用JSON格式进行传输。 ``` CString strRequest = _T("name=John&age=20"); CString strHeaders = _T("Content-Type: application/json\r\n"); CHttpFile* pHttpFile = pHttpConnect->OpenRequest(CHttpConnection::HTTP_VERB_POST, _T("/api/example"), nullptr, 1, nullptr, nullptr, INTERNET_FLAG_RELOAD); pHttpFile->AddRequestHeaders(strHeaders); pHttpFile->SendRequest(strRequest, strRequest.GetLength()); ``` 4. 发送HttpPost请求,并获取服务器返回的响应结果。 ``` CString strResponse; DWORD dwBytesRead = 0; CHAR buff[1024] = { 0 }; while (dwBytesRead = pHttpFile->Read(buff, 1024)) { strResponse += buff; } pHttpFile->Close(); pHttpConnect->Close(); session.Close(); ``` 通过以上方法可以完成在MFC中发送POST请求并传输JSON数据的操作。需要注意的是,请求头中需要设置Content-Type为application/json,同时请求体中的数据必须是JSON格式的字符串。此外,还需要注意HttpConnect连接、HttpPost请求和HttpFile对象的生命周期问题,确保在使用后及时进行关闭和销毁。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值