我从Google的AdWords网站获取数据,该网站包含多个具有相同id
的元素。
您能否解释为什么以下三个查询没有得到相同答案(2)?
HTML:
<div>
<span id="a">1</span>
<span id="a">2</span>
<span>3</span>
</div>
JS:
$(function() {
var w = $("div");
console.log($("#a").length); // 1 - Why?
console.log($("body #a").length); // 2
console.log($("#a", w).length); // 2
});
根据W3C规范,具有2个具有相同ID的元素不是有效的html。
[当您的CSS选择器只有一个ID选择器(并且不在特定上下文中使用)时,jQuery使用本机document.getElementById
方法,该方法仅返回具有该ID的第一个元素。
但是,在其他两个实例中,jQuery依赖于Sizzle选择器引擎(或querySelectorAll
,如果可用),该引擎显然选择了两个元素。结果可能因浏览器而异。
但是,在同一页面上绝对不能有两个具有相同ID的元素。如果您的CSS需要它,请改用一个类。
如果绝对必须通过重复的ID选择,请使用属性选择器:
$('[id="a"]');
看小提琴:http://jsfiddle.net/P2j3f/2/
注:如果可能的话,应使用标签选择器来限定该选择器,如下所示:
$('span[id="a"]');
jQuery的id
选择器仅返回一个结果。第二和第三条语句中的descendant
和multiple
选择器旨在选择多个元素。它类似于:
陈述1
var length = document.getElementById('a').length;
...产生一个结果。
陈述2
var length = 0;
for (i=0; i<document.body.childNodes.length; i++) {
if (document.body.childNodes.item(i).id == 'a') {
length++;
}
}
...产生两个结果。
陈述3
var length = document.getElementById('a').length + document.getElementsByTagName('div').length;
...也产生两个结果。
应该只有一个具有给定ID的元素。如果您遇到这种情况,请参阅我的答案的第二部分以获得选择。
[规范中未定义当您具有多个具有相同ID(非法HTML)的元素时浏览器的行为。您可以测试所有浏览器并找出它们的行为,但是使用此配置或依赖任何特定行为是不明智的。
如果要多个对象具有相同的标识符,请使用类。
<div>
<span class="a">1</span>
<span class="a">2</span>
<span>3</span>
</div>
$(function() {
var w = $("div");
console.log($(".a").length); // 2
console.log($("body .a").length); // 2
console.log($(".a", w).length); // 2
});
如果由于无法修复文档而要可靠地查看具有相同ID的元素,那么您将不得不进行自己的迭代,因为您不能依赖任何内置的DOM函数。
您可以这样做:
function findMultiID(id) {
var results = [];
var children = $("div").get(0).children;
for (var i = 0; i < children.length; i++) {
if (children[i].id == id) {
results.push(children[i]);
}
}
return(results);
}
或者,使用jQuery:
$("div *").filter(function() {return(this.id == "a");});
jQuery工作示例:http://jsfiddle.net/jfriend00/XY2tX/。
关于为什么会得到不同的结果,这与执行实际选择器操作的任何代码的内部实现有关。在jQuery中,您可以研究代码以找出给定版本在做什么,但是由于这是非法的HTML,因此无法保证它会随着时间的推移保持不变。根据我在jQuery中看到的内容,它首先检查选择器是否是一个简单的ID,例如#a
,如果是,则仅使用document.getElementById("a")
。如果选择器比该选择器复杂,并且存在querySelectorAll()
,则jQuery经常会将选择器传递给内置的浏览器功能,该功能将具有特定于该浏览器的实现。如果querySelectorAll()
不存在,则它将使用Sizzle选择器引擎手动找到具有自己实现的选择器。因此,您可以在同一浏览器系列中至少拥有三种不同的实现方式,具体取决于确切的选择器和浏览器的新颖程度。然后,各个浏览器都将具有自己的querySelectorAll()
实现。如果您想可靠地处理这种情况,则可能必须使用您自己的迭代代码,如上文所述。
每个id值在文档中只能使用一次。如果为多个元素分配了相同的ID,则使用该ID的查询将仅选择DOM中的第一个匹配元素。但是,不应依赖这种行为。包含多个具有相同ID的元素的文档无效。
顽皮的Google。但是他们甚至没有关闭我听到的<html>
和<body>
标签。问题是,为什么Misha的第二和第三查询返回2而不是1。
如果您有多个具有相同ID或相同名称的元素,只需将相同的类分配给这些多个元素并通过索引访问它们并执行所需的操作。
<div>
<span id="a" class="demo">1</span>
<span id="a" class="demo">2</span>
<span>3</span>
</div>
JQ:
$($(".demo")[0]).val("First span");
$($(".demo")[1]).val("Second span");
每个人都说“每个id值在一个文档中只能使用一次”,但是当我们有一个愚蠢的页面包含多个具有相同id的元素时,我们将如何获取所需的元素。如果我们使用JQuery'#duplicatedId'选择器,则仅获得第一个元素。要选择其他元素,您可以执行以下操作
$(“ [id = duplicatedId]”)
您将获得具有id = duplicatedId的所有元素的集合>
您可以简单地写$('span#a')。length来获取长度。