青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

隨筆-250  評論-20  文章-55  trackbacks-0

在如今的網絡應用中,文件的傳送是重要的功能之一,也是共享的基礎。一些重要的協議像HTTP,FTP等都支持文件的傳送。尤其是FTP,它的全稱就是“文件傳送協議”,當初的工程師設計這一協議就是為了解決網絡間的文件傳送問題,而且以其穩定,高速,簡單而一直保持著很大的生命力。作為一個程序員,使用這些現有的協議傳送文件相當簡單,不過,它們只適用于服務器模式中。這樣,當我們想在點與點之間傳送文件就不適用了或相當麻煩,有一種大刀小用的意味。筆者一直想尋求一種簡單有效,且具備多線程斷點續傳的方法來實現點與點之間的文件傳送問題,經過大量的翻閱資料與測試,終于實現了,現把它共享出來,與大家分享。
我寫了一個以此為基礎的實用程序(網絡傳圣,包含源代碼),可用了基于TCP/IP的電腦上,供大家學習。


(本文源代碼運行效果圖)


實現方法(VC++,基于TCP/IP協議)如下:
仍釆用服務器與客戶模式,需分別對其設計與編程。
服務器端較簡單,主要就是加入待傳文件,監聽客戶,和傳送文件。而那些斷點續傳的功能,以及文件的管理都放在客戶端上。

一、服務器端

首先介紹服務器端:
最開始我們要定義一個簡單的協議,也就是定義一個服務器端與客戶端聽得懂的語言。而為了把問題簡化,我就讓服務器只要聽懂兩句話,一就是客戶說“我要讀文件信息”,二就是“我準備好了,可以傳文件了”。
由于要實現多線程,必須把功能獨立出來,且包裝成線程,首先建一個監聽線程,主要負責接入客戶,并啟動另一個客戶線程。我用VC++實現如下:

DWORD WINAPI listenthread(LPVOID lpparam)
{
//由主函數傳來的套接字
  SOCKET  pthis=(SOCKET)lpparam;
//開始監聽
int rc=listen(pthis,30);
//如果錯就顯示信息
if(rc<0){
CString aaa;
aaa="listen錯誤\n";
AfxGetMainWnd()->SendMessageToDescendants(WM_AGE1,(LPARAM)aaa.GetBuffer(0),1);
aaa.ReleaseBuffer();
return 0;
}
//進入循環,并接收到來的套接字
while(1){
//新建一個套接字,用于客戶端
SOCKET s1;
s1=accept(pthis,NULL,NULL);
   //給主函數發有人聯入消息
CString aa;
aa="一人聯入!\n";
AfxGetMainWnd()->SendMessageToDescendants(WM_AGE1,(LPARAM)aa.GetBuffer(0),1);
aa.ReleaseBuffer();
DWORD dwthread;
//建立用戶線程
::CreateThread(NULL,0,clientthread,(LPVOID)s1,0,&dwthread);
}
return 0;
}
接著我們來看用戶線程:
先看文件消息類定義
struct fileinfo
{
int fileno;//文件號
int type;//客戶端想說什么(前面那兩句話,用1,2表示)
long len;//文件長度
int seek;//文件開始位置,用于多線程
char name[100];//文件名
};
用戶線程函數:
DWORD WINAPI clientthread(LPVOID lpparam)
{
//文件消息
fileinfo* fiinfo;
//接收緩存
char* m_buf;
m_buf=new char[100];
//監聽函數傳來的用戶套接字
SOCKET  pthis=(SOCKET)lpparam;
//讀傳來的信息
int aa=readn(pthis,m_buf,100);
//如果有錯就返回
if(aa<0){
closesocket (pthis);
return -1;
}
//把傳來的信息轉為定義的文件信息
fiinfo=(fileinfo*)m_buf;
CString aaa;
//檢驗客戶想說什么
switch(fiinfo->type)
{
//我要讀文件信息
case 0:
//讀文件
aa=sendn(pthis,(char*)zmfile,1080);
//有錯
if(aa<0){
closesocket (pthis);
return -1;
}
//發消息給主函數
aaa="收到LIST命令\n";
AfxGetMainWnd()->SendMessageToDescendants(WM_AGE1,(LPARAM)aaa.GetBuffer(0),1);
break;
//我準備好了,可以傳文件了
case 2:
//發文件消息給主函數
aaa.Format("%s  文件被請求!%s\n",zmfile[fiinfo->fileno].name,nameph[fiinfo->fileno]);
AfxGetMainWnd()->SendMessageToDescendants(WM_AGE1,(LPARAM)aaa.GetBuffer(0),1);
//讀文件,并傳送
readfile(pthis,fiinfo->seek,fiinfo->len,fiinfo->fileno);
//聽不懂你說什么
default:
aaa="接收協議錯誤!\n";
AfxGetMainWnd()->SendMessageToDescendants(WM_AGE1,(LPARAM)aaa.GetBuffer(0),1);
break;
}
return 0;
}
讀文件函數
void readfile(SOCKET  so,int seek,int len,int fino)
{
//文件名
CString myname;
myname.Format("%s",nameph[fino]);
CFile myFile;
//打開文件
myFile.Open(myname, CFile::modeRead | CFile::typeBinary|CFile::shareDenyNone);
//傳到指定位置 
myFile.Seek(seek,CFile::begin);
char m_buf[SIZE];
int len2;
int len1;
len1=len;
//開始接收,直到發完整個文件
while(len1>0){
len2=len>SIZE?SIZE:len;
myFile.Read(m_buf, len2);
int aa=sendn(so,m_buf,len2);
if(aa<0){
closesocket (so);
break;
}
len1=len1-aa;
len=len-aa;
}
myFile.Close();
}

