-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathstudyjQuery.txt
executable file
·4363 lines (3405 loc) · 122 KB
/
studyjQuery.txt
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
版本 3.2.1
注 未看标记为*****
jQuery 代码结构分析
首先定义一堆私有变量
记录变量,或者单看看不出意思的。
var class2type = {};
var toString = class2type.toString; // 猜测为toString.call(variable)
null -> [object Null]
undefiend -> [object Undefiend]
new Date() -> [object Date]
new ClassA() -> [obect Object]
之后补充个知识
typeof Object
// "function"
typeof Object()
// "object"
因为 Object 是一个构造函数
其后再补充 Function 与 Object
https://segmentfault.com/q/1010000002736664
https://segmentfault.com/q/1010000000249140
Object.__proto__ === Function.prototype
Function.__proto__.__proto__ === Object.prototype
一般__proto__ 为 constructor.proptypes
如果一个对象是通过Object.create函数构造出来的,.那其__proto__就不一定是.constructor.prototype了
目前感觉很绕,需抽空研究
var hasOwn = class2type.hasOwnProperty;
{}.hasOwnProperty 与 Object.prototype.hasOwnProperty 相同
因为 {} 与 new Object()
但是不要忘了 {} 与 Object.prototype 是不同的
例: hasOwnProperty 判断是否含有特定的自身属性
{}.hasOwnProperty('toString'); // false
Object.prototype.hasOwnProperty('toString'); // true
补充 for in 是会遍历原型链的,但是不会遍历不可枚举的属性
再补充 in 可以是不可枚举属性,也有原型链
'toString' in {}; // true
Object.getOwnPropertyNames 就可以输出 hasOwnProperty 的所有属性
Object.keys 是只能输出可枚举的
以上两个均是自身特定的,不会遍历原型链
var fnToString = hasOwn.toString;
var ObjectFunctionString = fnToString.call( Object );
如果要循环全部 需要一层一层原型链遍历
使用 Object.getPrototypeOf
类似这样 while (obj = Object.getPrototypeOf(obj));
var fnToString = hasOwn.toString;
直接调用 fnToString() 是会报错的,一开始用 FireBug 一直不知道错在哪,
一直报错 TypeError: Function.prototype.toString called on incompatible object
其后用 Chrome 报错 Uncaught TypeError: Function.prototype.toString requires that 'this' be a Function
明显多了,this 改动了,虽然不知道toString的源码是什么,但也能判断出来大概。
以 Object 做判断
var a = {b: 1};
var b = a.toString;
b();
FireBug 输出是 "[object Window]"
Chrome 输出是 "[object Undefined]"
可知源码必是用了 this 来输出结果
根据知乎 https://www.zhihu.com/question/19636194 方应杭的答案
this 三中调用形式
func(p1, p2)
obj.child.method(p1, p2)
func.call(context, p1, p2)
看第一种
function func() {
console.log(this)
}
func();
可等价为 func.call(undefiend) // 可简写为 func.call()
按理就是 undefined, 但浏览器中如果你传的 context 不是一个对象,那么 window 对象就是默认的 context。
(这条规则在 Node.js 和 strict 模式下会稍微不一样,不过那不是我们现在要讨论的)
看第二种
var obj = {
foo: function(){
console.log(this)
}
}
obj.foo();
可等价为 obj.foo.call(obj), this 就是 obj 了
看第三种
直接定了 context 了
现在可以知道了,举个例子
Object.prototype.test = function () {
console.log(this);
}
var a = {x: 1};
a.test(); // 此处this就是a
var b = a.test;
b(); // 此处this就是window
之后知乎里面一道 this 的题目
var a = 10;
function test() {
a = 5;
console.log(a);
console.log(this.a);
var a;
console.log(this.a);
console.log(a);
}
问:执行test()和new test() 返回值分别为啥?
此处涉及变量提前
a = 10;
var a;
console.log(a);
输出的是 10
答案是 5 10 10 5
5 undefined undefined 5
再补充箭头函数
箭头函数没有自己的this, 它的this是继承而来; 默认指向在定义它时所处的对象(宿主对象),而不是执行时的对象
function DOMEval( code, doc ) {
doc = doc || document;
var script = doc.createElement( "script" );
script.text = code;
doc.head.appendChild( script ).parentNode.removeChild( script );
}
appendChild 后立马 removeChild
单看认为是实现 once 功能
但自己目前对这样写的实际效果保持怀疑
jQuery = function( selector, context ) {
// The jQuery object is actually just the init constructor 'enhanced'
// Need init if jQuery is called (just allow error to be thrown if not included)
return new jQuery.fn.init( selector, context );
},
猜能猜到是创建一个jQuery对象
目前该函数尚未遇见,做个标记*****。
rtrim = /^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g;
https://imququ.com/post/bom-and-javascript-trim.html
\uFEFF。它是 ES5 新增的空白符,叫字节次序标记字符(Byte Order Mark),也就是 BOM 字符。
fcamelCase = function( all, letter ) {
return letter.toUpperCase();
};
作为 replace 的第二个参数
其后定义了 jQuery.fn
提前看
merge: function( first, second ) {
var len = +second.length,
j = 0,
i = first.length;
for ( ; j < len; j++ ) {
first[ i++ ] = second[ j ];
}
first.length = i;
return first;
},
就是把两个数组或类数组对象合并
pushStack: function( elems ) {
// Build a new jQuery matched element set
// constructor: jQuery,
var ret = jQuery.merge( this.constructor(), elems );
// Add the old object onto the stack (as a reference)
ret.prevObject = this;
// Return the newly-formed element set
return ret;
},
即创建一个新的jQuery对象,且放入新的elem,原先的扔在pervObject
目前可以取消链式调用的方法,如 eq find 等等都用了此函数,jQuery 的强大的链式调用关键方法
end: function () {
return this.prevObject || this.constructor();
}
做标记***** 后期 jQuery 的方法与 underscore 比较 (each, map 这些)
定义 jQuery 与 jQuery.fn 的 extend 方法
照着敲了一遍
jQuery.extend = jQuery.fn.extend = function () {
var options, name, src, copy, copyIsArray, clone,
target = arguments[ 0 ] || {},
i = 1,
length = arguments.length,
deep = false;
// 函数重载,判断是否深复制,默认浅复制
if ( typeof target === 'boolean' ) {
deep = target;
target = arguments[i] || {};
i++;
}
// 处理target为string或其他的情况,可能发生在深复制
if ( typeof target !== 'object' && !jQuery.isFunction ( target) ) {
target = {};
}
// 当只有一个参数时,extend jQuery 自己本身
if (i === length) {
target = this;
i--;
}
for ( ; i < length; i++ ) {
// 感觉此处不合理,用 != 而不是 !== 而且为了排除null与undefined
// 用 if (options) 更好
// for in 对null 和 undefined 也不会报错
if ( ( options = arguments[ i ] ) != null ) {
for ( name in options) {
src = target[ name ];
copy = options[ name ];
// 防止无限递归的对象
// var a = {b: 1}
// extend(a, {x: a});
if (target === copy ) {
continue;
}
// recurse 递归
if ( deep && copy && jQuery.isPlainObject( copy ) ||
( copyIsArray = Array.isArray( copy )) {
if ( copyIsArray ) {
// 目前不清楚为何要重置copyIsArray,完全没作用
// 然而是有很大作用的,因为此次为循环,且copyIsArray是“或”语句,或的话可能不会执行到,因此需要重置copyIsArray
copyIsArray = false;
clone = src && Array.isArray( src ) ? src : [];
} else {
clone = src && jQuery.isPlainObject( src ) ? src : {};
}
target[ name ] = jQuery.extend( deep, clone, copy);
} else if ( copy !== undefined ) {
target[ name ] = copy;
}
}
}
}
return target;
}
extend 函数补充知识点
关于typeof
typeof null === 'object';
typeof [1, 2, 3] === 'object';
// instaitated with 与...联系在一起
// All constructor function while instaitated with 'new' keyword will always be typeof 'object'
typeof new Number(100) === 'object';
// But there is a exception in case of Function constructor of Javascript
typeof new Function() === 'function';
typeof /s/ === 'function'; // Non-conform to ECMAScript 5.1
typeof /s/ === 'object'; // Conform to ECMAScript 5.1
关于 for in
var str = 'abc';
for (var key in str) {
console.log(key);
}
// 0
// 1
// 2
紧接着 extend jQuery 对象
一堆 isXXX 函数
此处也是够6,我应该不敢这么写,此处应该是算 jQuery 奇淫技巧之一。
extend 函数用到了需要 extend 的函数。
如 jQuery 本身没有 isFunction 但是 extend 函数用了 isFunction
isFunction 是需要 extend 的属性
expando 是 expandable object 的缩写,表示可扩展的对象
目前未知 noop 是干什么的,跟本没用到
介绍 isWindow
isWindow: function ( obj ) {
return obj != null && obj === obj.window
}
window 对象的 window 属性指向他自己本身,截取段 MDN 的解释
The window property of a window object points to the window object itself. Thus the following expressions all return the same window object
介绍 isNumeric
isNumeric: function ( obj ) {
// As of jQuery 3.0, isNumeric is limited to
// strings and numbers (primitives or objects)
// that can be coerced to finite numbers (gh-2662)
var type = jQuery.type( obj );
return ( type === 'number' || type === 'string') &&
// parseFloat NaNs numeric-cast false positives ("")
// ...but misinterprets leading-number strings, particularly hex literals ("0x...")
// subtraction forces infinities to NaN
!isNaN( obj - parseFloat( obj ) );
}
那堆英语我也不知道怎么翻译
先判断是否 数字 或 字符串
且 !isNaN( obj - parseFloat( obj ) )
一开始有个疑惑,为甚不直接 obj - obj
是为了排除 null '' Boolean
其后补充知识
减法会强制转换类型
null '' Boolean值 都会强制转换为数字
parseFloat 则没这种情况,但是 parseFloat 有其他问题
其会将'Ox15'这样的值转换为 0
会将 [6, 8] 数组转换成第一个数字(空数组没问题,对象也没问题)
isPlainObject: function ( obj ) {
var proto, Ctor;
if ( !obj || toString.call( obj ) !== '[object object]') {
return false;
}
proto = getProto( obj );
if ( !proto ) {
return true;
}
Ctor = hasOwn.call( proto, 'constructor' ) && proto.constructor;
return typeof Ctor === 'function' && fnToString.call( Ctor) === ObjectFunctionString
}
判断对象是否简单对象,即使用 {} 或 new Object 创建的对象
先看参数是否存在,排除了 null 这些, 其后 toString.call
不使用 typeof 因为 window Date document 这些全是 Object 使用 toString.call 可筛选这些
其后获得参数原型
若无原型,证明是 Object.create( null ) 创建的。继承在另一篇文里。
判断原型的 constructor 是否为 ƒ Object() { [native code] }
简单/朴素对象的原型 constructor 为 function
toString 为function Object() { [native code] }
function A() {}
var b = new A();
$.isPlainObject(b); // false
b 的 constructor 为 function A() {}
isEmptyObject: function ( obj ) {
var name;
for ( name in obj ) {
return false;
}
return true;
}
和自己之前写的判断是否空对象类似
type: function ( obj ) {
if ( obj == null ) {
return obj + '';
}
return typeof obj === 'object' || typeof === 'function' ?
class2type[ toString.call( obj ) ] || 'object' :
typeof obj;
}
jQuery.each( "Boolean Number String Function Array Date RegExp Object Error Symbol".split( " " ),
function( i, name ) {
class2type[ "[object " + name + "]" ] = name.toLowerCase();
} );
jQuery 的 type 遵从了一些 typeof, null 的类型还是 object
但也不遵从 如 Date, typeof 是 object, 变成 date
rmsPrefix = /^-ms-/,
rdashAlpha = /-([a-z])/g,
camelCase: function (string) {
return string.replace( rmsPrefix, 'ms-' ).replace( rdashAlpha, fcamelCase );
}
将 text-align 这样的值转换为 textAlign
肯定 css 函数里用了
之前做过标记
fcamelCase = function( all, letter ) {
return letter.toUpperCase();
};
为何来个all,就是参数而已
注意根据 RegExp 不同, replace 第二个参数 function 的 arguments 是不同的
var str = 'ascscabc123';
var a = /abc/g;
var b = /a(b)c/g;
str.replace(a, function () {
console.log(arguments); // abc, 5, ascscabc123
});
str.replace(b, function () {
console.log(arguments); // abc, b, 5, ascscabc123
});
若匹配,多一个括号,就多一个 argument
如果是 /a(b)(c)/g 则是 abc, b, c, 5, ascscabc123
each: function ( obj, callback ) {
var length, i = 0;
if ( isArrayLike( obj ) ) {
length = obj.length;
for ( ; i < length; i++ ) {
if ( callback.call( obj[ i ], i, obj[i] ) === false) {
break;
}
}
} else {
for ( i in obj ) {
if ( callback.call( obj[ i ], i, obj[ i ] ) === false ) {
break;
}
}
}
}
与原生的 each 很大不同
首先 arguments 只有两个, 依次为 i item
原生的有三个,为 item i array
之后此处是 callback.call( obj[ i ], i, obj[ i ] ) 证明 this 是 item
而原生的 this 是 window (无论是否箭头函数)
jQuery 的 each 可中断, 原生不可
var a = [{x: 3}, {x: 4}, {x : 5}];
jQuery.each(a, function (i, item) {
item.x = 666;
return false;
}); // [{x: 666}, {x: 4}, {x : 5}]
makeArray 把类数组对象 变成数组
grep linux的检索命令,也就是筛选出符合的 item, 就是filter
proxy 改变函数上下文,就是 bind
proxy: function ( fn, context ) {
var tmp, args, proxy;
if ( typeof context === 'string' ) {
tmp = fn[ context ];
context = fn;
fn = tmp;
}
if ( !jQuery.isFunction( fn) ) {
return undefined;
}
// 模拟 bind
args = slice.call( arguments, 2);
proxy = function () {
return fn.apply( context || this, args.concat( slice.call( arguments ) ) );
}
// Set the guid of unique handler to the same of original handler, so it can be removed
proxy.guid = fn.guid = fn.guid || jQuery.guid++;
return proxy;
}
isArrayLike 判断是否数组/类数组对象
function isArrayLike( obj ) {
var length = !!obj && 'length' in obj && obj.length;
type = jQuery.type( obj );
if ( type === 'function' || jQuery.isWindow( obj ) ) {
return false;
}
return type === 'array' || length === 0 || typeof length === 'number' && length > 0 && ( length - 1) in obj;
}
function 也有 length 属性,为形参个数
function a(x, y, z) {}
console.log(a.length); // 3
window 对象也有 length 属性, length 为 0
此处判断对象是否类数组对象 三个条件
typeof length === 'number'
&& length > 0
&& ( length - 1) in obj;
booleans = "checked|selected|async|autofocus|autoplay|controls\
|defer|disabled|hidden|ismap|loop|multiple|open|readonly\
|required|scoped",
autoplay video audio 音频及视频自动播放
controls video audio 音频视频控件
loop video audio 音频视频循环播放
defer script 仅适用于外部脚本,即使用了 src 属性,当页面加载完成后,才执行脚本
hidden * 对元素隐藏,比如可以在提交表单时,提交非用户直接输入的数据
ismap img 没用过没见过,直接复制解释 当用户在 ismap 图像上单击了某处时,浏览器会自动把鼠标的 x、y 位置(相对于图像的左上角)发送到服务器端
multiple input select 多选,input 多个文件,select 多选
open detail HTML5 的新标签,如同下拉展现
scoped style 限定 style 的作用区域,只有 Firefox 支持
whitespace = "[\\x20\\t\\r\\n\\f]",
\x20 表示空格
\t 制表符
\r 回车符
\n 换行符
\f 换页符
identifier = "(?:\\\\.|[\\w-]|[^\0-\\xa0])+",
先解释 ?: 非捕获正则
前面曾说过
var str = 'ascscabc123';
var a = /abc/g;
var b = /a(b)c/g;
str.replace(a, function () {
console.log(arguments); // abc, 5, ascscabc123
});
str.replace(b, function () {
console.log(arguments); // abc, b, 5, ascscabc123
});
若匹配,多一个括号,就多一个 argument
如果是 /a(b)(c)/g 则是 abc, b, c, 5, ascscabc123
此处若是使用非匹配正则
var c = /a(?:b)c/g;
str.replace(c, function () {
console.log(arguments); // abc, 5, ascscabc123
});
预匹配,有肯定(?=x)和否定(?!x)两种形式,表示匹配结果后紧跟着x或不能有x(注意匹配结果中不含x)
'abc'.match(/\w/g)结果为["a","b","c"]
'abc'.match(/\w(?=b)/g)结果为["a"]
'abc'.match(/\w(?!b)/g)结果为["b","c"]
四根斜杠 即 两根斜杠
之后用在 new RegExp 则表示一根
. 任意单个字符,换行符除外
[] 查找在方括号内任意一个字符
[\\w-] \w 任意一个字母或数字或下划线 -中划线
[^\0-\\xa0])+ 除了 \0 到 \xa0 的一个或多个字符
具体标识符为何这样写,懒得看文档,太多英文看得晕 ***** 后期看
http://www.w3.org/TR/CSS21/syndata.html#value-def-identifier
attributes = "\\[" + whitespace + "*(" + identifier + ")(?:" + whitespace +
// Operator (capture 2)
"*([*^$|!~]?=)" + whitespace +
// "Attribute values must be CSS identifiers [capture 5] or strings [capture 3 or capture 4]"
"*(?:'((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\"|(" + identifier + "))|)" + whitespace +
"*\\]",
// \[(?:whitespace)*((?:identifier))(?:(?:whitespace)*([*^$|!~]?=)(?:whitespace)*(?:'((?:\\.|[^\\'])*)'|\"((?:\\.|[^\\\"])*)\"|((?:identifier)))|)(?:whitespace)*\]
// https://jex.im/regulex/#!flags=&re=%5C%5B(%3F%3Awhitespace)*((%3F%3Aidentifier))(%3F%3A(%3F%3Awhitespace)*(%5B*%5E%24%7C!~%5D%3F%3D)(%3F%3Awhitespace)*(%3F%3A'((%3F%3A%5C%5C.%7C%5B%5E%5C%5C'%5D)*)'%7C%5C%22((%3F%3A%5C%5C.%7C%5B%5E%5C%5C%5C%22%5D)*)%5C%22%7C((%3F%3Aidentifier)))%7C)(%3F%3Awhitespace)*%5C%5D
attributes = "\\[whitespace*(identifier)\
(\
?:whitespace*([*^$|!~]?=)whitespace*\
(\
?:'((?:\\\\.|[^\\\\'])*)'|\
\"((?:\\\\.|[^\\\\\"])*)\"|\
(identifier)\
)|\
)whitespace*\\]";
pseudos = ":(" + identifier + "=)(?:\\((" +
// To reduce the number of selectors needing tokenize in the preFilter, prefer arguments:
// 1. quoted (capture 3; capture 4 or capture 5)
"('((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\")|" +
// 2. simple (capture 6)
"((?:\\\\.|[^\\\\()[\\]]|" + attributes + ")*)|" +
// 3. anything else (capture 2)
".*" +
")\\)|)",
// 伪元素匹配
// :((?:identifier)=)(?:\((('((?:\\.|[^\\'])*)'|\"((?:\\.|[^\\"])*)")|((?:\\\\.|[^\\\\()[\\]]|(?:attributes))*)|.*)\)|)
// https://jex.im/regulex/#!flags=&re=%3A((%3F%3Aidentifier)%3D)(%3F%3A%5C((('((%3F%3A%5C%5C.%7C%5B%5E%5C%5C'%5D)*)'%7C%5C%22((%3F%3A%5C%5C.%7C%5B%5E%5C%5C%22%5D)*)%22)%7C((%3F%3A%5C%5C%5C%5C.%7C%5B%5E%5C%5C%5C%5C()%5B%5C%5C%5D%5D%7C(%3F%3Aattributes))*)%7C.*)%5C)%7C)
pseudos = ":(identifier=)\
(\
?:\\(\
(\
(\
'(\
(\
?:\\\\.|[^\\\\']\
)*\
)\
'|\"((?:\\\\.|[^\\\\\"])*)\"\
)|\
((?:\\\\.|[^\\\\()[\\]]|attributes)*)|\
.*)\
\\)\
|)",
rtrim = new RegExp( "^" + whitespace + "+|((?:^|[^\\\\])(?:\\\\.)*)" + whitespace + "+$", "g" ),
该正则分两部分
^(?:whitespace)+ 或 ((?:^|[^\\])(?:\\.)*)whitespace+$
以空格 1~多次开头
或者 (非\\开头加空格)以空格结尾
以几个实例读懂上面那个正则
(?:^|[^\\]) 会匹配一个不是 '\\' 的任意字符,匹配空字符
若是 则不匹配,只是个 空字符串, (?:\\.) 会匹配'\\字符'
var a = /(?:(^|5)(\\.)*)$/;
var b = '6'
console.log(b.match(a))
var c = /^/;
var d = '7';
console.log(d.match(c));
var e = /^|[^\\]/;
var f = '\\';
console.log(f.match(e));
var g = /((?:^|[^\\])(?:\\.)*)$/;
var h = '\\';
console.log(h.match(g))
var i = /(^)|([^\\])/;
var j = '\\';
console.log(j.match(i));
var k = /\\./;
var l = '\\.';
console.log(l.match(k))
var m = /(?:(^|[^\\])(\\.)*)$/;
var n = '6'
console.log(n.match(m))
var o = /(?:(^|[^\\])(\\.)*)$/;
var p = '\\5'
console.log(p.match(o))
之后看 Sizzle 函数
Sizzle( selector, context, results, seed ) *****坐标记,后面两个参数未知是何?
要求 context 的 nodeType 是 1 或 9 或 11
记忆几个关键的nodeType
1 element
2 attribute
3 text
9 document
11 documentFragment
Sizzle 会优先使用 window 自带的 如 getElementById getElementsByTagNamegetElementsByClassName querySelectorAll
不行再调用 select 函数。
Sizzle 里面有众多前面未有的变量及函数,需后期观看 *****
定义一个 缓存函数
/**
* Create key-value caches of limited size
* @returns {function(string, object)} Returns the Object data after storing it on itself with
* property name the (space-suffixed) string and (if the cache is larger than Expr.cacheLength)
* deleting the oldest entry
*/
function createCache() {
var keys = [];
function cache( key, value ) {
// Use (key + " ") to avoid collision with native prototype properties (see Issue #157)
if ( keys.push( key + " " ) > Expr.cacheLength ) {
// Only keep the most recent entries
delete cache[ keys.shift() ];
}
return (cache[ key + " " ] = value);
}
return cache;
}
原有注释虽然讲得明明白白,但是未知其作用。
关键未知Expr是啥 *****
/**
* Mark a function for special use by Sizzle
* @param {Function} fn The function to mark
*/
function markFunction( fn ) {
fn[ expando ] = true;
return fn;
}
这个也未知作用
其中 expando = "sizzle" + 1 * new Date()
目前认为可以作为函数是否已经使用来对待。
其后有定义一个 assert 方法。在关于前端测试中经常可看到改函数
/**
* Support testing using an element
* @param {Function} fn Passed the created element and returns a boolean result
*/
function assert( fn ) {
var el = document.createElement("fieldset");
try {
return !!fn( el );
} catch (e) {
return false;
} finally {
// Remove from its parent by default
if ( el.parentNode ) {
el.parentNode.removeChild( el );
}
// release memory in IE
el = null;
}
}
try catch finally
其中 finally 会在执行完 try 和 catch 后执行
今日在写 async await 中就遇到 try catch
不得不说 async await 实在太好用了
function test(time) {
return new Promise((resolve, reject) => {
setTimeout(() => {
resolve('ok')
// reject('error')
}, time)
})
}
async function run() {
console.log('run begin');
try {
var result = await test(1500);
console.log(result);
} catch (e) {
console.log(e);
} finally {
console.log('run end');
}
}
run();
接着定义一个 addHandle 方法 为指定的属性添加相同的方法,此处同 createCache 关键未知Expr是啥 *****
/**
* Adds the same handler for all of the specified attrs
* @param {String} attrs Pipe-separated list of attributes
* @param {Function} handler The method that will be applied
*/
function addHandle( attrs, handler ) {
var arr = attrs.split("|"),
i = arr.length;
while ( i-- ) {
Expr.attrHandle[ arr[i] ] = handler;
}
}
接着定以 createInputPseudo createButtonPseudo createDisabledPseudo createPositionalPseudo
其中 createPositionalPseudo 用到了上面提过的 markFunction 整个函数都不知道意义,为位置伪造?也没传 element
/**
* Returns a function to use in pseudos for positionals
* @param {Function} fn
*/
function createPositionalPseudo( fn ) {
return markFunction(function( argument ) {
argument = +argument;
return markFunction(function( seed, matches ) {
var j,
matchIndexes = fn( [], seed.length, argument ),
i = matchIndexes.length;
// Match elements found at the specified indexes
while ( i-- ) {
if ( seed[ (j = matchIndexes[i]) ] ) {
seed[j] = !(matches[j] = seed[j]);
}
}
});
});
}
/**
* Checks a node for validity as a Sizzle context
* @param {Element|Object=} context
* @returns {Element|Object|Boolean} The input node if acceptable, otherwise a falsy value
*/
function testContext( context ) {
return context && typeof context.getElementsByTagName !== "undefined" && context;
}
测试上下文? 这函数也够绝 return context && typeof context.getElementsByTagName !== "undefined" && context;
链式调用。最后返回 context 但若不存在就返回 false
从未接触过 XML
isXML = Sizzle.isXML = function( elem ) {
// documentElement is verified for cases where it doesn't yet exist
// (such as loading iframes in IE - #4833)
var documentElement = elem && (elem.ownerDocument || elem).documentElement;
return documentElement ? documentElement.nodeName !== "HTML" : false;
};
其中 ownerDocument 返回 document 对象
<p id="a">123</p>
console.log(document.getElementById('a').documentElement) // undefined
其后一个 setDocument ,400行代码的大函数
直接复制他人写的解释
Sizzle的setDocument函数,根据当前文档设置文档相关的变量,参数element or document,返回current document
这个函数主要任务是测试浏览器对相关函数的支持,从而写出兼容的ID,TAG,CLASS的find和filter函数;
测试浏览器对querySelectorAll函数的支持程度,建立对不支持的情况过滤的正则表达式rbuggyQSA;
重写兼容的contains和sortOrder方法。
/**
* Sets document-related variables once based on the current document
* @param {Element|Object} [doc] An element or document object to use to set the document
* @returns {Object} Returns the current document
*/
setDocument = Sizzle.setDocument = function( node ) {
var hasCompare, subWindow,
doc = node ? node.ownerDocument || node : preferredDoc;