javascript - 使用 ajax 加载字典文件,不要让 iPhone Mobile Safari 崩溃

标签 javascript jquery iphone mobile-safari

我有一个 Web 应用程序,我在其中(通过 ajax)将字典文件 (1MB) 加载到 javascript 数组中。我找到了 Mobile Safari crashes after 10 seconds 的原因.但现在我想知道如何解决这个问题?

在上面的链接中,答案建议使用 setInterval,但这意味着我必须将字典文件分成多个部分,然后一个一个地加载它们。这当然可以做到,但考虑到互联网速度,我必须制作很多 block ,并且太多的请求将永远加载页面(如果我制作的 block 太大,一些移动用户可能会发生这种情况将无法在给定的 10 秒内下载该 block )。

所以,我的问题是:有没有人遇到过这种问题,您是如何解决的?对正确方向的普遍插入表示赞赏。

编辑: 这是我用来加载字典的 js 代码:

var dict = new Trie();

$.ajax({
    url: 'data/dictionary_342k_uppercase.txt',
    async: true,
    success: function (data) {
        var words = data.split('\n');
        for (var i = words.length - 1; i >= 0; i--) {
            dict.insert(words[i]);
        }           
    },
    error: function(){
        $('#loading-message').text("Problem s rječnikom");
    }
});

特里.js:

function Trie () {
  var ALPHABET_SIZE = 30;
  var ASCII_OFFSET = 'A'.charCodeAt();

  this.children = null;
  this.isEndOfWord = false;

  this.contains = function (str) {
    var curNode = this;

    for (var i = 0; i < str.length; i++) {
      var idx = str.charCodeAt(i) - ASCII_OFFSET;
      if (curNode.children && curNode.children[idx]) {
        curNode = curNode.children[idx];
      } else {
        return false;
      }
    }

    return curNode.isEndOfWord;
  }

  this.has = function (ch) {
    if (this.children) {
      return this.children[ch.charCodeAt() - ASCII_OFFSET] != undefined;
    }
    return false;
  }

  this.next = function (ch) {
    if (this.children) {
      return this.children[ch.charCodeAt() - ASCII_OFFSET];
    }
    return undefined;
  }

  this.insert = function (str) {
    var curNode = this;

    for (var i = 0; i < str.length; i++) {
      var idx = str.charCodeAt(i) - ASCII_OFFSET;

      if (curNode.children == null) {
        curNode.children = new Array(ALPHABET_SIZE);
        curNode = curNode.children[idx] = new Trie();
      } else if (curNode.children[idx]) {
        curNode = curNode.children[idx];
      } else {
        curNode = curNode.children[idx] = new Trie();
      }
    }

    curNode.isEndOfWord = true;
    return curNode;
  }
}

最佳答案

一旦您开始使用 JS 进行处理,这是一个非常常见的问题。如果 Mobile Safari 问题是原因,那么您要做的是找出此处的 CPU 时间。

我假设它是 dict.insert() 循环而不是 data.split() 调用(这会更难管理)。

这里的想法是将 dict.insert() 循环拆分为可以在顺序循环中异步调用的功能 block (这就是 setupBuildActions 函数做)。在第一个 block 之后,每个后续 block 都通过 setTimeout 调用,这有效地重置了 JS 运行时中的函数时间计数器(这似乎是杀死你的进程的原因)。

使用 Sequencer 函数意味着您还可以控制函数的运行顺序(它们始终按照它们在此处生成的顺序运行,并且不会安排两个或多个函数同时执行)。这比在没有回调的情况下触发数千个 setTimeout 调用要有效得多。您的代码保留对执行顺序的控制(这也意味着您可以在执行期间进行更改)并且 JS 运行时不会因计划执行请求而重载。

您可能还想在 https://github.com/michiel/sequencer-js 查看节点项目更多测序示例和http://ejohn.org/blog/how-javascript-timers-work/有关不同平台上的 setTimeout 的解释。

var dict = new Trie();

// These vars are accessible from all the other functions we're setting up and
// running here

var BLOCKSIZE     = 500;
var words         = [];
var buildActions  = [];

function Sequencer(funcs) {
  (function() {
    if (funcs.length !== 0) {
      funcs.shift()(arguments.callee);
    }
  })();
}

// Build an Array with functions that can be called async (using setTimeout)

function setupBuildActions() {
  for (var offset=0; offset<words.length; offset+= BLOCKSIZE) {
    buildActions.push((function(offset) {
      return function(callback) {
        for (var i=offset; i < offset + BLOCKSIZE ; i++) {
          if (words[i] !== null) { // ugly check for code brevity
            dict.insert(words[i]);
          }
        }           
        // This releases control before running the next dict.insert loop
        setTimeout(callback, 0);
      };
    })(offset));
  }
}

$.ajax({
    url: 'data/dictionary_342k_uppercase.txt',
    async: true,
    success: function (data) {
      // You might want to split and setup these calls 
      // in a setTimeout if the problem persists and you need to narrow it down
      words = data.split('\n');
      setupBuildActions();
      new Sequencer(buildActions);
    },
    error: function(){
      $('#loading-message').text("Problem s rječnikom");
    }
});

关于javascript - 使用 ajax 加载字典文件,不要让 iPhone Mobile Safari 崩溃,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16036433/

相关文章:

jquery - 我怎样才能改变图例 jquery flot 的形状

iphone - 在 UITabBarController 之前显示 XIB?

javascript - 在 asyncStorage 中存储数组状态对象

javascript - 如何增加关闭日期取决于选择下拉日期选择器

javascript - Windows 边栏环境中的代码分析

javascript - 使用 jQuery 在浏览器中播放 .m3u 播放列表的方法

iphone - NSData 显示为字符串

iphone - 将 slider 值增加 10

javascript - 如何获得准确的元件价格?

javascript - 有没有办法为按钮数组添加事件类 onClick?