鏈嶅姟鍣ㄥ疄鐜幫細
鏈嶅姟鍣ㄤ嬌鐢–++銆傛敞鎰忓畠鐨勭粨鏋勶細綾?ChatRoom 瀹炵幇浜嗗ぇ閮ㄥ垎鐨勫簲鐢ㄩ昏緫銆備負浜嗘敮鎸佹帹妯″瀷涓庢媺妯″瀷錛屾湇鍔″櫒瀹炵幇浜嗙被ChatSession 鍜岀被 PollingChatSession銆?ChatRoom 璋冪敤 ChatRoomCallbackAdapter 瀵硅薄鐨?send 鍑芥暟鏉ヤ紶閫掑鎴鋒秷鎭紝璇ュ璞¢殣钘忎簡涓ょ妯″瀷涔嬮棿鐨勫樊寮傘?br>
ChatRoom 瀹炵幇錛?br>
ChatRoom鏄竴涓櫘閫氱殑C++瀵硅薄錛岃屼笉鏄竴涓猄ervant.
// C++
class ChatRoomCallbackAdapter { /*
*/ };
typedef IceUtil::Handle<ChatRoomCallbackAdapter> ChatRoomCallbackAdapterPtr;
class ChatRoom : public IceUtil::Shared
{
public:
void reserve(const string&);
void unreserve(const string&);
void join(const string&, const ChatRoomCallbackAdapterPtr&);
void leave(const string&);
Ice::Long send(const string&, const string&);
private:
typedef map<string, ChatRoomCallbackAdapterPtr> ChatRoomCallbackMap;
ChatRoomCallbackMap _members;
set<string> _reserved;
IceUtil::Mutex _mutex;
};
typedef IceUtil::Handle<ChatRoom> ChatRoomPtr;
鎴愬憳_reserverd鏄竴涓瓧絎︿覆闆嗗悎錛屽畠瀛樺偍宸茬粡寤虹珛鍥炶瘽錛屼絾鏄繕娌℃湁鍔犲叆鑱婂ぉ瀹ょ殑瀹㈡埛鍚嶃俖members瀛樺偍褰撳墠鑱婂ぉ瀹ょ殑鎵鏈夌敤鎴鳳紙宸茬粡璋冪敤榪噅oin鍑芥暟鐨勭敤鎴鳳級銆?br>
鎴愬憳鍑芥暟 reserve 鍜?unreserve 緇存姢 _reserved 闆嗗悎銆?/p>
// C++
void
ChatRoom::reserve(const string& name)
{
IceUtil::Mutex::Lock sync(_mutex);
if(_reserved.find(name) != _reserved.end() || _members.find(name) != _members.end())
{
throw string("The name " + name + " is already in use.");
}
_reserved.insert(name);
}
void
ChatRoom::unreserve(const string& name)
{
IceUtil::Mutex::Lock sync(_mutex);
_reserved.erase(name);
}
join鎿嶄綔娣誨姞鐢ㄦ埛鍒拌亰澶╁銆?/p>
// C++
void
ChatRoom::join(const string& name, const ChatRoomCallbackAdapterPtr& callback)
{
IceUtil::Mutex::Lock sync(_mutex);
IceUtil::Int64 timestamp = IceUtil::Time::now().toMilliSeconds();
_reserved.erase(name);
Ice::StringSeq names;
ChatRoomCallbackMap::const_iterator q;
for(q = _members.begin(); q != _members.end(); ++q)
{
names.push_back((*q).first);
}
callback->init(names);
_members[name] = callback;
UserJoinedEventPtr e = new UserJoinedEvent(timestamp, name);
for(q = _members.begin(); q != _members.end(); ++q)
{
q->second->join(e);
}
}
send瀹炵幇錛屽悓join瀹炵幇闈炲父綾諱技錛?/p>
// C++
Ice::Long
ChatRoom::send(const string& name, const string& message)
{
IceUtil::Mutex::Lock sync(_mutex);
IceUtil::Int64 timestamp = IceUtil::Time::now().toMilliSeconds();
MessageEventPtr e = new MessageEvent(timestamp, name, message);
for(ChatRoomCallbackMap::iterator q = _members.begin(); q != _members.end(); ++q)
{
q->second->send(e);
}
return timestamp;
}
綾?ChatRoomCallbackAdapter
// C++
class ChatRoomCallbackAdapter : public IceUtil::Shared
{
public:
virtual void init(const Ice::StringSeq&) = 0;
virtual void join(const UserJoinedEventPtr&) = 0;
virtual void leave(const UserLeftEventPtr&) = 0;
virtual void send(const MessageEventPtr&) = 0;
};
鎺ㄦā寮?CallbackAdapter 瀹炵幇錛?
class SessionCallbackAdapter : public ChatRoomCallbackAdapter
{
public:
SessionCallbackAdapter(const ChatRoomCallbackPrx& callback, const ChatSessionPrx& session) : _callback(callback), _session(session)
{
}
void init(const Ice::StringSeq& users)
{
_callback->init_async(new AMICallback<AMI_ChatRoomCallback_init>(_session), users);
}
void join(const UserJoinedEventPtr& e)
{
_callback->join_async(new AMICallback<AMI_ChatRoomCallback_join>(_session),
e->timestamp,
e->name);
}
void leave(const UserLeftEventPtr& e)
{
_callback->leave_async(new AMICallback<AMI_ChatRoomCallback_leave>(_session),
e->timestamp,
e->name);
}
void send(const MessageEventPtr& e)
{
_callback->send_async(new AMICallback<AMI_ChatRoomCallback_send>(_session),
e->timestamp,
e->name,
e->message);
}
private:
const ChatRoomCallbackPrx _callback;
const ChatSessionPrx _session;
};
鐪嬩竴涓婼essionCallbackAdapter鐨勫洓涓垚鍛樺嚱鏁幫紝褰撳紓姝ヨ皟鐢ㄥ畬鎴愭椂錛岄兘浣跨敤綾籄MICallback鏉ユ帴鏀墮氱煡銆傚畠鐨勫畾涔夊涓嬶細
template<class T> class AMICallback : public T
{
public:
AMICallback(const ChatSessionPrx& session) : _session(session)
{
}
virtual void ice_response()
{
}
virtual void ice_exception(const Ice::Exception&)
{
try
{
_session->destroy(); // Collocated
}
catch(const Ice::LocalException&)
{
}
}
private:
const ChatSessionPrx _session;
};
褰撶敤鎴峰洖璋冩搷浣滄姏鍑哄紓甯革紝鏈嶅姟鍣ㄧ珛鍗抽攢姣佸鎴蜂細璇濓紝鍗蟲妸璇ョ敤鎴瘋刀鍑鴻亰澶╁銆傝繖鏄洜涓猴紝涓鏃﹀鎴風殑鍥炶皟瀵硅薄鍑虹幇浜嗕竴嬈″紓甯革紝瀹冧互鍚庝篃灝變笉鍙兘鍐嶆甯搞?br>
鎺ㄦā寮忎細璇濆垱寤?/span>錛?br>
鐜板湪鏉ョ湅涓涓嬩細璇濆垱寤恒傛帹妯″紡鐨勫鎴蜂嬌鐢℅lacier2錛屾墍浠ヨ浣跨敤Glacier2鐨勪細璇濆垱寤烘満鍒躲侴lacier2 鍏佽鐢ㄦ埛閫氳繃鎻愪緵涓涓狦lacier2::SessionManager瀵硅薄鐨勪唬鐞嗘潵鑷畾涔変細璇濆垱寤烘満鍒躲傞氳繃璁劇疆Glacier2.SessionManager灞炴ф潵閰嶇疆Gloacier2錛屽氨鍙互浣跨敤鑷繁鐨勪細璇濈鐞嗗櫒銆備細璇濈鐞嗗櫒闄や簡涓涓猼rivial鏋勯犲嚱鏁幫紙璁劇疆鑱婂ぉ瀹ゆ寚閽堬級錛屽彧鏈変竴涓搷浣滐紝create錛孏lacier2璋冪敤瀹冩潵浠g悊搴旂敤鐨勪細璇濆垱寤恒?create 鎿嶄綔蹇呴』榪斿洖涓涓細璇濅唬鐞嗭紙綾誨瀷涓篏lacier2::Session*錛夈傚疄鐜板涓嬶細
Glacier2::SessionPrx
ChatSessionManagerI::create(const string& name,
const Glacier2::SessionControlPrx&,
const Ice::Current& c)
{
string vname;
try
{
vname = validateName(name);
_chatRoom->reserve(vname);
}
catch(const string& reason)
{
throw CannotCreateSessionException(reason);
}
Glacier2::SessionPrx proxy;
try
{
ChatSessionIPtr session = new ChatSessionI(_chatRoom, vname);
proxy = SessionPrx::uncheckedCast(c.adapter->addWithUUID(session));
Ice::IdentitySeq ids;
ids.push_back(proxy->ice_getIdentity());
sessionControl->identities()->add(ids);
}
catch(const Ice::LocalException&)
{
if(proxy)
{
proxy->destroy();
}
throw CannotCreateSessionException("Internal server error");
}
return proxy;
}
棣栧厛璋冪敤涓涓畝鍗曠殑甯姪鍑芥暟 validateName, 鏉ユ鏌ヤ紶閫掔殑鐢ㄦ埛鍚嶆槸鍚﹀寘鍚潪娉曞瓧絎︼紝騫舵妸瀹冭漿涓哄ぇ鍐欙紝鐒跺悗璋冪敤 reserver鍑芥暟鎶婂畠鍔犲埌鑱婂ぉ瀹ょ殑_reserved闆嗗悎涓傛垜浠鐩戣榪欎簺鎿嶄綔鎶涘嚭鐨勬秷鎭紝騫舵妸瀹冭漿鍖栦負Glacide2::CannotCreateSessionException寮傚父錛屽嵆鍦╟reate鎿嶄綔鐨勫紓甯歌鑼冨0鏄庣殑寮傚父銆?br> 鎺ョ潃瀹炰緥鍖栦竴涓狢hatSessionI瀵硅薄錛堣涓嬮潰錛夋潵鍒涘緩浼氳瘽銆傛敞鎰忚繖涓細璇濅嬌鐢║UID浣滀負瀵硅薄鏍囪瘑錛屾墍浠ヤ繚璇佹爣璇嗙鍞竴銆?br> 鏈鍚庯紝娣誨姞榪欎釜鏂板垱寤虹殑浼氳瘽鏍囪瘑錛孏llacier2鍙氳繃瀹冩潵杞彂緇忚繃榪欎釜浼氳瘽鐨勮姹傘傚疄闄呬笂錛?#8220;鍙漿鍙戠粡榪囪繖涓細璇濈殑騫朵笖鍙埌榪欎釜浼氳瘽鐨勮姹?#8221;錛岃繖鏄竴縐嶅畨鍏ㄧ殑鍔炴硶錛氬鏋滄湁鎭舵剰瀹㈡埛鑳界寽鍑哄彟涓涓鎴蜂細璇濈殑鏍囪瘑錛屽畠涔熶笉鑳藉悜鍒殑瀵硅薄鍙戦佽姹傦紙鍙兘鍦ㄩ櫎浜嗚亰澶╂湇鍔″櫒涔嬪鐨勬湇鍔″櫒涓婏級銆傚鏋滃嚭閿欙紝灝遍攢姣佸垰鍒涘緩鐨勪細璇濆璞★紝榪欐牱閬垮厤浜嗚祫婧愭硠闇層?br> 榪欏氨鏄埄鐢℅lacier2鍒涘緩浼氳瘽鐨勫叏閮ㄣ傚鏋滀綘甯屾湜浣跨敤Glacier2鐨勮璇佹満鍒訛紝鍙互璁劇疆灞炴lacier2.PermissionsVerifier涓烘墽琛岃璇佺殑瀵硅薄浠g悊銆傦紙Glacier2鎻愪緵涓涓唴緗殑鏉冮檺楠岃瘉鍣紝NullPermissionsVerifier錛屽彲浠ユ鏌ョ敤鎴峰悕鍜屽瘑鐮侊級銆?br> 鍥撅細浼氳瘽鍒涘緩浜や簰鍥撅紙鐣ワ級
ChatSessionI綾誨疄鐜頒簡ChatSession鎺ュ彛銆?br>
class ChatSessionI : public ChatSession
{
public:
ChatSessionI(const ChatRoomPtr&, const string&);
virtual void setCallback(const ChatRoomCallbackPrx&, const Ice::Current&);
virtual Ice::Long send(const string&, const Ice::Current&);
virtual void destroy(const Ice::Current&);
private:
const ChatRoomPtr _chatRoom;
const string _name;
ChatRoomCallbackAdapterPtr _callback;
bool _destroy;
IceUtil::Mutex _mutex;
};
typedef IceUtil::Handle<ChatSessionI> ChatSessionIPtr;
鏋勯犲嚱鏁拌緗亰澶╁鍜岀敤鎴峰悕錛屽茍鎶奯destroy璁劇疆涓篎alse.
鐢變簬Glacier2::create鎿嶄綔涓嶅厑璁鎬紶閫掍唬鐞嗭紝蹇呴』鎶婂垱寤轟細璇濆拰璁劇疆鍥炶皟鍒嗘垚涓ゆ銆傝繖鏄痵etCallback鐨勫疄鐜幫紱
void
ChatSessionI::setCallback(const ChatRoomCallbackPrx& callback, const Ice::Current& c)
{
IceUtil::Mutex::Lock sync(_mutex);
if(_destroy)
{
throw Ice::ObjectNotExistException(__FILE__, __LINE__);
}
if(_callback || !callback)
{
return;
}
Ice::Context ctx;
ctx["_fwd"] = "o";
_callback = new SessionCallbackAdapter(callback->ice_context(ctx),
ChatSessionPrx::uncheckedCast(
c.adapter->createProxy(c.id)));
_chatRoom->join(_name, _callback);
}
娉ㄦ剰錛屽湪浣跨敤join浼犻掍唬鐞嗕箣鍓嶏紝鍚戝鎴蜂唬鐞嗘坊鍔犱簡涓涓間負 "o" 鐨刜fwd涓婁笅鏂囥傚畠鎻愮ずGlacier浣跨敤鍗曞悜璋冪敤鏉ヨ漿鍙戝鎴峰洖璋冦傝繖鏍鋒瘮鍙屽悜璋冪敤鏇村姞鏈夋晥銆傚洜涓烘墍鏈夌殑鍥炶皟鎿嶄綔鍧囦負void榪斿洖鍊鹼紝鎵浠ュ彲浠ュ崟鍚戣皟鐢ㄣ?br> 鏈嶅姟鍣ㄧ殑鍥炶皟涓烘櫘閫氱殑鍙屽悜璋冪敤銆傝繖鏍峰綋鍑洪敊鏃跺彲浠ラ氱煡鏈嶅姟鍣ㄣ傚綋瀹㈡埛绔嚭閿欐椂錛岃繖涓緇撴潫瀹㈡埛浼氳瘽寰堟湁鐢ㄣ?br>
涓鏃﹀鎴瘋皟鐢ㄤ簡setCallback錛屽氨鍙互鎺ユ敹鑱婂ぉ瀹ょ殑鍚勭琛屼負閫氱煡銆備笅涓簊end瀹炵幇錛?/p>
Ice::Long
ChatSessionI::send(const string& message, const Ice::Current&)
{
IceUtil::Mutex::Lock sync(_mutex);
if(_destroy)
{
throw Ice::ObjectNotExistException(__FILE__, __LINE__);
}
if(!_callback)
{
throw InvalidMessageException("You cannot send messages until you joined the chat.");
}
string;
try
{
msg = validateMessage(message);
}
catch(const string& reason)
{
throw InvalidMessageException(reason);
}
return _chatRoom->send(_name, msg);
}
瀹㈡埛瑕佺寮鑱婂ぉ瀹わ紝鍙璋冪敤 destory.
void
ChatSessionI::destroy(const Ice::Current& c)
{
IceUtil::Mutex::Lock sync(_mutex);
if(_destroy)
{
throw Ice::ObjectNotExistException(__FILE__, __LINE__);
}
try
{
c.adapter->remove(c.id);
if(_callback == 0)
{
_chatRoom->unreserve(_name);
}
else
{
_chatRoom->leave(_name);
}
}
catch(const Ice::ObjectAdapterDeactivatedException&)
{
// No need to clean up, the server is shutting down.
}
_destroy = true;
}

]]>