Merge pull request #659 from nofficalfs/Dev-GoLikeChannel
Add `lib.channel` to give a method of message receiving.
This commit is contained in:
commit
2f5c2320cb
102
game/game.js
102
game/game.js
|
@ -363,6 +363,108 @@ new Promise(resolve=>{
|
||||||
}
|
}
|
||||||
}],
|
}],
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* **无名杀频道推送机制**
|
||||||
|
*
|
||||||
|
* 鉴于`Javascript`的特性及自身对所需功能的思考,这是一个参考`Golang`的`channel`设计的、完全和`go channel`不一样的异步消息传递对象
|
||||||
|
*
|
||||||
|
* 当且仅当接收方和发送方均存在时进行消息传递,完全保证信息传递的单一性(发送方/接收方一旦确定则无法更改)和准确性(发送方必然将消息发送给接收方)
|
||||||
|
*
|
||||||
|
* 若存在发送方/接收方时调用`send`/`receive`,将报错
|
||||||
|
*
|
||||||
|
* 若需要异步/不报错发送信息,请等待`lib.actor`
|
||||||
|
*
|
||||||
|
* @template T
|
||||||
|
* @example
|
||||||
|
* // 创建一个频道
|
||||||
|
* const channel = new lib.channel();
|
||||||
|
*
|
||||||
|
* // 从某个角落接收channel发出的消息,若无消息则等待
|
||||||
|
* const message = await channel.receive();
|
||||||
|
*
|
||||||
|
* // 从某个角落向channel发消息,若无消息接收则等待
|
||||||
|
* await channel.send(item);
|
||||||
|
*/
|
||||||
|
channel: class {
|
||||||
|
/**
|
||||||
|
* @template TValue
|
||||||
|
* @callback PromiseResolve
|
||||||
|
* @param {TValue} value
|
||||||
|
* @returns {void}
|
||||||
|
*/
|
||||||
|
constructor() {
|
||||||
|
/**
|
||||||
|
* @type {"active" | "receiving" | "sending"}
|
||||||
|
*/
|
||||||
|
this.status = "active";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @type {PromiseResolve<T> | [T, PromiseResolve<void>] | null}
|
||||||
|
*/
|
||||||
|
this._buffer = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 向该频道发送消息,在消息未被接受前将等待
|
||||||
|
*
|
||||||
|
* @param {T} value - 要发送的消息
|
||||||
|
* @returns {Promise<void>}
|
||||||
|
*/
|
||||||
|
send(value) {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
switch (this.status) {
|
||||||
|
case "sending":
|
||||||
|
// TODO: handle the error.
|
||||||
|
reject(new Error());
|
||||||
|
break;
|
||||||
|
case "receiving":
|
||||||
|
/**
|
||||||
|
* @type {PromiseResolve<T>}
|
||||||
|
*/
|
||||||
|
const buffer = this._buffer;
|
||||||
|
this._buffer = null;
|
||||||
|
buffer(value);
|
||||||
|
this.status = "active";
|
||||||
|
resolve();
|
||||||
|
break ;
|
||||||
|
case "active":
|
||||||
|
this.status = "sending";
|
||||||
|
this._buffer = [value, resolve];
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 接收频道所发送的消息,若无消息发送则等待
|
||||||
|
*
|
||||||
|
* @returns {Promise<T>} 接收到的消息
|
||||||
|
*/
|
||||||
|
receive() {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
switch (this.status) {
|
||||||
|
case "receiving":
|
||||||
|
// TODO: handle the error.
|
||||||
|
reject(new Error());
|
||||||
|
break;
|
||||||
|
case "sending":
|
||||||
|
/**
|
||||||
|
* @type {[T, PromiseResolve<void>]}
|
||||||
|
*/
|
||||||
|
const buffer = this._buffer;
|
||||||
|
this._buffer = null;
|
||||||
|
resolve(buffer[0]);
|
||||||
|
this.status = "active";
|
||||||
|
buffer[1]();
|
||||||
|
break ;
|
||||||
|
case "active":
|
||||||
|
this.status = "receiving";
|
||||||
|
this._buffer = resolve;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
/**
|
/**
|
||||||
* **无名杀消息推送库**
|
* **无名杀消息推送库**
|
||||||
*
|
*
|
||||||
|
|
Loading…
Reference in New Issue