在Arkts中用ts实现更适合原生宝宝的Promise.WithResolver和LazyInit/Synchronize

开发过程中被各种async弄到破防,因此手撸了几个工具类保证代码稳定性,话不多说直接上代码,文件后缀需要改成.ts比如PromiseUtil.ts

export function Promise_withResolver<T>() {
  let resolve: (value: T) => void = undefined
  let reject: (err: Error) => void = undefined
  const promise = new Promise<T>((res, rej) => {
    resolve = res
    reject = rej
  })
  return {
    promise: promise,
    resolve: resolve,
    reject: reject
  }
}

该代码复刻js最新Promise的API,将resolver和reject从Promise中拆除,使数据的观察和触发分离,上场景
扫码组件初始化需要XComponent组件onLoad及权限判断成功后才开始扫码,扫码尺寸需要匹配XComponent宽高

private areaCallback = Promise_withResolver<Area>()
private loadCallback = Promise_withResolver<boolean>()
private permissionCallback = Promise_withResolver<boolean>()
private checkInitCallback = Promise.all([this.areaCallback.promise, this.loadCallback.promise, this.permissionCallback.promise])
build(){
          XComponent({
            id: 'componentId',
            type: 'surface',
            controller: this.mXComponentController
          })
            .onAreaChange((_, newValue) => {
              this.areaCallback.resolve(newValue)
            })
            .onLoad(() => {
              this.loadCallback.resolve(true)
            })
            .size(FullScreen)
}

aboutToAppear(): void {
    this.checkInitCallback.then((value) => {
      this.surfaceId = this.mXComponentController.getXComponentSurfaceId();
      this.cameraHeight = value[0].height as number
      this.cameraWidth = value[0].width as number
      if (value[2]) {
        this.vm.changeState(SCAN)
      } else {
        this.vm.changeState(NO_PERMISSION)
      }
    })
}

在进行页面的初始化时有很多场景触发接口去请求默认数据,相同的请求经常重复发送,所以想起了Synchronize原理,ts中实现如下

export function LazyInit<T>(block: () => Promise<T>) {
  const waitQueue: Array<(value: T) => void> = []
  let init: boolean | undefined = undefined
  let value: T
  return () => {
    if (init === undefined) {
      return new Promise<T>((res, rej) => {
        init = false
        block().then((value) => {
          init = true
          res(value)
          while (waitQueue.length != 0) {
            waitQueue.pop()?.(value)
          }
        })
      })
    } else if (init) {
      return Promise.resolve(value)
    } else {
      const resolver = Promise_withResolver<T>()
      waitQueue.push(resolver.resolve)
      return resolver.promise
    }
  }
}

直接上场景

  waitAndReturn() : Promise<string>{
    return new Promise((res) => {
      setTimeout(res, 5000, "123")
    })
  }

  private data = LazyInit(async () => {
    return await this.waitAndReturn()
  })
    Logger.debug("gscgsc1", `start`)
    this.data().then((value : string) => {
      Logger.debug("gscgsc1", `${value}`)
    })
    this.data().then((value : string) => {
      Logger.debug("gscgsc2", `${value}`)
    })
    this.data().then((value : string) => {
      Logger.debug("gscgsc3", `${value}`)
    })
    this.data().then((value : string) => {
      Logger.debug("gscgsc4", `${value}`)
    })
    this.data().then((value : string) => {
      Logger.debug("gscgsc5", `${value}`)
    })

运行截图如下


上面的LazyInit返回的是Promise,有时我们需要在全局变量初始化时进行懒加载声明,类似kotlin的by lazy,将上面的功能简化一下如下

export function byLazy<T>(block: () => T): () => T {
  let value: T | undefined
  return () => {
    if (!value) {
      value = block()
    }
    return value
  }
}

上才艺

  private n = byLazy(() => {
    Logger.debug("gscgsc", `math random`)
    return Math.random()
  })
    Logger.debug("gscgsc", 'start')
    Logger.debug("gscgsc", `${this.n()}`)
    Logger.debug("gscgsc", `${this.n()}`)
    Logger.debug("gscgsc", `${this.n()}`)
    Logger.debug("gscgsc", 'end')

日志如下


破费科特
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,542评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,596评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,021评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,682评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,792评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,985评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,107评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,845评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,299评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,612评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,747评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,441评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,072评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,828评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,069评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,545评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,658评论 2 350

推荐阅读更多精彩内容