成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

FileZilla 源代碼分析5

系統 Linux
FileZilla是一個免費開源的FTP客戶端軟件,分為客戶端版本和服務器版本,具備所有的FTP軟件功能。可控性、有條理的界面和管理多站點的簡化方式使得Filezilla客戶端版成為一個方便高效的FTP客戶端工具,而FileZilla Server則是一個小巧并且可靠的支持FTP&SFTP的FTP服務器軟件。

FileZilla是一種快速、可信賴的FTP客戶端以及服務器端開放源代碼程式,具有多種特色、直覺的接口。本文就給大家分析下FileZilla的源代碼。

  < type="text/javascript"> < src="http://pagead2.googlesyndication.com/pagead/show_ads.js" type="text/javascript"> < type="text/javascript"> < src="http://pagead2.googlesyndication.com/pagead/show_ads.js" type="text/javascript">

  在CServer的Create()中,創建對象CListenSocket來監聽21端口,來看看具體的代碼實現:

  CListenSocket *pListenSocket = new CListenSocket(this, ssl);

  if (!pListenSocket->Create(nPort, SOCK_STREAM, FD_ACCEPT, NULL) || !pListenSocket->Listen())

  基本上分三步:

  1、new CListenSocket:沒有什么特別的,基本就是初始化成員變量

  2、Create

  注:在所有的代碼中,先不看大量的if (m_pFirstLayer)這種代碼,這是CAsyncSocketExLayer的機制。

  Create其實調用的是父類CAsyncSocketEx的create()方法,這個方法中第一件事情就是建立m_spAsyncSocketExThreadDataList鏈、分發線程t_AsyncSocketExThreadData以及CAsyncSocketEx之間的關系,CAsyncSocketEx的create()方法首先調用InitAsyncSocketExInstance(),下面是CAsyncSocketEx::InitAsyncSocketExInstance()代碼片斷:

  DWORD id=GetCurrentThreadId();

  ...

  //Get thread specific data

  if (m_spAsyncSocketExThreadDataList) // 這個鏈已經建立了

  {

  t_AsyncSocketExThreadDataList *pList=m_spAsyncSocketExThreadDataList;

  while (pList) // 遍歷鏈

  {

  ASSERT(pList->pThreadData);

  ASSERT(pList->pThreadData->nInstanceCount>0);

  if (pList->pThreadData->nThreadId==id) // 對當前線程已經有分發線程了,就把當前socket的分發由這個分發線程來代理

  {

  m_pLocalAsyncSocketExThreadData=pList->pThreadData;

  m_pLocalAsyncSocketExThreadData->nInstanceCount++; // 多了這一個socket

  break;

  }

  pList=pList->pNext;

  }

  //Current thread yet has no sockets

  if (!pList) // 當前線程還沒有分發線程,則創建一個

  {

  //Initialize data for current thread

  pList=new t_AsyncSocketExThreadDataList;

  pList->pNext=m_spAsyncSocketExThreadDataList;

  m_spAsyncSocketExThreadDataList=pList;

  m_pLocalAsyncSocketExThreadData=new t_AsyncSocketExThreadData;

  m_pLocalAsyncSocketExThreadData->nInstanceCount=1; // 只掛了當前的socket

  m_pLocalAsyncSocketExThreadData->nThreadId=id; // 這個分發線程的threadID

  m_pLocalAsyncSocketExThreadData->m_pHelperWindow=new CAsyncSocketExHelperWindow(m_pLocalAsyncSocketExThreadData); // 為這個分發線程創建CAsyncSocketExHelperWindow

  m_spAsyncSocketExThreadDataList->pThreadData=m_pLocalAsyncSocketExThreadData;

  }

  }

  else // 如果分發線程鏈還沒有創建,則創建一個

  { //No thread has instances of CAsyncSocketEx; Initialize data

  m_spAsyncSocketExThreadDataList=new t_AsyncSocketExThreadDataList;

  m_spAsyncSocketExThreadDataList->pNext=0;

  m_pLocalAsyncSocketExThreadData=new t_AsyncSocketExThreadData; // 第一個分發線程

  m_pLocalAsyncSocketExThreadData->nInstanceCount=1; // 只掛了當前的socket

  m_pLocalAsyncSocketExThreadData->nThreadId=id; // 這個分發線程的threadID

  m_pLocalAsyncSocketExThreadData->m_pHelperWindow=new CAsyncSocketExHelperWindow(m_pLocalAsyncSocketExThreadData); // 為這個分發線程創建CAsyncSocketExHelperWindow

  m_spAsyncSocketExThreadDataList->pThreadData=m_pLocalAsyncSocketExThreadData;

  ...

  }

  下面看一個創建CAsyncSocketExHelperWindow的過程:

  CAsyncSocketExHelperWindow(CAsyncSocketEx::t_AsyncSocketExThreadData* pThreadData)

  {

  // m_pAsyncSocketExWindowData是一個t_AsyncSocketExWindowData數組,

  // 數組的每一個元素代表了一個CAsyncSocketEx,即要服務的socket

  //Initialize data

  m_pAsyncSocketExWindowData = new t_AsyncSocketExWindowData[512]; //Reserve space for 512 active sockets

  memset(m_pAsyncSocketExWindowData, 0, 512*sizeof(t_AsyncSocketExWindowData));

  m_nWindowDataSize=512; // 當前數組的大小,這是可自動擴充的,不過最大不能超過一個最大值

  m_nSocketCount=0; // 當前數組中CAsyncSocketEx的數量

  m_nWindowDataPos=0; // 如果要加一個新的CAsyncSocketEx進來,加到數組的哪個位置

  m_pThreadData = pThreadData; // 這個CAsyncSocketExHelperWindow對應的分發線程,兩者一一對應的

  // 下面創建一個標準的窗口,不過并不顯示出來

  //Create window

  WNDCLASSEX wndclass;

  wndclass.cbSize=sizeof wndclass;

  wndclass.style=0;

  wndclass.lpfnWndProc=WindowProc;

  wndclass.cbClsExtra=0;

  wndclass.cbWndExtra=0;

  wndclass.hInstance=GetModuleHandle(0);

  wndclass.hIcon=0;

  wndclass.hCursor=0;

  wndclass.hbrBackground=0;

  wndclass.lpszMenuName=0;

  wndclass.lpszClassName=_T("CAsyncSocketEx Helper Window");

  wndclass.hIconSm=0;

  RegisterClassEx(&wndclass);

  m_hWnd=CreateWindow(_T("CAsyncSocketEx Helper Window"), _T("CAsyncSocketEx Helper Window"), 0, 0, 0, 0, 0, 0, 0, 0, GetModuleHandle(0));

  ASSERT(m_hWnd);

  SetWindowLongPtr(m_hWnd, GWL_USERDATA, (LONG)this);

  };

  在調用完InitAsyncSocketExInstance()之后,CAsyncSocketEx的create()方法然后:

  SOCKET hSocket = socket(m_SocketData.nFamily, nSocketType, 0); // 這是真正的socket api,建立一個socket

  if (hSocket == INVALID_SOCKET)

  return FALSE;

  m_SocketData.hSocket = hSocket;

  AttachHandle(hSocket); // 將當前創建的socket加到分發線程管理中,這樣可以讓分發線程來負責這個socket的消息

  AttachHandle()調用了CAsyncSocketExHelperWindow的AddSocket方法:

  BOOL AddSocket(CAsyncSocketEx *pSocket, int &nSocketIndex)

  {

  ...

  //Search for free slot

  // 從m_nWindowDataPos開始搜索, 共搜m_nWindowDataSize個位置

  // 由于是下面的模運算i%m_nWindowDataSize,因此到達數組尾時,從重從繞回來,即收遍數組的每一個位置

  for (int i=m_nWindowDataPos;i<(m_nWindowDataSize+m_nWindowDataPos);i++)

  {

  // 注意模運算

  if (!m_pAsyncSocketExWindowData[i%m_nWindowDataSize].m_pSocket) // 這個位置是空的

  {

  m_pAsyncSocketExWindowData[i%m_nWindowDataSize].m_pSocket=pSocket;

  nSocketIndex=i%m_nWindowDataSize; // 在list中的pos

  m_nWindowDataPos=(i+1)%m_nWindowDataSize; // 以后從下一個搜索位置開臺

  m_nSocketCount++;

  return TRUE;

  }

  }

  ...

  }

  即在CAsyncSocketExHelperWindow管理的socket數組中,加上這次的這個CAsyncSocketEx。

  CAsyncSocketEx的create()方法然后:

  if (!AsyncSelect(lEvent))

  {

  Close();

  return FALSE;

  }

  這里AsyncSelect()方法里調用了windows socket api: WSAAsyncSelect(),這個方法可以讓windows在CAsyncSocketEx指定的socket上,當socket事件accept, read, write等發生時,發送消息到CAsyncSocketExHelperWindow中的窗口hWnd,然后CAsyncSocketExHelperWindow再通過回調函數WindowProc將消息發回到負責處理這個消息的CAsyncSocketEx上(這部分下面再詳細分析)。

  繼續CAsyncSocketEx的create()方法:

  if (!Bind(nSocketPort, lpszSocketAddress))

  {

  Close();

  return FALSE;

  }

  Bind()實際上調用了socket api: bind()方法,實現了local address和socket的綁定。

  3、Listen

  create完以后,就是listen,這個比較簡單,直接調用了socket api: listen(),在指定地址、端口進行監聽。

  程序運行到這里,核心的類已經初始化完成了,下面分析當socket有活動時,消息是如何從CAsyncSocketExHelperWindo分發到CAsyncSocketEx的。

  < type="text/javascript"> < src="http://pagead2.googlesyndication.com/pagead/show_ads.js" type="text/javascript">

