在Asp.net中调用异步方法--使用信号量

本文介绍如何解决ASP.NET应用中与XMPP服务异步通信的问题,使用AutoResetEvent来确保回调函数在页面呈现前执行,实现线程间的同步。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

    时间耦合:并发和次序(事件在时间中的相对位置,即“塔”必须在“嘀”之前发生)
    有些库可能只提供了异步方法,而ASP.net确是同步的,这次就遇到一个问题:页面显示出来以后才会执行回调函数。而我需要的流程是:在回调函数中执行验证,然后才能呈现页面。Mutex,AutoResetEvent提供了通过信号量来协调线程执行步骤的方法。
    XmppClientConnection是agsxmppJabber库中的类,调用Open会立即返回客户端(ie)呈现该页面,而不管是否成功,同时会在另一个线程会执行登陆,新建帐户的操作,成功后会触发回调事件,这样一来页面呈现后才会执行回调,不符合我们要的逻辑。我们把调用Open的线程叫做:Jabber线程,把执行登陆的线程叫做:Jabber线程的辅助线程。

    我最初的想法是使用Moniter,代码:

None.gifprivate object objlock=new object();
None.gif        
public void RegisterJab(string username, string password, string server)
ExpandedBlockStart.gifContractedBlock.gif        
dot.gif{
InBlock.gif
InBlock.gif            _connection.Server 
= server;
InBlock.gif            _connection.Username 
= username;
InBlock.gif            _connection.Password 
= password;
InBlock.gif            _connection.Port 
= 80;
InBlock.gif            _connection.UseSSL 
= false;
InBlock.gif            _connection.AutoResolveConnectServer 
= true;
InBlock.gif            _connection.ConnectServer 
= null;
InBlock.gif            _connection.SocketConnectionType 
= agsXMPP.net.SocketConnectionType.Direct;
InBlock.gif            _connection.UseStartTLS 
= true;
InBlock.gif            _connection.RegisterAccount 
= true;
InBlock.gif            Moniter.Enter(objlock);
InBlock.gif            _connection.Open();
InBlock.gif            Moniter.Wait(objlock);
InBlock.gif            _connection.Close();
InBlock.gif
ExpandedBlockEnd.gif        }

None.gif        
private void XmppCon_OnRegistered(object sender)
ExpandedBlockStart.gifContractedBlock.gif        
dot.gif{
InBlock.gif            IsSuccessfull 
= true;
InBlock.gif            Moniter.Exit(objlock);
ExpandedBlockEnd.gif        }


    在执行Moniter.Exit()时会抛出异常:SynchronizationLockException,因为Jabber辅助线程并不是锁的拥有者.发现Moniter很像临界区,并不适处理这种情况合。
    后来,转到了Mutex,Mutex: 是同步基元,它只向一个线程授予对共享资源的独占访问权。如果一个线程获取了互斥体,则要获取该互斥体的第二个线程将被挂起,直到第一个线程释放该互斥体。
    Mutex很合适这个功能的实现,可是还有没有更简便的方法呢?那就是AutoResetEvent:允许线程通过发信号互相通信。通常,此通信涉及线程需要独占访问的资源。最重要的是他提供了线程间通讯的方法,这样可以更灵活的控制线程的调用步骤,我们用到的就是信号量。
代码:

None.gifnamespace LoginBase
ExpandedBlockStart.gifContractedBlock.gif
dot.gif{
InBlock.gif    
public class Register
ExpandedSubBlockStart.gifContractedSubBlock.gif    
dot.gif{
InBlock.gif        XmppClientConnection _connection;
InBlock.gif        
static AutoResetEvent myResetEvent;
InBlock.gif        
public bool IsUsed;
InBlock.gif        
public Register()
ExpandedSubBlockStart.gifContractedSubBlock.gif        
dot.gif{
InBlock.gif
InBlock.gif            _connection 
= new XmppClientConnection();
InBlock.gif            _connection.SocketConnectionType 
= agsXMPP.net.SocketConnectionType.Direct;
InBlock.gif            _connection.OnLogin 
+= new ObjectHandler(XmppCon_OnLogin);
InBlock.gif            _connection.OnRegisterError 
+= new OnXmppErrorHandler(XmppCon_OnRegErr);
InBlock.gif            _connection.OnRegistered 
+= new ObjectHandler(XmppCon_OnRegistered);
InBlock.gif
ExpandedSubBlockEnd.gif        }

InBlock.gif        
public bool IsSuccessfull = false;
InBlock.gif        
public void RegisterJab(string username, string password, string server)
ExpandedSubBlockStart.gifContractedSubBlock.gif        
dot.gif{
InBlock.gif
InBlock.gif            _connection.Server 
= server;
InBlock.gif            _connection.Username 
= username;
InBlock.gif            _connection.Password 
= password;
InBlock.gif            _connection.Port 
= 80;
InBlock.gif            _connection.UseSSL 
= false;
InBlock.gif            _connection.AutoResolveConnectServer 
= true;
InBlock.gif            _connection.ConnectServer 
= null;
InBlock.gif            _connection.SocketConnectionType 
= agsXMPP.net.SocketConnectionType.Direct;
InBlock.gif            _connection.UseStartTLS 
= true;
InBlock.gif            _connection.RegisterAccount 
= true;
InBlock.gif            myResetEvent 
= new AutoResetEvent(false);
InBlock.gif            _connection.Open();
InBlock.gif            myResetEvent.WaitOne(
20 * 1000true);
InBlock.gif            _connection.Close();
InBlock.gif
ExpandedSubBlockEnd.gif        }

InBlock.gif        
private void XmppCon_OnRegistered(object sender)
ExpandedSubBlockStart.gifContractedSubBlock.gif        
dot.gif{
InBlock.gif            IsSuccessfull 
= true;
InBlock.gif            myResetEvent.Set();
ExpandedSubBlockEnd.gif        }

InBlock.gif
InBlock.gif        
private void XmppCon_OnLogin(object sender)
ExpandedSubBlockStart.gifContractedSubBlock.gif        
dot.gif{
InBlock.gif            IsSuccessfull 
= true;
InBlock.gif            myResetEvent.Set();
InBlock.gif
InBlock.gif
ExpandedSubBlockEnd.gif        }

InBlock.gif        
private void XmppCon_OnRegErr(object sender, Element e)
ExpandedSubBlockStart.gifContractedSubBlock.gif        
dot.gif{
InBlock.gif            
//errCode如果是409则已经存在用户
InBlock.gif
            IsSuccessfull = false;
InBlock.gif            Element xn 
= e.SelectSingleElement("error");
InBlock.gif            
if (xn.Attribute("code"== "409")
InBlock.gif                IsUsed 
= true;
InBlock.gif            myResetEvent.Set();
InBlock.gif
ExpandedSubBlockEnd.gif        }

InBlock.gif
InBlock.gif
ExpandedSubBlockEnd.gif    }

InBlock.gif
ExpandedBlockEnd.gif}

None.gif
None.gif

    先设置为非终止状态,然后进入Jabber线程,阻塞Asp线程,并且等待,超时时间为20秒。如果触发了回调事件,则设置状态为终止,asp线程继续执行。
    成功完成同步,这样一来,必须等到Jabber辅助线程执行完,Asp线程才会继续下去。
注:RegisterAsyncTask,也可以实现类似功能。好像它和异步页有很强的关联,用它实现这个类就没有了通用性。详细的可以看Msdn

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值