本帖最后由 天若有情 于 2014-6-16 14:02 编辑
3 \+ _8 a X8 U3 Q# q- W9 B0 S3 _( y/ ^7 F1 @8 e: K. m" c
本文将向大家介绍如何托管内部WCF服务和公共WCF服务,为了托管内部WCF服务,需要建立一个内部端点,并使用内部角色通信,与在输入端点上托管一个外部服务最主要的区别是内部端点不具有负载均衡特性,而输入端点是挂钩在负载均衡器上的,具有负载均衡功能。
8 q0 ~ R7 V' w; C+ q G 托管内部WCF服务0 o. e4 f( b% |' m/ y1 ]: m! |+ Q; X
其实要托管一个内部WCF服务很简单,唯一需要注意的是传递给 ServiceHost 的基地址不同,因为端口号和IP地址要等到运行时才知道,因此需要创建一个主机,动态地传递这些信息给它。
0 p3 m5 o. L5 h public override bool OnStart()# k/ i f5 h4 p5 p8 \1 P& x6 A
{' C# y8 ~) v6 R; r: B$ y6 D
// 设置最大并发连接数
3 y" g( F* P* h ServicePointManager.DefaultConnectionLimit = 12;" r+ d2 I* R1 Q3 n" A( Z2 L
DiagnosticMonitor.Start(“DiagnosticsConnectionString”);
* Y8 s' U" o! k, t4 u% D4 @ // For information on handling configuration changes* y, C- U5 \6 \1 _& R# K5 _5 n
// see the MSDN topic at =166357.( ~( y3 d2 a5 ^; ?' l
RoleEnvironment.Changing += RoleEnvironmentChanging;
9 }% V0 P7 c Y, u/ l1 l$ i4 t StartWCFService();
+ w9 g) c- y: c6 O. r return base.OnStart();. F+ E/ F: l5 R. W( [/ h! i
}- ~- F$ u, V! [! y
private void StartWCFService()
1 J2 S, L1 h/ s( K3 h( A {
9 L: ~$ ?" w7 \ h2 G var baseAddress = String.Format(& w. r8 n H1 C" P! Q: C
“net.tcp://{0}”,0 B# d! u; g- K
RoleEnvironment.CurrentRoleInstance.InstanceEndpoints[雨林木风系统“EchoService”].IPEndpoint
/ K V0 a+ i! ^' x/ f );
( J" t" ^: P8 j% b( r0 G var host = new ServiceHost(typeof(EchoService), new Uri(baseAddress));# _0 ?. o! y7 v- C$ V9 U4 K
host.AddServiceEndpoint(typeof(IEchoService), new NetTcpBinding(SecurityMode.None), “echo”);
, ?8 X7 a: ?8 S# y1 ~; _: M host.Open();
; B |' R# a- K+ k' P7 H" q" R 使用内部WCF服务
: D/ l4 m. X2 T; z% l 我想从我另一个托管的服务调用这个服务,下面就是调用这个服务的所有代码:: X4 l+ X Z; C; m& X
protected void Button1_Click(object sender, EventArgs e)( I8 d1 `3 P" t; o
{
' @# y0 d$ a( m var factory = new ChannelFactory(new NetTcpBinding(SecurityMode.None));
1 S" T/ [' B. o( [* t z1 ^ var channel = factory.CreateChannel(GetRandomEndpoint());
3 f" q$ ~1 T' O F4 i, }/ W Label1.Text = channel.Echo(TextBox1.Text);4 y3 Q' A6 [: `% e8 W }
}0 J1 k6 K7 g1 ]5 y( {: r
private EndpointAddress GetRandomEndpoint()/ X9 T, J2 B2 p8 ]
{$ O% T4 v7 Y; e: I6 I$ q
var endpoints = RoleEnvironment.Roles[“WorkerHost”].Instances8 @* E( b5 m$ [8 y, ~0 d, U
.Select(i =》 i.InstanceEndpoints[“EchoService”])" [/ z, C/ ?: b# q7 { N
.ToArray();! B7 F/ A) S" I, w$ r& P, W7 ]
var r = new Random(DateTime.Now.Millisecond);
. U7 e- w1 f6 d8 y return new EndpointAddress(
4 C' @8 W6 k, p5 p! A. e$ P6 q String.Format() k) ?5 J" p1 }% S2 e! N
“net.tcp://{0}/echo”,
# Z2 y4 K5 _7 f! L. c8 B endpoints[r.Next(endpoints.Count() - 1)].IPEndpoint)
, B7 h" b% W! e1 t3 i7 ?# t/ @ );
j- ~; E: B3 c7 E }
2 x' R0 c3 y" T! Z 这里唯一要注意的是查询F abric ,确定 WorkerHost 角色中实现了 EchoService 端点,深度系统官网并随机给它们路由请求的所有端点,本来不需要路由请求,我这样做是因为内部端点没有负载均衡功能,我希望在每个 WorkerHost 实例上均匀地分配负载。
7 E! J6 F* C& }8 o 我发现一个技巧,就是不需要缓存你找到的 IPEndpoint ,因为它已经缓存在API调用中,但根据最佳实践,你应该缓存你的 ChannelFactory 。" y. d' m* k6 {1 J% D8 v
托管公共WCF服务2 ^1 K# @& A$ v0 f( m
托管公共WCF服务也很简单,唯一需要注意的是要使用一个新的行为,为MEX端点处理负载均衡,此外,在你的服务上需要包括一个类属性处理地址过滤不匹配问题。: @- C( Z8 ^8 X1 p
|