自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

服務(wù)主機(jī)的路由器與負(fù)載均衡和實(shí)現(xiàn)思路

網(wǎng)絡(luò) 路由交換
WCF中間件的路由功能是在客戶端與服務(wù)端之間加入中介服務(wù),用來轉(zhuǎn)發(fā)它們之間的消息。實(shí)現(xiàn)消息的轉(zhuǎn)發(fā)可以修改WCF服務(wù)消息頭的內(nèi)容,重新指定服務(wù)地址即可。

路由器介紹及演示

WCF中間件的路由功能是在客戶端與服務(wù)端之間加入中介服務(wù),用來轉(zhuǎn)發(fā)它們之間的消息。實(shí)現(xiàn)消息的轉(zhuǎn)發(fā)可以修改WCF服務(wù)消息頭的內(nèi)容,重新指定服務(wù)地址即可,那給消息頭指定的服務(wù)地址從哪來,需要給路由器配置服務(wù)端地址目錄,路由器與服務(wù)端肯定不是一對(duì)一的,路由器可以指定多個(gè)服務(wù)端,而路由器把客戶端連接指定給哪個(gè)服務(wù)端這里就有一個(gè)算法,算法的優(yōu)劣就決定了中間件負(fù)載均衡的能力。

下面演示了中間件的路由功能,把Out目錄中的程序復(fù)制6份,分別改名如下,3個(gè)客戶端,1個(gè)路有中間件,2個(gè)服務(wù)中間件,還要修改每個(gè)程序的相關(guān)配置;這樣先啟動(dòng)路由中間件Router和服務(wù)中間件WCFservser1、WCFServer2,然后分別啟動(dòng)2個(gè)客戶端程序,路由中間件和服務(wù)中間件就會(huì)顯示客戶端的連接信息。3個(gè)客戶端會(huì)有2個(gè)分配到一個(gè)服務(wù)中間件,一個(gè)客戶端分配到另外一個(gè)服務(wù)中間件,不會(huì)說3個(gè)客戶端都分配到1個(gè)服務(wù)中間件,這是由路由中間件的負(fù)載均衡算法決定的;

 

 

 

 

路由功能的實(shí)現(xiàn)

框架增加了一個(gè)路由服務(wù)對(duì)象Router,用它來攔截客戶端發(fā)送的消息,攔截方法ProcessMessage(Message requestMessage);

首先根據(jù)路由目錄結(jié)合負(fù)載均衡的算法取得服務(wù)地址endpointAddress,然后創(chuàng)建WCF通道并綁定新的服務(wù)地址,調(diào)用服務(wù)端的方法;

 

  1. /// <summary> 
  2.         /// 截獲從Client端發(fā)送的消息轉(zhuǎn)發(fā)到目標(biāo)終結(jié)點(diǎn)并獲得返回值給Client端 
  3.         /// </summary> 
  4.         /// <param name="requestMessage"></param> 
  5.         /// <returns></returns> 
  6.         public Message ProcessMessage(Message requestMessage) 
  7.         { 
  8.             //Binding binding = null
  9.             EndpointAddress endpointAddress = null
  10.             GetServiceEndpoint(requestMessage, out endpointAddress); 
  11.             IDuplexRouterCallback callback = OperationContext.Current.GetCallbackChannel<IDuplexRouterCallback>(); 
  12.             NetTcpBinding tbinding = new NetTcpBinding("netTcpExpenseService_ForSupplier"); 
  13.             using (DuplexChannelFactory<IRouterService> factory = new DuplexChannelFactory<IRouterService>(new InstanceContext(null, new DuplexRouterCallback(callback)), tbinding, endpointAddress)) 
  14.             { 
  15.  
  16.                 factory.Endpoint.Behaviors.Add(new MustUnderstandBehavior(false)); 
  17.                 IRouterService proxy = factory.CreateChannel(); 
  18.  
  19.                 using (proxy as IDisposable) 
  20.                 { 
  21.                     // 請(qǐng)求消息記錄 
  22.                     IClientChannel clientChannel = proxy as IClientChannel; 
  23.                     //Console.WriteLine(String.Format("Request received at {0}, to {1}\r\n\tAction: {2}", DateTime.Now, clientChannel.RemoteAddress.Uri.AbsoluteUri, requestMessage.Headers.Action)); 
  24.                     if (Convert.ToInt32(HostSettingConfig.GetValue("debug")) == 1) 
  25.                         hostwcfMsg(DateTime.Now, String.Format("路由請(qǐng)求消息發(fā)送:  {0}", clientChannel.RemoteAddress.Uri.AbsoluteUri)); 
  26.                     // 調(diào)用綁定的終結(jié)點(diǎn)的服務(wù)方法 
  27.                     Message responseMessage = proxy.ProcessMessage(requestMessage); 
  28.  
  29.                     // 應(yīng)答消息記錄 
  30.                     //Console.WriteLine(String.Format("Reply received at {0}\r\n\tAction: {1}", DateTime.Now, responseMessage.Headers.Action)); 
  31.                     //Console.WriteLine(); 
  32.                     //hostwcfMsg(DateTime.Now, String.Format("應(yīng)答消息: {0}", responseMessage.Headers.Action)); 
  33.                     return responseMessage; 
  34.                 } 
  35.             } 
  36.         } 

