使用WinInet类,连接HTTP服务器,在下载大文件时,程序会未响应

sherlockcsdn_1 2015-01-14 09:48:46

DWORD WINAPI CUpdateDlg::DownloadFileFromWeb(LPVOID lParam)
{
CUpdateDlg *pDlg=(CUpdateDlg *)lParam;
pDlg->strFileURLInServer != "";
pDlg->strFileLocalFullPath != "";
CInternetSession session;
CHttpConnection* pHttpConnection = NULL;
CHttpFile* pHttpFile = NULL;
CString strServer, strObject;
INTERNET_PORT wPort;
BOOL bReturn = FALSE;

DWORD dwType;
const int nTimeOut = 2000;
session.SetOption(INTERNET_OPTION_CONNECT_TIMEOUT, nTimeOut); //重试之间的等待延时
session.SetOption(INTERNET_OPTION_CONNECT_RETRIES, 1); //重试次数
char* pszBuffer = NULL;



try
{
AfxParseURL(pDlg->strFileURLInServer, dwType, strServer, strObject, wPort);
pHttpConnection = session.GetHttpConnection(strServer, wPort);
pHttpFile = pHttpConnection->OpenRequest(CHttpConnection::HTTP_VERB_GET, strObject);
if(pHttpFile->SendRequest() == FALSE)
return FALSE;
DWORD dwStateCode;

pHttpFile->QueryInfoStatusCode(dwStateCode);
if(dwStateCode == HTTP_STATUS_OK)
{
HANDLE hFile = CreateFile(pDlg->strFileLocalFullPath, GENERIC_WRITE,
FILE_SHARE_WRITE, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL,
NULL); //创建本地文件
if(hFile == INVALID_HANDLE_VALUE)
{
pHttpFile->Close();
pHttpConnection->Close();
session.Close();
return FALSE;
}

char szInfoBuffer[1000]; //返回消息
DWORD dwFileSize = 0; //文件长度
DWORD dwInfoBufferSize = sizeof(szInfoBuffer);
BOOL bResult = FALSE;
bResult = pHttpFile->QueryInfo(HTTP_QUERY_CONTENT_LENGTH,
(void*)szInfoBuffer,&dwInfoBufferSize,NULL);

dwFileSize = atoi(szInfoBuffer);
const int BUFFER_LENGTH = 1024 * 10;
pszBuffer = new char[BUFFER_LENGTH]; //读取文件的缓冲
DWORD dwWrite, dwTotalWrite;
dwWrite = dwTotalWrite = 0;
UINT nRead = pHttpFile->Read(pszBuffer, BUFFER_LENGTH); //读取服务器上数据

ofstream fileOut("d:\\TestOut.txt", ios::app);
CString sDate;
time_t curUtcTime;
struct tm *curTime;

while(nRead > 0)
{

time(&curUtcTime);
curTime = localtime(&curUtcTime);
sDate.Format("%02d:%02d:%02d", curTime->tm_hour, curTime->tm_min, curTime->tm_sec);
fileOut << sDate << "start write" << endl;

WriteFile(hFile, pszBuffer, nRead, &dwWrite, NULL); //写到本地文件
ZeroMemory(pszBuffer, BUFFER_LENGTH);
dwTotalWrite += dwWrite;

time(&curUtcTime);
curTime = localtime(&curUtcTime);
sDate.Format("%02d:%02d:%02d", curTime->tm_hour, curTime->tm_min, curTime->tm_sec);
fileOut << sDate << "start read" << endl;

nRead = pHttpFile->Read(pszBuffer, BUFFER_LENGTH);

pDlg->nPecent += 1000 * nRead / ((float)dwFileSize);
//m_progressDownload.SetPos((int)nPecent);
//Sleep(5);
}

pDlg->nPecent = 1000;
//GetDlgItem(CUpdateDlg.m_hWnd, uid)->SetPos((int)nPecent);
//m_progressDownload.SetPos((int)nPecent);

delete[]pszBuffer;
pszBuffer = NULL;
CloseHandle(hFile);
bReturn = TRUE;
}
}
catch(CInternetException* e)
{
TCHAR tszErrString[256];
e->GetErrorMessage(tszErrString, sizeof(tszErrString));
TRACE(_T("Download XSL error! URL: %s,Error: %s"), pDlg->strFileURLInServer, tszErrString);
e->Delete();
}
catch(...)
{
}

if(pszBuffer != NULL)
{
delete[]pszBuffer;
}
if(pHttpFile != NULL)
{
pHttpFile->Close();
delete pHttpFile;
}
if(pHttpConnection != NULL)
{
pHttpConnection->Close();
delete pHttpConnection;
}
session.Close();
return bReturn;
}

我把这个函数创建了线程,但运行下载大文件时还是会出现程序未响应这种类似阻塞现象的情况。在未响应的过程中,文件还是一直在下载,最后也能将文件下载完成。但我需要在下载过程中同时进行其他操作。请问,怎么会产生这种情况的?
...全文
174 5 打赏 收藏 转发到动态 举报
AI 作业
写回复
用AI写文章
5 条回复
切换为时间正序
请发表友善的回复…
发表回复
阿先森 2015-02-01
  • 打赏
  • 举报
回复
感谢邀请。 在新线程里面,禁止使用MFC窗口类,如CDialog,CSpliderCtrl,CProgressBarCtrl等等。
sherlockcsdn_1 2015-01-15
  • 打赏
  • 举报
回复
我把问题解决了,这个函数我之前确实是单独开一个线程运行。函数本身没有问题,阻塞原因在于另一个函数是while(1)运行,由于是MFC程序,时间一长命令处理不过来,造成了阻塞现象,程序才会未响应。现在去掉了while(1)循环,程序就通畅了。多谢各位帮助,结贴。
zgl7903 2015-01-15
  • 打赏
  • 举报
回复
Sleep(0) 切换一下 给其它线程运行的机会
弱水垂钓 2015-01-15
  • 打赏
  • 举报
回复
引用 1 楼 worldy 的回复:
CUpdateDlg::DownloadFileFromWeb 应该在一个独立的线程中调用
下载文件就不要放到主线程去做,会造成消息阻塞的,所以没有响应
worldy 2015-01-14
  • 打赏
  • 举报
回复
CUpdateDlg::DownloadFileFromWeb 应该在一个独立的线程中调用

18,363

社区成员

发帖
与我相关
我的任务
社区描述
VC/MFC 网络编程
c++c语言开发语言 技术论坛(原bbs)
社区管理员
  • 网络编程
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