如何在JavaScript中使用具有Promise字符串标签

huangapple go评论77阅读模式
英文:

How to use it with promise string label in javascript

问题

  1. 我正在尝试使用如下的 Promise
  2. ```typescript
  3. let promiseArray: [string, Promise<unknown>][] = [];
  4. for(const a of array)
  5. {
  6. const promise = new Promise(() => {this.repository.getRepository<a.entity>.find()});
  7. promiseArray.push([a.entityName, promise]);
  8. }

上述代码的结果是:

  1. result : [
  2. ['EntityName', [{},{},{}]],
  3. ['EntityName2', [{},{},{}]],
  4. ....
  5. ]

但我不知道如何将 promiseArray 应用到 Promise.all

  1. await Promise.all(promiseArray)
  2. .then((res) => {
  3. console.log(res);
  4. })
  5. .catch((e) => console.error(e));

我尝试了上述方法,但没有起作用,也尝试了 promiseArray.map((pm)=>pm[1]),但我无法映射出 pm[0] 的值。

英文:

I am trying to use a promise as below.

  1. let promiseArray: [string, Promise&lt;unknown&gt;][] = [];
  2. for(const a of array)
  3. {
  4. const promise = new Promise(() =&gt; {this.repository.getRepository&lt;a.entity&gt;.find()});
  5. promiseArray.push([a.entityName, promise]);
  6. }

And the result of the code above is:

  1. result : [
  2. [&#39;EntityName&#39;, [{},{},{}]],
  3. [&#39;EntityName2&#39;, [{},{},{}]],
  4. ....
  5. ]

But I don't know how can I apply promiseArray to promise.all.

  1. await Promise.all(promiseArray)
  2. .then((res) =&gt; {
  3. console.log(res);
  4. })
  5. .catch((e) =&gt; console.error(e));

I tried the above, but it didn't work, and I tried promiseArray.map((pm)=&gt;pm[1]), but I couldn't map the value of pm[0].

答案1

得分: 1

看起来你在填充数组时出了一些问题。根据你的代码,似乎你在使用TypeORM,并且.find()是异步的(返回一个Promise)。所以,你不需要再包装它成另一个Promise。尝试这样填充数组:

  1. let promiseArray: [string, Promise<unknown>][] = [];
  2. for(const a of array)
  3. {
  4. const promise = this.repository.getRepository<a.entity>.find();
  5. promiseArray.push([a.entityName, promise]);
  6. }

然后,你只需要使用.map()来获取数组中的第二个值,即.find()返回的Promise:

  1. await Promise.all(promiseArray.map(el => el[1]));

如果你需要将.entityName与Promise的结果一起传递,那么可以将每个元素映射到它的.find() Promise,然后在.then()中将实体名称添加到结果中。这样为array中的每个元素创建了一个单独的Promise,可以简单地传递给Promise.all()

  1. const promiseArray: [Promise<[string, unknown]>][] = array.map(el => this.repository.getRepository<a.entity>.find().then(res => [el.entityName, res]));
  2. const results = await Promise.all(promiseArray);
英文:

It looks like you're populating your array wrong. From your code, it looks like you're using TypeORM, and that .find() is async (returns a promise). So, you don't need to wrap it in another promise. Try filling the array like this:

  1. let promiseArray: [string, Promise&lt;unknown&gt;][] = [];
  2. for(const a of array)
  3. {
  4. const promise = this.repository.getRepository&lt;a.entity&gt;.find();
  5. promiseArray.push([a.entityName, promise]);
  6. }

Then, all you need to do is a .map() to get the second value in the array, which is the promise returned by .find():

  1. await Promise.all(promiseArray.map(el =&gt; el[1]));

If you need to pass the .entityName along with the promise's result, then map each element to its .find() promise, then add the entity name to the result in a .then(). This creates a single promise per element in array and can be simply passed to Promise.all():

  1. const promiseArray: [Promise&lt;[string, unknown]&gt;][] = array.map(el =&gt; this.repository.getRepository&lt;a.entity&gt;.find().then(res =&gt; [el.entityName, res]));
  2. const results = await Promise.all(promiseArray);

答案2

得分: 1

Promise.all 接受一个 Promise 数组,但你有一个数组类型为 -

  1. Array&lt;[String, Promise]&gt;

你需要以不同方式构建 Promise 数组 -

  1. const promiseArray: Promise&lt;Array&lt;[string, unknown]&gt;&gt; = []
  2. for (const a of array) {
  3. const promise = new Promise((resolve, reject) =&gt; {
  4. // ...
  5. })
  6. .then(result =&gt; [a.entityName, result]) // ✅
  7. promiseArray.push(promise)
  8. }

现在你可以使用 Promise.all -

  1. Promise.all(promisesArray).then(...)

结果为 -

  1. [
  2. [&#39;EntityName&#39;, &lt;promise result&gt;],
  3. [&#39;EntityName2&#39;, &lt;promise result&gt;],
  4. ...
  5. ]

这里有一个可运行的示例 -

<!-- begin snippet: js hide: false console: true babel: false -->

<!-- language: lang-js -->

  1. const randPromise = () =&gt;
  2. new Promise(r =&gt;
  3. setTimeout(r, 1000, Math.random() * 100 | 0)
  4. )
  5. const labels = [&quot;a&quot;, &quot;b&quot;, &quot;c&quot;, &quot;d&quot;]
  6. const promises = []
  7. for (const label of labels)
  8. promises.push(randPromise().then(value =&gt;
  9. [label, value]
  10. ))
  11. Promise.all(promises)
  12. .then(JSON.stringify)
  13. .then(console.log)
  14. .catch(console.error)
  15. // [[&quot;a&quot;,53],[&quot;b&quot;,25],[&quot;c&quot;,22],[&quot;d&quot;,53]]

<!-- end snippet -->

不使用 for..of 循环,许多人发现使用数组的 map 函数更容易 -

<!-- begin snippet: js hide: false console: true babel: false -->

<!-- language: lang-js -->

  1. const randPromise = () =&gt;
  2. new Promise(r =&gt;
  3. setTimeout(r, 1000, Math.random() * 100 | 0)
  4. )
  5. const labels = [&quot;a&quot;, &quot;b&quot;, &quot;c&quot;, &quot;d&quot;]
  6. Promise all(labels.map(label =&gt;
  7. randPromise().then(value =&gt; [label, value])
  8. ))
  9. .then(JSON.stringify)
  10. .then(console.log)
  11. .catch(console.error)
  12. // [[&quot;a&quot;,90],[&quot;b&quot;,20],[&quot;c&quot;,76],[&quot;d&quot;,60]]

<!-- end snippet -->

每个 then 表达式都有一个等效的 async..await 表示。这种方法使用更少的嵌套 lambda,因此更容易理解 -

<!-- begin snippet: js hide: false console: true babel: false -->

<!-- language: lang-js -->

  1. const randPromise = () =&gt;
  2. new Promise(r =&gt;
  3. setTimeout(r, 1000, Math.random() * 100 | 0)
  4. )
  5. const labels = [&quot;a&quot;, &quot;b&quot;, &quot;c&quot;, &quot;d&quot;]
  6. Promise all(labels.map(async label =&gt;
  7. [label, await randPromise()]
  8. ))
  9. .then(JSON.stringify)
  10. .then(console.log)
  11. .catch(console.error)
  12. // [[&quot;a&quot;,90],[&quot;b&quot;,20],[&quot;c&quot;,76],[&quot;d&quot;,60]]

<!-- end snippet -->

英文:

Promise.all accepts an array of promises, but you have an array of -

  1. Array&lt;[String, Promise]&gt;

You will need to construct the array of promises differently -

  1. const promiseArray: Promise&lt;Array&lt;[string, unknown]&gt;&gt; = []
  2. for (const a of array) {
  3. const promise = new Promise((resolve, reject) =&gt; {
  4. // ...
  5. })
  6. .then(result =&gt; [a.entityName, result]) // ✅
  7. promiseArray.push(promise)
  8. }

Now you can use Promise.all -

  1. Promise.all(promisesArray).then(...)

Results in -

  1. [
  2. [&#39;EntityName&#39;, &lt;promise result&gt;],
  3. [&#39;EntityName2&#39;, &lt;promise result&gt;],
  4. ...
  5. ]

Here's a functioning example -

<!-- begin snippet: js hide: false console: true babel: false -->

<!-- language: lang-js -->

  1. const randPromise = () =&gt;
  2. new Promise(r =&gt;
  3. setTimeout(r, 1000, Math.random() * 100 | 0)
  4. )
  5. const labels = [&quot;a&quot;, &quot;b&quot;, &quot;c&quot;, &quot;d&quot;]
  6. const promises = []
  7. for (const label of labels)
  8. promises.push(randPromise().then(value =&gt;
  9. [label, value]
  10. ))
  11. Promise.all(promises)
  12. .then(JSON.stringify)
  13. .then(console.log)
  14. .catch(console.error)
  15. // [[&quot;a&quot;,53],[&quot;b&quot;,25],[&quot;c&quot;,22],[&quot;d&quot;,53]]

<!-- end snippet -->

Instead of using the for..of loop, many find it easier to use an array's map function -

<!-- begin snippet: js hide: false console: true babel: false -->

<!-- language: lang-js -->

  1. const randPromise = () =&gt;
  2. new Promise(r =&gt;
  3. setTimeout(r, 1000, Math.random() * 100 | 0)
  4. )
  5. const labels = [&quot;a&quot;, &quot;b&quot;, &quot;c&quot;, &quot;d&quot;]
  6. Promise.all(labels.map(label =&gt;
  7. randPromise().then(value =&gt; [label, value])
  8. ))
  9. .then(JSON.stringify)
  10. .then(console.log)
  11. .catch(console.error)
  12. // [[&quot;a&quot;,90],[&quot;b&quot;,20],[&quot;c&quot;,76],[&quot;d&quot;,60]]

<!-- end snippet -->

Every then expression has an equivalent async..await representation. This approach uses fewer nested lambdas and therefore a bit easier to digest -

<!-- begin snippet: js hide: false console: true babel: false -->

<!-- language: lang-js -->

  1. const randPromise = () =&gt;
  2. new Promise(r =&gt;
  3. setTimeout(r, 1000, Math.random() * 100 | 0)
  4. )
  5. const labels = [&quot;a&quot;, &quot;b&quot;, &quot;c&quot;, &quot;d&quot;]
  6. Promise.all(labels.map(async label =&gt;
  7. [label, await randPromise()]
  8. ))
  9. .then(JSON.stringify)
  10. .then(console.log)
  11. .catch(console.error)
  12. // [[&quot;a&quot;,90],[&quot;b&quot;,20],[&quot;c&quot;,76],[&quot;d&quot;,60]]

<!-- end snippet -->

huangapple
  • 本文由 发表于 2023年2月24日 11:18:29
  • 转载请务必保留本文链接:https://go.coder-hub.com/75552319.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定