#p#負(fù)載均衡的實(shí)現(xiàn)

負(fù)載均衡實(shí)現(xiàn)代碼在Router對(duì)象中的GetServiceEndpoint方法中,定義了RegistrationList對(duì)象用來存儲(chǔ)客戶端列表,在消息頭中增加了兩個(gè)標(biāo)識(shí)routerID和CMD,routerID用來識(shí)別客戶端,值是客戶端創(chuàng)建發(fā)送到路由中間件,每個(gè)客戶端只有一個(gè)routerID;CMD用來客戶端發(fā)送給路由中間件的命令標(biāo)識(shí),這里只用到了一個(gè)就是”Quit”就是卸載路由中間件中的RegistrationList客戶端列表;

解決了識(shí)別客戶端的問題,那平均算法每個(gè)客戶端分配到哪個(gè)服務(wù)中間件就很簡(jiǎn)單了,RoundRobinCount就記錄每個(gè)服務(wù)中間件對(duì)應(yīng)的客戶端個(gè)數(shù),哪個(gè)服務(wù)中間件數(shù)量少新的客戶端就分配給它;

 

  1. private void GetServiceEndpoint(Message requestMessage,out EndpointAddress endpointAddress) 
  2.         { 
  3.  
  4.             string ns = "http://www.3yxx.com/"
  5.             string routerID = GetHeaderValue("routerID", ns); 
  6.             string cmd = GetHeaderValue("CMD", ns); 
  7.             string contractNamespace = requestMessage.Headers.Action.Substring(0, requestMessage.Headers.Action.LastIndexOf("/")); 
  8.  
  9.             
  10.  
  11.             RegistrationInfo regInfo = null
  12.  
  13.             if (Router.RoundRobinCount.ContainsKey(routerID)) 
  14.             { 
  15.                 int key = Router.RoundRobinCount[routerID]; 
  16.                 regInfo = Router.RegistrationList[key]; 
  17.                 if (cmd == "Quit") 
  18.                 { 
  19.                     regInfo.ClientNum -1
  20.                 } 
  21.             } 
  22.             else 
  23.             { 
  24.                 //根據(jù)指定的協(xié)議名稱空間從注冊(cè)表容器中得到注冊(cè)項(xiàng)列表 
  25.                 var results = from item in Router.RegistrationList 
  26.                               where item.Value.ContractNamespace.Contains(contractNamespace) 
  27.                               orderby item.Value.ClientNum ascending 
  28.                               select item; 
  29.                 if (results.Count<KeyValuePair<int, RegistrationInfo>>() > 0) 
  30.                 { 
  31.                     var val = results.First<KeyValuePair<int, RegistrationInfo>>(); 
  32.                     Router.RoundRobinCount.Add(routerID, val.Key); 
  33.                     val.Value.ClientNum += 1; 
  34.                     regInfo = val.Value; 
  35.                 } 
  36.             } 
  37.  
  38.             Uri addressUri = new Uri(regInfo.Address); 
  39.  
  40.             //binding = CustomBindConfig.GetRouterBinding(addressUri.Scheme); 
  41.             endpointAddress = new EndpointAddress(regInfo.Address); 
  42.             //重設(shè)Message的目標(biāo)終結(jié)點(diǎn) 
  43.             requestMessage.Headers.To = new Uri(regInfo.Address); 
  44.  
  45.             hostwcfRouter(RegistrationList.Values.ToList()); 
  46.         } 

