C#中通過(guò)LRU實(shí)現(xiàn)通用高效的超時(shí)連接探測(cè)
編寫網(wǎng)絡(luò)通訊都要面對(duì)一個(gè)問(wèn)題,就是要把很久不存活的死連接清除,如果不這樣做那死連接最終會(huì)占用大量?jī)?nèi)存影響服務(wù)運(yùn)作!在實(shí)現(xiàn)過(guò)程中一般都會(huì)使用ping,pong原理,通過(guò)ping,pong來(lái)更新連接的時(shí)效性,最后通過(guò)掃描連接列表來(lái)清除掉。雖然這種做法比較簡(jiǎn)單,但很難抽取出通用性的封裝,掃描整個(gè)列表復(fù)雜度也比較高。以下講解如何通過(guò)LRU算法實(shí)現(xiàn)一個(gè)通用高效的探測(cè)超時(shí)連接功能類。
什么是LRU
在這里還是要大概介紹一下LRU,LRU算法的設(shè)計(jì)原則是:如果一個(gè)數(shù)據(jù)在最近一段時(shí)間沒(méi)有被訪問(wèn)到,那么在將來(lái)它被訪問(wèn)的可能性也很小.也就是說(shuō),當(dāng)限定的空間已存滿數(shù)據(jù)時(shí),應(yīng)當(dāng)把最久沒(méi)有被訪問(wèn)到的數(shù)據(jù)淘汰.當(dāng)然在這里并不需要使用到自動(dòng)淘汰機(jī)制,只需要把未位到達(dá)超時(shí)的連接清除即可。
在C#中如何實(shí)現(xiàn)LRU
C#并不存在這樣的數(shù)據(jù)結(jié)構(gòu),不過(guò)有一個(gè)結(jié)構(gòu)很適合實(shí)現(xiàn)LRU,這個(gè)結(jié)構(gòu)就是LinkedList雙向鏈表,通過(guò)以下結(jié)構(gòu)圖就容易理解通過(guò)LinkedList實(shí)現(xiàn)LRU