服務器端最要的功能各技術就是這些,下面介紹客戶端。

二、客戶端

客戶端最重要,也最復雜,它負責線程的管理,進度的記錄等工作。

大概流程如下:
先連接服務器,接著發送命令1(給我文件信息),其中包括文件長度,名字等,然后根據長度決定分幾個線程下載,并初使化下載進程,接著發送命令2(可以給我傳文件了),并記錄文件進程。最后,收尾。
這其中有一個十分重要的類,就是cdownload類,定義如下:

class cdownload
{
public:
void createthread();//開線程
DWORD finish1();//完成線程
int sendlist();//發命令1
downinfo doinfo;//文件信息(與服務器定義一樣)
int startask(int n);開始傳文件n
long m_index;
BOOL good[BLACK];
int  filerange[100];
CString fname;
CString fnametwo;
UINT threadfunc(long index);//下載進程
int sendrequest(int n);//發文件信息
cdownload(int thno1);
virtual ~cdownload();
};
下面先介紹sendrequest(int n),在開始前,向服務器發獲得文件消息命令,以便讓客戶端知道有哪些文件可傳
int cdownload::sendrequest(int n)
{
//建套接字
sockaddr_in local;
SOCKET m_socket;
int rc=0;
//初使化服務器地址
local.sin_family=AF_INET;
local.sin_port=htons(1028);
local.sin_addr.S_un.S_addr=inet_addr(ip);
m_socket=socket(AF_INET,SOCK_STREAM,0);
int ret;
//聯接服務器
ret=connect(m_socket,(LPSOCKADDR)&local,sizeof(local));
//有錯的話
if(ret<0){
AfxMessageBox("聯接錯誤");
closesocket(m_socket);
return -1;
}
//初使化命令
fileinfo fileinfo1;
fileinfo1.len=n;
fileinfo1.seek=50;
fileinfo1.type=1;
//發送命令
int aa=sendn(m_socket,(char*)&fileinfo1,100);
if(aa<0){
closesocket(m_socket);
return -1;
}
//接收服務器傳來的信息
aa=readn(m_socket,(char*)&fileinfo1,100);
if(aa<0){
closesocket(m_socket);
return -1;
}
//關閉
shutdown(m_socket,2);
closesocket(m_socket);
return 1;
}
有了文件消息后我們就可以下載文件了。在主函數中,用法如下:
//下載第clno個文件,并為它建一個新cdownload類
down[clno]=new cdownload(clno);
//開始下載,并初使化
type=down[clno]->startask(clno);
//建立各線程
createthread(clno);
下面介紹開始方法:
//開始方法
int cdownload::startask(int n)
{
//讀入文件長度
doinfo.filelen=zmfile[n].length;
//讀入名字
fname=zmfile[n].name;
CString tmep;
//初使化文件名
tmep.Format("\\temp\\%s",fname);
//給主函數發消息
CString aaa;
aaa="正在讀取 "+fname+" 信息,馬上開始下載。。。\n";
AfxGetMainWnd()->SendMessageToDescendants(WM_AGE1,(LPARAM)aaa.GetBuffer(0),1);
aaa.ReleaseBuffer();
//如果文件長度小于0就返回
if(doinfo.filelen<=0) return -1;
//建一個以.down結尾的文件記錄文件信息
CString m_temp;
m_temp=fname+".down";
doinfo.name=m_temp;
FILE* fp=NULL;
CFile myfile;
//如果是第一次下載文件,初使化各記錄文件
if((fp=fopen(m_temp,"r"))==NULL){
filerange[0]=0;
//文件分塊
for(int i=0;i<BLACK;i++)
{
if(i>0)
filerange[i*2]=i*(doinfo.filelen/BLACK+1);
filerange[i*2+1]=doinfo.filelen/BLACK+1;
}
filerange[BLACK*2-1]=doinfo.filelen-filerange[BLACK*2-2];
myfile.Open(m_temp,CFile::modeCreate|CFile::modeWrite | CFile::typeBinary);
//寫入文件長度
myfile.Write(&doinfo.filelen,sizeof(int));
myfile.Close();
 
CString temp;
for(int ii=0;ii<BLACK;ii++){
//初使化各進程記錄文件信息(以.downN結尾)
temp.Format(".down%d",ii);
m_temp=fname+temp;
myfile.Open(m_temp,CFile::modeCreate|CFile::modeWrite | CFile::typeBinary);
//寫入各進程文件信息
myfile.Write(&filerange[ii*2],sizeof(int));
myfile.Write(&filerange[ii*2+1],sizeof(int));
myfile.Close();
}
((CMainFrame*)::AfxGetMainWnd())->m_work.m_ListCtrl->AddItemtwo(n,2,0,0,0,doinfo.threadno);
}
else{
//如果文件已存在,說明是續傳,讀上次信息
CString temp;
 
m_temp=fname+".down0";
if((fp=fopen(m_temp,"r"))==NULL)
return 1;
else fclose(fp);
int bb;
bb=0;
//讀各進程記錄的信息
for(int ii=0;ii<BLACK;ii++)
{
temp.Format(".down%d",ii);
m_temp=fname+temp;
 
myfile.Open(m_temp,CFile::modeRead | CFile::typeBinary);
myfile.Read(&filerange[ii*2],sizeof(int));
myfile.Read(&filerange[ii*2+1],sizeof(int));
myfile.Close();
bb = bb+filerange[ii*2+1];
CString temp;
}
if(bb==0) return 1;
doinfo.totle=doinfo.filelen-bb;
 
((CMainFrame*)::AfxGetMainWnd())->m_work.m_ListCtrl->AddItemtwo(n,2,doinfo.totle,1,0,doinfo.threadno);
}
 	//建立下載結束進程timethread,以管現各進程結束時間。
DWORD dwthread;
::CreateThread(NULL,0,timethread,(LPVOID)this,0,&dwthread);
return 0;
}
下面介紹建立各進程函數,很簡單:
void CMainFrame::createthread(int threadno)
{
DWORD dwthread;
//建立BLACK個進程
for(int i=0;i<BLACK;i++)
{
m_thread[threadno][i]=	::CreateThread(NULL,0,downthread,(LPVOID)down[threadno],0,&dwthread);
}
}
downthread進程函數
DWORD WINAPI downthread(LPVOID lpparam)
{
cdownload* pthis=(cdownload*)lpparam;
//進程引索+1
InterlockedIncrement(&pthis->m_index);
//執行下載進程
pthis->threadfunc(pthis->m_index-1);
return 1;
}
下面介紹下載進程函數,最最核心的東西了
UINT cdownload::threadfunc(long index)
{
//初使化聯接
sockaddr_in local;
SOCKET m_socket;
int rc=0;
 
local.sin_family=AF_INET;
local.sin_port=htons(1028);
local.sin_addr.S_un.S_addr=inet_addr(ip);
m_socket=socket(AF_INET,SOCK_STREAM,0);
int ret;
//讀入緩存
char* m_buf=new char[SIZE];
int re,len2;
fileinfo fileinfo1;
//聯接
ret=connect(m_socket,(LPSOCKADDR)&local,sizeof(local));
//讀入各進程的下載信息
fileinfo1.len=filerange[index*2+1];
fileinfo1.seek=filerange[index*2];
fileinfo1.type=2;
fileinfo1.fileno=doinfo.threadno;
 
re=fileinfo1.len;
 
//打開文件 
CFile destFile;
FILE* fp=NULL;
//是第一次傳的話
if((fp=fopen(fname,"r"))==NULL)
destFile.Open(fname, CFile::modeCreate|CFile::modeWrite | CFile::typeBinary|CFile::shareDenyNone);
else
//如果文件存在,是續傳
destFile.Open(fname,CFile::modeWrite | CFile::typeBinary|CFile::shareDenyNone);
//文件指針移到指定位置
destFile.Seek(filerange[index*2],CFile::begin);
//發消息給服務器,可以傳文件了
sendn(m_socket,(char*)&fileinfo1,100);
CFile myfile;
CString temp;
temp.Format(".down%d",index);
m_temp=fname+temp;
 	//當各段長度還不為0時
while(re>0){
len2=re>SIZE?SIZE:re;
 
//讀各段內容
int len1=readn(m_socket,m_buf,len2);
//有錯的話
if(len1<0){
closesocket(m_socket);
break;
}
 
//寫入文件
destFile.Write(m_buf, len1);
//更改記錄進度信息
filerange[index*2+1]-=len1;
filerange[index*2]+=len1;
//移動記錄文件指針到頭
myfile.Seek(0,CFile::begin);
//寫入記錄進度
myfile.Write(&filerange[index*2],sizeof(int));
myfile.Write(&filerange[index*2+1],sizeof(int));
//減去這次讀的長度
re=re-len1;
//加文件長度
doinfo.totle=doinfo.totle+len1;
};
//這塊下載完成,收尾
 
myfile.Close();
destFile.Close();
delete [] m_buf;
shutdown(m_socket,2);
 
 
if(re<=0) good[index]=TRUE;
return 1;
}

