Rumah > hujung hadapan web > tutorial js > Bagaimana untuk meningkatkan prestasi petua code_javascript javascript

Bagaimana untuk meningkatkan prestasi petua code_javascript javascript

WBOY
Lepaskan: 2016-05-16 16:06:16
asal
1135 orang telah melayarinya

Pada asalnya saya akan meringkaskan artikel prestasi kod ini selepas menulis artikel kod kebolehselenggaraan, tetapi ia telah ditangguhkan selama beberapa hari pada asalnya saya memutuskan untuk mengemas kini artikel setiap hari kerana saya berhutang terlalu banyak dan tidak meringkaskannya sebelum ini . , jika anda tidak meringkaskan apa yang telah anda pelajari, anda akan melupakannya dengan cepat akan meninggalkan kesan yang lebih mendalam pada otak anda, terutamanya kebolehselenggaraan kod, prestasi dan sebagainya , Maka anda hebat! Berikut ialah cadangan untuk pemula: ringkaskan lagi perkara yang telah anda pelajari, kerana ini sebenarnya mempelajari pengetahuan baharu! Okey, mari pergi ke topik kami: cara meningkatkan prestasi kod JS.

1. Optimumkan interaksi DOM

DOM berkait rapat dengan halaman kami Apabila penyemak imbas memaparkan halaman, ini bermakna memaparkan operasi dan interaksi DOM yang dihuraikan memakan banyak masa kerana mereka sering perlu memaparkan semula keseluruhan halaman atau sebahagian daripadanya . Tambahan pula, beberapa operasi yang kelihatan halus mungkin mengambil banyak masa untuk dilakukan, kerana DOM perlu memproses banyak maklumat, jadi kami harus mengoptimumkan operasi berkaitan DOM sebanyak mungkin untuk mempercepatkan pemaparan halaman oleh penyemak imbas! Mengapakah sesetengah operasi DOM menjejaskan prestasi halaman? Anda boleh menyemak beberapa artikel yang saya tulis tentang prinsip penyemak imbas:

ok, kami mempunyai beberapa cara utama untuk mengoptimumkan operasi DOM:

1.1 Kemas kini di tapak yang diminimumkan

Apakah kemas kini langsung DOM: Paparan sebahagian halaman yang sudah dipaparkan dalam DOM perlu dikemas kini dengan segera. Walau bagaimanapun, setiap perubahan, sama ada memasukkan satu aksara atau keseluruhan serpihan, disertakan dengan penalti prestasi tertentu, kerana penyemak imbas perlu mengira semula dimensi yang tidak terkira banyaknya untuk dikemas kini (baca ini:). Oleh itu, lebih banyak kemas kini langsung dilakukan, lebih lama masa yang diperlukan untuk kod dilaksanakan, dan sebaliknya, lebih cepat kod dilaksanakan, seperti berikut:

var list = document.getElementById('mylist'),
      item,
      i;
for(i = 0; i < 10; i++){
  item = document.creatElement('li');
  list.appendChild(item);
  item.appendChild(document.creatTextNode('item' + i));
}
Salin selepas log masuk

Kod ini menambahkan 10 item ke senarai mylist Jika item tidak ditambahkan, 2 kemas kini di tapak diperlukan: menambah elemen dan menambah nod teks, jadi operasi ini memerlukan 20 kemas kini di tapak. Setiap kemas kini akan Kehilangan prestasi, dapat dilihat bahawa kod tersebut berjalan agak perlahan.

Penyelesaian adalah dengan menggunakan pemecahan dokumen untuk menukar elemen DOM secara tidak langsung:

var list = document.getElementById('mylist'),
      fragment = document.creatDocumentFragment(),
      item,
      i;
for(i = 0; i < 10; i++){
  item = document.creatElement('li');
  fragment .appendChild(item);
  item.appendChild(document.creatTextNode('item' + i));
}
list.appendChild(fragment);
Salin selepas log masuk

Kod seperti ini hanya perlu dikemas kini secara langsung sekali. Ingat, apabila menghantar serpihan dokumen ke appendChild(), hanya nod anak dalam serpihan dokumen akan ditambahkan pada elemen sasaran, bukan serpihan itu sendiri.

