ArkTS 高性能编程实践指南

ArkTS 高性能编程实践指南

一、引言

在应用开发中,性能是至关重要的因素。本文聚焦于 ArkTS 高性能编程实践,提供一系列在应用性能敏感场景下的编程建议,旨在帮助开发者打造高效能的应用。通过遵循这些建议,开发者能够在实现业务功能的同时,优化代码性能,提升用户体验。

二、声明与表达式

(一)使用 const 声明不变的变量

对于在后续过程中不会发生改变的变量,推荐使用 const 进行声明。例如:

const index = 10000; // 该变量在后续过程中未发生改变,建议声明成常量

这样做有助于编译器进行优化,提高代码执行效率。

(二)number 类型变量避免整型和浮点型混用

在 ArkTS 中,number 类型在运行时优化会区分整型和浮点型数据。因此,建议在变量声明后不要改变其数据类型。例如:

let intNum = 1;
intNum = 1.1;  // 该变量在声明时为整型数据,建议后续不要赋值浮点型数据

let doubleNum = 1.1;
doubleNum = 1;  // 该变量在声明时为浮点型数据,建议后续不要赋值整型数据

保持数据类型的一致性可以避免不必要的类型转换开销,提升性能。

(三)数值计算避免溢出

某些数值计算可能导致溢出,进而使引擎进入慢速的溢出逻辑分支处理,影响性能。在进行加法、减法、乘法、指数运算等操作时,应确保数值在 INT32_MAXINT32_MIN 范围内;对于 &(and)、>>>(无符号右移)等运算操作,数值应小于等于 INT32_MAX。开发者在编写代码时需要特别注意这些情况,避免因溢出而引发性能问题。

(四)循环中常量提取,减少属性访问次数

在循环中,如果存在不随循环迭代而改变的常量访问操作,可以将其提取到循环外部。例如:

class Time {
  static start: number = 0;
  static info: number[] = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12];
}

function getNum(num: number ): number {
  let total: number = 348;
  for (let index: number = 0x8000; index > 0x8; index >>= 1) {
    // 此处会多次对 Time 的 info 及 start 进行查找,并且每次查找出来的值是相同的
    total += ((Time.info[num - Time.start] & index)!== 0)? 1 : 0;
  }
  return total;
}

// 优化后代码
class Time {
  static start: number = 0;
  static info: number[] = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12];
}

function getNum(num: number ): number {
  let total: number = 348;
  const info = Time.info[num - Time.start];  // 从循环中提取不变量
  for (let index: number = 0x8000; index > 0x8; index >>= 1) {
    if ((info & index)!= 0) {
      total++;
    }
  }
  return total;
}

通过这种优化,可以大幅减少属性访问次数,显著提升性能。

三、函数

(一)建议使用参数传递函数外的变量

使用闭包会带来额外的闭包创建和访问开销。在性能敏感场景中,推荐使用参数传递函数外的变量来替代闭包。例如:

let arr = [0, 1, 2];
function foo(): number {
  return arr[0] + arr[1];
}
foo();

// 优化后
let arr = [0, 1, 2];
function foo(array: number[]): number {
  return array[0] + array[1];
}
foo(arr);

这种方式可以减少不必要的开销,提高函数执行效率。

(二)避免使用可选参数

函数的可选参数在函数内部使用时需要进行非空值判断,这会造成额外的开销。例如:

function add(left?: number, right?: number): number | undefined {
  if (left!= undefined && right!= undefined) {
    return left + right;
  }
  return undefined;
}

// 优化后,根据业务需要将函数参数声明为必须参数,并可使用默认参数
function add(left: number = 0, right: number = 0): number {
  return left + right;
}

通过避免使用可选参数,可以简化函数逻辑,提升性能。

四、数组

(一)数值数组推荐使用 TypedArray

在纯数值计算的场景中,推荐使用 TypedArray 数据结构。例如:

// 优化前
const arr1 = new Array<number>([1, 2, 3]);
const arr2 = new Array<number>([4, 5, 6]);
let res = new Array<number>(3);
for (let i = 0; i < 3; i++) {
  res[i] = arr1[i] + arr2[i];
}

// 优化后
const typedArray1 = new Int8Array([1, 2, 3]);
const typedArray2 = new Int8Array([4, 5, 6]);
let res = new Int8Array(3);
for (let i = 0; i < 3; i++) {
  res[i] = typedArray1[i] + typedArray2[i];
}

TypedArray 能够提供更高效的数值处理能力,提升计算性能。

(二)避免使用稀疏数组

当运行时分配超过 1024 大小的数组或处理稀疏数组时,会采用 hash 表的方式存储元素,这种方式相比于偏移访问数组元素速度较慢。因此,在开发过程中应尽量避免数组变为稀疏数组。例如:

// 直接分配 100000 大小的数组,运行时会处理成用 hash 表来存储元素
let count = 100000;
let result: number[] = new Array(count);

// 创建数组后,直接在 9999 处赋值,会变成稀疏数组
let result: number[] = new Array();
result[9999] = 0;

注意数组的创建和赋值方式,防止出现稀疏数组情况。

(三)避免使用联合类型数组

应避免在数值数组中混合使用整型数据和浮点型数据,以及使用联合类型数组。例如:

let arrNum: number[] = [1, 1.1, 2];  // 数值数组中混合使用整型数据和浮点型数据
let arrUnion: (number | string)[] = [1, 'hello'];  // 联合类型数组

// 优化后,将相同类型的数据放置在同一数组中
let arrInt: number[] = [1, 2, 3];
let arrDouble: number[] = [0.1, 0.2, 0.3];
let arrString: string[] = ['hello', 'world'];

保持数组元素类型的一致性有助于提高数组操作的性能。

五、异常

避免频繁抛出异常

创建异常时会构造异常的栈帧,这会造成性能损耗。在性能敏感场景下,如 for 循环语句中,应避免频繁抛出异常。例如:

// 优化前
function div(a: number, b: number): number {
  if (a <= 0 || b <= 0) {
    throw new Error('Invalid numbers.')
  }
  return a / b
}

function sum(num: number ): number {
  let sum = 0
  try {
    for (let t = 1; t < 100; t++) {
      sum += div(t, num)
    }
  } catch (e) {
    console.log(e.message)
  }
  return sum
}

// 优化后
function div(a: number, b: number): number {
  if (a <= 0 || b <= 0) {
    return NaN
  }
  return a / b
}

function sum(num: number ): number {
  let sum = 0
  for (let t = 1; t < 100; t++) {
    if (t <= 0 || num <= 0) {
      console.log('Invalid numbers.')
    }
    sum += div(t, num)
  }
  return sum
}

通过合理处理错误情况,减少异常抛出次数,可以提升代码性能。

遵循以上 ArkTS 高性能编程实践建议,开发者能够在开发过程中有效优化代码性能,提升应用的整体质量和用户体验。在实际项目中,应根据具体业务场景和性能需求,灵活运用这些技巧,打造高效稳定的应用程序。

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

推荐阅读更多精彩内容