算法還原
白盒還原
- 直接扣算法, 或者是標准算法
- 理解對方 js 的意思, 能夠翻譯成其他語言, 可能會占用較長的分析時間
黑盒還原
- 直接整體調用(加密復雜, 更新頻繁, 整體關聯度高)
- 不需要關注算法邏輯, 需要模擬瀏覽器環境, 需要對抗環境檢測
rpc 調用
- 算法復雜度高, 瀏覽器環境難以模擬
- 找到算法位置, 暴露出來, 直接 rpc 調用, 需要保證瀏覽器狀態(內存泄漏, 保活)
瀏覽器自動化
- 無法逆向
- 接近真人, 但是有大量的自動化痕跡;
基本數據類型
- 數值 (Number): 整數和小數
- 字符串 (String): 文本
- 布爾值(Boolean): 布爾值, true 表示真, false 表示假
- undefined: 未定義, 或者不存在
- null: 表示空值
- 對象 (Object): 各種值組成的集合
原始類型
- 數值
- 字符串
- 布爾值
原始類型就是最基本的數據類型, 不能再進行細分;
undefined 和 null 一般看成是兩個特殊值;
合成類型
- 對象
- 狹義的對象 Object
- 數組 Array
- 函數 Function
一個對象往往是由多個類型的值組成, 可以看成是一個存放各種值的容器
查看類型
- typeof: 返回一個值的數據類型
- instanceof: 表示對象是否是某個構造函數的實例
- Object.prototype.toString
typeof
可以用來檢查一個未聲明的變量, 而不報錯;
JS
// 基本數據類型
var tmp1 = "字符串";
var tmp2 = 1;
var tmp3 = 1.1;
var tmp4 = true;
// 特殊類型
var tmp5= undefined;
var tmp6 = null; // null 是一個 object
// 對象
var tmp7 = {}; // 對象
var tmp8 = []; // 數組
var tmp9 = function(){}; // 函數
console.log("typeof(tmp1)", typeof (tmp1));
console.log("typeof(tmp2)", typeof (tmp2));
console.log("typeof(tmp3)", typeof (tmp3));
console.log("typeof(tmp4)", typeof (tmp4));
console.log("typeof(tmp5)", typeof (tmp5));
console.log("typeof(tmp6)", typeof (tmp6));
console.log("typeof(tmp7)", typeof (tmp7));
console.log("typeof(tmp8)", typeof (tmp8));
console.log("typeof(tmp9)", typeof (tmp9));
// function 因為有類object 的操作, 所以也屬於 object
console.log("tmp9.name", tmp9.name); // 獲取函數名
//typeof(tmp1) string
//typeof(tmp2) number
//typeof(tmp3) number
//typeof(tmp4) boolean
//typeof(tmp5) undefined
//typeof(tmp6) object
//typeof(tmp7) object
//typeof(tmp8) object
//typeof(tmp9) function
//tmp9.name tmp9
null undefined 和布爾值
null 和 undefined 的區別
- null 表示一個空對象, undefined 表示未定義;
- null 轉為數值的時候為 0, undefined 轉為數值的實收為 NaN;
boolean
布爾值表示真和假, true 表示真, false 表示假;
下列運算符會返回布爾值:
- ! (not)
- 相等運算符: =, , !, !=
- 比較運算符: >=, <=, <, >
表示 false 的值
在自動數據轉換中, 下列值會表示 false:
- undefined
- null
- false
- 0
- NaN
- “” 或者 ‘’ 空字符串
其他的值都會被當成 true;
空數組[]和空對象{}對應的布爾值都是 true;
數值
在 js 中, 所有的數值都是 64 位浮點數的形式進行存儲的, 也就是說在 js 底層, 沒有整數只有浮點數;
因為浮點數精度的問題, js 在進行浮點數運算的時候經常會出現問題:
JS
console.log(0.1 + 0.2);
// 0.30000000000000004
進制
- 十進制: 沒有前導, 直接用數值表示
- 八進制: 有前綴 0o 或者 0O
- 十六進制: 有前綴 0x 或者 0X
- 二進制: 有前綴 0b 或者 0B
默認情況下, js 內部會將八進制, 十六進制, 二進制轉為十進制;
NaN
NaN 是 js 中的特殊值, 表示非數字(Not a Number), 主要出現在字符串解析成數字出錯的時候;
- NaN 不是獨立的數據類型, 它是一個特殊值, 它的數據類型依然是 Number
- NaN 不等於任何值, 包括它本身 (不等於本身可以用來檢測某個值是否是 NaN)
- NaN 和任何數運算, 得到的結果都是 NaN;
Infinity
Infinity 用來表示無窮, 一般出現在兩種場景下:
- 正數的數值太大, 或者負數的數值太小;
- 非 0 的數除以 0, 得到 Infinity
- js 中數值正向溢出或者負向溢出或者被 0 除都不會報錯, 所以單純的數學運算幾乎沒有可能拋出異常
- Infinity 大於一切數值(除了 NaN); -Infinity 小於一切數值(除了 NaN)
- Infinity 和 NaN 比較, 總是返回 false;
全局 api
parseInt(string[,radix])
parseInt(string[,radix])
將字符串解析成數值; 如果入參非字符串, 則會調用入參的toString
方法轉換為字符串再進行轉換; 如果設置了第二個參數 radix, 則會將字符串按指定的 radix 進制轉換成十進制; 返回數值或者 NaN
JS
var a = '0xf';
console.log(a, 16); // 將 16 進制的 0xf 轉為十進制
// 15
parseFloat(string)
parseFloat(string)
將字符串入參解析成浮點數; 返回浮點數或者 NaN ;
JS
var a = "4.567";
console.log(parseFloat(a)); // 4.567
// 當入參有非法字符, 則只保留合法部分進行轉換
var b = "4.567abcd";
console.log(parseFloat(b)); // 4.567
// 第二個小數點忽略
var c = "1.2.3";
console.log(parseFloat(c)); // 1.2
// 起始為非法字符則直接返回 NaN
var d = "aaa1.2"; // NaN
console.log(parseFloat(d));
isNaN()
判斷某個入參是否是 NaN; 可以利用 NaN 的不等性來進行判斷
JS
var a = NaN;
if (a != a ){
console.log("it is NaN");
}
isFinite()
判斷某個入參是否是 Infinity;
字符串
- 字符串和數組相似, 都支持使用[]運算符來通過指定索引來獲取值;
- length: 可以獲取字符串的長度
- 字符串不能通過[]運算符和索引來修改字符串的值
字符集
js 中使用的字符集為 Unicode 字符集, 所有的字符串都使用 Unicode 表示;
JS
var f\u006F\u006F = 'abc';
console.log(f\u006F\u006F);
base64 轉碼
- 瀏覽器:
- btoa(): 任意值轉為 base64 編碼
- atob(): base64 值解碼
- 非 ASCII 碼(如中文)要轉碼之后再 base64;
encodeURIComponent
decodeURIComponent
- Nodejs
var base64encode = Buffer.from("js").toString("base64");
var base64decode = Buffer.from(base64encode,'base64').toString();
對象
- 對象就是一組鍵值對 key-value 的集合, 是一種無序的復合數據集合
- 對象的每個鍵名又稱為屬性 property; 它的值可以是任意數據類型;
- 如果一個對象的某個屬性的值是函數, 則通常將這個屬性稱為該對象的方法, 可以像函數一樣調用這個方法;
- 屬性可以動態創建, 不必在對象聲明的時候就全部定義;
對象引用
- 如果不同的變量名指向同一個對象, 那么他們都是對這個對象的引用; 也就是說這些變量都指向了同一個內存地址, 修改其中任意一個的值都會影響到其他的變量;
- 如果取消某一個變量對於原對象的引用, 不會影響到其他引用該對象的變量
- 這種引用只局限於對象, 如果兩個變量指向同一個原始類型的值, 這些變量都是對原始類型的值的拷貝, 改變任意變量都不會影響其他變量
屬性查看
Obejct.keys()
可以查看該對象自身的屬性, 繼承來的屬性無法查看
JS
var a = {
hello: function(){
consoel.log("hi");
},
table: [1, 2, 3],
name: "kevin",
age: 21,
married: false
}
console.log(Object.keys(a));
//[ 'hello', 'table', 'name', 'age', 'married' ]
屬性刪除
- delete 用於刪除對象的屬性, 刪除成功后返回 true
- 刪除一個不存在的屬性, delete 不會報錯, 而且返回 true
- 只有一種情況, delete 命令會返回 false; 那就是刪除一條存在的屬性, 但是這條屬性被定義成不能刪除; (定義該屬性不能刪除 defineProperty)
- 只能刪除對象自身的屬性, 繼承來的屬性不能刪除
JS
var a = {
hello: function(){
consoel.log("hi");
},
table: [1, 2, 3],
name: "kevin",
age: 21,
married: false
}
delete a.age, delete a.hello;
console.log(Object.keys(a));
// [ 'table', 'name', 'married' ]
屬性存在判斷
- in 運算符可以用於檢查對象是否包含某個屬性; 檢查的是鍵名, 存在這個屬性返回 true, 不存在則返回 false;
hasOwnProperty()
: 判斷某個屬性是否是該對象自身的屬性;
JSvar a = { hello: function(){ consoel.log("hi"); }, table: [1, 2, 3], name: "kevin", age: 21, married: false}console.log(a.hasOwnProperty('table'));if ("table" in a) { console.log("table is property of a");}
屬性遍歷
- for in 循環可以用於遍歷對象的全部屬性; 不僅可以遍歷對象自身的屬性, 還可以遍歷對象繼承來的屬性;
- 如果只想遍歷對象自身的屬性, 可以配合
hasOwnProperty()
進行篩選
JS
var a = {
hello: function(){
consoel.log("hi");
},
table: [1, 2, 3],
name: "kevin",
age: 21,
married: false
}
for (const aKey in a) {
// 使用 hasOwnProperty 進行篩選
if (a.hasOwnProperty(aKey)) {
console.log(aKey);
}
}
函數
函數聲明
js 中函數聲明有三種方式
- 使用 function 申明
function a(){}
- 函數表達式
var a = function(){}
- Function 構造函數
var a = Function("a","b", "return a+b")
或者var a = new Function("a","b", "return a+b")
這兩種方式效果一樣
函數是一等公民
js 將函數看成是一個值, 與其他數據類型一樣, 凡是可以使用其他數據類型的地方都可以使用函數, 例如:
- 可以將函數賦值給變量或者對象的屬性
- 可以將函數作為參數傳遞給其他函數
- 可以將函數作為其他函數的返回值
函數變量名提升
js 中全局變量名存在變量提升, 函數內部的局部變量也存在變量提升;
JSfunction outer(){ console.log(a); // undefined 說明全局變量存在變量提升 console.log(b); // undefined 說明局部變量存在變量提升 var b = 2;}outer();var a = 1;
js 中函數的聲明也存在變量提升, 可以先調用該方法, 再定義該方法
JSb();function b(){ console.log("b called");}
函數的屬性和方法
- name (屬性): 返回函數的名字
- length(屬性): 返回函數預期傳入的形參數量
- toString()方法: 返回函數的字符串源碼
函數作用域
- 作用域 scope 是指變量存在的范圍
- es5 中 js 只有兩個作用域
- 全局作用域: 變量在整個程序中一直存在, 所有地方都可以讀取到該變量
- 函數作用域: 變量只在函數內部存在
- es6 中新增了塊級作用域
- es5 中 js 只有兩個作用域
- 函數外部聲明的變量就是全局變量, 它可以在函數內部讀取到;
- 在函數內部聲明的變量就是局部變量, 函數外部無法讀取;
- 函數本身的作用域就是其聲明時所在的作用域, 與其運行時的作用域無關;
- 函數內部聲明的函數, 作用域綁定函數內部(閉包)
函數參數省略
js 中函數的參數不是必須的, 允許省略函數的參數;
函數的 length 屬性只和函數聲明時形參的個數有關, 和實際調用時傳入的參數個數無關;
參數傳遞方式
- 函數的參數如果是原始數據類型 (數值, 字符串, 布爾), 參數傳遞使用按值傳遞的方式, 在函數內部修改參數的值不會影響函數外部
- 如果函數的參數是復合數據類型(數組, 對象, 其他函數), 參數的傳遞方式是按址傳遞, 傳入的是引用的地址, 因此在函數內部修改參數, 會影響到原始值;
arguments 對象
- 因為 js 允許函數有不定數目的參數, 所以需要在函數體的內部可以讀取到所有參數, 這就是 arguments 對象的由來
- arguments 對象包含了函數運行時的所有參數, 這是一個類數組對象,
arguments[0]
就是第一個參數; - arguments 對象只能在函數內部使用
- arguments.length 可以獲取函數調用時入參的真正個數
- arguments.callee 屬性可以獲取對應的原函數
- arguments 對象是一個類數組對象, 如果要讓他使用真正的數組方法, 需要將 arguments 轉換成數組:
Array.prototype.slice.call(arguments)
- 新建數組, 遍歷 arguments 將元素 push 到新數組中;
閉包
- 要理解閉包首先要理解 js 的作用域; 前面提到的 js 在 es5 中只有兩種作用域:
- 全局作用域
- 函數作用域
- 在函數的內部可以全局作用域的變量
- js 中特有的鏈式作用域結構, 子級會向上一級一級尋找所有父級的變量, 父級的所有變量對於子級來說都是可見的, 反之不成立;
JSvar a = 1;var b = 2;function f(){ var b = 3; console.log(a,b); function f1(){ var b = 4; console.log(a,b); } f1();}f();// 1 3// 1 4
鏈式作用域查找
子級會優先使用自己的作用域, 如果變量存在則使用, 不存在則會依次向上尋找, 直至全局作用域;
閉包定義
- 閉包可以簡單理解成定義在一個函數內部的函數
- 閉包最大的特點就是它可以記住自己誕生的環境, 本質上, 閉包就是將函數內部和函數外部連接起來的橋梁;
- 閉包最大的用處有兩個
- 可以直接讀取到外層函數內部的變量
- 可以讓這些變量始終保存在內存中, 閉包讓自己誕生的環境一直存在
通過閉包實現簡單的計數器
JSfunction count(){ var count = 0; function f(){ count++; console.log("count", count); } return f;}f = count();f();f();f();
立即調用函數表達式
js 中有三種立即調用函數的方式
var f = function(){}();
(function(){}())
(function(){})()
通常情況下, 只對匿名函數使用這種立即執行的表達式, 這樣有兩個目的:
- 不必為函數命名, 避免污染全局環境
- 立即調用函數的內部會形成單獨的作用域, 可以封裝一些外部無法讀取的私有變量
eval 命令
- eval 可以接受一個字符串, 並將字符串當做代碼執行
- eval 沒有自己的作用域, 都是使用當前運行的作用域, 所以 eval 會修改當前作用域下的變量的值
- eval 的本質是在當前作用域中, 注入代碼, 經常用於混淆和反爬
eval 別名調用
eval 的別名調用在 nodejs 下無法跑通, 需要在瀏覽器下運行;
需要注意, 在 eval 通過別名調用的時候, 作用域永遠是全局作用域;
JSvar a = 1;var e = eval;(function(){ var a = 2; e("console.log(a);"); // eval 在別名調用的時候使用全局作用域}())
數組
- 數組 Array 是按次序排列的一組值
- 每個值的位置都有對應的索引
- 數組使用 [ ] 來表示
- 任何類型的數據都可以放入數組中
- 本質上, 數組是特殊的對象, typeof 查看數組的類型返回的是 object
- Object.keys() 可以返回數組的鍵名(索引)
數組的屬性
- length: 表示數組的元素個數, 這個屬性是可寫的, 可以直接修改數組的 length 屬性, 來實現清空數組或刪除數組中元素的效果
- 數組本質上是特殊的對象, 支持使用點操作符對數組添加屬性;
JSvar a = [1, 1.1, true, {}, [], "hello", null, undefined];console.log("a.length", a.length);a.name = "add a.name property";for (const aKey in a) { console.log(aKey, a[aKey]);}console.log("Object.keys(a)", Object.keys(a));a.length = 0;console.log("a", a);console.log("a['name']", a['name']);console.log("a.name", a.name);console.log("a[0]", a[0]);
數組循環
- for in 循環
- for 循環
- while 循環
- forEach : 只有數組才有該方法; 該方法接受一個回調函數, 回調函數入參為 value 和 key;
JSvar a = [1, 1.1, true, {}, [], "hello", null, undefined];// for infor (const aKey in a) { console.log("aKey:", aKey, "value:", a[aKey]);}console.log("-------------------------------")// forfor (var i = 0; i <= a.length; i++) { console.log("index:", i, "value:", a[i]);}console.log("-------------------------------")// whilevar index = 0;while (index <= a.length) { console.log("index:", index, "value:", a[index]); index++;}console.log("-------------------------------")// forEacha.forEach(function (value,key) { console.log("key:", key, "value:", value);})
數組空值
js 中的數組支持空值, 出現空值時會占用該索引位, 但是遍歷的時候不會遍歷該索引的值
JSvar a = [1, 2, 3, , 5];a.forEach(function (value,key) { console.log("key", key, "value", value);})// key 0 value 1// key 1 value 2// key 2 value 3// key 4 value 5
類數組對象
- 如果一個對象的所有鍵名都是正整數或者 0, 且有 length 屬性, 那么這個對象就是類數組對象
- 典型的類數組對象有 arguments 對象, 字符串, 以及大部分的 dom 元素集
- 數組的 slice 方法可以將類似數組的對象變成真正的數組
var arr = Array.prototype.slice.call(arrayLike);
- 除了將類數組對象轉成真正的數組, 還可以使用 call() 將數組的方法直接放到類數組對象上使用
Array.prototype.forEach.call(arrayLike, function(){});
數據類型轉換
自動數據類型轉換
其他類型轉字符串
當+
加號作為操作符, 且操作數中含有字符串時, 會自動將另一個操作數轉為字符串;
規則如下:
- 字符串+基礎數據類型: 會直接將基礎數據類型轉為和字面量相同的字符串
- 字符串+復合數據類型: 復合數據類型會先調用
valueOf()
方法, 如果該方法返回基礎數據類型則將其轉為字符串, 如果返回的是復合數據類型, 則調用toString()
方法, 如果返回的是基礎數據類型則將其轉為字符串, 如果不是則報錯;
JS
// 基礎類型
// 自動數據類型轉換
// + 字符串
// 1. 字符串 + 基礎數據類型: 會直接將基礎數據類型轉為和字面量相同的字符串
var tmp1 = "" + 3;
console.log("tmp1", tmp1); // tmp1 "3"
var tmp2 = "" + true;
console.log("tmp2", tmp2); // tmp2 "true"
var tmp3 = "" + undefined;
console.log("tmp3", tmp3); // tmp3 "undefined"
var tmp4 = "" + null;
console.log("tmp4", tmp4); // tmp4 "null"
// 字符串+復合數據類型: 復合數據類型會先調用 valueOf 方法, 如果此方法返回的是引用類型, 則再調用 toString()方法, 最后將返回值轉為字符串類型
var tmp5 = [1, 2, 3] + "";
console.log("tmp5", tmp5); // tmp5 1,2,3
var tmp6 = {} + "";
console.log("tmp6", tmp6); // tmp6 [object Object]
// 重寫 toString 方法
var o = {
toString: function () {
return 1;
}
}
var tmp7 = o + "";
console.log("tmp7", tmp7) // tmp7 "1"
// 重寫 valueOf 方法
o.valueOf = function () {
return 2;
}
var tmp8 = "" + o;
console.log("tmp8", tmp8); // tmp8 2
var a = {
valueOf: function () {
return {}
},
toString: function () {
return "toString"
}
};
console.log("" + a); // toString
其他類型轉布爾值
數值轉布爾值
數值在邏輯判斷條件下會自動轉成布爾值; +-0 和 NaN 為 false, 其他數值都是 true;
JS
if (0) {
console.log("0 is true");
}else{
console.log("0 is false");
}
if (NaN) {
console.log("NaN is true");
}else{
console.log("NaN is false");
}
if (-0) {
console.log("-0 is true");
} else {
console.log("-0 is false")
}
// 0 is false
// NaN is false
// -0 is false
字符串轉布爾值
空字符串””或者’’為 false, 其他都是 true
undefined 和 null 轉布爾值
undefined 和 null 轉為布爾值都是 false
對象轉布爾值
只有對象為 null 或者 undefined 時, 轉為布爾值才是 false; 其他情況下(包括空對象{}和空數組[])轉為布爾值都是 true;
JS
var o = {};
if(o){
console.log("{} is true");
}else{
console.log("[] is true");
}
// {} is true ; 空數組[] 同理
其他類型轉為數值
一元操作符+
和-
都會觸發其他類型轉為數值;
數學運算符操作的兩個操作數都不是字符串時, 也會觸發其他類型轉為數值的操作;
轉化規則如下:
- 字符串轉數值: 空字符串轉為 0, 非空字符串轉為對應的數字; 不合法則返回 NaN;
- 布爾值轉數值: true 為 1, false 為 0;
- null 轉為數值為 0
- undefined 轉數值為 NaN
- 對象轉數值會先調用
valueOf()
方法, 如果其返回值是基礎數據類型, 則將其轉為數值返回, 如果是復合數據類型, 則會調用toString()
方法, 再將其轉為數值, 如果不合法則返回 NaN;
強制類型轉換
- Number(): 將其他類型轉為數值; 將對象轉為數值時, 會先調用對象的
valueOf()
方法, 不滿足條件再調用toString()
方法 - String(): 將其他類型轉為字符串; 將對象轉為字符串時, 會先調用對象的
toString()
方法, 不滿足條件時再調用對象的valueOf()
方法; - Boolean(): 將其他類型轉為布爾值
異常處理
Error 對象
Error 對象通常包含常用的三個屬性, 且必須包含 message 屬性;
- name: 異常名
- message: 異常提示信息
- stack: 異常調用棧信息
JS
var e = new Error("自定義異常觸發");
e.name = "自定義異常名稱";
console.log(e.name);
console.log(e.message);
console.log(e.stack);
/*
自定義異常名稱
自定義異常觸發
自定義異常名稱: 自定義異常觸發
at Object.<anonymous> (/Users/zhangyang/codes/antiJs/js_learn/_03_基本數據類型(下)/tmp.js:201:9)
at Module._compile (internal/modules/cjs/loader.js:1085:14)
at Object.Module._extensions..js (internal/modules/cjs/loader.js:1114:10)
at Module.load (internal/modules/cjs/loader.js:950:32)
at Function.Module._load (internal/modules/cjs/loader.js:790:14)
at Function.executeUserEntryPoint [as runMain] (internal/modules/run_main.js:76:12)
at internal/main/run_main_module.js:17:47
*/
try catch finally
手動拋出異常並捕獲
JS
try {
var e = new Error("message 自定義異常");
e.name = "自定義異常名稱";
throw e;
} catch (e) {
console.log("e.name", e.name, "e.message", e.message);
console.log("e.stack", e.stack);
}
throw
throw
語句用來拋出用戶定義的異常, 當前函數的執行將被停止, throw
之后的代碼不會被執行; 並且代碼將會進入調用棧中第一個catch
塊中; 如果沒有catch
塊, 程序將會終止;
JStry{ console.log("before throw error"); throw new Error("throw error"); console.log("after throw error");}catch(e){ console.log("catch error", e.message);}// before throw error// catch error throw error
try/catch/finally
try 的三種聲明形式:
- try…catch
- try…finally
- try…catch…finally
try/catch 主要用於捕獲異常, try/catch 語句包含一個 try 塊, 和至少一個 catch 塊或者一個 finally 塊;
-
try: try 塊中放入可能會產生異常的語句或者函數
-
catch: catch 塊中包含要執行的語句, 當 try 塊中拋出異常時, catch 塊會捕獲這個異常, 並執行 catch 塊中的代碼; 如果 try 塊中沒有異常拋出, 則 catch 塊會被跳過;
-
finally: finally 塊在 try 塊和 catch 塊之后執行, 無論是否有異常產生, finally 塊都會執行; 當 finally 塊中有異常產生時, 會覆蓋掉 try 塊中的異常信息;
JS try { try { throw new Error("error1"); } finally { throw new Error("error2"); } } catch (e) { console.log("e.message", e.message); } // e.message error2
-
如果 finally 塊中返回一個值, 那么這個值將會作為整個 try/catch/finally 的返回值, 無論在 try 和 catch 有沒有 return, 都是返回 finally 中的值;
JS function test() { try { throw new Error("error1"); return 1 }catch (e) { throw new Error("error2"); return 2; }finally { return 3; } } console.log("test()", test()); // test() 3
對象詳解和 hook
Object 靜態方法和實例方法
- js 中所有其他對象都是繼承自 Object ; 即所有對象都是 Object 的實例;
- Object 的原生方法分為兩類: Object 本身的方法和 Object 的實例方法
- Object 本身的方法: 直接定義在 Object 上的方法, 相當於 java 的靜態方法
- Object 的實例方法: 定義在 Object 的原型對象 Object.prototype 上的方法; 相當於 java 中的動態方法, 可以被 Object 對象直接使用
Object 的靜態方法
所謂靜態方法, 就是設置在 Object 類上的方法;
- Object.keys(): 遍歷對象的屬性, 返回可枚舉的屬性名; 這個方法只能查看對象自身的屬性, 繼承來的屬性無法查看
- Object.getOwnpeopertyNames(): 遍歷對象的屬性, 可以返回不可枚舉的屬性名;
- Object.getOwnPropertyDescriptors(): 獲取對象所有屬性的描述對象;
- Object.defineProperty(): 通過描述對象, 定義某個屬性, 可以重寫 get 和 set 方法來進行 hook
- Object.defineProperties(): 通過描述對象, 定義多個屬性, 可以重寫 set 和 get 方法來進行 hook
- Object.getPrototypeOf(): 返回參數對象的原型, 可以用於環境監測;
- Object.setPrototypeOf(): 為參數對象設置原型, 返回該參數對象; 它接收兩個參數, 第一個是現有對象, 第二個是原型對象;
JS
function Test(a, b, c) {
this.a = a;
this.b = b;
this.c = c;
};
var test = new Test(1, 2, 3);
Object.defineProperty(test,"d",{
configurable: false, // 是否可以刪除 默認 false 不可刪除
enumerable: false, // 是否是可迭代字段
value: "4",
writable: true,
// 使用Object.defineProperty() 定義對象屬性時,如已設置 set 或 get, 就不能設置 writable 和 value 中的任何一個了
// get: function(){
// return this.d;
// },
// set: function (value) {
// this.d = value;
// return this.d;
// }
})
// 遍歷對象的屬性, 返回可枚舉的屬性名, 僅可查看變量本身的屬性, 不能查看繼承來的屬性
console.log(Object.keys(test)); // [ 'a', 'b', 'c' ]
// 遍歷對象的屬性, 可以返回不可枚舉的類型
console.log(Object.getOwnPropertyNames(test)); // [ 'a', 'b', 'c', 'd' ]
// 獲取對象的所有屬性的描述對象
console.log(Object.getOwnPropertyDescriptors(test));
/*
* {
a: { value: 1, writable: true, enumerable: true, configurable: true },
b: { value: 2, writable: true, enumerable: true, configurable: true },
c: { value: 3, writable: true, enumerable: true, configurable: true },
d: {
value: '4',
writable: true,
enumerable: false,
configurable: false
}
}
*/
// 設置對象的原型
Object.setPrototypeOf(test, Test);
// 獲取對象的原型對象, 可以用於原型檢測
console.log(Object.getPrototypeOf(test));
Object 的實例方法
- Object.prototype.valueOf(): 返回當前對象對應的值
- Object.prototype.toString(): 返回當前對象對應的字符串形式
- 因為實例對象可能自定義了 toString 方法, 覆蓋了 Object.prototype.toString 方法; 所以為了得到類型字符串, 最好使用 Object.prototype.toString 方法, 通過 call 的方式可以在任意對象上調用這個方法, 幫助判斷;
- Object.prototype.hasOwnProperty(): 判斷某個屬性是否是當前對象自身的屬性, 可以區分某個屬性是對象自身的還是繼承來的;
- Object.prototype.isPrototypeOf(): 判斷當前對象是否是另一個對象的原型;
- Object.prototype.propertyIsEnumerable(): 判斷某個屬性是否是可枚舉的
構造函數
- 典型的面向對象的變成語言都有類的概念, 所謂類就是對象的模板, 對象就是類的實例;
- js 不是基於類的, 而是基於構造函數和原型鏈;
- 構造函數就是一個簡單的函數, 當函數和 new 關鍵字配合使用, 就可以返回一個實例對象;
JS
var init = function (a, b, c) {
this.a = a;
this.b = b;
this.c = c;
};
var initObj = new init(1, 2, 3);
console.log("initObj", initObj);
創建對象的方式
- 直接賦值,
var obj = {};
或者var obj=Object({});
(強制轉為對象) - new 關鍵字創建:
var obj = new Object();
- Object.create(arg, property): 以傳入的對象的構造函數作為模板, 可以生成實例對象; 一般用於在拿不到構造函數時, 直接通過現有的實例作為參數, 來構造新的實例;
這三種創建對象方式的區別:
- 字面量和 new 關鍵字創建的對象是 Object 的實例, 原型指向 Object.prototype; 繼承內置對象 Object
- Object.create(arg, property) 創建的對象的原型取決入參 arg, arg 為 null 時, 創建的是空對象; arg 為指定對象時, 則新對象的原型指向指定對象, 繼承指定對象;
JSvar init = function (a, b, c) { this.a = a; this.b = b; this.c = c;};var obj = new init(1, 2, 3);var obj2 = Object.create(obj,{ "p": { "value": "p1" }})console.log("obj2", obj2); // obj2 init {}console.log("obj2.p", obj2.p);// obj2.p p1
原型對象 Object.prototype
- js 的繼承機制的設計思想是: 原型對象上的所有屬性和方法, 都能被實例對象共享; 也就是說, 如果屬性和方法定義在原型對象上, 那么所有的實例都可以共享這些屬性和方法;
- js 規定, 每個對象都有一個
__proto__
指向原型對象 - js 規定, 每個函數除了有
__proto__
之外, 還有一個prototype
屬性;__proto__
指向函數原型 Function.prototype,prototype
指向一個原型對象; - 原型對象上定義的屬性和方法不是實例對象自身的, 所以修改原型對象上的屬性和方法, 這種變動會影響到所有實例屬性;
- 當實例本身沒有某個屬性或者方法的時候, 它會到原型對象上尋找該屬性和方法, 類似 java 的雙親委派機制;
prototype
和__proto__
區別
JS
function test(){
console.log("test");
}
console.log(test.prototype.__proto__.__proto__); // null
console.log(test.__proto__.__proto__.__proto__); // null
JS
function DTA(boss, employee) {
this.boss = boss;
this.employee = employee;
}
DTA.prototype.room = {
"roomNum": 1
}
DTA.prototype.name = "kevin";
DTA.prototype.dowork = function () {
return this.boss +" " + this.employee +" "+ "do work now";
}
var dta = new DTA("kevin", "rub");
var dta2 = new DTA("kevin2", "rub2");
// 修改 DTA.prototype 原型上的屬性, 會影響到所有的實例
DTA.prototype.name = "change";
console.log(dta2.__proto__ === DTA.prototype); // true
console.log(dta2.name); // change
dta2.room.roomNum = 2; // 從其他對象上修改原型對象的屬性, 也會影響其他對象
dta2.__proto__.name = "change2";
console.log(dta.name); // change2
console.log("dta.boss", dta.boss);
console.log("dta.employee", dta.employee);
console.log("dta.dowork()", dta.dowork());
console.log("dta.room.roomNum", dta.room.roomNum); // 2
原型鏈
- js 規定, 所有對象都有自己的原型對象 prototype;
- 任何一個對象都可以充當其他對象的原型;
- 由於原型也是一個對象, 所以它也有自己的原型對象, 最終形成一個原型鏈, 對象 –> 原型 –> 另一個原型…
constructor
- 原型對象 property 有一個 constructor 屬性, 這個屬性指向 prototype 對象所在的構造函數
- constructor 屬性的作用是: 可以得知某一個實例對象, 到底是由哪一個構造函數產生的
- 有了 constructor 屬性, 就可以找到構造函數, 可以通過一個實例對象來新建另一個實例
- constructor 還可以用來設置一些環境監測:
document.constructor === HTMLDocument
navigator.constructor === Navigator
- 經常和
toString()
配合進行檢測, 這樣在 node 中運行不會報錯,更難發現;navigator.constructor.toString() === "function Navigator() { [native code] }"
JS
// 函數的 .prototype.constructor 等於函數本身
function test() {};
console.log(test.prototype.constructor === test);
// 環境監測 document.constructor === HTMLDocument
function HTMLDocument() {};
var document = new HTMLDocument();
console.log(document.constructor === HTMLDocument);
var document2 = new document.__proto__.constructor();
console.log(document2.constructor === HTMLDocument);
// 環境監測 navigator.constructor === Navigator
function Navigator() {};
var navigator = new Navigator();
console.log(navigator.constructor === Navigator);
var navigator2 = new navigator.constructor();
console.log(navigator2.constructor === Navigator); // 不帶__proto__也可以, 默認會自動去原型鏈上查找
// 環境監測 配合 toString()進行檢測,node 下不會報錯, 更難發現
// navigator.constructor.toString() === "function Navigator() { [native code] }"
function Navigator() {}
Navigator.toString = function () { // Navigator.prototype.constructor.toString
return "function Navigator() { [native code] }"
}
var navigator = new Navigator();
console.log(navigator.constructor.toString());
console.log(navigator.constructor.toString() === "function Navigator() { [native code] }");
hook cookie
簡易版 hook cookie
JS
var cookie = document.cookie;
Object.defineProperty(document, 'cookie', {
get: function(){
console.log("getter:" + cookie);
return cookie;
},
set: function(value){
console.log("setter:" + cookie);
if (value.indexOf("targetCookie") > -1){
debugger;
}
cookie = value;
}
})
油猴 hook cookie
JS
// ==UserScript==
// @name Hook Cookie
// @namespace http://tampermonkey.net/
// @version 0.1
// @description try to take over the world!
// @author You
// @include *
// @grant none
// @run-at document-start
// ==/UserScript==
(function () {
'use strict';
console.log("hook cookies start ...");
var cookie_cache = document.cookie;
Object.defineProperty(document, 'cookie', {
get: function () {
console.log("get cookie:" + cookie_cache);
return cookie_cache;
},
set: function (val) {
console.log('Setting cookie', val);
// 填寫cookie名
if (val.indexOf('w_token') != -1) {
debugger;
}
var cookie = val.split(";")[0];
var ncookie = cookie.split("=");
var flag = false;
var cache = cookie_cache.split("; ");
cache = cache.map(function (a) {
if (a.split("=")[0] === ncookie[0]) {
flag = true;
return cookie;
}
return a;
})
cookie_cache = cache.join("; ");
if (!flag) {
cookie_cache += cookie + "; ";
}
return cookie_cache;
}
});
})();
object copy
對象拷貝需要做到兩件事:
- 確保拷貝后的對象, 與原對象具有相同的原型
- 確保拷貝后的對象, 與原對象具有同樣的實例屬性
JS
function copyObj(obj){
return Object.create(
Object.getPrototypeOf(obj),
Object.getOwnPropertyDescriptors(obj)
)
}
function kevin(a){
this.a = a;
}
var kevinObj = new kevin(1);
var kevinObj2 = copyObj(kevinObj);
Array 對象
- isArray(): 返回一個布爾值, 表示參數是否是一個數組, 可以彌補 typeof 運算符的不足
- push(), pop()
- push: 數組的末端添加一個或者多個元素, 並返回添加新元素后的數組長度
- pop: 刪除數組的最后一個元素, 並返回該元素
- shift(), unshift()
- shift: 用於刪除數組的第一個元素, 並返回該元素
- unshift: 用於在數組的第一個位置添加元素, 並返回添加新元素后數組長度
- join(): 以指定的參數作為分隔符, 將數組中所有成員連接成一個字符串並返回, 如果不指定分隔符, 則默認使用逗號分隔
- concat(): 用於多個數組合並, 將新數組的成員添加到原數組成員的后面, 然后返回一個新數組, 原數組保持不變
- reverse(): 用於顛倒排列數組元素, 返回改變后的數組, 該方法將會改變原數組
- slice(): 提取目標數組的一部分, 返回一個新數組, 原數組保持不變
- splice(): 刪除原數組的一部分, 並可以在刪除的位置添加新的數組成員, 返回值是被刪除的元素, 該方法會改變原數組;
- sort(): 對數組成員進行排序, 默認是按照字典順序排序, 排序后原數組將改變
- map(): 將數組的所有成員依次傳入回調函數中, 然后把回調函數每一次執行結果組成一個新數組返回
- forEach(): 與 map()類似, 無返回值, 只是操作原數組
- filter(): 用於過濾數組成員, 滿足條件的成員組成一個新數組返回;
- indexOf(): 返回給定元素在數組中第一次出現的位置, 如果沒有出現則返回-1;
包裝對象
三種原始類型的值在一定條件下也會自動轉為對象, Boolean(), String()和 Number();
當使用 new 關鍵字的時候就會創建一個新的包裝對象, 當不使用 new 關鍵字時就是強制類型轉換;
Math
- Math.abs(): 返回入參的絕對值
- Math.ceil(): 返回入參向上取整之后的值
- Math.floor(): 返回入參向下取整之后的值
- Math.max(): 返回數組中的最大值
- Math.min(): 返回數組中的最小值
- Math.random(): 返回 0-1 的偽隨機值
- Math.round(): 返回入參四舍五入后的整數
Date
創建一個新的 Date 對象的唯一方式是通過 new 操作符;
JSnew Date();new Date(value);new Date(dateString);new Date(year, monthIndex [, day [, hours [, minutes [, seconds [, milliseconds]]]]]);
- Date.now(): 返回當前時間戳
- Date.parse(): 解析時間字符串, 返回時間戳
控制台 API
- inspect(obj): 打開相關面板並選中相應的元素, 展示它的細節
- getEventListeners(): 返回一個對象, 該對象的鍵為事件, 值為數組; 數組的成員為該事件的回調函數;
- keys(), values(): 返回一個數組, 包含 object 的所有鍵名和值
- monitorEvents(object,[,events]): 監聽特定對象上發生的特定事件; 事件發生時, 會返回一個 Event 對象; 包含該事件的相關信息; unmonitorEvents 方法用來停止監聽;
- monitorEvents: 允許監聽同一大類的事件, 所有事件可以分為四大類:
- mouse: mousedown, mouseup, click, dblclick, mousemove, mouseover, mouseout, mousewheel
- key: keydown, keyup, keypress, textInput
- touch: touchstart, touchmove, touchend, touchcancel
- control: resize, scroll, zoom, focus, blur, select, change, submit, reset
- monitorEvents: 允許監聽同一大類的事件, 所有事件可以分為四大類:
- copy(obj): 復制特定對象到剪貼板
- debugger: 斷點
this 關鍵字
- this 就是屬性或者方法當前所在的對象
- this 主要使用場景:
- 全局環境使用 this, 它指的就是頂層對象 window
- 構造函數中使用 this, 它指的就是實例對象
- 如果對象的方法包含 this, this 的指向就是方法運行時所在的對象; 該方法賦值給另一個對象, 就會改變 this 的指向;
綁定 this 的方法
- Function.prototype.call(thisValue, arg1, arg2 …) : 函數實例的 call 方法, 可以指定函數內部的 this 的指向(即函數執行時所在的作用域); 然后在指定作用域中調用該函數
- Function.prototype.apply(thisValue, [arg1, arg2. ..]): apply 和 call 方法類似, 都是可以改變 this 的指向; 唯一區別就是它接受一個數組作為函數執行時的參數;
- Function.prototype.bind(thisValue, arg1, arg2. ..): 將函數體內的 this 綁定到某個對象, 然后返回一個新函數;
JSvar d = new Date();console.log(d.getTime());var getTime = d.getTime;getTime(); // 直接調用 error, this is not a Date objectvar d_getTime = getTime.bind(d, []);console.log(d_getTime());var d = new Date();var getTime = d.getTime;var result = getTime.apply(d, []);console.log(result);var d = new Date();var getTime = d.getTime;var result = getTime.call(d);console.log(result);
es6 部分新特性
es6 是 js 語言的下一代標准, 在 2015 年 6 月正式發布; 它的目標是讓 js 語言可以用來編寫復雜的大型應用程序; es 和 js 的關系是: es 是 js 的規格, js 是 es 的實現;
塊級作用域
es6 中提出了兩個新的變量聲明命令: let 和 const; let 可以完全取代 var; var 命令存在變量提升的效果, let 命令則不存在這個問題; 所以盡量使用 let, 減少 var 的使用;
在全局環境中使用 var 聲明變量, 會直接將變量定義到全局環境中, 在全局環境中使用 let 和 const 是不會直接將變量定義到全局環境上的;
JS
// 在瀏覽器環境下運行
var a = 10;
let b = 20;
const c = 30;
console.log(window.a); // 10
console.log(window.b); // undefined
console.log(window.c); // undefined
在 es5 中, 只有函數作用域和全局作用域; 在 es6 中新增了塊級作用域; 比如典型的 for 循環; 使用 var 會導致變量泄漏到全局環境中, 使用 let 就不會存在該問題;
JS
for (var i = 0; i < 10; i++){
console.log(i);
}
console.log(i);
for (let a = 0; a < 10; a++){
console.log(a);
}
console.log(a); // undefined
屬性簡潔表達
JS
// es5
// var team = {company: "dta"}
// es6
const company = "dta"; // 先定義屬性
const team = {company}; // 再定義對象
console.log(team);
// es6
let boss = "Dta boss";
let Dta = {
[boss]: "roysue"
}
console.log("Dta['Dta boss']", Dta['Dta boss']);
對象方法簡潔表達
JS// es5var team = { dowork: function(){ return "work!"; }}// es6var team = { dowork(){ return "work"; }}
字符串模板
在 es6 中新增了動態的字符串模板, 使用反引號表示, 使用${}進行填充;
JS
var name = "kevin";
console.log(`hi, ${name}`);
解構賦值
數組解構
JS
const arr = [1,2,3,4];
const [first, second] = arr; // first = arr[0], second = arr[1];
console.log(first, second); // 1,2
對象解構
JS
var obj = {
firstName: "chen",
lastName: "guilin"
}
function getFullName(obj){
const {firstName, lastName} = obj;
console.log(firstName, lastName);
}
getFullName(obj); // chen guilin
快速復制對象
JS
let Dta = {
name: "kevin",
age: 29,
dowork(){
return "work";
}
}
let newDta = {...Dta}; // 用於取出參數對象的所有可遍歷屬性, 拷貝到當前對象中
console.log("newDta", newDta);
// 其實是調用了 Object.assign()方法
let roysue = {};
let r1ysue = Object.assign({}, roysue);
// 等同於
let r2ysue = {...roysue};
// 其他快速克隆對象的方法
var obj = {};
const clone1 = {
__proto__: Object.getPrototypeOf(obj),
...obj
};
const clone2 = Object.assign(
Object.create(Object.getPrototypeOf(obj),
obj
);
const clone3 = Object.create(
Object.getPrototypeOf(obj),
Object.getOwnPropertyDescriptors(obj)
)
數組創建對象
JS
function processInput(){
let [a,b,c,d] = [1,2,3,4];
return {a,b,c,d}
}
console.log(processInput()); // {a:1, b:2, c:3, d:4};
箭頭函數
立即執行函數可以寫成箭頭函數的形式:
JS
(() => {
console.log("welcome to the DTA");
})();
在那些使用匿名函數當做參數的場景下, 盡量使用箭頭函數作為替代, 因為使用箭頭函數更加簡潔, 而且箭頭函數在定義函數的時候就綁定了 this; 不需要在執行的時候考慮 this 的問題;
JS
let a = [1, 2, 3].map((x) => {
return x * x;
});
console.log("a", a); // [1,4,9]
function Test(){
this.s1 = 0;
this.s2 = 0;
setInterval(()=>{this.s1++;}, 1000);
setInterval(function () {
this.s2++;
}, 1000);
}
var test = new Test();
setInterval(() => {
console.log(test.s1);
}, 3000); // 2 5 ...
setInterval(() => {
console.log(test.s2);
}, 3000); // 0 0 ...
模塊的引入
JS
// commonJs 模塊引入 node 還是使用這個
let {stat, exists, readfile} = require("fs");
// 等同於
let _fs = require("fs");
let stat = _fs.stat;
let exists = _fs.exists;
let readfile = _fs.readfile;
// es6 模塊
import {stat, exists, readfile} from 'fs';
在 js 中引入其他文件中的函數或者對象
JS
// 當前目錄下新建 testImport.js 文件
var dta = {
name: "dta",
version : '1.0.0',
boss: "roysue",
dowork: function (name) {
console.log(`${this.name} is working`);
}
}
module.exports = dta; // 將當前對象導出
// 在其他文件中引入
let dta = require("./testImport");
console.log(dta);
dta.dowork();
// 在當前目錄下新建 testImport.js 文件
function add(x,y){
return x+y;
}
exports.add = add;
// 在其他文件中引入
let {add} = require("./testImport");
var result = add(1,2);
console.log("result",result);
鏈式判斷運算符
如果讀取對象內部的某個屬性, 往往需要判斷一下該對象是否存在這屬性; 比如 message.body.user.firstName
, 安全的寫法為:
JSconst firstName = (message && message.body && message.body.user && message.body.user.firstName) || 'default';
這樣寫相對比較麻煩, 在 es6 中可以使用鏈式判斷運算符進行判斷;
JS
let Dta = {
name: "kevin",
age: 29,
dowork(){
return "work";
},
room: {
r1: {
l: "l1"
}
}
}
console.log(Dta.room.r1.l); // l1
console.log(Dta?.room?.r1?.l?.f || "default"); // default
null 判斷運算符
在讀取對象屬性的時候, 如果該屬性是 null 或者 undefined, 可以通過 || 設置默認值const roysue = Dta.Boss.roysue || 'roysue'
但是這樣寫的話在, 當Dta.Boss.roysue
的值為空字符串或者 false 或者 0 的時候也會觸發默認值;
為了避免這個錯誤, es6 中新增了 null 判斷運算符 ??; 只有當運算符的左側值為 null 或者 undefined 時, 才會返回右側的默認值
JS
const roysue = Dta.Boss.roysue ?? 'roysue';
對象的新增方法
- Object.is(a,b) 用來比較兩個值是否是嚴格相等; 與嚴格比較符 === 基本一致;
- Object.assign(target, source1, source2): 用於對象的合並, 將原對象 source 的所有可枚舉屬性, 全部復制到 target 對象中;
- 注意點:
- 淺拷貝: 如果原對象某個屬性是對象, 那么目標對象拷貝得到的屬性時這個對象屬性的引用;
- 同名屬性替換: 一旦遇到同名的屬性, Object.assign()處理的方法是替換, 而不是添加
- 常見用途:
- 為對象添加屬性
- 為對象添加方法
- 克隆對象: 只能克隆對象自身的值, 不能克隆它繼承的值
- 合並多個對象
- 注意點:
- Object.getOwnPropertyDescriptors(): 返回指定對象的所有自身屬性的描述對象(非繼承來的屬性)
__proto__
: 該屬性可以用來讀取和設置當前對象的原型對象(prototype);
類的表示
在 es5 中, 類的表示通常是:
JS
function Dta(boss, employee) {
this.boss = boss;
this.employee = employee;
}
Dta.prototype.toString = function () {
return '(' + this.boss + ',' + this.employee + ')';
}
var dta = new Dta('kevin', 'rub');
console.log(dta.toString());
在 es6 中, 為了讓類的概念更接近 java 和 c++的類, 新增了 class 關鍵字;
JS
class Dta{
constructor(boss, employee) {
this.boss = boss;
this.employee = employee;
}
toString(){
return '(' + this.boss + ',' + this.employee + ')';
}
}
var dta = new Dta("kevin", "rub");
console.log(dta.toString());
Proxy 和 Reflect
- Proxy 可以理解成, 在目標對象之前架設一層攔截, 外界對該對象的訪問, 都必須先通過這一層攔截; 因此提供了一種機制, 可以對外界的訪問進行過濾和改寫; Proxy 這個詞的原意是代理, 在這里可以理解成 Proxy 為一個代理器;
- Proxy 支持的 13 中攔截操作
- get(target, propertyKey, receiver): 攔截對象屬性的讀取, 比如: proxy.foo 或者 proxy[‘foo’];
- set(target, propertyKey, propertyValue, receiver): 攔截對象屬性的設置, 比如 proxy.foo=1或者proxy[‘foo’]=1; 返回布爾值
- has(target, propertyKey): 攔截 propertyKey in proxy 的操作, 返回布爾值;
- deleteProperty(target, propertyKey): 攔截 delete proxy[propertyKey]的操作, 返回布爾值;
- ownKeys(target): 攔截 Object.getOwnPropertyNames(), Obejct.getOwnPropertySymbols(),Obejct.keys(),for..in 循環; 返回數組, 該方法返回目標對象所有自身的屬性的屬性名; 而 Object.keys()僅返回目標對象自身可遍歷的屬性名;
- getOwnPropertyDescriptor(target, propertyKey): 攔截 Object.getOwnPropertyDescriptor(); 返回屬性的描述對象
- defineProperty(target, propertyKey, propertyDesc): 攔截 Object.defineProperty()和 Object.defineProperties(), 返回布爾值;
- preventExtensions(target): 攔截 Object.preventExtensions()返回布爾值
- getPrototypeOf(target): 攔截 Object.getPrototypeOf(), 返回一個對象;
- isExtensible(target): 攔截 Object.isExtensible(), 返回布爾值;
- setPrototypeOf(target, proto): 攔截 Object.setPrototypeOf(), 返回布爾值;
- apply(target, object, args): 攔截 proxy 實例作為函數調用的操作; 比如 proxy(..args), proxy.call(obj, args), proxy.apply(…) ;
- construct: 攔截 proxy 實例作為構造函數調用的操作, 比如 new proxy(…args);
- Proxy 支持的 13 中攔截操作
- Reflect 對象和 Proxy 對象一樣, 是 es6 為了操作對象而提供的新 API; 大部分與 Object 對象的同名方法的作用都是相同的; 而且 Reflect 與 Proxy 對象的方法是一一對應的;
- Reflect 設計的目的:
- 將 Object 對象的一些明顯屬於語言內部的方法(Object.defineProperty)放在 Reflect 對象上; 現階段某些方法同時在 Object 和 Reflect 對象上部署, 未來新的方法只能部署在 Reflect 對象上; 也就是說 Reflect 可以拿到語言內部的方法
- 修改某些 Object 方法的返回結果, 讓其變得更合理
- 讓 Object 操作都變成函數行為
- Reflect 對象的方法和 Proxy 對象的方法一一對應;
- Reflect 靜態方法
- Reflect.get(target, propertyName, receiver): 查找並返回 target 對象的 name 屬性, receiver 綁定 this;
- Reflect.set(target, propertyName, propertyValue, receiver): 設置 target 對象的 propertyName 屬性等於 propertyValue
- Reflect.has(obj, propertyName): 判斷 propertyName 是否存在在對象中, 相當於 in
- Reflect.deleteProperty(obj, propertyName): 方法等同於 delete obj[propertyName]; 用於刪除對象的屬性
- Reflect.construct(target, args): 等同於 new target(args); 調用構造函數的方法
- Reflect.getPrototypeOf(obj): 讀取對象的
__proto__
屬性; 對應 Object.getPrototypeOf() - Reflect.setPrototypeOf(obj, newProto): 設置目標對象的原型, 對應 Object.setPrototypeOf();
- Reflect.apply(func, thisArg, args): 等同於 Function.prototype.apply.call(func, thisArg, args)
- Reflect.defineProperty(target, propertyKey, attributes): 等同於 Obejct.defineProperty();
- Reflect.getOwnPropertyDescriptor(target, propertyKey): 等同於 Object.getOwnPropertyDescriptor();
- Reflect.isExtensible(target): 對應 Object.isExtensible()表示當前對象是否可擴展;
- Reflect.preventExtensions(target): 對用 Object.preventExtensions()讓一個對象變為不可擴展
- Reflect.ownKeys(target): 返回對象的所有屬性, 可以返回 Symbol 對象;
- Reflect 設計的目的:
代理器案例
JS
// Proxy 和 Reflect
let obj = new Proxy({}, {
get(target, p, receiver) {
console.log(`get ${p}`);
return Reflect.get(target, p, receiver);
},
set(target, p, value, receiver) {
console.log(`set key: ${p}, value: ${value}`);
return Reflect.set(target, p, value, receiver);
}
});
obj.name = "kevin";
obj.name;
Proxy 作為其他對象的原型
Proxy 實例也可以作為其他對象的原型對象, 可以用來攔截原型鏈的訪問:
JS
var proxy = new Proxy({}, {
get: function(target, propertyKey, receiver){
return 35;
}
})
var obj = Object.create(proxy);
console.log(obj.time); // 35
Proxy 攔截函數調用
JS
// Proxy 攔截函數調用
function test(a, b) {
return a + b;
}
test = new Proxy(test, {
apply(target, thisArg, argArray) {
console.log("thisArg", thisArg);
console.log("target", target);
console.log("argArray", argArray);
let result = Reflect.apply(target, thisArg, argArray);
console.log("result", result);
return result;
}
})
test(1, 2);
/*
thisArg undefined
target [Function: test]
argArray [ 1, 2 ]
result 3
* */
Proxy 監控構造函數的調用
JS
// Proxy 監控構造函數
function Test(a,b) {
this.a = a;
this.b = b;
return a + b;
}
Test = new Proxy(Test, {
construct(target, argArray, newTarget) {
console.log("target", target);
console.log("argArray", argArray);
console.log("newTarget", newTarget);
var result = Reflect.construct(target, argArray, newTarget);
console.log("result", JSON.stringify(result));
return result
}
})
let test = new Test(1, 2);
console.log("test", test);
/*
target [Function: Test]
argArray [ 1, 2 ]
newTarget [Function: Test]
result {"a":1,"b":2}
test Test { a: 1, b: 2 }
*/
window 和 navigator Proxy 實戰
采用 Proxy 幫助補充環境
JS
let mywindow = {};
let mynavigator = {};
let rawstringify = JSON.stringify;
JSON.stringify = function (Object){
if((Object?.value ?? Object) === global){
return "global"
}else{
return rawstringify(Object)
}
}
function checkproxy(){
//Object.keys(window)
window.a = {
"b":{
"c":{
"d":123
}
}
}
window.a.b.c.d = 456
window.a.b
window.btoa("123")
window.atob.name
"c" in window.a
delete window.a.b
Object.defineProperty(window, "b",{
value:"bbb"
})
Object.getOwnPropertyDescriptor(window,"b")
Object.getPrototypeOf(window)
Object.setPrototypeOf(window,{"dta":"dta"})
// for (let windowKey in window) {
// windowKey
// }
Object.preventExtensions(window)
Object.isExtensible(window)
}
function getMethodHandler(WatchName){
let methodhandler = {
apply(target, thisArg, argArray) {
let result = Reflect.apply(target, thisArg, argArray)
console.log(`[${WatchName}] apply function name is [${target.name}], argArray is [${argArray}], result is [${result}].`)
return result
},
construct(target, argArray, newTarget) {
var result = Reflect.construct(target, argArray, newTarget)
console.log(`[${WatchName}] construct function name is [${target.name}], argArray is [${argArray}], result is [${JSON.stringify(result)}].`)
return result;
}
}
return methodhandler
}
function getObjhandler(WatchName){
let handler = {
get(target, propKey, receiver) {
let result = Reflect.get(target, propKey, receiver)
if (result instanceof Object){
if (typeof result === "function"){
console.log(`[${WatchName}] getting propKey is [${propKey}] , it is function`)
//return new Proxy(result,getMethodHandler(WatchName))
}else{
console.log(`[${WatchName}] getting propKey is [${propKey}], result is [${JSON.stringify(result)}]`);
}
return new Proxy(result,getObjhandler(`${WatchName}.${propKey}`))
}
console.log(`[${WatchName}] getting propKey is [${propKey}], result is [${result}]`);
return result;
},
set(target, propKey, value, receiver) {
if(value instanceof Object){
console.log(`[${WatchName}] setting propKey is [${propKey}], value is [${JSON.stringify(value)}]`);
}else{
console.log(`[${WatchName}] setting propKey is [${propKey}], value is [${value}]`);
}
return Reflect.set(target, propKey, value, receiver);
},
has(target, propKey){
var result = Reflect.has(target, propKey);
console.log(`[${WatchName}] has propKey [${propKey}], result is [${result}]`)
return result;
},
deleteProperty(target, propKey){
var result = Reflect.deleteProperty(target, propKey);
console.log(`[${WatchName}] delete propKey [${propKey}], result is [${result}]`)
return result;
},
getOwnPropertyDescriptor(target, propKey){
var result = Reflect.getOwnPropertyDescriptor(target, propKey);
console.log(`[${WatchName}] getOwnPropertyDescriptor propKey [${propKey}] result is [${JSON.stringify(result)}]`)
return result;
},
defineProperty(target, propKey, attributes){
var result = Reflect.defineProperty(target, propKey, attributes);
console.log(`[${WatchName}] defineProperty propKey [${propKey}] attributes is [${JSON.stringify(attributes)}], result is [${result}]`)
return result
},
getPrototypeOf(target){
var result = Reflect.getPrototypeOf(target)
console.log(`[${WatchName}] getPrototypeOf result is [${JSON.stringify(result)}]`)
return result;
},
setPrototypeOf(target, proto){
console.log(`[${WatchName}] setPrototypeOf proto is [${JSON.stringify(proto)}]`)
return Reflect.setPrototypeOf(target, proto);
},
preventExtensions(target){
console.log(`[${WatchName}] preventExtensions`)
return Reflect.preventExtensions(target);
},
isExtensible(target){
var result = Reflect.isExtensible(target)
console.log(`[${WatchName}] isExtensible, result is [${result}]`)
return result;
},
ownKeys(target){
var result = Reflect.ownKeys(target)
console.log(`[${WatchName}] invoke ownkeys, result is [${JSON.stringify(result)}]`)
return result
},
apply(target, thisArg, argArray) {
let result = Reflect.apply(target, thisArg, argArray)
console.log(`[${WatchName}] apply function name is [${target.name}], argArray is [${argArray}], result is [${result}].`)
return result
},
construct(target, argArray, newTarget) {
var result = Reflect.construct(target, argArray, newTarget)
console.log(`[${WatchName}] construct function name is [${target.name}], argArray is [${argArray}], result is [${JSON.stringify(result)}].`)
return result;
}
}
return handler;
}
const window = new Proxy(Object.assign(global,mywindow), getObjhandler("window"));
const navigator = new Proxy(Object.create(mynavigator), getObjhandler("navigator"));
checkproxy()
module.exports = {
window,
navigator
}
異步操作與 Ajax
單線程模型
- 單線程模型指的是: js 只在一個線程上運行, 也就是說, js 同時只能執行一個任務, 其他任務都必須在后面排隊等待
- js 只在一個線程上運行, 不代表 js 引擎只有一個線程, 實際上 js 引擎有多個線程, 單個腳本只能在一個線程上運行(主線程); 其他線程都在后台配合主線程;
同步任務和異步任務
- 同步任務是那些沒有被引擎掛起, 在主線程上排隊執行的任務; 只有前一個任務執行完畢, 才能執行后一個任務;
- 異步任務是那些被引擎放在一邊, 不進入主線程, 而是會進入任務隊列的任務; 只有引擎認為某些異步任務可以執行了(比如 Ajax 操作從服務器得到了響應), 該任務(采用回調函數的形式) 才會進入到主線程進行執行; 排在異步任務后面的代碼不用等待異步任務結束就可以馬上開始運行, 所以說異步任務不會阻塞其他任務代碼;
Ajax 操作可以被當做同步任務處理, 也可以當做異步任務處理, 由參數 async 控制; 如果是同步任務, 主線程就會一直等待 Ajax 操作返回結果, 才會繼續執行后面的代碼; 如果是異步任務, 那么主線程發出 Ajax 請求后會直接向下執行其他代碼, 等到 Ajax 操作有了結果, 引擎會其對應的回調函數放入主線程進行處理;
任務隊列和事件循環
- js 運行時, 除了正在運行的主線程, 引擎還提供了一個任務隊列(task queue); 任務隊列中是各種需要當前程序處理的異步任務;
- 首先, 主線程會執行所有的同步任務, 等到同步任務全部執行完成, 主線程才會去任務隊列中尋找異步任務;
- 如果異步任務滿足執行條件, 則異步任務重新進入到主線程中執行, 此時異步任務就變成了同步任務; 當這個異步任務執行完成, 下一個異步任務再進入主線程進行執行, 一旦任務隊列被清空, 程序就結束運行;
- 事件循環指的是只要同步任務執行完成, 引擎就會不斷去檢查那些掛起的異步任務是不是滿足進入主線程的條件; 這種循環機制就叫做事件循環, 是一種程序結構, 用於等待和發送消息和事件;
異步操作模式
- 回調函數
- 回調函數是異步操作最簡單的方法
- 回調函數的優點是簡單, 容易理解和實現; 缺點是不利於代碼的閱讀和維護; 多個部分之間高度耦合; 在多個回調函數相互嵌套的情況下, 會導致程序結構混亂, 難以閱讀和追蹤;
- 事件監聽
- 采用事件驅動模式, 異步任務的執行不取決於代碼的順序, 而是取決於某個事件是否發生;
- 優點是比較容易理解, 可以綁定多個事件, 每個事件可以指定多個回調函數; 可以解耦, 實現模塊化; 缺點是整個程序都要變成事件驅動型, 運行流程不清晰, 閱讀難以看出主流程;
- 發布/訂閱
- 這種方式的性質和事件監聽類似; 但是可以通過查看消息, 了解存在多少信號, 每個信號有多少個訂閱者, 從而監控程序的運行;
定時器
定時器是 js 中提供定時執行代碼的功能; 主要由setTimeout()
和setInterval()
這兩個函數來完成; 它們可以向任務隊列中添加定時任務;
運行機制
- setTimeout 和 setInterval 的運行機制, 是將指定的代碼移出本輪事件循環, 等到下一輪事件循環, 再檢測是否滿足指定時間, 如果滿足就執行對應的代碼, 如果不滿足則繼續等待;
- setTimeout 和 setInterval 指定的回調函數, 必須等到本輪事件循環的所有同步任務都執行完, 才會開始執行; 由於前面的任務到底需要多少時間執行是不確定的, 所以沒有辦法保證 setTimeout 和 setInterval 指定的任務一定會按照預定時間執行;
- setTimeout(f,0); 也不會立即執行, f 方法會被移出本輪事件循環;
setTimeout
function setTimeout(callback, ms, ...args)
- setTimeout 函數用於指定某個函數或者某段代碼在多少毫秒之后執行; 它返回一個整數, 該整數表示定時器的編號, 之后可以通過這個編號取消該定時器;
- 需要注意: node 中如果回調函數是對象的方法, 那么 setTimeout 會使方法內部的 this 關鍵字指向全局環境, 而不是定義這個方法時的那個對象;
- 為了防止該問題, 可以將對象的方法放在一個函數內
- 使用 bind 方法, 將對象的方法 bind 到對象上
JS
// setTimeout()
var x = "from window";
var obj = {
x: "from obj",
y: function () {
console.log(this.x);
}
}
setTimeout(obj.y, 1000); // node 環境下輸出 undefined, 瀏覽器下輸出 from window
// 將該方法放在一個函數中, 防止 this 指向全局環境
setTimeout(function () {
obj.y(); // node 和瀏覽器下都輸出 from obj;
}, 1000);
setTimeout(obj.y.bind(obj), 1000); // node 和 瀏覽器下都輸入 from obj
setTimeout 也可以實現指定間隔時間執行, 相當於下面的 setInterval 方法; 區別在於 setTimeout 會考慮自身的運行時間,可以保證嚴格的間隔時間;
JS
var i = 1;
function f() {
console.log(i);
i++;
setTimeout(f, 1000);
}
setTimeout(f, 1000);
setInterval
- setInterval 函數的用法和 setTimeout 完全一致, 區別僅僅是在以 setInterval 指定某個任務每隔一段時間就執行一次, 也就是無限次的定時執行;
- setInterval 指定的是開始執行的間隔時間, 並不考慮每次任務執行本身所消耗的時間, 所以兩次執行之間的間隔會小於指定的時間;
JS
var i = "from global";
var obj = {
a : 1,
i: "from obj",
f: function () {
console.log("this.i", this.i);
}
}
setInterval(obj.f, 1000); // node 下輸出為 undefined, 瀏覽器下輸出為 from global
setInterval(function () {
obj.f();
}, 1000) // from obj
setInterval(obj.f.bind(obj), 1000); // from obj
清除定時器
- 可以使用 clearTimeout()和 clearInterval()方法來清除定時器
- setTimeout()和 setInterval()都返回一個整數值, 該整數值表示定時器編號; 將這個編號傳入 clearTimeout()和 clearInterval()就可以取消對應的定時器;
- 需要注意, setTimeout 和 setInterval 返回的定時器編號的數值都是連續的且單調遞增的, 可以利用這一點取消所有的定時器 (過一部分檢測)
- 在 node 環境下, setTimeout 和 setInterval 返回的不是定時器編號, 而是一個定時器對象;
JS
// 清除所有定時器
(function(){
var gid = setInterval(clearAllTimeouts,0);
console.log(gid);
function clearAllTimeouts(){
var id = setTimeout(function(){},0);
while (id > 0){
if (id !== gid){
clearTimeout(id);
}
id--;
}
}
})()
Promise 對象
- Promise 對象是 js 的異步操作解決方案, 為異步操作提供了統一的接口; 它起到了代理作用, 充當異步操作與回調函數之間的中介; 讓異步操作具備同步操作的接口; Promise 可以讓異步操作寫起來就像在寫同步操作的流程, 而不必一層層嵌套回調函數;
- Promise 是一個對象, 也是一個構造函數;
JS
var objPromise = function (input) {
return new Promise(function (resolve, reject) {
var obj = {};
obj.name = input;
setTimeout(resolve, 1000, obj.name);
console.log("繼續執行");
});
}
function resolve(name) {
console.log("name", name);
return name;
}
function resolve2(name) {
console.log("name2", name);
return name;
}
objPromise("kevin").then(resolve).then(resolve2).catch((error)=>{
console.log("error", error);});
使用 Promise 的時候, 業務邏輯代碼都在 newPromise 和 then 的回調方法中;
Ajax 請求
- Ajax 是一種在無需重新加載整個頁面的情況下, 能夠更新部分網頁的技術;
- Ajax = 異步 js 和 xml
- 傳統的頁面如果需要更新內容, 必須要重新加載整個頁面; Ajax 可以使網頁實現異步更新, 在不加載整個頁面的情況下,對頁面的部分進行更新;
JS
// 創建 XMLHttpRequest 對象
var xmlhttp = new XMLHttpRequest();
// 和服務器建立連接
xmlhttp.open("method", "url", async);
/*
method: 請求的類型, GET POST
url: url 地址
async: true(異步), false(同步);
*/
// 發送請求
xmlhttp.send(string); // string 僅用於 post 請求
在異步 async 為 true 時, 需要設置 XMLHttpRequest 對象的onreadystatechange 一個回調函數; 或者設置 XMLHttpReuqest 對象的 onload 一個回調函數;
JS
xmlhttp.onreadystatechange = function(){
if (xmlhttp.readyState === 4) && (xmlhttp.status === 200){
console.log(xmlhttp.responseText);
}
}
// or
xmlhttp.onload = function(){
if (xmlhttp.readyState === 4) && (xmlhttp.status === 200){
console.log(xmlhttp.responseText);
}
}
簡單示例
python 服務端代碼
PYTHON
# -*- coding: UTF-8 -*-
__author__ = "Kevin"
__time__ = "2021/8/26"
__blog__ = "https://kevinspider.github.io/"
from fastapi import FastAPI
from fastapi.middleware.cors import CORSMiddleware
app = FastAPI()
origins = ["*"]
app.add_middleware(
CORSMiddleware,
allow_origins=origins,
allow_credentials=True,
allow_methods=["*"],
allow_headers=["*"],
)
@app.get("/")
async def root():
return {"message": "Hello World"}
html 請求頁面
HTML
<!DOCTYPE html>
<html lang="en" xmlns:v-on="http://www.w3.org/1999/xhtml">
<head>
<meta charset="UTF-8">
<title>Demo</title>
<script src="../snip/vue.js"></script>
</head>
<body>
<div id="root">
<button id="request" v-on:click="promiseOnloadRequest">發送請求</button>
</div>
</body>
<script>
var vue = new Vue({
el: "#root",
methods: {
request: function () {
// 發送請求
var xmlhttp = new XMLHttpRequest();
// 設置回調函數
xmlhttp.onreadystatechange = function () {
if (xmlhttp.readyState == 4 && xmlhttp.status == 200) {
console.log("success: ", xmlhttp.status);
console.log("xmlhttp.responseText: ", xmlhttp.responseText);
}
}
xmlhttp.open("GET", "http://127.0.0.1:8000", true);
xmlhttp.send()
},
promiseRequest: function () {
var promise = new Promise(function (resolve, reject) {
var xmlhttp = new XMLHttpRequest();
xmlhttp.onreadystatechange = function () {
if ((xmlhttp.readyState === 4) && (xmlhttp.status == 200)) {
resolve(xmlhttp.responseText);
}
}
xmlhttp.open("GET", "http://127.0.0.1:8000", true);
xmlhttp.send()
})
function f(res) {
console.log("use promiseRequest");
console.log("res", res);
}
promise.then(f).catch((error) => {
console.log(error);
})
},
promiseOnloadRequest: function () {
var promise = new Promise(function (resolve, reject) {
var xml = new XMLHttpRequest();
xml.open("GET", "http://127.0.0.1:8000", true);
xml.onload = resolve;
xml.send()
});
function parseResponse(res) {
console.log("promiseOnloadRequest", res);
}
function parseError(error) {
console.log("error", error);
}
promise.then(parseResponse).catch(parseError);
}
}
});
</script>
</html>
Dom 和瀏覽器模型
Dom 簡介
- Dom 是 js 操作網頁的接口, 全稱是文檔對象模型; 它的作用是將網頁轉為一個 js 對象, 從而可以通過用腳本進行各種操作(增刪改查)
- 瀏覽器會根據 Dom 模型, 將結構化的文檔(html 或 XML) 解析成一系列的節點, 再由這些節點組成一個樹狀結構; 所有的節點和最終的樹狀結構都有規范的對外接口;
- Dom 是一個接口規范, 可以用任何語言實現; 嚴格來說 Dom 不是 js 語法的一部分, 但是 Dom 操作是 js 最常見的操作, 離開了 Dom, js 就無法控制網頁; 另一方面, js 也是最常用於操作 Dom 的語言;
節點
- Dom 的最小組成單位是節點(node); 文檔的樹形結構就是由各種不同類型的節點組成;
- 節點一共有七種:
- Document: 整個文檔樹的頂層節點
- DocumentType: doctype 標簽
<! DOCTYPE html>
- Element: 網頁的各種 HTML 標簽, 比如
<body> <a>
等 - Attr: 網頁元素的屬性 比如
class="right"
- Text: 標簽之間或標簽包含的文本
- Comment: 注釋
- DocumentFragment: 文檔的片段
- 瀏覽器提供一個原生的節點對象 Node, 上面這七種節點都繼承了 Node, 因此具有一些相同的屬性和方法
事件
事件名稱參照: https://developer.mozilla.org/zh-CN/docs/Web/Events
- 事件的本質是程序各個組成部分之間的一種通信方式, 也是異步編程的一種實現; Dom 支持大量的事件
- Dom 的事件操作(監聽和觸發) 都定義在 EventTarget 接口; 所有節點對象都部署在這個接口, 其他一些需要事件通信的瀏覽器內置對象(比如 XMLHttpRequest, AudioNode, AudioContext等)也都部署了這個接口;
- EventTarget 接口主要提供了三個方法
- addEventListener(type, listener[,useCapture]): 綁定事件的監聽函數
- removeEventListener(type, listener[,useCapture]): 移除事件的監聽函數
- dispatchEvent(event): 觸發事件
事件傳播
- 一個事件發生后, 會在子元素和父元素之間傳播, 這種傳播分成三個階段:
- 第一個階段: 從 window 對象傳導到目標節點 (上層傳到底層), 稱為捕獲階段;
- 第二個階段: 在目標節點上觸發, 稱為目標階段
- 第三個階段: 從目標節點傳導回 window 對象 (從底層傳回上層); 稱為冒泡階段;
- 三種階段的傳播模型, 讓同一個事件會在多個節點上觸發;
HTML
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>節點</title>
</head>
<body>
<div id="div">
<p id="p">點擊觸發</p>
</div>
</body>
<script>
var phases = {
"1": "capture", // 捕獲階段
"2": "target", // 目標階段
"3": "bubble" // 冒泡階段
}
var div = document.getElementById("div");
var p = document.getElementById("p");
function callback(event) {
let tag = event.currentTarget.tagName;
let phase = phases[event.eventPhase];
console.log("tag", tag);
console.log("phase", phase);
}
// 當 addEventListener 第三個參數設置為 true 時, 會先觸發父節點的事件, phase 為 capture; 再觸發 target 目標, phase 為 target;
// div.addEventListener("click", callback, true);
// p.addEventListener("click", callback, true);
// 當 addEventListener 第三個參數設置為 false 時, 會先觸發子節點的事件, phase 為 target; 再觸發冒泡, 回傳到父節點, phase 為 bubble;
div.addEventListener("click", callback, false);
p.addEventListener("click", callback, false);
</script>
</html>
Event 對象
- 事件發生后, 會產生一個事件對象, 作為參數傳遞給監聽函數; 瀏覽器原生提供了一個 Event 對象, 所有的事件都是這個對象的實例, 或者說所有的事件都繼承了 Event.prototype 對象;
- Event 對象本身就是一個構造函數, 可以用來生成一個新的事件實例;
var event = new Event(type, options);
- Event 構造函數接受兩個參數, 第一個參數 type 是字符串, 表示事件的名稱; 第二個參數 options 是一個對象, 表示事件對象的配置; 該對象主要有兩個屬性:
- bubbles: 布爾值, 可選, 默認是 false; 表示事件對象是否冒泡;
- cancelable: 布爾值, 可選, 默認是 false; 表示事件是否可以被取消, 即能否用
Event.proventDefault()
取消這個事件; 一旦事件被取消, 就好像沒有發生過一樣, 不會觸發瀏覽器對該事件的默認行為;
HTML
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>節點</title>
</head>
<body>
<div id="div">
<p id="p">點擊觸發</p>
</div>
</body>
<script>
var phases = {
"1": "capture", // 捕獲階段
"2": "target", // 目標階段
"3": "bubble" // 冒泡階段
}
var div = document.getElementById("div");
var p = document.getElementById("p");
function callback(event) {
let tag = event.currentTarget.tagName;
let phase = phases[event.eventPhase];
console.log("tag", tag);
console.log("phase", phase);
}
var event = new Event("look", {
bubbles: true,
cancelable: true
});
p.addEventListener("click", callback, false);
p.dispatchEvent(event);
</script>
</html>
監聽函數
-
瀏覽器的事件模型, 就是通過監聽函數(listener)對事件作出反應; 事件發生后, 瀏覽器監聽到了這個事件, 就會執行對應的監聽函數;
-
三種方式為事件綁定監聽函數
-
HTML 的 on -屬性
HTML <div id="button" onclick="btnClick()">點擊觸發</div> <script> function btnClick(event) { console.log("event", event); console.log("button clicked"); } </script>
-
元素節點的事件屬性
HTML <div id="button">點擊觸發</div> <script> var button = document.getElementById("button"); button.onclick = function (event) { console.log("event", event); console.log("按鈕點擊"); } </script>
-
EventTarget.addEventListener()
HTML <div id="button">點擊觸發</div> <script> var button = document.getElementById("button"); function btnClick(event) { console.log("event", event); console.log("addEventListener", "clicked"); } button.addEventListener("click", btnClick); </script>
-
window
瀏覽器中, window 對象指的是當前的瀏覽器窗口; 它也是當前頁面的頂層對象, 它也是當前頁面的頂層對象, 即最高一層的對象, 所有其他對象都是它的下屬, 一個變量如果未聲明, 那么默認就是頂層對象的一個屬性;
Document
- document 節點對象代表整個文檔, 每個網頁都有自己的 document 對象; window.document 屬性就指向了這個對象; 只要瀏覽器開始載入 HTML 文檔, 該對象就存在了, 可以直接使用;
- document 對象有不同的辦法可以獲取
- 正常的網頁, 直接使用 document 或者 window.document 獲取
- iframe 框架里面的網頁, 使用 iframe 節點的 contentDocuemnt 屬性
- Ajax 操作返回的文檔, 使用 XMLHttpRequest 對象的 responseXML 屬性;
- 內部節點的 ownerDocument 屬性
- document 對象繼承了 EventTarget 接口和 Node 接口; 還 mixin 了 ParentNode 接口;
Document 常用方法
- document.open(): 清除當前文檔的所有內容, 讓文檔處於可寫狀態, 供 document.write()方法寫入內容
- document.write(): 寫入內容
- document.close(): 關閉文檔
- document.querySelector(): 接受一個 css 選擇器作為參數, 返回匹配到該選擇器的元素節點; 如果有多個節點滿足匹配條件, 則返回第一個匹配的節點; 如果沒有發現匹配的節點, 返回 null;
- document.querySelectorAll(): 接受一個 css 選擇器作為參數, 返回匹配到該選擇器的所有元素節點數組; 如果沒有發現匹配的節點, 返回 null;
- document.getElementsByClassName(): 返回一個類似數組的對象, 包括了所有 class 名字符合指定條件的元素;
- document.getElementsByName(): 返回一個類似數組的對象, 包含了所有 name 屬性滿足條件的元素;
- document.getElementById(): 返回匹配指定 id 的元素
navigator
navigator 對象的屬性:
- navigator.userAgent: 瀏覽器請求頭
- navigator.plugins: 返回一個類數組的對象, 成員是 plugin;
- navigator.platform: 返回用戶的操作系統信息
- navigator.onLine: 表示當前用戶在線還是離線
- navigator.languages: 表示當前瀏覽器語言
- navigator.language: 表示當前瀏覽器的首選語言
- navigator.geolocation: 返回一個 Geolocation 地理位置對象
- navigator.cookieEnabled: 表示瀏覽器的 Cookie功能是否開啟
navigator 對象的方法
- navigator.javaEnabled(): 表示瀏覽器是否能運行 Java Applet 小程序
- navigator.sendBeacon(): 用於向服務器異步發送數據
- navigator.deviceMemory(): 返回當前計算機的內存數量(單位為 G)
- navigator.connection(): 包含當前網絡連接的相關信息;
screen
- screen.height: 瀏覽器窗口所在的屏幕的高度(單位像素)
- screen.width: 瀏覽器窗口所在的屏幕的寬度(單位像素)
- screen.availHeight: 瀏覽器窗口可用的屏幕高度(單位像素); 因為部分空間可能不可以用, 比如系統的任務欄或者 mac 系統屏幕底部的 Dock 區;這個屬性等於 height 減去那些被系統占用的組件的高度;
- screen.availWidth: 瀏覽器窗口可用的屏幕寬度(單位像素);
- screen.pixelDepth: 整數, 表示屏幕的色彩位數, 比如 24 表示屏幕提供 24 位的色彩
- screen.colorDepth: screen.pixelDepth 的別名, 嚴格來說, colorDepth 表示應用程序的顏色深度, pixelDepth 表示屏幕的顏色深度, 絕大多數情況下, 它們都是相同的一件事;
- screen.orientation: 返回一個對象, 表示屏幕的方向;
cookie
- cookie 是服務器保存在瀏覽器的一小段文本信息, 一般大小不能超過 4KB; 瀏覽器每次向服務器發出請求, 會自動附上這段信息
- cookie 主要保存狀態信息, 一般用於:
- 對話 session 管理; 保存登錄, 購物車等需要記錄的信息
- 個性化信息: 保存用戶偏好, 比如網頁的字體大小, 背景色等;
- 追蹤用戶: 記錄和分析用戶的行為和偏好
cookie 的生成
- 服務器如果希望在瀏覽器保存 cookie, 就需要在 http 的 response 中設置 Set-Cookie 字段;
- document.cookie 用於讀寫當前頁面的 cookie; 讀取的時候, 會返回當前網頁中所有的不含 HttpOnly 屬性的 cookie 信息;
- document.cookie 屬性時可寫的; 可以通過對 document.cookie 設置值來添加 cookie;
document.cookie = "dta=dta; expires=Mon, Aug 30 2021 15:44:17 GMT; path=/; domain=www.dtasecurity.cn"
- 刪除現有的 cookie 的唯一方法, 就是設置它的 expires 屬性為一個過去的時間;
storage
- window.sessionStorage 和 window.localStorage 接口都實現了 Storage 接口;
- window.sessionStorage 保存的數據用於瀏覽器的一次會話; 當會話結束(通常是窗口關閉), 數據被清空;
- localStorage 保存的數據長期存在, 下一次訪問該網站的時候, 網頁可以直接讀取以前的數據
- 除了保存數據的時間長短不同, 這個兩個對象其他方面都基本一致
Storage 屬性
- Storage.length: 返回保存的數據個數
Storage 方法
- Storage.setItem(): 接受兩個參數, 一個是鍵名, 一個是要保存的數據; 寫入不一定要用這個方法, 也可以直接賦值
- Storage.getItem(): 用於讀取數據, 它只有一個參數, 就是鍵名;
- Storage.removeItem(): 用於清除某個鍵名對應的鍵值
- Storage.clear(): 用於清除所有保存的數據
- Storage.key(): 接受一個整數作為參數, 返回該位置對應的鍵名; 結合 Storage.length 屬性和 Storage.key()方法, 可以遍歷所有的鍵;
history
- window.history 屬性指向 History 對象, 它表示當前窗口的瀏覽歷史;
- History.length: 當前窗口訪問過的網址數量(包括當前頁面)
- History.state: History 堆棧最上層的狀態值, 通常是 undefined;
- History.back(): 移動到上一個網址, 等同於點擊瀏覽器的后退鍵
- History.forward(): 移動到下一個網址, 等同於點擊瀏覽器的前進鍵;
- History.go(): 接受一個整數作為參數, 以當前網址作為基准, 移動到參數指定的網址
- History.pushState(): 在歷史記錄中添加一條記錄
- History.replaceState(): 修改 History 對象的當前記錄;
location
location 對象是瀏覽器提供的原生對象, 提供 URL 相關的信息和操作方法; 通過 window.location 或者 document.location 都可以獲得這個對象;
location 屬性
- location.href: 整個 URL
- location.protocol: 當前 URL 的協議, 包括冒號(😃
- location.host: 主機, 如果端口不是協議默認的 80 和 443, 則還會返回冒號和端口號
- location.hostname: 主機名, 不包括端口號;
- location.port: 端口號
- location.pathname: URL 的路徑部分, 從根路徑/開始
- location.search: 查詢字符串部分, 從?開始;
- location.hash: 片段字符串部分, 從#開始
- location.username: 域名簽名的用戶名
- location.password: 域名前面的密碼;
- location.origin: URL 的協議, 主機名和端口
location 方法
-
location.assign(): 接受一個 URL 字符串參數, 讓瀏覽器立即跳轉到新的 URL;
-
location.replace(): 同上, 但是會在瀏覽器的歷史 history 里面刪除當前網址;
-
location.reload(): 使得瀏覽器重新加載當前網址, 相當於刷新
-
location.toString(): 返回整個 URL 字符串, 相當於讀取了 location.href 屬性;
-
location.encodeURI(): 會將除了元字符和語義字符之外的字符都進行轉義, 一般用於轉義整個 url;
JS var url = "https://www.baidu.com?q=你好"; encodeURI(url); // "https://www.baidu.com?q=%E4%BD%A0%E5%A5%BD"
-
location.encodeURIComponent(): 會將除了語義字符之外的所有字符都轉義, 一般用於轉義部分 url;
JS var url = "https://www.baidu.com?q=你好"; encodeURIComponent(url); // "https%3A%2F%2Fwww.baidu.com%3Fq%3D%E4%BD%A0%E5%A5%BD"
-
location.decodeURI(): encodeURI()方法的逆運算
-
location.decodeURIComponent(): encodeURIComponent()方法的逆運算
滑塊
hook 定位
比較常見的通過 Canvas 繪制滑塊驗證碼的 hook 點:
JS
(function() {
'use strict';
// Your code here...
console.log("hook CanvasRenderingContext2D");
var _drawImage = CanvasRenderingContext2D.prototype.drawImage;
var _putImageData = CanvasRenderingContext2D.prototype.putImageData;
CanvasRenderingContext2D.prototype.drawImage = function(){
console.log("CanvasRenderingContext2D.prototype.drawImage", arguments);
debugger;
return _drawImage.apply(this, arguments);
};
CanvasRenderingContext2D.prototype.putImageData = function(){
console.log("CanvasRenderingContext2D.prototype.putImageData", arguments);
debugger;
return _putImageData.apply(this, arguments);
}
})();
通過 hook 追蹤堆棧, 找到 js 中是如何對正序數組進行還原的;
hook 獲取軌跡
JS
// ==UserScript==
// @name hook trail
// @namespace http://tampermonkey.net/
// @version 0.1
// @description try to take over the world!
// @author You
// @include *
// @grant none
// @run-at document-start
// ==/UserScript==
(function () {
let _addEventListener = EventTarget.prototype.addEventListener;
window.flag = false;
window.zedtail = [];
window.start_x;
window.start_y;
window.start_t;
EventTarget.prototype.addEventListener = function () {
let eventname = arguments[0];
let eventfunc = arguments[1];
let neweventfunc = function (events) {
if (events.type === "mousedown") {
window.zedtail = [];
window.start_x = events.clientX;
window.start_y = events.clientY;
window.start_t = +new Date;
window.flag = true;
} else if (events.type === "mousemove") {
if (window.flag) {
let movex = parseInt(events.clientX - window.start_x);
let movey = parseInt(events.clientY - window.start_y);
let movet = (new Date).getTime() - window.start_t;
console.log([movex, movey, movet]);
window.zedtail.push([movex, movey, movet]);
}
} else if (events.type === "mouseup") {
console.log(window.zedtail);
window.flag = false;
}
eventfunc(events);
};
console.log(eventname, eventfunc.toString());
return _addEventListener.apply(this, [eventname, neweventfunc]);
}
})();
完整代碼
通過正序數組將亂序圖片還原成正序圖片
PYTHON
# -*- coding: UTF-8 -*-
__author__ = "Kevin"
__time__ = "2021/9/1"
__blog__ = "https://kevinspider.github.io/"
"""
e.canvasCtx.drawImage(
e.img,
30 * a, sx可選 需要繪制到目標上下文中的,image的矩形(裁剪)選擇框的左上角 X 軸坐標。
0, sy可選 需要繪制到目標上下文中的,image的矩形(裁剪)選擇框的左上角 Y 軸坐標。
30, sWidth可選 需要繪制到目標上下文中的,image的矩形(裁剪)選擇框的寬度。如果不說明,整個矩形(裁剪)從坐標的sx和sy開始,到image的右下角結束。
400, sHeight可選 需要繪制到目標上下文中的,image的矩形(裁剪)選擇框的高度。
30 * t[a] / 1.5, dx image的左上角在目標canvas上 X 軸坐標。
0, dy image的左上角在目標canvas上 Y 軸坐標。
20, image 在目標canvas上繪制的寬度。 允許對繪制的image進行縮放。 如果不說明, 在繪制時image寬度不會縮放。
200) dHeight可選 image在目標canvas上繪制的高度。 允許對繪制的image進行縮放。 如果不說明, 在繪制時image高度不會縮放。
"""
import json
import math
import random
import requests
import base64
import cv2 as cv
import numpy as np
import matplotlib.pyplot as plt
from PIL import Image
# 1.獲取滑塊圖片
def getImg():
headers = {
'Connection': 'keep-alive',
'Pragma': 'no-cache',
'Cache-Control': 'no-cache',
'Accept': 'application/json, text/plain, */*',
'User-Agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.159 Safari/537.36',
'Origin': 'http://www.dtasecurity.cn:30080',
'Referer': 'http://www.dtasecurity.cn:30080/',
'Accept-Language': 'zh,zh-CN;q=0.9',
}
params = (
('w', '400'),
('h', '200'),
)
response = requests.get('http://www.dtasecurity.cn:35555/picture', headers=headers, params=params, verify=False)
data = json.loads(response.text)
# 驗證碼 id , 之后需要回傳進行驗證
sid = data['sid']
# 加密之后的正序數組, 需要解密
c = data['c']
p1_path = "./p1.jpeg"
p2_path = "./p2.jpeg"
# 亂序圖片 ""
p1 = data['p1'].split(',')[-1]
# 滑塊圖片 ""
p2 = data['p2'].split(',')[-1]
with open(p1_path, 'wb') as f:
f.write(base64.b64decode(p1))
with open(p2_path, 'wb') as f:
f.write(base64.b64decode(p2))
# 滑塊對應的 y 坐標
y = data['y']
return sid, c, p1_path, p2_path, y
# 2. python 還原 js 中獲取真實排序數組的方法
def decodeKey(input):
order_list = []
for each in input:
order_list.append(ord(each) ^ 66)
print(order_list)
return order_list
# 3. 將下載下來的亂序圖片還原
def getRightImg(path, order_list, offset):
right_path = "./right.jpg"
wrong_img = Image.open(path)
width, height = wrong_img.size
# 初始化正確順序的列表容器
right_order = []
for i in range(len(order_list)):
right_order.append(0)
# 遍歷傳入的數組, 按數組指定的順序將每個片段插入到列表容器中
for index, i in enumerate(order_list):
box = (index * offset, 0, (index + 1) * offset, height)
tmp_img = wrong_img.crop(box)
right_order[i] = tmp_img
# 將排序之后的片段重新寫入圖片
x = y = 0
right_img = Image.new("RGB", (width, height), 255)
for img in right_order:
right_img.paste(img, (x, y))
x += offset
# right_img.show()
right_img.save(right_path)
return right_path
# 4. 灰度 高斯濾波器(降噪) 邊緣檢測 調參, 模板匹配
def getDistance(right_path, p2_path):
# 模板匹配
def find_template(canny_img, canny_slide_img):
result = cv.matchTemplate(canny_img, canny_slide_img, cv.TM_CCOEFF_NORMED)
min_val, max_val, min_loc, max_loc = cv.minMaxLoc(result)
return min_val, max_val, min_loc, max_loc
# 獲取灰度圖片 cv.IMREAD_GRAYSCALE
gray_img = cv.imread(right_path, cv.IMREAD_GRAYSCALE)
gray_slide_img = cv.imread(p2_path, cv.IMREAD_GRAYSCALE)
# 去除噪聲 高斯模糊
gs_img = cv.GaussianBlur(gray_img, (5, 5), 0)
gs_slide_img = cv.GaussianBlur(gray_slide_img, (5, 5), 0)
# 邊緣檢測
canny_img = cv.Canny(gs_img, 25, 45)
canny_slide_img = cv.Canny(gs_slide_img, 25, 45)
result = find_template(canny_img, canny_slide_img)
print(result)
return result
# 通過 tracebar 進行參數調節 mac 上就沒成功過
# def tracebar(x):
# threshold1 = cv.getTrackbarPos("threshold1", "test")
# threshold2 = cv.getTrackbarPos("threshold2", "test")
# canny_img = cv.Canny(gs_img, threshold1, threshold2)
# cv.imshow("canny_img", canny_img)
# cv.namedWindow("test")
# cv.createTrackbar("threshold1", "test", 0, 255, tracebar)
# cv.createTrackbar("threshold2", "test", 0, 255, tracebar)
# 顯示圖片測試
# cv.imshow("gray_img", gray_img)
# cv.imshow("gs_img", gs_img)
# cv.waitKey()
# cv.destroyAllWindows()
# 5. 獲取軌跡(jshook打印 or 直接還原編碼后的軌跡信息)
# 油猴 hook 腳本 ./hookTrail.js
# 6. 畫出軌跡圖像 找緩動函數相同形狀
def show_plt():
for i in range(1, 6):
trail_list = eval(f"trail_{i}")
print(f"trail_{i}")
x_trail = []
y_trail = []
t_trail = []
for trail in trail_list:
x = trail[0]
y = trail[1]
t = trail[2]
x_trail.append(x)
y_trail.append(y)
t_trail.append(t)
print(np.diff(x_trail))
# 打印 t 軸
# print(t_trail)
plt.plot(t_trail, x_trail)
plt.show()
# 7. 繪制緩動函數 找到符合形狀的作用域
# 8. 根據圖片調整 np.linspace 的作用域, 最終調整為 -0.5 到 1;
# 9. 找到作用域內最大值 最小值 上下移動 *距離系數
# 10. 替換時間(t)軸為自己的
# 11. 高斯函數增加波動
# 緩動函數速查表 https://www.xuanfengge.com/easeing/easeing/
# 對應實現: https://github.com/gdsmith/jquery.easing
def show_easeOutQuint(distance):
def func(x):
return (1 - pow(1 - x, 5)) + 6.59375 # 6.59375 最小值, 整體上移
size = 400
# 作用域 -0.5 到 1, 400 個點
x = np.linspace(-0.5, 1, size)
# 獲取 y 的最大值和最小值, 修改 func 函數, 讓圖片整體向上移動到 x 軸上方
print(func(-0.5), func(1))
# y 的最大值為最終要滑動的距離, 所以 func(1)的值要等於傳入的滑動距離, func(1) * distance/y當前最大值
# y = [distance/7.59375 * func(i) for i in x]
y = [func(i) for i in x]
plt.plot(x, y)
plt.show()
# 將 x 軸換成我們的時間 t, 打印時間 t 查看大概的時間區間
t = np.linspace(200, 3000, size)
# 使用高斯函數對 y 增加噪點波動
delta_pt = abs(np.random.normal(scale=1.1, size=size))
for index in range(len(delta_pt)):
change_y = int(x[index] + delta_pt[index])
if (index + 1 < size and y[index + 1] > change_y):
y[index] += change_y
# y[index] = int(y[index])
print("np.diff(y)", np.diff(y))
plt.plot(t, y)
plt.show()
# 10. 細節修改 最終版本
def get_trail(move_distence, show=False):
def easeOutQuint(x):
return (1 - math.pow(1 - x, 5))
def __set_pt_time(_dist):
if _dist < 100:
__need_time = int(random.uniform(500, 1500))
else:
__need_time = int(random.uniform(1000, 2000))
__end_pt_time = []
__move_pt_time = []
__pos_z = []
total_move_time = __need_time * random.uniform(0.8, 0.9)
start_point_time = random.uniform(110, 200)
__start_pt_time = [int(start_point_time)]
sum_move_time = 0
_tmp_total_move_time = total_move_time
while True:
delta_time = random.uniform(15, 20)
if _tmp_total_move_time < delta_time:
break
sum_move_time += delta_time
_tmp_total_move_time -= delta_time
__move_pt_time.append(int(start_point_time + sum_move_time))
last_pt_time = __move_pt_time[-1]
__move_pt_time.append(int(last_pt_time + _tmp_total_move_time))
sum_end_time = start_point_time + total_move_time
other_point_time = __need_time - sum_end_time
end_first_ptime = other_point_time / 2
while True:
delta_time = random.uniform(110, 200)
if end_first_ptime - delta_time <= 0:
break
end_first_ptime -= delta_time
sum_end_time += delta_time
__end_pt_time.append(int(sum_end_time))
__end_pt_time.append(int(sum_end_time + (other_point_time / 2 + end_first_ptime)))
__pos_z.extend(__start_pt_time)
__pos_z.extend(__move_pt_time)
__pos_z.extend(__end_pt_time)
return __pos_z
def __get_pos_y(point_count):
_pos_y = []
start_y = random.randint(-1, 1)
end_y = random.randint(-13, -5)
x = np.linspace(start_y, end_y, point_count)
for _, val in enumerate(x):
_pos_y.append(int(val))
return _pos_y
time_list = __set_pt_time(move_distence)
trail_length = len(time_list)
t = np.linspace(-0.5, 1, trail_length) # t
# -6.59375 1.0 先進行向上平移, + 6.59375 => 平移后最大值變為 1.0 + 6.59375 = 7.59375
print(easeOutQuint(-0.5), easeOutQuint(1))
mult = move_distence / 7.59375
x = [int(mult * (easeOutQuint(i) + 6.59375)) for i in t]
y = __get_pos_y(trail_length)
# t=-0.5 x=-6.59375
# t=1 x=7.59375
delta_pt = abs(np.random.normal(scale=3, size=trail_length))
for index in range(len(delta_pt)):
change_x = int(x[index] + delta_pt[index])
if index + 1 < trail_length and x[index + 1] > change_x:
x[index] = change_x
if show:
delta_t = [i for i in range(trail_length)]
plt.plot(delta_t, delta_pt, color='green')
plt.plot(time_list, x, color='red')
plt.show()
result = []
print(x[-1] - x[0])
for idx in range(trail_length):
result.append([x[idx], y[idx], time_list[idx]])
return result
# 11. 提交數據加密
def encodeData(trail):
postStr = ""
for each in trail:
x = each[0]
y = each[1]
t = each[2]
tmp = f"{x},{y},{t}"
postStr += base64.b64encode(tmp.encode()).decode() + "*"
postStr = postStr.rstrip("*")
return postStr
# 12. 提交滑塊
def postData(postStr, sid):
headers = {
'Connection': 'keep-alive',
'Pragma': 'no-cache',
'Cache-Control': 'no-cache',
'Accept': 'application/json, text/plain, */*',
'User-Agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.159 Safari/537.36',
'Content-Type': 'application/json;charset=UTF-8',
'Origin': 'http://www.dtasecurity.cn:30080',
'Referer': 'http://www.dtasecurity.cn:30080/',
'Accept-Language': 'zh,zh-CN;q=0.9',
}
data = {
"sid": sid,
"trail": postStr
}
response = requests.post('http://www.dtasecurity.cn:35555/slide',
headers=headers, json=data, verify=False)
print(response.text)
if __name__ == '__main__':
sid, c, p1_path, p2_path, y = getImg()
print(sid, c, p1_path, p2_path, y)
order_list = decodeKey(c)
right_path = getRightImg(p1_path, order_list, 30)
min_val, max_val, min_loc, max_loc = getDistance(right_path, p2_path)
distance = max_loc[0]
trail = get_trail(distance / (30 / 20), True) # 30 原始圖片截取寬度 20 寫入的時候的寬度
postStr = encodeData(trail)
postData(postStr,sid)
# show_plt()
# distance = 100
# show_easeOutQuint(100)
# result = get_trail(100, True)
# print(result)
文章作者: Kevin
文章鏈接: http://example.com/antijs/
版權聲明: 本博客所有文章除特別聲明外,均采用 CC BY-NC-SA 4.0 許可協議。轉載請注明來自 凡牆總是門!