WCF客戶端配置和中間件配置還有路由地址配置

如果部署的時(shí)候不使用中間件的路由功能,那客戶端配置服務(wù)地址直接指定服務(wù)端WCF地址就行了,而如果啟用路由功能,那客戶端就配置路由中間件的WCF地址,路由中間件再配置路由目錄,對(duì)應(yīng)服務(wù)端;

客戶端WCF配置和服務(wù)端WCF配置還有一個(gè)地方值得注意,就是netTcpBinding節(jié)點(diǎn)的配置;***配置為取消服務(wù)器憑據(jù)認(rèn)證,因?yàn)槿绻慌渲脼镹one,當(dāng)客戶端斷開連接后再連接的時(shí)候就會(huì)一些安全性驗(yàn)證,導(dǎo)致連接報(bào)錯(cuò),所以對(duì)WCF安全性方面的配置沒有吃透的話還是先這樣配置好;

1)客戶端App.Config配置

 

 

 

 

2)路由中間件App.Config配置和路由目錄RouterBill.xml配置

 

 

 

 

3)服務(wù)中間件App.Config配置

 

 

5.總結(jié)

本章我們?cè)敿?xì)講解了EFW框架中的WCF中間件的路由功能和負(fù)載均衡的實(shí)現(xiàn),代碼很簡(jiǎn)單,但深入理解卻沒那么容易,我也只是略懂點(diǎn)皮毛,參考了網(wǎng)上資料把功能實(shí)現(xiàn)而已,而想要做成專業(yè)級(jí)別的中間件是有一個(gè)過程的,所以不只是我,也需要有興趣的人一起完善它;

路由實(shí)例程序下載 :http://pan.baidu.com/s/1eQ8FscE

注意:實(shí)例中的配置文件中的IP地址192.168.1.3修改為你本機(jī)的IP地址;

 

責(zé)任編輯:林琳 來源: 博客園精華區(qū)
相關(guān)推薦

2009-12-18 16:05:03

智能型負(fù)載均衡

2013-03-12 09:36:49

路由器負(fù)載均衡路由技術(shù)

2013-01-16 15:47:33

路由器IP地址負(fù)載均衡

2010-05-10 15:12:21

路由器負(fù)載均衡

2010-04-25 18:23:24

負(fù)載均衡路由器

2019-02-21 09:18:27

服務(wù)路由負(fù)載均衡微服務(wù)

2009-12-08 17:49:17

2010-05-04 18:10:07

路由器負(fù)載均衡

2009-04-09 09:35:00

多WAN口寬帶路由器負(fù)載均衡

2018-11-27 12:56:09

負(fù)載均衡應(yīng)用路由

2024-05-13 18:35:06

負(fù)載均衡主機(jī)端口

2009-01-10 18:53:01

服務(wù)器ServerDNS

2009-07-22 10:25:37

2009-12-09 14:47:54

WAN口寬帶路由器

2010-05-05 22:40:21

apache服務(wù)器負(fù)載均衡

2010-04-22 16:31:09

配置網(wǎng)絡(luò)負(fù)載均衡

2010-11-19 12:53:53

梭子魚負(fù)載均衡

2010-04-22 23:33:48

負(fù)載均衡設(shè)置

2011-11-22 21:26:59

pfSense配置Web服務(wù)器負(fù)載均衡

2013-10-24 09:43:39

路由器
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)