通過文章完整的描述,大家應該知道了FileZilla 源代碼,希望對大家有幫助!

【編輯推薦】

責任編輯:趙鵬 來源: 網絡轉載
相關推薦

2011-02-23 14:54:58

FileZilla

2011-02-23 14:16:43

FileZilla

2011-02-23 14:39:27

FileZilla

2011-02-23 15:11:27

FileZilla

2011-02-23 15:33:42

FileZilla

2011-02-23 14:26:28

FileZilla

2011-02-23 15:21:06

FileZilla

2011-02-23 15:26:01

FileZilla

2011-02-23 13:47:33

FileZilla

2011-03-01 16:32:58

FileZilla

2011-03-01 16:25:37

FileZilla

2011-03-01 16:01:08

FileZilla

2011-03-01 16:19:27

FileZilla

2023-12-26 14:35:37

2015-08-28 09:38:51

Linux源代碼分析工具

2018-05-25 14:16:55

NFS源代碼線程

2015-08-26 17:38:47

Linux源代碼

2009-07-02 13:59:35

JSP后臺

2011-08-24 15:42:38

LUA源代碼

2020-02-24 09:25:33

代碼開發工具
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产精品一区久久久 | 欧美在线日韩 | 天天天久久久 | jdav视频在线观看免费 | 日韩av视屏| 久久九精品 | 91精品国产综合久久久久 | 国产欧美日韩一区 | 欧美视频免费 | 国产视频一区二区在线观看 | 国产午夜av片 | 久久亚洲欧美日韩精品专区 | 成人精品视频在线 | 四虎在线观看 | 久久精品99 | 成人视屏在线观看 | 一区二区三区电影网 | 国产精品国产三级国产aⅴ中文 | 国内精品久久久久久 | 91久久视频 | av一区在线观看 | 亚洲国产aⅴ成人精品无吗 欧美激情欧美激情在线五月 | 毛片一区二区 | 欧洲一区二区在线 | 国产午夜久久久 | 国产激情91久久精品导航 | 中文字幕亚洲区 | 欧美一级视频免费看 | 一区二区三区免费 | 黄色毛片在线看 | 殴美成人在线视频 | 日韩第一夜| 精品一二三区 | 欧美九九 | 欧美视频在线一区 | 日本在线免费 | 久久国产精品一区 | 国产成人精品视频在线观看 | 欧美另类日韩 | 国产精品69av | 色久伊人 |