Sekarang, anda harus memahami betapa kesalnya anda terhadap penyemak imbas jika anda menggunakan gelung untuk terus menambah, memadam, menyemak dan mengubah suai nod DOM `(∩_∩)′ .

1.2 Menggunakan innerHTML

Sebagai tambahan kepada kaedah gabungan createElement() dan appendChild() yang digunakan dalam kod di atas untuk mencipta elemen DOM, ia juga boleh dibuat dengan memberikan nilai kepada innerHTML. Untuk perubahan DOM yang kecil, kecekapan kedua-dua kaedah itu sebenarnya hampir sama, tetapi untuk perubahan kepada sejumlah besar nod DOM, yang kedua adalah lebih pantas daripada yang pertama! Kenapa cubit?

Kerana apabila kami menetapkan nilai kepada innerHTML, penghurai HTML akan dibuat di latar belakang, dan kemudian panggilan DOM dalaman digunakan untuk mencipta struktur DOM, bukannya panggilan DOM berdasarkan Javascript, kerana kaedah dalaman disusun agak daripada ditafsirkan, jadi kelajuan pelaksanaan kod adalah lebih pantas!

Tulis semula contoh di atas menggunakan innerHTML:

  var list = document.getElementById('mylist'),
       html = '', //声明一个空字符串
        i;
  for(i = 0; i < 10; i++){
    html += '<li>item' + i + '</li>';
  }
  list.innerHTML = html; // 这里记得innerHTML后面的HTML四个字母都要大写!
Salin selepas log masuk

这种方式同样也只进行了一次的现场更新,并且性能要比上一种方式要好!虽然在字符串的链接上有点性能损失。

1.3 使用事件代理/事件委托

事件处理程序为web应用提供交互能力,因此许多开发人员会不分青红皂白地向页面中添加大量的处理程序,有个问题就是一个页面上的事件处理程序数量将直接关系到页面的整体运行性能。为什么捏?

首先,事件处理程序对应至少一个函数,JS中每个函数都是对象,都会占用内存,内存中的对象越多,性能就越差。

其次,我们必须事先指定所有事件处理程序,这就导致了DOM访问次数增多,会延迟整个页面的交互就绪时间,页面响应用户操作变得相对缓慢。

所以减少事件处理程序同样也可以让我们的页面更牛畅!使用事件委托势在必得啊!

事件委托的原理其实就是事件冒泡,只指定一个事件处理程序就可以管理某一类型操作的所有事件。例如:click事件会一直冒泡到document层次,也就是说我们不必为每个元素添加事件,只需在较高的层次的元素上添加事件处理程序即可,然后利用事件对象(event)的属性或方法去判断当前点击的元素,然后做出相应的响应。这个我就不展开讲了,初学者可以自行查阅事件冒泡知识。

2.作用域很重要

说到作用域啊就很容易想到作用域链(scope chain),我们知道要搜索一个变量,所在的执行环境都要沿着这条作用域向上搜索这个变量,作用域链上有很多的变量,那么我们就得遍历,遍历就需要时间啊,而且你越往上查找所需时间越多,如果我们能减少这个时间,我们代码执行效率不是可以提高了吗?

好聪明啊,ok,我看看有哪些方式可以减少这个时间:

2.1 避免全局查找

这是性能优化的一重点,上面也说了,越往上查找时间越多,也就是说查找全局变量和函数比局部要多!看代码:

function updateUI(){
  var imgs = document.getElementByTagName('img');
  for(var i = 0 ,lng = imgs.length;i < lng;i ++){
    imgss[i].title = document.title + 'image' + i;
  }
  var msg = docuement.getElementById('msg');
  msg.innerHTML = 'update complete.';
}
Salin selepas log masuk

这代码很正常呀!我之前也经常这么做滴。但是我们细心可以发现,这段代码有三处引用了全局变量document,如果我们的页面很多图片,那么在for循环中的document就会被执行上百次,而每次都要需要在作用域链中查找,时间都去哪了,我还没......停!。

我们可以通过在函数中创建一个局部变量保存对document的引用,这样,我们在函数里任何地方引用document都不用跑到全局变量去找了。这样就改进了代码的性能,看代码:

function updateUI(){
  var doc = document; // 将document保存在局部变量doc中
  var imgs = doc.getElementByTagName('img');
  for(var i = 0 ,lng = imgs.length;i < lng;i ++){
    imgss[i].title = doc.title + 'image' + i;
  }
  var msg = doc.getElementById('msg');
  msg.innerHTML = 'update complete.';
}
Salin selepas log masuk

所以啊,我们在开发中,如果在函数中会经常用到全局变量,把它保存在局部变量中!

2.2 避免使用with语句

用with语句延长了作用域,查找变量同样费时间,这个我们一般不会用到,所以不展开了。解决方法还是和上面的例子一样,将全局变量保存在局部变量中!

3.优化循环

循环在编程中可谓家常便饭,在js中也随处可见,循环体会反复地执行同一段代码,执行时间一直累加,所以能够对循环体的代码进行优化也可以大大减少执行时间!如何优化?四种方式。

3.1 减值迭代

我们写迭代器(循环条件)的时候一般都这样(var i = 0;i < 10;i ++),从0开始,增加到某个特定值。然而在很多情况下,如果在循环中使用减值迭代器效率更高。我测试了下,如果循环体不复杂的话,两者差不多!

//增值迭代 --效率较低
for(var i = 0;i < items.length;i++){
  doSomething(items[i]); 
}
//减值迭代 --效率较高
for(var i = items.length - 1;i >= 0;i--){
  doSomething(items[i]); 
}
Salin selepas log masuk

3.2 简化终止条件

由于每次循环都会计算终止条件,所以必须保证它的执行尽可能地块。这里主要是避免其他DOM元素及其属性的的查找。

 //看终止条件,每次循环都需要查询items及其length属性
for(var i = 0;i < items.length;i++){
  doSomething(items[i]); 
}

//将items.length的值保存在局部变量lng中。
for(var i = 0,lng = items.length;i < lng;i++){
  doSomething(items[i]); 
}
Salin selepas log masuk

3.3 简化循环体

原因和上面以上的,所以在循环体内避免大量的密集的操作。

这其实和上面讲的:1.1 最小化现场更新 。是一样的优化方式。可以倒回去看看。

4.基本的算法优化

在计算机中,算法的复杂度用O表示。下面是javascript中几种常见的算法类型:

O(1) :常数,不管有多少值,执行的时间都是恒定的,比如简单值和存储在变量中的值。
O(log n):对数,总的执行时间和数量有关,但不一定要获取每一个值,如:二分法查找
O(n) :线性,总执行时间和数量直接相关,如:遍历
O(n*n) :平方,总执行时间和数量有关,每个值至少获取N次,如:插入排序
ok,有了上面的知识,我们就可以对javascript进行一些算法上的优化了。看代码:

var value = 5;
var sum = value + 10;
alert(sum);
Salin selepas log masuk

这段代码进行了4次常量值的查找:数字5,变量value,数字10,变量sum,这段代码的算法复杂度就是O(1)。又如:

var value = [10,5];
var sum = value[0] + value[1];
alert(sum);
Salin selepas log masuk

在javascript中访问数组元素也是一个O(1)操作,和简单的变量查找效率一样。再看:

var value = {one:10,two:10};
var sum = value.one + value.two;
alert(sum);
Salin selepas log masuk

要表达的是访问对象上的属性要比访问数组和变量的效率低。因为这是一个O(n)操作。你需要在对象的原型链中查找该属性,所花时间较多。

好了,看完这个是不是感觉眼前一片光明啊。其实我们前面所讲的要把经常用到的全局属性保存在一个局部变量中就是根据这个原理了,访问全局属性是一个O(n)的操作,而访问变量是一个O(1)的操作,大声告诉我,挖掘机哪家强啊!

5.最小化语句数

前面讲的优化差不多都是和精简优化语句有关的,是的,我觉得代码的质量和数量就是性能的评判标准。前面讲了一些代码质量相关的优化,这里就讲讲代码数量的优化。

5.1 精简变量声明

//用了5条语句声明5个变量
var count = 5;
var color = 'red';
var values = [1,2,3];
var now = new Date();

//用了1条语句声明5个变量,注意每个变量用逗号隔开
var count = 5,
  color = 'red',
  values = [1,2,3],
  now = new Date();

Salin selepas log masuk

5.2 使用数组和对象字面量

// 创建两个对象 ----不好的方式
//one 四条语句
var values = new Array();
values[0] = 123;
values[1] = 456;
values[2] = 789;
//two 四条语句
var person = new Object();
person.name = 'jozo';
person.age = 21;
person.sayName = function(){
  alert(this.name);
};
// 创建两个对象 ----推荐的方式
//one 1条语句
var values = [123,456,789]
//two 1条语句
var person = {
  name : 'jozo',
  age : 21,
  sayName : function(){
  alert(this.name);
};
Salin selepas log masuk

6.其他

写累了,如有不正确的地方请指正哦,还有一些其他的优化,下次文章继续!

Label berkaitan:
sumber:php.cn
Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn
Tutorial Popular
Lagi>
Muat turun terkini
Lagi>
kesan web
Kod sumber laman web
Bahan laman web
Templat hujung hadapan