嵌套的异步方法怎么返回一个值? - V2EX
V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
推荐关注
Meteor
JSLint - a Javascript code quality tool
jsFiddle
D3.js
WebStorm
推荐书目
Javascript 权威指南第 5 版
Closure: The Definitive Guide
lbfeng
V2EX    Javascript

嵌套的异步方法怎么返回一个值?

  •  
  •   lbfeng 2021-02-09 11:09:40 +08:00 2263 次点击
    这是一个创建于 1705 天前的主题,其中的信息可能已经有所发展或是发生改变。

    puppeteer 官方文档里的一段代码想改一下 https://pptr.dev/#?product=Puppeteer&version=v7.0.1&show=api-pagewaitforselectorselector-options

    const puppeteer = require('puppeteer'); (async () => { const browser = await puppeteer.launch(); const page = await browser.newPage(); let currentURL; page .waitForSelector('img') .then(() => console.log('First URL with image: ' + currentURL)); for (currentURL of ['https://example.com', 'https://google.com', 'https://bbc.com']) { await page.goto(currentURL); } await browser.close(); })(); 

    简单的 print 第一个有 img 的网站。想改成

    const puppeteer = require('puppeteer'); const get_available_item = async (page) => { let currentURL; const urls = [ 'https://example.com', 'https://example.com', 'https://baidu.com' ] page.waitForSelector('img').then( () => { console.log('First URL with image: ' + currentURL) } ); for (currentURL of urls) { await page.goto(currentURL, {waitUntil: 'load'}); } }; const main = async () => { const browser = await puppeteer.launch({ headless: false, product: 'firefox', defaultViewport: { width: 1366, height: 768 } }); const page = await browser.newPage(); await get_available_item(page) await browser.close(); } main() 

    想让get_available_item返回currentURL,试了一种用 callback 返回的方法

    const puppeteer = require('puppeteer'); const get_available_item = async (page, callback) => { let currentURL; const urls = [ 'https://example.com', 'https://example.com', 'https://baidu.com' ] page.waitForSelector('img').then( () => { callback(currentURL) } ); for (currentURL of urls) { await page.goto(currentURL, {waitUntil: 'load'}); } }; const main = async () => { let url const browser = await puppeteer.launch({ headless: false, product: 'firefox', defaultViewport: { width: 1366, height: 768 } }); const page = await browser.newPage(); await get_available_item(page, (res) => { url = res }) console.log(`result is ${url}`) await browser.close(); } main() 

    有比 callback 更简洁的方法么?

    9 条回复    2021-02-10 03:15:21 +08:00
    cyrbuzz
        1
    cyrbuzz  
       2021-02-09 11:29:39 +08:00
    试试用订阅发布模型。

    window.addEventListener('getWithImageUrl', (_url) => url = _url); window.dispatchEvent(new Event('getWithImage'))
    azcvcza
        2
    azcvcza  
       2021-02-09 11:32:04 +08:00
    能在你想返回的地方加点注释吗。。
    musi
        3
    musi  
       2021-02-09 11:45:49 +08:00
    既然已经用 async 了,把 callback 改成 promise 后用 await
    lzdyes
        4
    lzdyes  
       2021-02-09 12:10:51 +08:00
    你都用 async await 了还不会包装个 Promise 吗,返回值 <string>Promise
    lzdyes
        5
    lzdyes  
       2021-02-09 12:27:33 +08:00
    const get_available_item = async (page: string):Promise<string> => {
    return new Promise<string>( ( reject) =>{
    // ...
    reject(currentURL)
    })
    }

    const currentURL = await get_available_item(page)
    oxromantic
        6
    oxromantic  
       2021-02-09 12:47:20 +08:00
    @lzdyes 为什么你第一个参数起名叫 reject ????
    azcvcza
        7
    azcvcza  
       2021-02-09 13:48:12 +08:00
    @oxromantic 实际上都是按参数位置调的,叫 reject 一样按 resolve 用,就是迷惑了点
    lzdyes
        8
    lzdyes  
       2021-02-09 14:12:27 +08:00
    啊哈,脑抽写错了 不过不影响 lz 自己改成 resolve 吧
    lbfeng
        9
    lbfeng  
    OP
       2021-02-10 03:15:21 +08:00
    @cyrbuzz 这个真没想到。
    @lzdyes 你这个可能不 work 。new Promise 里的 function 一般情况下是 sync function,但因为有 await 必须用 async 。

    ```Javascript
    const operation1 = Promise.resolve(5)
    const operation2 = Promise.resolve(15)
    const publishResult = () => Promise.reject(`Can't publish`)

    let p = new Promise((resolve, reject) => {
    (async () => {
    try {
    const op1 = await operation1;
    const op2 = await operation2;

    if (op2 == null) {
    throw new Error('Validation error');
    }

    const res = op1 + op2;
    const result = await publishResult(res);
    resolve(result)
    } catch (err) {
    reject(err)
    }
    })()
    });

    (async () => {
    await p;
    })().catch(e => console.log("Caught: " + e));
    ```

    这个例子 work,但有 anti pattern
    关于     帮助文档     自助推广系统     博客     API     FAQ     Solana     5989 人在线   最高记录 6679       Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 25ms UTC 06:25 PVG 14:25 LAX 23:25 JFK 02:25
    Do have faith in what you're doing.
    ubao snddm index pchome yahoo rakuten mypaper meadowduck bidyahoo youbao zxmzxm asda bnvcg cvbfg dfscv mmhjk xxddc yybgb zznbn ccubao uaitu acv GXCV ET GDG YH FG BCVB FJFH CBRE CBC GDG ET54 WRWR RWER WREW WRWER RWER SDG EW SF DSFSF fbbs ubao fhd dfg ewr dg df ewwr ewwr et ruyut utut dfg fgd gdfgt etg dfgt dfgd ert4 gd fgg wr 235 wer3 we vsdf sdf gdf ert xcv sdf rwer hfd dfg cvb rwf afb dfh jgh bmn lgh rty gfds cxv xcv xcs vdas fdf fgd cv sdf tert sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf shasha9178 shasha9178 shasha9178 shasha9178 shasha9178 liflif2 liflif2 liflif2 liflif2 liflif2 liblib3 liblib3 liblib3 liblib3 liblib3 zhazha444 zhazha444 zhazha444 zhazha444 zhazha444 dende5 dende denden denden2 denden21 fenfen9 fenf619 fen619 fenfe9 fe619 sdf sdf sdf sdf sdf zhazh90 zhazh0 zhaa50 zha90 zh590 zho zhoz zhozh zhozho zhozho2 lislis lls95 lili95 lils5 liss9 sdf0ty987 sdft876 sdft9876 sdf09876 sd0t9876 sdf0ty98 sdf0976 sdf0ty986 sdf0ty96 sdf0t76 sdf0876 df0ty98 sf0t876 sd0ty76 sdy76 sdf76 sdf0t76 sdf0ty9 sdf0ty98 sdf0ty987 sdf0ty98 sdf6676 sdf876 sd876 sd876 sdf6 sdf6 sdf9876 sdf0t sdf06 sdf0ty9776 sdf0ty9776 sdf0ty76 sdf8876 sdf0t sd6 sdf06 s688876 sd688 sdf86