开发者问题收集

使用 JSON 时出现 TypeError

2015-07-05
265

JSON:

{
    "Item 1":[
        {
            "Item 1.1":2,
            "Item 1.2":3
        }
    ],
    "Item 2":[
        {
            "Item 2.1":2,
            "Item 2.2":23
        }
    ]
}

JS:

$.getJSON(jsonPath, function(data, status) {
    var tocHTML = [];
    function eachObj(obj, i) {
        $.each(obj, function(key, value) {
            ++i;
            if (i) {
                tocHTML.push('<li class="item">');
            } else {
                tocHTML.push('<li>');
            }
            if (value != null && Array.isArray(value)) {
                tocHTML.push('<a data-page="' + value[0] + '">' + key + '</a>');
                tocHTML.push('<ul>');
                eachObj(value[1]);
                tocHTML.push('</ul>');
            } else {
                tocHTML.push('<a data-page="' + value + '">' + key + '</a>');
            }
            tocHTML.push('</li>');
        });
    }
    eachObj(data, 0);
    $('#tableOfContents ul.list').html(tocHTML.join(""));
});

错误:

[Error] TypeError: undefined is not an object (evaluating 'e.length')
    each (app.js, line 2)
    eachObj (_init.js, line 223)
    (anonymous function) (_init.js, line 233)
    each (app.js, line 2)
    eachObj (_init.js, line 223)
    (anonymous function) (_init.js, line 241)
    c (app.js, line 3)
    fireWith (app.js, line 3)
    n (app.js, line 4)
    (anonymous function) (app.js, line 4)

无法弄清楚错误内容,它甚至引用了一个单独的 app.js 文件。

2个回答

当代码发现外部迭代中某个条目的值是数组时,它会执行此操作:

            tocHTML.push('<a data-page="' + value[0] + '">' + key + '</a>');
            tocHTML.push('<ul>');
            eachObj(value[1]);
            tocHTML.push('</ul>');

该代码旨在迭代 value[1] ,因此它应该是一个对象或数组。但是,在您的 JSON 中:

"Item 1":[
    {
        "Item 1.1":2,
        "Item 1.2":3
    }
],

Item 1 的数组值只有一个条目;在这种情况下, value[1] 将为 undefined 。这意味着您正在使用 undefined 作为第一个参数调用 eachObj() ,这将导致 jQuery 的 $.each() 遇到该问题。

Pointy
2015-07-05

尝试使用 $.map()

var data = {
  "Item 1": [{
    "Item 1.1": 2,
    "Item 1.2": 3
  }],
  "Item 2": [{
    "Item 2.1": 2,
    "Item 2.2": 23
  }]
};

var ul = $("<ul class=list />");

ul.appendTo("body");

$.each(data, function(key, value) {

  var html = "<li class=item>" 
             + "<a>" + key + "</a>" + "<ul>" 
             + $.map(value, function(vals, keys) {
                 return $.map(vals, function(v, k) {
                   return "<li><a data-page=" + v + ">" + k + "</a></li>"
                 })
               }).join("") + "</ul>" + "</li>"

  ul.append(html)

});

/*
This is the expected result: stackoverflow.com/questions/31127879/

<ul class="list">
    <li class="item">
        <a data-page="9">Item 1</a>
        <ul>
            <li><a data-page="19">Item 1.1</a></li>
            <li><a data-page="29">Item 1.2</a></li>
            <li><a data-page="39">Item 1.3</a></li>
        </ul>
    </li>
    <li class="item"><a data-page="49">Item 2</a></li>
    <li class="item">
        <a data-page="59">Item 3</a>
        <ul>
            <li><a data-page="69">Item 3.1</a></li>
            <li><a data-page="79">Item 3.2</a></li>
            <li><a data-page="89">Item 3.3</a></li>
        </ul>
    </li>
</ul>
*/
<script src="https://ajax.googleapis.com/ajax/libs/jquery/1.11.1/jquery.min.js"></script>
guest271314
2015-07-05