摘要:一返回值共有種二變量聲明方法聲明一個(gè)變量,可以將其初始化為一個(gè)值聲明一個(gè)塊級(jí)作用域變量,可以將其初始化一個(gè)值聲明一個(gè)只讀的常量使用的注意事項(xiàng)不支持塊級(jí)作用域存在變量提升舉例由于變量聲明提升,這段代碼相當(dāng)于使用的注意事項(xiàng)塊級(jí)作用域
一、type of
返回值共有7種:undefined, object, boolean, number, string, symbol, function
typeof undefined === "undefined"; typeof true === "boolean"; typeof 123 === "number"; typeof "username" === "string"; typeof {team : "75team"} === "object" typeof symbol() === "symbol"; typeof null === "object"; typeof function() {} === "function";二、變量聲明方法
var - 聲明一個(gè)變量,可以將其初始化為一個(gè)值
const - 聲明一個(gè)塊級(jí)作用域變量,可以將其初始化一個(gè)值
let - 聲明一個(gè)只讀的常量
var不支持塊級(jí)作用域
var存在變量提升
舉例:
console.log(a === undefined); var a = 10; function foo(){ console.log([a, i]); var a = 20; for(var i = 0; i < a; i++){ //do sth. } console.log([a, i]); } foo();
//---console: true [undefined, undefined] [20, 20]
由于變量聲明提升,這段代碼相當(dāng)于
var a; console.log(a === undefined); a = 10; function foo(){ var a, i; console.log([a, i]); a = 20; for(i = 0; i < a; i++){ //do sth. } console.log([a, i]); } foo();2.2 使用let的注意事項(xiàng)
塊級(jí)作用域
同一個(gè)作用域下只能聲明一次
暫存死區(qū)(temporal dead zone)
循環(huán)中的let作用域
瀏覽器兼容性
塊級(jí)作用域示例:
{ let x = 10; console.log("x is " + x); } console.log(typeof x); let i = 20; for(let i = 0; i < 10; i++){ console.log(i); } console.log(i); var y = 3; var y = 4; console.log(y); // let z = 3; // var z = 4; // console.log(z);
//---console: "x is 10" "undefined" 0 1 2 3 4 5 6 7 8 9 20 4
暫存死區(qū)示例:
let x = 10; function foo(){ console.log(x); let x = 20; //如果這一句改成 var 會(huì)怎樣? return x * x; } console.log(foo());
//---console: "ReferenceError: x is not defined"
循環(huán)中的let作用域示例:
var buttons = document.querySelectorAll("button"); for(var i = 0; i < buttons.length; i++){ buttons[i].onclick = evt => console.log("點(diǎn)擊了第 " + i + "個(gè)按鈕"); }
//---console: "點(diǎn)擊了第 5個(gè)按鈕" "點(diǎn)擊了第 5個(gè)按鈕" "點(diǎn)擊了第 5個(gè)按鈕" "點(diǎn)擊了第 5個(gè)按鈕"
這里是因?yàn)閏lick這個(gè)方法是異步的,只有在點(diǎn)擊之后,js才會(huì)去執(zhí)行這段代碼,這時(shí)for循環(huán)中的 i 已經(jīng)循環(huán)完,所以會(huì)產(chǎn)生點(diǎn)擊哪個(gè)按鈕都返回 5 。解決這類問題常利用‘閉包’,或者使用let
var buttons = document.querySelectorAll("button"); for(var i = 0; i < buttons.length; i++){ (function (i) { buttons[i].onclick = evt => console.log("點(diǎn)擊了第 " + i + "個(gè)按鈕"); })(i) }2.2 使用const的注意事項(xiàng)
const聲明的變量不可以再次被改變
其他使用與let一樣
示例:
const BUFFER_SIZE = 1024; let buffer = new ArrayBuffer(BUFFER_SIZE); console.log(buffer); let data = new Uint16Array(buffer); let data2 = new Uint8Array(buffer); data[0] = 0xff06; console.log(data2[0], data2[1]);
//---console: [object ArrayBuffer] 6 255
注意:給變量賦值為對(duì)象時(shí),對(duì)象引用不能變,但是對(duì)象里的屬性值可以改變。
// A的地址沒有改變,這樣也是可以的。 const A = {a: 1}; A.a = 2; console.log(A); // A: {a: 2} // 如何不讓A被操作呢? const A = Object.freeze({a: 1}); A.a = 2; console.log(A); // A: {a: 1}三、嚴(yán)格模式-"use strict";
(function(){ //"use strict"; var x = y = 0; //do sth. })(); console.log([typeof x, typeof y]);
//---console: ["undefined", "number"]
但是在ES6與‘use strict’某些時(shí)候會(huì)有沖突
(function f( j = 777 ) { "use strict"; // do sth. }) ERROR: Uncaught SyntaxError: Illegal "use strict" directive in function with non-simple parameter list四、抽象相等 4.1嚴(yán)格相等于非嚴(yán)格相等
javascript
console.log([null !== undefined, null == undefined]); //true, true console.log(["1" == 1, [] == 0, "" == 0, 0 == false, 1 == true]); console.log([NaN != NaN]);
Console
[true, true] [true, true, true, true, true] [true]五、Boolean 類型
true 和 false
0、""、null、undefined 被隱式轉(zhuǎn)換為 false,其他轉(zhuǎn)為 true
建議采用嚴(yán)格比較,可以通過 !! 將非 boolean 值轉(zhuǎn)為 boolean
布爾操作符 && 和 || 并不會(huì)轉(zhuǎn)換類型
&& 和 || 的短路特性
比較操作符總是返回 boolean 類型
5.1 Boolean類型的應(yīng)用javascript
var result = 1 && 2; console.log([result, !!result]); var list = [1,,2,,,4,5,0,9]; list = list.filter(item => item); console.log(list); function showTip(tip){ tip = tip || "This is a tip"; console.log(tip); var label = document.createElement("label"); label.className = "tip"; label.innerHTML = tip; document.body.appendChild(label); } document.querySelector("button").onclick = showTip.bind(null, "");
console
"This is a tip" "This is a tip" "This is a tip" ...
示例2:
JavaScript
/** * options -> type:x、y、xy or function */ function applyAnimate(el, duration, options, easing){ var startTime = Date.now(); if(typeof el === "string"){ el = document.querySelector(el); } duration = duration || 1000; options = options || { property: "x", distance: 100 }; easing = easing || function(p){return p;}; requestAnimationFrame(function update(){ var now = Date.now(); var p = (now - startTime)/duration; var ep = easing(p); if(typeof options !== "function"){ var attr = options.property, distance = options.distance; var translate = []; if(attr.indexOf("x") >= 0){ translate.push("translateX(" + distance * ep + "px)"); } if(attr.indexOf("y") >= 0){ translate.push("translateY(" + distance * ep + "px)"); } el.style.transform = translate.join(" "); }else{ options(el, ep, p); } if(p <= 1){ requestAnimationFrame(update); } }); } document.querySelector(".ball").onclick = function(){ applyAnimate(".ball"); /*applyAnimate(".ball", 1000, function(el, ep, p){ const distance = 100, pi2 = 2 * Math.PI; el.style.transform = "translate(" + distance * Math.cos(pi2 * ep) + "px," + distance * Math.sin(pi2 * ep) + "px)"; });*/ }六、Number
整數(shù) -2^53 ~ 2^53
小數(shù)精度 Number.EPSILON
Infinity、Number.MAX_VALUE、Number.MIN_VALUE
浮點(diǎn)數(shù)有精度問題
2進(jìn)制,8進(jìn)制,十六進(jìn)制
+0和-0
6.1 Number類型的應(yīng)用javascript
//----------- console.log(0.2 + 0.4); //WTF!! IEEE754 console.log(((0.2 + 0.4) * 100).toPrecision(2)); console.log(((0.2 + 0.4) * 100).toFixed(2)); //---------- const ball = document.getElementById("ball"); var distance = 100, duration = 2000; ball.onclick = function(){ var startTime = Date.now(); requestAnimationFrame(function update(){ var p = (Date.now() - startTime) / duration; ball.style.transform = "translateX(" + distance * p + "px)" console.log(p,distance); if(p <= 1){ //不應(yīng)當(dāng)用相等比較浮點(diǎn)數(shù)??! requestAnimationFrame(update); } }); }
console
0.6000000000000001 "60" "60.00"
示例2:
JavaScript
console.log([Number.MAX_SAFE_INTEGER, Number.MIN_SAFE_INTEGER]); var bigInteger = Number.MAX_SAFE_INTEGER + 1; console.log([bigInteger, bigInteger + 1, bigInteger + 2]); //WTF!! console.log(1234567890000000000000000000); //科學(xué)計(jì)數(shù)法 console.log(Math.pow(2, 53) === bigInteger); console.log([Number.isSafeInteger(bigInteger), Number.isSafeInteger(bigInteger - 1)]); //----------- console.log([Number.MAX_VALUE, Number.MIN_VALUE]); console.log([Number.MAX_VALUE + 1, Number.MAX_VALUE * 2]); console.log([Number.EPSILON]); console.log(0.99 - 1e-17 === 0.99); console.log(0.99 - Number.EPSILON === 0.99);
console
[9007199254740991, -9007199254740991] [9007199254740992, 9007199254740992, 9007199254740994] 1.23456789e+27 true [false, true] [1.7976931348623157e+308, 5e-324] [1.7976931348623157e+308, Infinity] [2.220446049250313e-16] true false七、String
引號(hào)規(guī)范
轉(zhuǎn)義字符
字符串與字符
字符串與類型轉(zhuǎn)換
常用字符串操作
模板字符串
7.1 引號(hào)規(guī)范與轉(zhuǎn)義javascript
var text = "This is a text."; //規(guī)范建議默認(rèn)使用雙引號(hào) var html = "This is a paragraph
"; console.log(text); console.log(html); var text2 = "我所做的餡餅 是全天下 最好吃的"; console.log(text2); var text3 = "if(a){ console.log(b); }"; console.log(text3); var text4 = "u5947u821eu56e2"; console.log(text4);
Console
"This is a text." "7.2 處理字符This is a paragraph
" "我所做的餡餅 是全天下 最好吃的" "if(a){ console.log(b); }" "奇舞團(tuán)"
JavaScript
var str = "my string"; console.log(str.charAt(5)); var charArray = Array.from(str); //str.split(""); console.log(charArray); console.log(str.charCodeAt(5), str.charCodeAt(6)); console.log(String.fromCharCode(114, 105)); console.log(String.fromCharCode(...charArray.map(c=>c.charCodeAt(0))));
Console
"r" ["m", "y", " ", "s", "t", "r", "i", "n", "g"] 114 105 "ri" "my string"7.3 字符串類型轉(zhuǎn)換 7.3.1 隱式類型轉(zhuǎn)換
console.log([1+2, "1"+2, "1"-2]); // console: [3, "12", -1] // 注釋:減法不會(huì)發(fā)生隱式類型轉(zhuǎn)換,可以用減零的方式將字符串轉(zhuǎn)化成數(shù)字,如:console.log("1"- 0 + 2);
console.log(parseInt("100abc", 2), Number("0b100")); // console: 4 4 // 轉(zhuǎn)整數(shù)
console.log(parseFloat("12.3e10xx")); // console: 123000000000 // 轉(zhuǎn)浮點(diǎn)數(shù), parseFloat會(huì)盡可能的將字符串中的值轉(zhuǎn)化成浮點(diǎn)數(shù),然后將轉(zhuǎn)化不了的字符串忽略,這里是將“xx”省略。
// 封裝對(duì)象的 toString 方法 var foo = { toString(){ return "foo"; } }; console.log(foo + " bar"); // console: "foo bar"7.5 常用字符串操作
javascript
const a = "hello"; const b = "WORLD"; const c = "!"; console.log(a + " " + b + c); //字符串連接 // 注釋:字符串可以用“+”相加,但在某些瀏覽器內(nèi)性能不好,可采用數(shù)組的join方法拼接字符串。 // console.log([a, b, c].join("")); console.log(a.toUpperCase() + " " + b.toLowerCase() + c); //轉(zhuǎn)大小寫 console.log(a.split("").reverse().join("")); //逆序字符串 console.log([a.slice(2,3), a.substr(2,3)]); //截取子串 // 注釋: slice和substr的參數(shù)含義不一樣 const d = a + " " + b + c; console.log(d.indexOf(b)); //字符串查找 console.log(d.replace(a, a.toUpperCase()));
Console
"hello WORLD!" "HELLO world!" "olleh" ["l", "llo"] 6 "HELLO WORLD!"7.6 模板字符串(ES6)
const tpl1 = `我所做的餡餅 是全天下 最好吃的`; console.log([tpl1, typeof tpl1]); { let who = "月影", what = "月餅"; const tpl2 = `${who}所做的${what} 是全天下 最好吃的`; console.log(tpl2); }
["我所做的餡餅 是全天下 最好吃的", "string"] "月影所做的月餅 是全天下 最好吃的"7.7 模板字符串高級(jí)用法
JavaScript
let output = (a, b) => `${a} + $ is ${a + b}`; console.log(output(3, 5)); let formatedList = (data) => `
Console
"3 + 5 is 8" "
對(duì)象的屬性
值和引用
類型與構(gòu)造器
內(nèi)置類型
對(duì)象的高級(jí)屬性
class
8.1 對(duì)象的屬性屬性名規(guī)則:可以是有效字符串或者任意可轉(zhuǎn)換為有效字符串的類型???????
屬性的訪問和遍歷???????
8.2 對(duì)象創(chuàng)建//創(chuàng)建對(duì)象 { let myObj = new Object(); myObj.name = "akira"; myObj.birthday = "12-29"; console.log(myObj); } //然而上面這么寫的是菜鳥,普通青年這么寫 { let myObj = { name: "akira", birthday: "12-29" }; console.log(myObj); } //有些二逼青年: { let myObj = Object.create({name: "akira", birthday: "21-29"}); console.log(myObj); }8.3 屬性訪問
javascript
// 對(duì)象屬性是有效字符串,屬性訪問可以通過.和[] { let myObj = { name: "akira", birthday: "12-29" }; console.log([myObj.name, myObj["birthday"]]); } { // []屬性訪問的好處是可以計(jì)算 const conf = { adapter: "sqlite", db: { sqlite: { //... }, mysql: { //... } } } let dbSetting = conf.db[conf.adapter]; } { // 在 ES6 中字面量的 key 也支持屬性計(jì)算 // 比如可以給對(duì)象定義一個(gè)變量key let process = {env: {}}; const ENV = process.env.JSBIN_ENV || "development"; const conf = { [ENV]: true }; console.log([conf.development, conf.production]); //ES5中只能這樣實(shí)現(xiàn): var ENV = process.env.JSBIN_ENV || "development"; var conf = {}; conf[ENV] = true; console.log([conf.development, conf.production]); }
Console
["akira", "12-29"] [true, undefined]8.4 屬性訪問
javscript
let point = { x : 100, y : 100, getLength : function(){ let {x, y} = this; return Math.sqrt(x * x + y * y); } } console.log(point.getLength()); //用 for...in 遍歷 for(let key in point){ console.log([key, point[key]]); } //用 Object.keys 遍歷 Object.keys(point).forEach((key) => console.log([key, point[key]]));
Console
141.4213562373095 ["x", 100] ["y", 100] ["getLength", function(){ let {x, y} = this; return Math.sqrt(x * x + y * y); }] ["x", 100] ["y", 100] ["getLength", function(){ let {x, y} = this; return Math.sqrt(x * x + y * y); }]8.5 值和引用
值類型與引用類型
基本類型對(duì)應(yīng)的包裝類型
對(duì)象的拷貝
let x = 20, y = 30; function foo(a, b){ a++; b++; console.log([a, b]); } foo(x, y); console.log([x, y]); // 注意:Number是值類型,當(dāng)把x,y作為參數(shù)傳遞給function的時(shí)候,只是傳遞了x,y的副本,所以執(zhí)行完foo(x,y), x,y的值并沒有變化。 const obj = {x: 20, y: 30}; function foo2(obj){ obj.x++; obj.y++; console.log(obj); } foo2(obj); console.log(obj); // 注意:Object是引用類型,它是將obj里的引用自加,所以函數(shù)執(zhí)行完,obj里的值會(huì)發(fā)生變化。
Console
[21, 31] [20, 30] [object Object] { x: 21, y: 31 } [object Object] { x: 21, y: 31 }九、基本類型的包裝類
var str = "Hello World"; var strObj = new String(str); // 注釋:如果寫new是包裝類型,如果不寫new是值類型 console.log([str, strObj]); // console: ["Hello World", "Hello World"] console.log([typeof str, typeof strObj]); // console: ["string", "object"] console.log([str instanceof String, strObj instanceof String]); // console: [false, true] var n = new Number(10); console.log([typeof n, typeof ++n]); // console: ["object", "number"] console.log(Object.prototype.toString.call(10)); // console: [object Number]十、對(duì)象的拷貝
let conf = { adapter: "sqlite", db: { sqlite: { name: "xxx.sqlite" }, mysql: { name: "xxx", username: "work", passwd: "******" } } }; //直接引用 let conf2 = conf; conf2.adapter = "mysql"; console.log(conf.adapter); // console: "mysql" //ES5 淺拷貝 conf.adapter = "sqlite"; let copied = Object.assign({}, conf); copied.adapter = "mysql"; console.log(conf.adapter); // console: "sqlite" console.log(copied.adapter); // console: "mysql" copied.db.sqlite.name = "yyy.sqlite"; console.log(conf.db.sqlite.name); // console: "yyy.sqlite" //深拷貝 function deepCopy(des, src){ for(var key in src){ let prop = src[key]; if(typeof prop === "object"){ des[key] = des[key] || {}; deepCopy(des[key], prop); }else{ des[key] = src[key]; } } return des; } let deepCopied = deepCopy({}, conf); deepCopied.db.sqlite.name = "zzz.sqlite"; console.log([deepCopied.db.sqlite.name, conf.db.sqlite.name]); // console: ["zzz.sqlite", "yyy.sqlite"]十一、對(duì)象的類型和構(gòu)造器
new 和 constructor
prototype
instanceOf
ES6 class
//假設(shè) JS 沒有 "new" 操作符,我們?cè)撊绾螌?shí)現(xiàn)創(chuàng)建對(duì)象? function defineClass(initializer, proto){ return function f(...args){ let obj = Object.create(proto); //f.prototype = proto; //just let instanceof make sense obj.constructor = initializer; obj.constructor(...args); return obj; } } var Point = defineClass(function(x, y){ this.x = x; this.y = y; }, { getLength: function(){ let {x, y} = this; return Math.sqrt(x * x + y * y); } }); var p = Point(3, 4); console.log([p.getLength(), p instanceof Point, p instanceof Object]); //所以 "new" 其實(shí)是一種語法糖十二、原型鏈
// __proto__ 暴力構(gòu)建原型鏈 var a = {x : 1}, b = {y : 2}, c = {z : 3}; b.__proto__ = a; c.__proto__ = b; console.log(c); /** * console: * [object Object] { * x: 1, * y: 2, * z: 3 * } */
//使用 Object.create 構(gòu)建原型鏈 var a = {x : 1}; var b = Object.create(a); // b繼承對(duì)象a b.y = 2; var c = Object.create(b); // c繼承對(duì)象b c.z = 3; console.log(c); /** * console: * [object Object] { * x: 1, * y: 2, * z: 3 * } */
//使用構(gòu)造器方式 function A() { this.x = 1; } function B() { this.y = 2; } B.prototype = new A(); function C() { this.z = 3; } C.prototype = new B(); var c = new C(); console.log(c); /** * console: * [object Object] { * x: 1, * y: 2, * z: 3 * } */12.1 原型鏈有什么弊端?
設(shè)計(jì)原型鏈?zhǔn)菫榱舜a復(fù)用,但是原型鏈有額外構(gòu)造器調(diào)用的問題。
12.2 問題:額外的構(gòu)造器調(diào)用?javascript
//原型繼承 /** * abstract point */ function Point(components){ console.log("Point constructor called"); this.components = components; } Point.prototype = { getDimension: function(){ return this.components.length; }, getLength: function(){ var sum = 0, components = this.components; for(var i = 0; i < components.length; i++){ sum += Math.pow(components[i], 2); } return Math.sqrt(sum); } }; function Point2D(x, y){ Point.call(this, [x, y]); } Point2D.prototype = new Point(); Point2D.prototype.getXY = function(){ var components = this.components; return { x: components[0], y: components[1] }; }
Console
"Point constructor called" "Point constructor called" ["(3,4)", 5, true] // 注意:這里調(diào)用了兩次構(gòu)造器解決方法 1:通用化父類構(gòu)造器
//原型繼承 /** * abstract point */ function Point(dimension){ console.log("Point constructor called"); this.dimension = dimension; } Point.prototype = { setComponents: function(){ components = [].slice.call(arguments); if(this.dimension !== components.length){ throw new Error("Dimension not match!"); } this.components = components; }, getDimension: function(){ return this.dimension; }, getLength: function(){ var sum = 0, components = this.components; for(var i = 0; i < components.length; i++){ sum += Math.pow(components[i], 2); } return Math.sqrt(sum); } }; function Point2D(x, y){ this.setComponents(x, y); } Point2D.prototype = new Point(2); Point2D.prototype.getXY = function(){ var components = this.components; return { x: components[0], y: components[1] }; } Point2D.prototype.toString = function(){ return "(" + this.components + ")"; } var p = new Point2D(3, 4); console.log([p+"", p.getLength(), p instanceof Point]);
Console
"Point constructor called" ["(3,4)", 5, true]
注釋:但這并不總是好的,尤其是在多重繼承的時(shí)候
解決方法 2:延遲父類構(gòu)造器調(diào)用//原型繼承 /** * abstract point */ function Point(components){ console.log("Point constructor called"); this.components = components; } Point.prototype = { getDimension: function(){ return this.components.length; }, getLength: function(){ var sum = 0, components = this.components; for(var i = 0; i < components.length; i++){ sum += Math.pow(components[i], 2); } return Math.sqrt(sum); } }; function Point2D(x, y){ Point.call(this, [x, y]); } Point2D.prototype = Object.create(Point.prototype); //function PointT(){}; //PointT.prototype = Point.prototype; //Point2D.prototype = new PointT(); Point2D.prototype.getXY = function(){ var components = this.components; return { x: components[0], y: components[1] }; } Point2D.prototype.toString = function(){ return "(" + this.components + ")"; } var p = new Point2D(3, 4); console.log([p+"", p.getLength(), p instanceof Point]);
Console
"Point constructor called" ["(3,4)", 5, true]十三、javascript內(nèi)置類型
Object
Function
Array
Date
Regex
Error
Math
ArrayBuffer
Promise
DataView
Map
Set
TypedArray // Uint16Array(buffer) Uint8Array(buffer)
Proxy(ES6)// 處理異步
十四、為什么不建議修改 Object.prototype 和 Array.prototype 14.1 在數(shù)組上添加方法會(huì)被for...in枚舉出來Array.prototype.remove = function(item) { var idx = this.indexOf(item); if(idx >= 0){ return this.splice(idx, 1)[0]; } return null; } var arr = [1, 2, 3]; arr.remove(2); //perfect?? console.log(arr); for(var i in arr){ if(!Number.isNaN(i-0)){ console.log(i + ":" + arr[i]); }else{ console.log(i + "是什么鬼?"); } } /** * [1, 3] * "0:1" * "1:3" * "remove是什么鬼?" */ // 注釋:利用for in遍歷數(shù)組或?qū)ο蟮暮锰幨强梢园阎禐閡ndefined的過濾掉,它會(huì)把數(shù)組上可枚舉的屬性都遍歷出來。解決這一問題的方法
使用 ES5 新特性 defineProperty
規(guī)定禁止使用 for...in 操作數(shù)組,只用 for 或 Array.prototype.forEach
規(guī)定禁止往數(shù)組上隨便添加方法
示例:
Object.defineProperty(Array.prototype, "remove", { value : function(item) { var idx = this.indexOf(item); if(idx >= 0) { return this.splice(idx, 1); } return null; }, // enumerable: true // 當(dāng)且僅當(dāng)該屬性的enumerable為true時(shí),該屬性才能夠出現(xiàn)在對(duì)象的枚舉屬性中。默認(rèn)為 false。 }); var arr = [1, 2, 3]; arr.remove(2); console.log(arr); for(var i in arr) { if(!Number.isNaN(i-0)) { console.log(i + ":" + arr[i]); }else{ console.log(i + "是什么鬼?"); } } /** * console: * [1, 3] * "0:1" * "1:3" */14.2 getter / setter
function Point2D(x, y){ this.x = x; this.y = y; } Object.defineProperty(Point2D.prototype, "length", { get: function() { let {x, y} = this; return Math.sqrt(x * x + y * y); }, set: function(len) { let arc = Math.atan2(this.y, this.x); this.x = len * Math.cos(arc); this.y = len * Math.sin(arc); } }); Object.defineProperty(Point2D.prototype, "arc", { get: function(){ let {x, y} = this; return 180 * Math.atan2(y, x) / Math.PI; }, set: function(arc){ arc = Math.PI * arc / 180; let len = this.length; this.x = len * Math.cos(arc); this.y = len * Math.sin(arc); } }); let p = new Point2D(1, 1); console.log([p.length, p.arc]); // [1.4142135623730951, 45] p.length *= 2; console.log([p.x, p.y]); // [2.0000000000000004, 2] p.arc = 90; console.log([p.x, p.y]); // [1.7319121124709868e-16, 2.8284271247461903]14.3 雙向綁定
const view = { nameEl: document.getElementById("name"), ageEl: document.getElementById("age"), submitBtn: document.getElementById("confirm") } view.submitBtn.addEventListener("click", function(evt){ console.log("你要提交的數(shù)據(jù)是:" + [user.name, user.age]); evt.preventDefault(); }); function User(name, age){ this.name = name; this.age = age; } User.prototype.bind = function(view){ view.nameEl.addEventListener("change", evt => { this.name = evt.target.value; }); view.ageEl.addEventListener("change", evt => { this.age = evt.target.value; }); } Object.defineProperty(User.prototype, "name", { set: function(name){ view.nameEl.value = name; }, get: function(){ return view.nameEl.value; } }); Object.defineProperty(User.prototype, "age", { set: function(name){ var ageOptions = Array.from(view.ageEl.options) .map(item => item.innerHTML); if(ageOptions.indexOf(name) === "-1"){ throw new Error("無效的年齡格式"); } view.ageEl.value = name; }, get: function(){ return view.ageEl.value; } }); var user = new User("akira", "80后"); user.bind(view);
演示 demo
歡迎關(guān)注文章內(nèi)容來源:360奇舞團(tuán)前端星課程。
奇舞周刊: https://weekly.75team.com/
博客:
奇舞團(tuán)博客:https://75team.com/
月影的博客:https://www.h5jun.com/
屈屈的博客:https://imququ.com/
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/98220.html
摘要:變量與常量變量定義變量是存儲(chǔ)數(shù)據(jù)信息的容器。中,變量是弱類型,也被稱為松散類型。三類型布爾類型只有兩個(gè)值均為小寫如果大寫活著大小寫混合,識(shí)別為另一個(gè)變量名稱。 JavaScript簡(jiǎn)介 JavaScript是一門跨平臺(tái)、面向?qū)ο蟮妮p量級(jí)腳本語言,運(yùn)行于JavaScript解釋器或者引擎中。JavaScript中,包括三個(gè)主要的部分1.核心語法內(nèi)容(ECMAScript)2.文檔對(duì)象模型...
摘要:第十六天筆記基礎(chǔ)語法區(qū)分大小寫是一種區(qū)分大小寫的語言的關(guān)鍵字變量名函數(shù)名以及任何其他的標(biāo)識(shí)符必須適應(yīng)一致的大小寫形勢(shì)臥龍前端空格和換行并不是的語法內(nèi)容可有可無建議使用將代碼書寫的更加整潔美觀更有可讀性可選的分號(hào)分號(hào)可選項(xiàng)語法并不是強(qiáng)制要 第十六天筆記 JavaScript 基礎(chǔ)語法 區(qū)分大小寫 JavaScript是一種區(qū)分大小寫的語言 JavaScript的關(guān)鍵字 變量名 函...
摘要:的基礎(chǔ)語法的語法是區(qū)分大小寫的注意在中是區(qū)分大小寫的變量名稱不能是中文代碼示例一花一世界控制臺(tái)調(diào)用效果圖語法中的空格與換行注意并不是的語法內(nèi)容可有可無的建議使用因?yàn)檫@樣會(huì)使代碼看的更清晰利于閱讀代碼示例代碼一花一世界中的分號(hào)可選項(xiàng)語法并 JavaScript的基礎(chǔ)語法: JavaScript的語法是區(qū)分大小寫的 注意: 在JavaScript中是區(qū)分大小寫的 變量名稱不能是中文 代...
摘要:被識(shí)別為結(jié)束符。變量時(shí)被認(rèn)為時(shí)有名字的容器。常量一旦被聲明初始化,值并不能被改變。字符串可以看成右字符組成的數(shù)組,可能被誤認(rèn)為可變的。表達(dá)式?jīng)]有顯示的返回任何內(nèi)容。 Java script 簡(jiǎn)介 Javascript 是一門跨平臺(tái),面向?qū)ο蟮妮p量級(jí)腳本語言,運(yùn)行于Javascript解釋權(quán)/引擎 avascript內(nèi)置了一個(gè)包含一系列對(duì)象的標(biāo)準(zhǔn)庫,比如數(shù)組,日期,數(shù)字和一個(gè)語言元素合...
摘要:前綴規(guī)范每個(gè)局部變量都需要有一個(gè)類型前綴,按照類型可以分為表示字符串。例如,表示以上未涉及到的其他對(duì)象,例如,表示全局變量,例如,是一種區(qū)分大小寫的語言。布爾值與字符串相加將布爾值強(qiáng)制轉(zhuǎn)換為字符串。 基本概念 javascript是一門解釋型的語言,瀏覽器充當(dāng)解釋器。js執(zhí)行時(shí),在同一個(gè)作用域內(nèi)是先解釋再執(zhí)行。解釋的時(shí)候會(huì)編譯function和var這兩個(gè)關(guān)鍵詞定義的變量,編譯完成后從...
摘要:當(dāng)程序需要將值保存起來以備將來使用時(shí),便將其賦值給一個(gè)變量,值的類型稱作數(shù)據(jù)類型。運(yùn)算符鑒于是松散類型的,因此需要有一種手段來檢測(cè)給定變量的數(shù)據(jù)類型,就是負(fù)責(zé)提供這方面信息的運(yùn)算符。 當(dāng)程序需要將值保存起來以備將來使用時(shí),便將其賦值給一個(gè)變量,值的類型稱作數(shù)據(jù)類型。 變量 JavaScript 的變量是松散類型的,所謂松散類型就是可以用來保存任何類型的數(shù)據(jù)。換句話說,每個(gè)變量?jī)H僅是一個(gè)...
閱讀 799·2021-10-09 09:44
閱讀 704·2019-08-30 13:55
閱讀 3162·2019-08-29 15:07
閱讀 3228·2019-08-29 13:09
閱讀 2420·2019-08-29 11:10
閱讀 1297·2019-08-26 14:05
閱讀 3604·2019-08-26 13:57
閱讀 2212·2019-08-23 16:42