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