通過(guò)LinkedList的功能我們可以把活越項(xiàng)先移出來(lái),然后再把項(xiàng)移到頭部。在這里需要注意LinkedList的Remove方法,它有兩個(gè)重載版本,兩個(gè)版本的復(fù)雜度不一樣。一個(gè)是O(n)一個(gè)是O(1)所以使用上一定要注意,否則在數(shù)據(jù)多的情況下效率差別巨大(這些細(xì)節(jié)都可以通過(guò)源代碼來(lái)查看)!
代碼實(shí)現(xiàn)
前面已經(jīng)大概講述的原理,接下來(lái)要做的就是代碼實(shí)現(xiàn)了。第一步需要制訂一個(gè)基礎(chǔ)可控測(cè)對(duì)象規(guī)則接口,這樣就可以讓現(xiàn)有的已經(jīng)實(shí)現(xiàn)的功能實(shí)現(xiàn)它并可得到相關(guān)功能的支持。
public interface IDetector
{
double ActiveTime
{ get; set; }
LinkedListNode<IDetector> DetectorNode
{
get;
set;
}
}
接口定義了兩個(gè)屬性,一個(gè)是最近活越時(shí)間,另一個(gè)就是LinkedListNode<IDetector>這個(gè)屬性比交關(guān)鍵,通過(guò)LinkedListNode<IDetector>可以讓LinkedList在Remove時(shí)復(fù)雜度為O(1).接下來(lái)就要針對(duì)基于LRU算法處理超時(shí)制定一個(gè)應(yīng)用規(guī)則
public interface ILRUDetector
{
void Update(IDetector item);
void Detection(int timeout);
double GetTime();
Action<IList<IDetector>> Timeout { get; set; }
}
規(guī)則也是比較簡(jiǎn)單,Update用于更新跟蹤對(duì)象,一般在處理接受ping或pong包后進(jìn)行調(diào)用;Detection方法是探測(cè)超出指定時(shí)間的對(duì)象,時(shí)間當(dāng)位是毫秒,如果存在有超時(shí)的對(duì)象則觸發(fā)Timeout事件;GetTime是獲取探測(cè)器已經(jīng)運(yùn)行的時(shí)間單位毫秒!規(guī)則定好了那接著要做的事實(shí)就是要實(shí)現(xiàn)它:
class LRUDetector : ILRUDetector, IDisposable
{
public LRUDetector()
{
mTimeWatch = new System.Diagnostics.Stopwatch();
mTimeWatch.Restart();
}
private Buffers.XSpinLock xSpinLock = new Buffers.XSpinLock();
private System.Diagnostics.Stopwatch mTimeWatch;
private LinkedList<IDetector> mItems = new LinkedList<IDetector>();
public Action<IList<IDetector>> Timeout
{
get; set;
}
public void Detection(int timeout)
{
double time = GetTime();
List<IDetector> result = new List<IDetector>();
using (xSpinLock.Enter())
{
LinkedListNode<IDetector> last = mItems.Last;
while (last != null && (time - last.Value.ActiveTime) > timeout)
{
mItems.Remove(last);
result.Add(last.Value);
last.Value.DetectorNode = null;
last = mItems.Last;
}
}
if (Timeout != null && result.Count > 0)
Timeout(result);
}
public void Update(IDetector item)
{
using (xSpinLock.Enter())
{
if (item.DetectorNode == null)
item.DetectorNode = new LinkedListNode<IDetector>(item);
item.ActiveTime = GetTime();
if (item.DetectorNode.List == mItems)
mItems.Remove(item.DetectorNode);
mItems.AddFirst(item);
}
}
public void Dispose()
{
mItems.Clear();
}
public double GetTime()
{
return mTimeWatch.Elapsed.TotalMilliseconds;
}
}
代碼并不復(fù)雜,相信不用過(guò)多解釋也能看懂相關(guān)操作原理。
測(cè)試
既然功能已經(jīng)實(shí)現(xiàn),接下來(lái)就要對(duì)代碼進(jìn)行測(cè)試看運(yùn)行效果。測(cè)試代碼比較簡(jiǎn)單首先開(kāi)啟一個(gè)Timer定時(shí)執(zhí)行Detection,另外開(kāi)一個(gè)線程去調(diào)用Update方法
class Program
{
public class TestDetector : IDetector
{
public double ActiveTime { get; set; }
public string Name { get; set; }
public LinkedListNode<IDetector> DetectorNode { get; set; }
}
static void Main(string[] args)
{
LRUDetector lRUDetector = new LRUDetector();
lRUDetector.Timeout = (items) =>
{
foreach (TestDetector item in items)
Console.WriteLine($"{(item.Name)} timeout {lRUDetector.GetTime() - item.ActiveTime}ms");
};
System.Threading.Timer timer = null;
timer = new System.Threading.Timer(o =>
{
timer.Change(-1, -1);
lRUDetector.Detection(5000);
timer.Change(5000, 5000);
}, null, 5000, 5000);
System.Threading.ThreadPool.QueueUserWorkItem(o =>
{
int i = 0;
while (true)
{
System.Threading.Thread.Sleep(500);
i++;
TestDetector testDetector = new TestDetector();
testDetector.Name = "my name is " + i;
lRUDetector.Update(testDetector);
}
});
Console.Read();
}
}
運(yùn)行效果:

以上所述是小編給大家介紹的C#中通過(guò)LRU實(shí)現(xiàn)通用高效的超時(shí)連接探測(cè),希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
相關(guān)文章
.Net WInform開(kāi)發(fā)筆記(二)Winform程序運(yùn)行結(jié)構(gòu)圖及TCP協(xié)議在Winform中的應(yīng)用
中午沒(méi)事,把去年剛畢業(yè)那會(huì)畫(huà)的幾張圖翻出來(lái)了,大概介紹Winform應(yīng)用程序運(yùn)行的過(guò)程,以及TCP協(xié)議在Winform中的應(yīng)用。感興趣的朋友可以了解下;如果有Windows消息機(jī)制等基礎(chǔ),很好理解這兩張2013-01-01
C#實(shí)現(xiàn)的滾動(dòng)網(wǎng)頁(yè)截圖功能示例
這篇文章主要介紹了C#實(shí)現(xiàn)的滾動(dòng)網(wǎng)頁(yè)截圖功能,結(jié)合具體實(shí)例形式分析了C#圖形操作的相關(guān)技巧,需要的朋友可以參考下2017-07-07
C#中txt數(shù)據(jù)寫入的幾種常見(jiàn)方法
這篇文章主要給大家介紹了關(guān)于C#中txt數(shù)據(jù)寫入的幾種常見(jiàn)方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-10-10
C#實(shí)現(xiàn)拆分合并Word表格中的單元格
我們?cè)谑褂肳ord制作表格時(shí),由于表格較為復(fù)雜,只是簡(jiǎn)單的插入行、列并不能滿足我們的需要。要做一個(gè)完整的表格,很多時(shí)候需要將單元格進(jìn)行拆分或者合并。本文將詳細(xì)為您介紹在Word表格中拆分或合并單元格的思路及方法,希望對(duì)大家有所幫助2022-12-12

