Les derniers articles sont liés au développement de mini-programmes WeChat, alors quelqu'un a demandé : "Je ne comprends pas les mini-programmes, pouvez-vous écrire autre chose ?".En fait, vous n'avez pas besoin de prêter trop d'attention au « petit programme », car le « petit programme » n'est qu'un scénario de développement dans l'article. Les problèmes que nous résolvons réellement ne se rencontrent pas uniquement dans les petits programmes. , et les moyens pour résoudre le problème sont complètement Cela n'a rien à voir avec des mini programmes !
laisse un problème dans l'appel asynchrone du proxy encapsulant l'applet WeChat :
est comme wx.request()
Comment faire encapsuler cette situation où il y a une valeur de retour ?
Si vous devez annuler la demande pendant le processus de demande, la valeur de retour de wx.request()
sera utilisée :
const requestTask = wx.request(...); if (...) { // 因为某些原因需要取消这次请求 requestTask.abort(); }
Le awx.request()
encapsulé renverra un objet Promise, suivi de wx.request()
La valeur de retour d'origine n'a rien à voir avec cela. Si vous souhaitez pouvoir annuler la demande, vous devez faire ressortir la valeur de retour originale de wx.request()
. Que devez-vous faire ?
function wxPromisify(fn) { return async function (args) { return new Promise((resolve, reject) => { const originalResult = fn({ // ^^^^^^^^^^^^^^^^^^^^^^^ // 怎么把 originalResult 带出去? ...(args || {}), success: res => resolve(res), fail: err => reject(err) }); }); }; }
Ce n'est pas trop compliqué, voici plusieurs options :
{ promise, originalResult}
ou [promise, originalResult]
; awx.request(params, outBox = {})
, et attribue une valeur à outBox
lors du traitement : outBox.originalResult
; promise.originalResult = ...
. Du point de vue de l'utilisateur, la plupart du temps, la valeur de retour d'origine n'est pas nécessaire pour le moment, nous voulons absolument await awx.request()
au lieu de déconstruire d'abord, puis await
(ou ), la première méthode n’est donc pas facultative. then()
: wxPromisify()
, mais maintenant il devrait suffire d'ajouter une variable temporaire : return new Promise()
function wxPromisify(fn) { return async function (args) { const promise = new Promise((resolve, reject) => { // ^^^^^^^^^^^^^^^^ promise.originalResult = fn({ // ^^^^^^^^^^^^^^^^^^^^^^^^^ ...(args || {}), success: res => resolve(res), fail: err => reject(err) }); }); return promise; // ^^^^^^^^^^^^^^^ }; }
TypeError: Cannot set property 'originalResult' of undefined
était une variable locale accessible directement, il n'y avait donc aucun problème à l'utiliser dans sa sous-portée. Mais on ignore ici que cette sous-portée est dans le constructeur. Faisons une analyse grossière : promise
nécessite une fonction (supposée être appelée new Promise()
) comme paramètre, mais quel est le timing d'exécution de cette factory
? Notez qu'après que factory
a généré une instance Promise, nous n'appelons jamais activement aucune méthode de cette instance, nous pouvons donc conclure que new Promise()
est exécuté pendant le processus de construction. En d'autres termes, l'instance Promise n'a pas encore été générée pour le moment, et factory
fait référence à promise
. undefined
dans le processus de construction de l'instance Promise, et factory
exécute directement factory
dans le corps de la fonction, et vous pouvez obtenir la valeur de retour de fn
immédiatement, donc cette instance Promise construction Une fois terminé, vous pouvez obtenir la valeur de retour originale. fn
function wxPromisify(fn) { return async function (args) { let originalResult; // ^^^^^^^^^^^^^^^^^^^ const promise = new Promise((resolve, reject) => { originalResult = fn({ // ^^^^^^^^^^^^^^ ...(args || {}), success: res => resolve(res), fail: err => reject(err) }); }); promise.originalResult = originalResult; // ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ return promise; }; }
après new Promise()
, et cette "valeur" est générée dans le processus de promise.originalResult
, puis ajoutez une autre variable locale new Promise()
Faites-la simplement ressortir. originalResult
Faire cela ne produira pas lefunction wxPromisify(fn) { return async function (args) { let promise = new Promise(); // ^^^^^^^^^^^^^^^^^^^^^^^^^^^^ promise = new Promise((resolve, reject) => { // ^^^^^^^^^^ promise.originalResult = fn({ ... }); // ^^^^^^^^^^^^^^^^^^^^^^ }); return promise; }; }Copier après la connexion
susmentionné, mais l'objet Promise obtenu de l'extérieur ne porte pas TypeError
. La raison spécifique est la même que celle de la tentative échouée ci-dessus, je n'entrerai donc pas dans les détails. Juste un rappel : originalResult
Deux objets Promise sont générés ici.
Cette fois, la valeur de retour d'origine est ressortie en utilisant wx.request()
comme exemple. Le but principal de sa valeur de retour est de fournir le. .abort()
méthode de demande d'annulation. Ce scénario d'application est en fait similaire à la façon dont Axios gère « Annulation », vous pouvez donc aussi bien vous référer à la méthode implémentée par Axios via cancelToken
. L'essence de cancelToken
est la deuxième méthode mentionnée ci-dessus : passer l'objet "conteneur" pour faire ressortir les éléments nécessaires. Le faire ressortir via un objet Promise revient essentiellement à le faire ressortir via un objet "conteneur" spécial, donc je n'en dirai pas plus.
Tutoriel recommandé : "Programme WeChat Mini"
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!