Posted on 2011-01-19 17:08
點(diǎn)點(diǎn)滴滴 閱讀(809)
評(píng)論(0) 編輯 收藏 引用 所屬分類(lèi):
10 服務(wù)器
前面一直都在說(shuō)接收數(shù)據(jù)時(shí)的處理方法,我們應(yīng)該用專(zhuān)門(mén)的IO線(xiàn)程,接收到完整的消息包后加入到主線(xiàn)程的消息隊(duì)列,但是主線(xiàn)程如何發(fā)送數(shù)據(jù)還沒(méi)有探討過(guò)。
一般來(lái)說(shuō)最直接的方法就是邏輯線(xiàn)程什么時(shí)候想發(fā)數(shù)據(jù)了就直接調(diào)用相關(guān)的socket API發(fā)送,這要求服務(wù)器的玩家對(duì)象中保存其連接的socket句柄。但是直接send調(diào)用有時(shí)候有會(huì)存在一些問(wèn)題,比如遇到系統(tǒng)的發(fā)送緩沖區(qū)滿(mǎn)而阻塞住的情況,或者只發(fā)送了一部分?jǐn)?shù)據(jù)的情況也時(shí)有發(fā)生。我們可以將要發(fā)送的數(shù)據(jù)先緩存一下,這樣遇到未發(fā)送完的,在邏輯線(xiàn)程的下一次處理時(shí)可以接著再發(fā)送。
考慮數(shù)據(jù)緩存的話(huà),那這里這可以有兩種實(shí)現(xiàn)方式了,一是為每個(gè)玩家準(zhǔn)備一個(gè)緩沖區(qū),另外就是只有一個(gè)全局的緩沖區(qū),要發(fā)送的數(shù)據(jù)加入到全局緩沖區(qū)的時(shí)候同時(shí)要指明這個(gè)數(shù)據(jù)是發(fā)到哪個(gè)socket的。如果使用全局緩沖區(qū)的話(huà),那我們可以再進(jìn)一步,使用一個(gè)獨(dú)立的線(xiàn)程來(lái)處理數(shù)據(jù)發(fā)送,類(lèi)似于邏輯線(xiàn)程對(duì)數(shù)據(jù)的處理方式,這個(gè)獨(dú)立發(fā)送線(xiàn)程也維護(hù)一個(gè)消息隊(duì)列,邏輯線(xiàn)程要發(fā)數(shù)據(jù)時(shí)也只是把數(shù)據(jù)加入到這個(gè)隊(duì)列中,發(fā)送線(xiàn)程循環(huán)取包來(lái)執(zhí)行send調(diào)用,這時(shí)的阻塞也就不會(huì)對(duì)邏輯線(xiàn)程有任何影響了。
采用第二種方式還可以附帶一個(gè)優(yōu)化方案。一般對(duì)于廣播消息而言,發(fā)送給周?chē)婕业臄?shù)據(jù)都是完全相同的,我們?nèi)绻捎媒o每個(gè)玩家一個(gè)緩沖隊(duì)列的方式,這個(gè)數(shù)據(jù)包將需要拷貝多份,而采用一個(gè)全局發(fā)送隊(duì)列時(shí),我們只需要把這個(gè)消息入隊(duì)一次,同時(shí)指明該消息包是要發(fā)送給哪些socket的即可。有關(guān)該優(yōu)化的說(shuō)明在云風(fēng)描述其連接服務(wù)器實(shí)現(xiàn)的blog文章中也有講到,有興趣的可以去閱讀一下。