Filtering an array with a function that returns a promise

前端 未结 15 1247
遇见更好的自我
遇见更好的自我 2020-11-28 13:22

Given

let arr = [1,2,3];

function filter(num) {
  return new Promise((res, rej) => {
    setTimeout(() => {
      if( num === 3 ) {
        res(num);
         


        
相关标签:
15条回答
  • 2020-11-28 13:36

    Late to the game but since no one else mentioned it, Bluebird supports Promise.map which is my go-to for filters requiring aysnc processing for the condition,

    function filterAsync(arr) {
        return Promise.map(arr, num => {
            if (num === 3) return num;
        })
            .filter(num => num !== undefined)
    }
    
    0 讨论(0)
  • 2020-11-28 13:37

    A variant of @DanRoss's:

    async function filterNums(arr) {
      return await arr.reduce(async (res, val) => {
        res = await res
        if (await filter(val)) {
          res.push(val)
        }
        return res
      }, Promise.resolve([]))
    }
    

    Note that if (as in current case) you don't have to worry about filter() having side effects that need to be serialized, you can also do:

    async function filterNums(arr) {
      return await arr.reduce(async (res, val) => {
        if (await filter(val)) {
          (await res).push(val)
        }
        return res
      }, Promise.resolve([]))
    }
    
    0 讨论(0)
  • 2020-11-28 13:38

    For typescript folk (or es6 just remove type syntax)

    function mapAsync<T, U>(array: T[], callbackfn: (value: T, index: number, array: T[]) => Promise<U>): Promise<U[]> {
      return Promise.all(array.map(callbackfn));
    }
    
    async function filterAsync<T>(array: T[], callbackfn: (value: T, index: number, array: T[]) => Promise<boolean>): Promise<T[]> {
      const filterMap = await mapAsync(array, callbackfn);
      return array.filter((value, index) => filterMap[index]);
    }
    

    es6

    function mapAsync(array, callbackfn) {
      return Promise.all(array.map(callbackfn));
    }
    
    async function filterAsync(array, callbackfn) {
      const filterMap = await mapAsync(array, callbackfn);
      return array.filter((value, index) => filterMap[index]);
    }
    

    es5

    function mapAsync(array, callbackfn) {
      return Promise.all(array.map(callbackfn));
    }
    
    function filterAsync(array, callbackfn) {
      return mapAsync(array, callbackfn).then(filterMap => {
        return array.filter((value, index) => filterMap[index]);
      });
    }
    

    edit: demo

    function mapAsync(array, callbackfn) {
      return Promise.all(array.map(callbackfn));
    }
    
    function filterAsync(array, callbackfn) {
      return mapAsync(array, callbackfn).then(filterMap => {
        return array.filter((value, index) => filterMap[index]);
      });
    }
    
    var arr = [1, 2, 3, 4];
    
    function isThreeAsync(number) {
      return new Promise((res, rej) => {
        setTimeout(() => {
          res(number === 3);
        }, 1);
      });
    }
    
    mapAsync(arr, isThreeAsync).then(result => {
      console.log(result); // [ false, false, true, false ]
    });
    
    filterAsync(arr, isThreeAsync).then(result => {
      console.log(result); // [ 3 ]
    });

    0 讨论(0)
  • 2020-11-28 13:38

    Late to the party, and I know that my answer is similar to other already posted answers, but the function I'm going to share is ready for be dropped into any code and be used. As usual, when you have to do complex operations on arrays, reduce is king:

    const filterAsync = (asyncPred) => arr => 
      arr.reduce(async (acc,item) => {
        const pass = await asyncPred(item);
        if(pass) (await acc).push(item);
        return acc;
      },[]);
    

    It uses modern syntax so make sure your target supports it. To be 100% correct you should use Promise.resolve([]) as the initial value, but JS just doesn't care and this way it is way shorter.

    Then you can use it like this:

    var wait = ms => new Promise(resolve => setTimeout(resolve, ms));
    const isOdd = x => wait(1).then(()=>x%2);
    (filterAsync(isOdd)([1,2,3,4,4])).then(console.log) // => [1,3]
    
    0 讨论(0)
  • 2020-11-28 13:38

    There is a one liner to to do that.

    const filterPromise = (values, fn) => 
        Promise.all(values.map(fn)).then(booleans => values.filter((_, i) => booleans[i]));
    

    Pass the array into values and the function into fn.

    More description on how this one liner works is available here.

    0 讨论(0)
  • 2020-11-28 13:39

    asyncFilter method:

    Array.prototype.asyncFilter = async function(f){
        var array = this;
        var booleans = await Promise.all(array.map(f));
        return array.filter((x,i)=>booleans[i])
    }
    
    0 讨论(0)
提交回复
热议问题