首頁 >web前端 >js教程 >詳解JavaScript中Property和Attribute的區別

詳解JavaScript中Property和Attribute的區別

黄舟
黄舟原創
2017-03-11 15:23:401353瀏覽

property 和 attribute非常容易混淆,兩個單字的中文翻譯也都非常相近(property:屬性,attribute:特性),但實際上,二者是不同的東西,屬於不同的範疇。

  • property是DOM中的屬性,是JavaScript裡的物件;

  • attribute是HTML標籤上的特性,它的值只能夠是字串;

基於JavaScript分析property 和attribute

html中有這樣一段程式碼:

<input id="in_1" value="1" sth="whatever">

簡單的在html頁面上建立一個input輸入列(注意在這個標籤中加入了一個DOM中不存在的屬性「sth」),此時在JS執行如下語句

var in1 = document.getElementById(&#39;in_1&#39;);

執行語句

console.log(in1);

從console的列印結果,可以看到in1含有一個名為「attributes」的屬性,它的型別是NamedNodeMap,同時還有「id」和「value」兩個基本的屬性,但沒有「sth」這個自訂的屬性。

attributes: NamedNodeMap
value: "1"
id: "in_1"

有些console可能不會列印in1上的屬性,那麼可以執行以下命令列印要觀察的屬性:

console.log(in1.id);		// &#39;in_1&#39;
console.log(in1.value);		// 1
console.log(in1.sth);		// undefined

可以發現,標籤中的三個屬性,只有“id”和“value”會在in1上創建,而“sth”不會被創建。這是由於,每一個DOM物件都會有它預設的基本屬性,而在創建的時候,它只會建立這些基本屬性,我們在TAG標籤中自訂的屬性是不會直接放到DOM中的。

我們做一個額外的測試,創建另一個input標籤,並執行類似的操作:

#html:

<input id="in_2">

JS:

var in2 = document.getElementById(&#39;in_2&#39;);
console.log(in2);

從列印訊息中可以看到:

id: "in_2"
value: null

儘管我們沒有在TAG中定義“value”,但由於它是DOM預設的基本屬性,在DOM初始化的時候它照樣會被創建。由此我們可以得出結論:

  • DOM有其預設的基本屬性,而這些屬性就是所謂的「property」,無論如何,它們都會在初始化的時候再DOM物件上建立。

  • 如果在TAG對這些屬性進行賦值,那麼這些值就會作為初始值賦給DOM的同名property。

現在回到第一個input(「#in_1」),我們就會問,「sth」去哪裡了?別急,我們把attributes這個屬性印出來看看

console.log(in2);

上面有幾個屬性:

0: id
1: value
2: sth
length: 3
__proto__: NamedNodeMap

原來「sth」被放到了attributes這個物件裡面,這個物件按順序記錄了我們在TAG中定義的屬性和屬性的數量。此時,如果再將第二個input標籤的attributes印出來,就會發現只有一個「id」屬性,「length」為1。

從這裡就可以看出,attributes是屬於property的子集,它保存了HTML標籤上定義屬性。如果再進一步探索attitudes中的每一個屬性,會發現它們並不是簡單的對象,它是一個Attr類型的對象,擁有NodeType、NodeName等屬性。關於這一點,稍後再研究。 注意,印出attribute屬性不會直接得到物件的值,而是取得一個包含屬性名稱和值的字串,如:

console.log(in1.attibutes.sth);		// &#39;sth="whatever"&#39;

由此可以得出:

  • HTML標籤中定義的屬性和值會儲存該DOM物件的attributes屬性裡面;

  • ##這些attribute屬性的JavaScript中的類型是Attr,而不僅僅是保存屬性名稱和值這麼簡單;

那麼,如果我們更改property和attribute的值會出現什麼效果呢?執行如下語句:

in1.value = &#39;new value of prop&#39;;
console.log(in1.value);				// &#39;new value of prop&#39;
console.log(in1.attributes.value);	// &#39;value="1"&#39;

此時,頁面中的輸入欄的值變成了“new value of prop”,而propety中的value也變成了新的值,但attributes卻仍然是“ 1”。從這裡可以推斷,property和attribute的同名屬性的值並不是雙向綁定的。

如果反過來,設定attitudes中的值,效果會怎麼樣呢?

in1.attributes.value.value = &#39;new value of attr&#39;;
console.log(in1.value);				// &#39;new value of attr&#39;
console.log(in1.attributes.value);	// &#39;new value of attr&#39;

此時,頁面中的輸入列得到更新,property中的value也改變了。此外,執行下面語句也會得到相同的結果

in1.attributes.value.nodeValue = &#39;new value of attr&#39;;

由此,可下結論:

  • property能夠從attribute中得到同步;

  • attribute不會同步property上的值;

  • attribute與property之間的資料綁定是單向的,attribute->property;

  • 改變property和attribute上的任意值,都會將更新反映在HTML頁面中;

基於jQuery分析attribute和property

那麼jQuery中的attr和prop方法是怎樣的呢?

首先利用jQuery.prop來測試

$(in1).prop(&#39;value&#39;, &#39;new prop form $&#39;);

console.log(in1.value);				// &#39;new prop form $&#39;
console.log(in1.attributes.value);	// &#39;1&#39;

輸入欄的值更新了,但attribute並未更新。

然後用jQuery.attr來測試

$(in1).attr(&#39;value&#39;, &#39;new attr form $&#39;);

console.log(in1.value);				// &#39;new attr form $&#39;
console.log(in1.attributes.value);	// &#39;new attr form $&#39;

輸入欄的值更新了,同時property和attribute都更新了。

從上述測試的現象可以推斷,jQuery.attr和jQuery.prop基本和原生的操作方法效果一致,property會從attribute中獲取同步,然而attribute不會從property中獲取同步。那麼jQuery到底是如何實現的呢?

下面,我们来看看jQuery.attr和jQuery.prop的源码。

jQuery源码

$().prop源码

jQuery.fn.extend({
	prop: function( name, value ) {
		return access( this, jQuery.prop, name, value, arguments.length > 1 );
	},
	... // removeProp方法
});

$().attr源码

jQuery.fn.extend({
	attr: function( name, value ) {
		return access( this, jQuery.attr, name, value, arguments.length > 1 );
	},
	... // removeAttr方法
});

无论是attr还是prop,都会调用access方法来对DOM对象的元素进行访问,因此要研究出更多内容,就必须去阅读access的实现源码。

jQuery.access

// 这是一个多功能的函数,能够用来获取或设置一个集合的值
// 如果这个“值”是一个函数,那么这个函数会被执行

// @param elems, 元素集合
// @param fn, 对元素进行处理的方法
// @param key, 元素名
// @param value, 新的值
// @param chainable, 是否进行链式调用
// @param emptyGet,
// @param raw, 元素是否一个非function对象
var access = jQuery.access = function( elems, fn, key, value, chainable, emptyGet, raw ) {
	var i = 0,						// 迭代计数
		length = elems.length,		// 元素长度
		bulk = key == null;			// 判断是否有特定的键(属性名)

	// 如果存在多个属性,递归调用来逐个访问这些值
	if ( jQuery.type( key ) === "object" ) {
		chainable = true;
		for ( i in key ) {
			jQuery.access( elems, fn, i, key[i], true, emptyGet, raw );
		}

	// 设置一个值
	} else if ( value !== undefined ) {
		chainable = true;

		if ( !jQuery.isFunction( value ) ) {	// 如果值不是一个function
			raw = true;
		}

		if ( bulk ) {
			// Bulk operations run against the entire set
			// 如果属性名为空且属性名不是一个function,则利用外部处理方法fn和value来执行操作
			if ( raw ) {
				fn.call( elems, value );
				fn = null;

			// ...except when executing function values
			// 如果value是一个function,那么就重新构造处理方法fn
			// 这个新的fn会将value function作为回调函数传递给到老的处理方法
			} else {
				bulk = fn;
				fn = function( elem, key, value ) {
					return bulk.call( jQuery( elem ), value );
				};
			}
		}

		if ( fn ) {	// 利用处理方法fn对元素集合中每个元素进行处理
			for ( ; i < length; i++ ) {
				fn( elems[i], key, raw ? value : value.call( elems[i], i, fn( elems[i], key ) ) );
				// 如果value是一个funciton,那么首先利用这个函数返回一个值并传入fn
			}
		}
	}

	return chainable ?
		elems :			// 如果是链式调用,就返回元素集合

		// Gets
		bulk ?
			fn.call( elems ) :
			length ? fn( elems[0], key ) : emptyGet;
};

access方法虽然不长,但是非常绕,要完全读懂并不简单,因此可以针对jQuery.fn.attr的调用来简化access。

jQuery.fn.attr/ jQuery.fn.prop 中的access调用

$().attr的调用方式:

  • $().attr( propertyName ) // 获取单个属性

  • $().attr( propertyName, value ) // 设置单个属性

  • $().attr( properties ) // 设置多个属性

  • $().attr( propertyName, function ) // 对属性调用回调函数

prop的调用方式与attr是一样的,在此就不重复列举。为了简单起见,在这里只对第一和第二种调用方式进行研究。

调用语句:

access( this, jQuery.attr, name, value, arguments.length > 1 );

简化的access:

// elems 当前的jQuery对象,可能包含多个DOM对象
// fn jQuery.attr方法
// name 属性名
// value 属性的值
// chainable 如果value为空,则chainable为false,否则chainable为true

var access = jQuery.access = function( elems, fn, key, value, chainable, emptyGet, raw ) {

	var i = 0,						// 迭代计数
		length = elems.length,		// 属性数量
		bulk = false;				// key != null

	if ( value !== undefined ) {	// 如果value不为空,则为设置新值,否则返回该属性的值
		chainable = true;
		raw = true;				// value不是function

		if ( fn ) {	// fn为jQuery.attr
			for ( ; i < length; i++ ) {
				fn( elems[i], key, value);		// jQuery.attr(elems, key, value);
			}
		}
	}

	if(chainable) {			// value不为空,表示是get
		return elems;		// 返回元素实现链式调用
	} else {
		if(length) {		// 如果元素集合长度不为零,则返回第一个元素的属性值
			return fn(elems[0], key); 	// jQuery.attr(elems[0], key);
		} else {
			return emptyGet;		// 返回一个默认值,在这里是undefined
		}
	}
};

通过简化代码,可以知道,access的作用就是遍历上一个$调用得到的元素集合,对其调用fn函数。在jQuery.attr和jQuery.prop里面,就是利用access来遍历元素集合并对其实现对attribute和property的控制。access的源码里面有多段条件转移代码,看起来眼花缭乱,其最终目的就是能够实现对元素集合的变量并完成不同的操作,复杂的代码让jQuery的接口变得更加简单,能极大提高代码重用性,意味着减少了代码量,提高代码的密度从而使JS文件大小得到减少。

这些都是题外话了,现在回到$().attr和$().prop的实现。总的说,这两个原型方法都利用access对元素集进行变量,并对每个元素调用jQuery.prop和jQuery.attr方法。要注意,这里的jQuery.prop和jQuery.attr并不是原型链上的方法,而是jQuery这个对象本身的方法,它是使用jQuery.extend进行方法扩展的(jQuery.fn.prop和jQuery.fn.attr是使用jQuery.fn.extend进行方法扩展的)。

下面看看这两个方法的源码。

jQury.attr

jQuery.extend({
	attr: function( elem, name, value ) {
		var hooks, ret,
			nType = elem.nodeType;	// 获取Node类型

		// 如果 elem是空或者NodeType是以下类型
		// 		2: Attr, 属性, 子节点有Text, EntityReference
		// 		3: Text, 元素或属性中的文本内容
		// 		8: Comment, 注释
		// 不执行任何操作
		if ( !elem || nType === 3 || nType === 8 || nType === 2 ) {
			return;
		}

		// 如果支持attitude方法, 则调用property方法
		if ( typeof elem.getAttribute === strundefined ) {
			return jQuery.prop( elem, name, value );
		}

		// 如果elem的Node类型不是元素(1)
		if ( nType !== 1 || !jQuery.isXMLDoc( elem ) ) {
			name = name.toLowerCase();
			// 针对浏览器的兼容性,获取钩子函数,处理一些特殊的元素
			hooks = jQuery.attrHooks[ name ] ||
				( jQuery.expr.match.bool.test( name ) ? boolHook : nodeHook );
		}

		if ( value !== undefined ) {		// 如果value不为undefined,执行"SET"

			if ( value === null ) {			// 如果value为null,则移除attribute
				jQuery.removeAttr( elem, name );	

			} else if ( hooks && "set" in hooks && (ret = hooks.set( elem, value, name )) !== undefined ) {
				return ret;					// 使用钩子函数

			} else {						// 使用Dom的setAttribute方法
				elem.setAttribute( name, value + "" );		// 注意,要将value转换为string,因为所有attribute的值都是string
				return value;
			}

		// 如果value为undefined,就执行"GET"
		} else if ( hooks && "get" in hooks && (ret = hooks.get( elem, name )) !== null ) {
			return ret;			// 使用钩子函数

		} else {
			ret = jQuery.find.attr( elem, name );	// 实际上调用了Sizzle.attr,这个方法中针对兼容性问题作出处理来获取attribute的值

			// 返回获得的值
			return ret == null ?
				undefined :
				ret;
		}
	},

	...
});

从代码可以发现,jQuery.attr调用的是getAttribute和setAttribute方法。

jQeury.prop

jQuery.extend({

	...	
	prop: function( elem, name, value ) {
		var ret, hooks, notxml,
			nType = elem.nodeType;

		// 过滤注释、Attr、元素文本内容
		if ( !elem || nType === 3 || nType === 8 || nType === 2 ) {
			return;
		}

		notxml = nType !== 1 || !jQuery.isXMLDoc( elem );

		if ( notxml ) {		// 如果不是元素
			name = jQuery.propFix[ name ] || name;	// 修正属性名
			hooks = jQuery.propHooks[ name ];		// 获取钩子函数
		}

		if ( value !== undefined ) {		// 执行"SET"
			return hooks && "set" in hooks && (ret = hooks.set( elem, value, name )) !== undefined ?
				ret :						// 调用钩子函数
				( elem[ name ] = value );	// 直接对elem[name]赋值

		} else {							// 执行"GET"
			return hooks && "get" in hooks && (ret = hooks.get( elem, name )) !== null ?
				ret :				// 调用钩子函数
				elem[ name ];		// 直接返回elem[name]
		}
	},

	...
});

jQuery.prop则是直接对DOM对象上的property进行操作。

通过对比jQuery.prop和jQuery.attr可以发现,前者直接对DOM对象的property进行操作,而后者会调用setAttribute和getAttribute方法。setAttribute和getAttribute方法又是什么方法呢?有什么效果?

setAttribute和getAttribute

基于之前测试使用的输入框,执行如下代码:

in1.setAttribute(&#39;value&#39;, &#39;new attr from setAttribute&#39;);

console.log(in1.getAttribute(&#39;value&#39;));			// &#39;new attr from setAttribute&#39;
console.log(in1.value);							// &#39;new attr from setAttribute&#39;
console.log(in1.attributes.value);				// &#39;value="new attr from setAttribute"&#39;,实际是一个Attr对象

执行完setAttribute以后,就如同直接更改attributes中的同名属性;
而getAttribute的结果与访问property的结果一模一样,而不会像直接访问attritudes那样返回一个Attr对象。

特殊的例子

href

然而,是不是所有标签,所有属性都维持保持这样的特性呢?下面我们看看href这个属性/特性。

首先在html中创建一个标签:

<a href=&#39;page_1.html&#39; id=&#39;a_1&#39;></a>

在JS脚本中执行如下代码:

console.log(a1.href);	// &#39;file:///D:/GitHub/JS/html/test_01/page_1.html&#39;
console.log(a1.getAttribute(&#39;href&#39;));	// &#39;page_1.html&#39;

可以看到,property中保存的是绝对路径,而attribute中保存的是相对路径。那么,如果更改了这些值会发生什么情况呢?

更改attribute:

a1.setAttribute(&#39;href&#39;, &#39;page_2.html&#39;);		// 相对路径
console.log(a1.href);	// &#39;file:///D:/GitHub/JS/html/test_01/page_2.html&#39;
console.log(a1.getAttribute(&#39;href&#39;));	// &#39;page_2.html&#39;

a1.setAttribute(&#39;href&#39;, &#39;/page_3.html&#39;);	// 根目录路径
console.log(a1.href);						// &#39;file:///D:/page_3.html&#39;
console.log(a1.getAttribute(&#39;href&#39;));		// &#39;/page_3.html&#39;

更改property:

a1.href = &#39;home.html&#39;;	// 相对路径
console.log(a1.href);	// &#39;file:///D:/GitHub/JS/html/test_01/home.html&#39;
console.log(a1.getAttribute(&#39;href&#39;));	// &#39;home.html&#39;

a1.href = &#39;/home.html&#39;;	// 根目录路径
console.log(a1.href);	// &#39;file:///D:/home.html&#39;
console.log(a1.getAttribute(&#39;href&#39;));	// &#39;/home.html&#39;

从这里可以发现,href是特殊的属性/特性,二者是双向绑定的,更改任意一方,都会导致另一方的的值发生改变。而且,这并不是简单的双向绑定,property中的href永远保存绝对路径,而attribute中的href则是保存相对路径。

看到这里,attribute和property的区别又多了一点,然而,这又让人变得更加疑惑了。是否还有其他类似的特殊例子呢?

id

尝试改变property中的id:

	a1.id = &#39;new_id&#39;;
	console.log(a1.id);						// &#39;new_id&#39;
	console.log(a1.getAttribute(&#39;id&#39;));		// &#39;new_id&#39;

天呀,现在attribute中的id从property中的id发生了同步,数据方向变成了property attribute

disabled

再来看看disabled这个属性,我们往第一个添加“disabled”特性:

<input id="in_1" value="1" sth="whatever" disabled=&#39;disabled&#39;>	// 此时input已经被禁用了

然后执行下面的代码:

console.log(in1.disabled);		// true
in1.setAttribute(&#39;disabled&#39;, false);	// 设置attribute中的disabled,无论是false还是null都不会取消禁用
console.log(in1);				// true
console.log(in1.getAttribute(&#39;disabled&#39;));	// &#39;false&#39;

改变attributes中的disabled不会改变更改property,也不会取消输入栏的禁用效果。

如果改成下面的代码:

console.log(in1.disabled);		// true
in1.disabled = false;			// 取消禁用
console.log(in1.disabled);		// false
console.log(in1.getAttribute(&#39;disabled&#39;));	// null,attribute中的disabled已经被移除了

又或者:

console.log(in1.disabled);		// true
in1.removeAttribute(&#39;disabled&#39;);	// 移除attribute上的disabled来取消禁用
console.log(in1.disabled);		// false
console.log(in1.getAttribute(&#39;disabled&#39;));	// null,attribute中的disabled已经被移除了

可以发现,将property中的disabled设置为false,会移除attributes中的disabled。这样数据绑定又变成了,propertyattribute;

所以property和attritude之间的数据绑定问题并不能单纯地以“property”来说明。

总结

分析了这么多,对property和attribute的区别理解也更深了,在这里总结一下:

建立

  • DOM物件初始化時會在建立預設的基本property;

  • 只有在HTML標籤中定義的attribute才會被儲存在property的attributes屬性中;

  • attribute會初始化property中的同名屬性,但自訂的attribute不會出現在property中;

  • attribute的值都是字串#;

#資料綁定

  • attributes的資料會同步到property上,然而property的變更不會改變attribute;

  • 對於value,class這樣的屬性/特性,資料綁定的方向是單向的,attribute->property;

  • #對於id而言,資料綁定是雙向的,attribute;

  • 對於disabled而言,property上的disabled為false時,attribute上的disabled必定會並且存在,此時資料綁定可以認為是雙向的;

使用

  • #可以使用DOM的setAttribute方法同時更改attribute;

  • #直接存取attributes上的值會得到一個Attr對象,而透過getAttribute方法存取則會直接得到attribute的值;

  • 大多數情況(除非有瀏覽器相容性問題),jQuery.attr是透過setAttribute實現,而jQuery.prop則會直接存取DOM物件的property;

##到這裡為止,得出,property是DOM物件自身就擁有的屬性,而attribute是我們透過設定HTML標籤而賦予的特性,attribute和property的同名屬性/特性之間會產生一些特殊的資料聯繫,而這些聯繫會針對不同的屬性/特性有不同的區別。

事實上,在這裡,property和attribute之間的區別和聯繫難以用簡單的技術特性來描述,我在StackFlow上找到如下的回答,或者會更加接近真正的答案:

These words existed way before Computer Science came around.

Attribute is a quality or object that we attribute to someone or something. For example, the scepter is an attribute of power and statehood.

Property## is a quality that exists without any attribution. For example, clay has adhesive qualities; or, one of the properties of metals is electrical conductivity. Properties demonstrate themselves though physical phenomena without the need attribute them to somenone someken, somem somem some some somem some somem somem some somem some somem some somem some somem some somem somem some somem some somem some somem somem somem some some somem 可以在 somem some some some somem 到 some ” self-evident. In effect, you could say that a property is owned by someone or something.

To be fair though, in Computer Science these two words, at least for the most part , can be used interchangeably – but then again programmers usually don't hold degrees in English Literature and do not write or care much about grammar books
.

詳解JavaScript中Property和Attribute的區別

##最關鍵的兩句話:

  • attribute(特性),是我們賦予某個事物的特質或物件。

  • property(屬性),是早已存在的不需要外界所賦予的特質。

#

以上是詳解JavaScript中Property和Attribute的區別的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn