1# ArkTS高性能编程实践 2 3<!--Kit: ArkTS--> 4<!--Subsystem: ArkCompiler--> 5<!--Owner: @zhanyi819--> 6<!--Designer: @qyhuo32--> 7<!--Tester: @kirl75; @zsw_zhushiwei--> 8<!--Adviser: @zhang_yixin13--> 9 10## 概述 11 12本文提供应用性能敏感场景下的高性能编程建议,帮助开发者编写高性能应用。高性能编程实践是在开发过程中总结的一些高性能写法和建议。在实现业务功能时,应同步思考并理解高性能写法的原理,并将其应用于代码逻辑中。关于ArkTS编程规范,请参考[ArkTS编程规范](./arkts-coding-style-guide.md)。 13 14## 声明与表达式 15 16### 使用`const`声明不变的变量 17 18不变的变量推荐使用`const`声明。 19 20``` TypeScript 21const index = 10000; // 该变量在后续过程中未发生改变,建议声明成常量 22``` 23 24 25### `number`类型变量避免整型和浮点型混用 26 27针对`number`类型,运行时在优化时会区分整型和浮点型数据。建议避免在初始化后改变数据类型。 28 29``` TypeScript 30let intNum = 1; 31intNum = 1.1; // 该变量在声明时为整型数据,建议后续不要赋值浮点型数据 32 33let doubleNum = 1.1; 34doubleNum = 1; // 该变量在声明时为浮点型数据,建议后续不要赋值整型数据 35``` 36 37 38### 数值计算避免溢出 39 40常见的可能导致溢出的数值计算包括如下场景,溢出之后,会导致引擎走入慢速的溢出逻辑分支处理,影响后续的性能。 41 42- 针对加法、减法、乘法、指数运算等运算操作,应避免数值大于INT32_MAX(2147483647)或小于INT32_MIN(-2147483648)。 43 44- 针对&(and)、>>>(无符号右移)等运算操作,应避免数值大于INT32_MAX。 45 46 47### 循环中常量提取,减少属性访问次数 48 49如果常量在循环中不会改变,可以将其提取到循环外部,减少访问次数。 50 51``` TypeScript 52class Time { 53 static start: number = 0; 54 static info: number[] = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12]; 55} 56 57function getNum(num: number): number { 58 let total: number = 348; 59 for (let index: number = 0x8000; index > 0x8; index >>= 1) { 60 // 此处会多次对Time的info及start进行查找,并且每次查找出来的值是相同的 61 total += ((Time.info[num - Time.start] & index) !== 0) ? 1 : 0; 62 } 63 return total; 64} 65``` 66 67优化后的代码如下,可以将`Time.info[num - Time.start]`提取为常量,这样可以显著减少属性访问次数,提升性能。 68 69``` TypeScript 70class Time { 71 static start: number = 0; 72 static info: number[] = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12]; 73} 74 75function getNum(num: number): number { 76 let total: number = 348; 77 const info = Time.info[num - Time.start]; // 从循环中提取不变量 78 for (let index: number = 0x8000; index > 0x8; index >>= 1) { 79 if ((info & index) != 0) { 80 total++; 81 } 82 } 83 return total; 84} 85``` 86 87 88## 函数 89 90### 建议使用参数传递函数外的变量 91 92使用闭包会造成额外的开销。在性能敏感场景中,建议使用参数传递函数外的变量替代。 93 94``` TypeScript 95let arr = [0, 1, 2]; 96 97function foo(): number { 98 return arr[0] + arr[1]; 99} 100 101foo(); 102``` 103 104建议使用参数传递函数外部的变量,以替代使用闭包。 105``` TypeScript 106let arr = [0, 1, 2]; 107 108function foo(array: number[]): number { 109 return array[0] + array[1]; 110} 111 112foo(arr); 113``` 114 115 116### 避免使用可选参数 117 118函数的可选参数表示参数可能为`undefined`,在函数内部使用该参数时,需要进行非空值的判断,造成额外的开销。 119 120``` TypeScript 121function add(left?: number, right?: number): number | undefined { 122 if (left != undefined && right != undefined) { 123 return left + right; 124 } 125 return undefined; 126} 127``` 128 129根据业务需求,将函数参数声明为必选参数。考虑使用默认参数。 130``` TypeScript 131function add(left: number = 0, right: number = 0): number { 132 return left + right; 133} 134``` 135 136 137## 数组 138 139### 数值数组推荐使用TypedArray 140 141涉及纯数值计算时,推荐使用TypedArray数据结构。 142 143优化前的代码示例: 144``` TypeScript 145const arr1 = new Array<number>(1, 2, 3); 146const arr2 = new Array<number>(4, 5, 6); 147let res = new Array<number>(3); 148for (let i = 0; i < 3; i++) { 149 res[i] = arr1[i] + arr2[i]; 150} 151``` 152 153优化后的代码示例: 154``` TypeScript 155const typedArray1 = Int8Array.from([1, 2, 3]); 156const typedArray2 = Int8Array.from([4, 5, 6]); 157let res = new Int8Array(3); 158for (let i = 0; i < 3; i++) { 159 res[i] = typedArray1[i] + typedArray2[i]; 160} 161``` 162 163 164### 避免使用稀疏数组 165 166运行时在分配超过1024大小的数组或稀疏数组时,会采用hash表来存储元素。在该模式下,访问数组元素速度较慢。代码开发时应避免数组变成稀疏数组。 167 168``` TypeScript 169// 直接分配100000大小的数组,运行时会处理成用hash表来存储元素 170let count = 100000; 171let result: number[] = new Array(count); 172 173// 创建数组后,直接在9999处赋值,会变成稀疏数组 174let result: number[] = new Array(); 175result[9999] = 0; 176``` 177 178 179### 避免使用联合类型数组 180 181避免使用联合类型数组。避免在数值数组中混合使用整型数据和浮点型数据。 182 183``` TypeScript 184let arrNum: number[] = [1, 1.1, 2]; // 数值数组中混合使用整型数据和浮点型数据 185 186let arrUnion: (number | string)[] = [1, 'hello']; // 联合类型数组 187``` 188 189根据业务需求,将相同类型的数据放在同一数组中。 190``` TypeScript 191let arrInt: number[] = [1, 2, 3]; 192let arrDouble: number[] = [0.1, 0.2, 0.3]; 193let arrString: string[] = ['hello', 'world']; 194``` 195 196 197## 异常 198 199### 避免频繁抛出异常 200 201创建异常时会构造异常的栈帧,造成性能损耗。在性能敏感场景下,如`for`循环语句中,应避免频繁抛出异常。 202 203优化前的代码示例: 204 205``` TypeScript 206function div(a: number, b: number): number { 207 if (a <= 0 || b <= 0) { 208 throw new Error('Invalid numbers.'); 209 } 210 return a / b; 211} 212 213function sum(num: number): number { 214 let sum = 0; 215 try { 216 for (let t = 1; t < 100; t++) { 217 sum += div(t, num); 218 } 219 } catch (e) { 220 console.info(e.message); 221 } 222 return sum; 223} 224``` 225 226优化后的代码示例: 227 228``` TypeScript 229function div(a: number, b: number): number { 230 if (a <= 0 || b <= 0) { 231 return NaN; 232 } 233 return a / b; 234} 235 236function sum(num: number): number { 237 let sum = 0; 238 for (let t = 1; t < 100; t++) { 239 if (num <= 0) { 240 console.info('Invalid numbers.'); 241 } 242 sum += div(t, num); 243 } 244 return sum; 245} 246``` 247