有些库可能只提供了异步方法,而asp.net确是同步的,这次就遇到一个问题:页面显示出来以后才会执行回调函数。而我需要的流程是:在回调函数中执行验证,然后才能呈现页面。mutex,autoresetevent提供了通过信号量来协调线程执行步骤的方法。
xmppclientconnection是agsxmppjabber库中的类,调用open会立即返回客户端(ie)呈现该页面,而不管是否成功,同时会在另一个线程会执行登陆,新建帐户的操作,成功后会触发回调事件,这样一来页面呈现后才会执行回调,不符合我们要的逻辑。我们把调用open的线程叫做:jabber线程,把执行登陆的线程叫做:jabber线程的辅助线程。
我最初的想法是使用moniter,代码:
private object objlock=new object();
public void registerjab(string username, string password, string server)
{
_connection.server = server;
_connection.username = username;
_connection.password = password;
_connection.port = 80;
_connection.usessl = false;
_connection.autoresolveconnectserver = true;
_connection.connectserver = null;
_connection.socketconnectiontype = agsxmpp.net.socketconnectiontype.direct;
_connection.usestarttls = true;
_connection.registeraccount = true;
moniter.enter(objlock);
_connection.open();
moniter.wait(objlock);
_connection.close();
}
private void xmppcon_onregistered(object sender)
{
issuccessfull = true;
moniter.exit(objlock);
}
在执行moniter.exit()时会抛出异常:synchronizationlockexception,因为jabber辅助线程并不是锁的拥有者.发现moniter很像临界区,并不适处理这种情况合。
后来,转到了mutex,mutex: 是同步基元,它只向一个线程授予对共享资源的独占访问权。如果一个线程获取了互斥体,则要获取该互斥体的第二个线程将被挂起,直到第一个线程释放该互斥体。
mutex很合适这个功能的实现,可是还有没有更简便的方法呢?那就是autoresetevent:允许线程通过发信号互相通信。通常,此通信涉及线程需要独占访问的资源。最重要的是他提供了线程间通讯的方法,这样可以更灵活的控制线程的调用步骤,我们用到的就是信号量。
代码:
namespace loginbase
{
public class register
{
xmppclientconnection _connection;
static autoresetevent myresetevent;
public bool isused;
public register()
{
_connection = new xmppclientconnection();
_connection.socketconnectiontype = agsxmpp.net.socketconnectiontype.direct;
_connection.onlogin += new objecthandler(xmppcon_onlogin);
_connection.onregistererror += new onxmpperrorhandler(xmppcon_onregerr);
_connection.onregistered += new objecthandler(xmppcon_onregistered);
}
public bool issuccessfull = false;
public void registerjab(string username, string password, string server)
{
_connection.server = server;
_connection.username = username;
_connection.password = password;
_connection.port = 80;
_connection.usessl = false;
_connection.autoresolveconnectserver = true;
_connection.connectserver = null;
_connection.socketconnectiontype = agsxmpp.net.socketconnectiontype.direct;
_connection.usestarttls = true;
_connection.registeraccount = true;
myresetevent = new autoresetevent(false);
_connection.open();
myresetevent.waitone(20 * 1000, true);
_connection.close();
}
private void xmppcon_onregistered(object sender)
{
issuccessfull = true;
myresetevent.set();
}
private void xmppcon_onlogin(object sender)
{
issuccessfull = true;
myresetevent.set();
}
private void xmppcon_onregerr(object sender, element e)
{
//errcode如果是409则已经存在用户
issuccessfull = false;
element xn = e.selectsingleelement(“error”);
if (xn.attribute(“code”) == “409”)
isused = true;
myresetevent.set();
}
}
}
先设置为非终止状态,然后进入jabber线程,阻塞asp线程,并且等待,超时时间为20秒。如果触发了回调事件,则设置状态为终止,asp线程继续执行。
成功完成同步,这样一来,必须等到jabber辅助线程执行完,asp线程才会继续下去。
http://www.cnblogs.com/bluewater/archive/2006/08/14/476720.html