到這客戶端的主要模塊和機制已基本介紹完。希望好好體會一下這種多線程斷點續傳的方法。

posted on 2007-04-20 15:18 jay 閱讀(466) 評論(0)  編輯 收藏 引用 所屬分類: socket
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            欧美午夜精品理论片a级按摩 | 欧美日韩国产综合一区二区| 欧美一区二区三区精品| 亚洲婷婷免费| 亚洲欧美激情四射在线日| 午夜精品一区二区三区在线| 久久精品免费| 欧美国产精品va在线观看| 欧美日韩一区二区三区| 国产精品美女999| 一区二区三区**美女毛片| 亚洲在线国产日韩欧美| 一区二区三区久久网| 亚洲欧美日韩成人高清在线一区| 亚洲一卡久久| 久久精品免费电影| 亚洲第一视频网站| 日韩视频在线免费| 亚洲小说春色综合另类电影| 久久精品国产成人| 欧美日韩亚洲视频一区| 韩曰欧美视频免费观看| 一区二区三区精品| 久久一区国产| 一区二区久久久久久| 久久人人97超碰精品888| 欧美亚洲成人免费| 亚洲国产精品一区制服丝袜 | 亚洲视频精选| 久久亚洲精品中文字幕冲田杏梨| 欧美午夜视频| 亚洲精品中文字幕女同| 久久精品国产视频| 亚洲美女视频网| 麻豆国产精品777777在线| 国产精品黄色| 亚洲一区二区动漫| 亚洲国产一区二区三区高清| 亚洲经典三级| 久久噜噜亚洲综合| 欧美成在线观看| 国产欧美日韩在线视频| 91久久亚洲| 久久九九电影| 一区二区免费在线观看| 蜜臀久久久99精品久久久久久| 国产精品国产三级国产专播品爱网| 亚洲大胆女人| 久久99伊人| 亚洲综合丁香| 国产精品久久久久影院亚瑟| 99re热这里只有精品免费视频| 裸体一区二区| 久久九九全国免费精品观看| 国产精品视频精品| 欧美网站在线| 羞羞答答国产精品www一本 | 亚洲第一页在线| 久久国产一二区| 亚洲视频电影图片偷拍一区| 欧美国产视频一区二区| 亚洲精品一区二区三区99| 欧美成人久久| 免费亚洲一区二区| 亚洲精品国精品久久99热一| 欧美成人精品h版在线观看| 久久久水蜜桃av免费网站| 国产日韩在线看| 卡通动漫国产精品| 美女诱惑一区| 一区二区三区日韩精品| 日韩视频在线一区二区| 国产精品欧美一区喷水| 久久国产精品久久久久久电车| 欧美一区二区三区久久精品茉莉花 | 亚洲高清一二三区| 欧美国产日韩精品| 欧美巨乳在线观看| 亚洲一区二区免费视频| 亚洲欧美视频在线观看| 亚洲国产成人精品久久| 亚洲日本电影| 国产日韩成人精品| 欧美成人免费大片| 欧美日韩另类综合| 欧美在线免费视屏| 久久精品国产视频| 中国成人黄色视屏| 久久电影一区| 欧美 日韩 国产一区二区在线视频 | 亚洲福利视频网站| 亚洲精品极品| 国产三区精品| 夜夜嗨一区二区三区| 欧美刺激午夜性久久久久久久| 亚洲人成久久| 亚洲综合精品| 精品9999| 一区二区三区四区五区视频| 国产欧亚日韩视频| 欧美成人高清| 国产精品三级视频| 亚洲国产精品成人综合| 国产精品护士白丝一区av| 美日韩在线观看| 国产精品一区二区三区免费观看| 老司机免费视频一区二区三区| 巨胸喷奶水www久久久免费动漫| 99这里只有久久精品视频| 亚洲一区欧美激情| 日韩亚洲综合在线| 久久精品在这里| 销魂美女一区二区三区视频在线| 免费在线欧美视频| 久久久久女教师免费一区| 欧美视频一二三区| 91久久精品国产91久久| 国产一区二区中文字幕免费看| 99这里有精品| 99国产精品| 欧美电影免费观看高清| 欧美韩日精品| 有码中文亚洲精品| 欧美一区二区三区视频免费播放 | 中文日韩电影网站| 欧美va天堂在线| 欧美顶级少妇做爰| 精品成人在线观看| 午夜欧美大尺度福利影院在线看| 中文在线不卡视频| 欧美日韩伦理在线| 99视频+国产日韩欧美| 99ri日韩精品视频| 欧美黄色一区二区| 欧美日韩亚洲综合在线| 欧美淫片网站| 亚洲欧美日韩一区在线| 欧美日韩mp4| 亚洲啪啪91| 亚洲韩国精品一区| 欧美经典一区二区| 亚洲免费av片| 亚洲欧美日韩精品综合在线观看| 欧美午夜精品久久久| 亚洲午夜久久久久久尤物| 欧美一级淫片播放口| 国产精品视频大全| 欧美一级视频| 欧美二区在线播放| 一区二区三区www| 国产精品国产一区二区| 性欧美在线看片a免费观看| 久久久国产午夜精品| 极品尤物av久久免费看| 蘑菇福利视频一区播放| aaa亚洲精品一二三区| 亚洲欧美日韩在线观看a三区| 国产欧美日韩三级| 久久久久综合| 亚洲日本va午夜在线电影| 亚洲欧美日韩成人| 一区二区三区在线观看视频 | 亚洲一二三区视频在线观看| 午夜国产不卡在线观看视频| 国产欧美日韩免费看aⅴ视频| 久久久福利视频| 亚洲黄页一区| 亚洲在线国产日韩欧美| 好看的av在线不卡观看| 欧美成人亚洲| 欧美一区1区三区3区公司| 亚洲国产成人av| 久久国产精品久久久| 亚洲精品在线免费观看视频| 国产伦精品一区二区三| 欧美大胆a视频| 翔田千里一区二区| 亚洲国产欧美在线人成| 久久久久欧美精品| 亚洲视频一区在线观看| 亚洲成人资源| 国产一区二区三区精品欧美日韩一区二区三区 | 欧美精品自拍| 久久成人精品一区二区三区| 亚洲欧洲精品一区二区三区不卡| 久久国产欧美日韩精品| 中文在线不卡视频| 亚洲国产精品一区二区第四页av| 国产精品成人一区二区| 欧美精品一区三区在线观看| 久久精品视频在线播放| 亚洲欧美国产高清va在线播| 亚洲精品国产欧美| 欧美成人一区二区在线| 久久婷婷蜜乳一本欲蜜臀| 亚洲欧美卡通另类91av| 亚洲精品影院在线观看| 亚洲激情另类| 亚洲精品久久视频| 91久久精品国产91久久|