摘要:來源編程精解中文第三版翻譯項目原文譯者飛龍協(xié)議自豪地采用谷歌翻譯部分參考了編程精解第版確定編程語言中的表達式含義的求值器只是另一個程序。若文本不是一個合法程序,解析器應(yīng)該指出錯誤。
來源:ApacheCN『JavaScript 編程精解 中文第三版』翻譯項目原文:Project: A Programming Language
譯者:飛龍
協(xié)議:CC BY-NC-SA 4.0
自豪地采用谷歌翻譯
部分參考了《JavaScript 編程精解(第 2 版)》
確定編程語言中的表達式含義的求值器只是另一個程序。
Hal Abelson 和 Gerald Sussman,《計算機程序的構(gòu)造和解釋》
構(gòu)建你自己的編程語言不僅簡單(只要你的要求不要太高就好),而且對人富有啟發(fā)。
希望通過本章的介紹,你能發(fā)現(xiàn)構(gòu)建自己的編程語言其實并不是什么難事。我經(jīng)常感到某些人的想法聰明無比,而且十分復(fù)雜,以至于我都不能完全理解。不過經(jīng)過一段時間的閱讀和實驗,我就發(fā)現(xiàn)它們其實也并沒有想象中那么復(fù)雜。
我們將創(chuàng)造一門名為 Egg 的編程語言。這是一門小巧而簡單的語言,但是足夠強大到能描述你所能想到的任何計算。它允許基于函數(shù)的簡單抽象。
解析程序設(shè)計語言中最直觀的部分就是語法(syntax)或符號。解析器是一種程序,負責(zé)讀入文本片段(包含程序的文本),并產(chǎn)生一系列與程序結(jié)構(gòu)對應(yīng)的數(shù)據(jù)結(jié)構(gòu)。若文本不是一個合法程序,解析器應(yīng)該指出錯誤。
我們的語言語法簡單,而且具有一致性。Egg 中一切都是表達式。表達式可以是綁定名稱、數(shù)字,或應(yīng)用(application)。不僅函數(shù)調(diào)用屬于應(yīng)用,而且if和while之類的語言構(gòu)造也屬于應(yīng)用。
為了確保解析器的簡單性,Egg 中的字符串不支持反斜杠轉(zhuǎn)義符之類的元素。字符串只是簡單的字符序列(不包括雙引號),并使用雙引號包圍起來。數(shù)值是數(shù)字序列。綁定名由任何非空白字符組成,并且在語法中不具有特殊含義。
應(yīng)用的書寫方式與 JavaScript 中一樣,也是在一個表達式后添加一對括號,括號中可以包含任意數(shù)量的參數(shù),參數(shù)之間使用逗號分隔。
do(define(x, 10), if(>(x, 5), print("large"), print("small")))
Egg 語言的一致性體現(xiàn)在:JavaScript 中的所有運算符(比如>)在 Egg 中都是綁定,但是可以像其他函數(shù)一樣調(diào)用。由于語法中沒有語句塊的概念,因此我們需要使用do結(jié)構(gòu)來表示多個表達式的序列。
解析器的數(shù)據(jù)結(jié)構(gòu)用于描述由表達式對象組成的程序,每個對象都包含一個表示表達式類型的type屬性,除此以外還有其他描述對象內(nèi)容的屬性。
類型為"value"的表達式表示字符串和數(shù)字。它們的value屬性包含對應(yīng)的字符串和數(shù)字值。類型為"word"的表達式用于標(biāo)識符(名稱)。這類對象以字符串形式將標(biāo)識符名稱保存在name屬性中。最后,類型為"apply"的表達式表示應(yīng)用。該類型的對象有一個operator屬性,指向其操作的表達式,還有一個args屬性,持有參數(shù)表達式的數(shù)組。
上面代碼中> (x, 5)這部分可以表達成如下形式:
{ type: "apply", operator: {type: "word", name: ">"}, args: [ {type: "word", name: "x"}, {type: "value", value: 5} ] }
我們將這樣一個數(shù)據(jù)結(jié)構(gòu)稱為表達式樹。如果你將對象想象成點,將對象之間的連接想象成點之間的線,這個數(shù)據(jù)結(jié)構(gòu)將會變成樹形。表達式中還會包含其他表達式,被包含的表達式接著又會包含更多表達式,這類似于樹的分支重復(fù)分裂的方式。
我們將這個解析器與我們第 9 章中編寫的配置文件格式解析器進行對比,第 9 章中的解析器結(jié)構(gòu)很簡單:將輸入文件劃分成行,并逐行處理。而且每一行只有幾種簡單的語法形式。
我們必須使用不同方法來解決這里的問題。Egg 中并沒有表達式按行分隔,而且表達式之間還有遞歸結(jié)構(gòu)。應(yīng)用表達式包含其他表達式。
所幸我們可以使用遞歸的方式編寫一個解析器函數(shù),并優(yōu)雅地解決該問題,這反映了語言自身就是遞歸的。
我們定義了一個函數(shù)parseExpression,該函數(shù)接受一個字符串,并返回一個對象,包含了字符串起始位置處的表達式與解析表達式后剩余的字符串。當(dāng)解析子表達式時(比如應(yīng)用的參數(shù)),可以再次調(diào)用該函數(shù),返回參數(shù)表達式和剩余字符串。剩余的字符串可以包含更多參數(shù),也有可以是一個表示參數(shù)列表結(jié)束的右括號。
這里給出部分解析器代碼。
function parseExpression(program) { program = skipSpace(program); let match, expr; if (match = /^"([^"]*)"/.exec(program)) { expr = {type: "value", value: match[1]}; } else if (match = /^d+/.exec(program)) { expr = {type: "value", value: Number(match[0])}; } else if (match = /^[^s(),"]+/.exec(program)) { expr = {type: "word", name: match[0]}; } else { throw new SyntaxError("Unexpected syntax: " + program); } return parseApply(expr, program.slice(match[0].length)); } function skipSpace(string) { let first = string.search(/S/); if (first == -1) return ""; return string.slice(first); }
由于 Egg 和 JavaScript 一樣,允許其元素之間有任意數(shù)量的空白,所以我們必須在程序字符串的開始處重復(fù)刪除空白。 這就是skipSpace函數(shù)能提供的幫助。
跳過開頭的所有空格后,parseExpression使用三個正則表達式來檢測 Egg 支持的三種原子的元素:字符串、數(shù)值和單詞。解析器根據(jù)不同的匹配結(jié)果構(gòu)造不同的數(shù)據(jù)類型。如果這三種形式都無法與輸入匹配,那么輸入就是一個非法表達式,解析器就會拋出異常。我們使用SyntaxError而不是Error作為異常構(gòu)造器,這是另一種標(biāo)準(zhǔn)錯誤類型,因為它更具體 - 它也是在嘗試運行無效的 JavaScript 程序時,拋出的錯誤類型。
接下來,我們從程序字符串中刪去匹配的部分,將剩余的字符串和表達式對象一起傳遞給parseApply函數(shù)。該函數(shù)檢查表達式是否是一個應(yīng)用,如果是應(yīng)用則解析帶括號的參數(shù)列表。
function parseApply(expr, program) { program = skipSpace(program); if (program[0] != "(") { return {expr: expr, rest: program}; } program = skipSpace(program.slice(1)); expr = {type: "apply", operator: expr, args: []}; while (program[0] != ")") { let arg = parseExpression(program); expr.args.push(arg.expr); program = skipSpace(arg.rest); if (program[0] == ",") { program = skipSpace(program.slice(1)); } else if (program[0] != ")") { throw new SyntaxError("Expected "," or ")""); } } return parseApply(expr, program.slice(1)); }
如果程序中的下一個字符不是左圓括號,說明當(dāng)前表達式不是一個應(yīng)用,parseApply會返回該表達式。
否則,該函數(shù)跳過左圓括號,為應(yīng)用表達式創(chuàng)建語法樹。接著遞歸調(diào)用parseExpression解析每個參數(shù),直到遇到右圓括號為止。此處通過parseApply和parseExpression互相調(diào)用,實現(xiàn)函數(shù)間接遞歸調(diào)用。
因為我們可以使用一個應(yīng)用來操作另一個應(yīng)用表達式(比如multiplier(2)(1)),所以parseApply解析完一個應(yīng)用后必須再次調(diào)用自身檢查是否還有另一對圓括號。
這就是我們解析 Egg 所需的全部代碼。我們使用parse函數(shù)來包裝parseExpression,在解析完表達式之后驗證輸入是否到達結(jié)尾(一個 Egg 程序是一個表達式),遇到輸入結(jié)尾后會返回整個程序?qū)?yīng)的數(shù)據(jù)結(jié)構(gòu)。
function parse(program) { let {expr, rest} = parseExpression(program); if (skipSpace(result.rest).length > 0) { throw new SyntaxError("Unexpected text after program"); } return expr; } console.log(parse("+(a, 10)")); // → {type: "apply", // operator: {type: "word", name: "+"}, // args: [{type: "word", name: "a"}, // {type: "value", value: 10}]}
程序可以正常工作了!當(dāng)表達式解析失敗時,解析函數(shù)不會輸出任何有用的信息,也不會存儲出錯的行號與列號,而這些信息都有助于之后的錯誤報告。但考慮到我們的目的,這門語言目前已經(jīng)足夠優(yōu)秀了。
求值器(evaluator)在有了一個程序的語法樹之后,我們該做什么呢?當(dāng)然是執(zhí)行程序了!而這就是求值器的功能。我們將語法樹和作用域?qū)ο髠鬟f給求值器,執(zhí)行器就會求解語法樹中的表達式,然后返回整個過程的結(jié)果。
const specialForms = Object.create(null); function evaluate(expr, scope) { if (expr.type == "value") { return expr.value; } else if (expr.type == "word") { if (expr.name in scope) { return scope[expr.name]; } else { throw new ReferenceError( `Undefined binding: ${expr.name}`); } } else if (expr.type == "apply") { let {operator, args} = expr; if (operator.type == "word" && operator.name in specialForms) { return specialForms[operator.name](expr.args, scope); } else { let op = evaluate(operator, scope); if (typeof op == "function") { return op(...args.map(arg => evaluate(arg, scope))); } else { throw new TypeError("Applying a non-function."); } } } }
求值器為每一種表達式類型都提供了相應(yīng)的處理邏輯。字面值表達式產(chǎn)生自身的值(例如,表達式100的求值為數(shù)值100)。對于綁定而言,我們必須檢查程序中是否實際定義了該綁定,如果已經(jīng)定義,則獲取綁定的值。
應(yīng)用則更為復(fù)雜。若應(yīng)用有特殊形式(比如if),我們不會求解任何表達式,而是將表達式參數(shù)和環(huán)境傳遞給處理這種形式的函數(shù)。如果是普通調(diào)用,我們求解運算符,驗證其是否是函數(shù),并使用求值后的參數(shù)調(diào)用函數(shù)。
我們使用一般的 JavaScript 函數(shù)來表示 Egg 的函數(shù)。在定義特殊格式fun時,我們再回過頭來看這個問題。
evaluate的遞歸結(jié)構(gòu)類似于解析器的結(jié)構(gòu)。兩者都反映了語言自身的結(jié)構(gòu)。我們也可以將解析器和求值器集成到一起,在解析的同時求解表達式,但將其分離為兩個階段使得程序更易于理解。
這就是解釋 Egg 所需的全部代碼。這段代碼非常簡單,但如果不定義一些特殊的格式,或向環(huán)境中添加一些有用的值,你無法使用該語言完成很多工作。
特殊形式specialForms對象用于定義 Egg 中的特殊語法。該對象將單詞和求解這種形式的函數(shù)關(guān)聯(lián)起來。目前該對象為空,現(xiàn)在讓我們添加if。
specialForms.if = (args, scope) => { if (args.length != 3) { throw new SyntaxError("Wrong number of args to if"); } else if (evaluate(args[0], scope) !== false) { return evaluate(args[1], scope); } else { return evaluate(args[2], scope); } };
Egg 的if語句需要三個參數(shù)。Egg 會求解第一個參數(shù),若結(jié)果不是false,則求解第二個參數(shù),否則求解第三個參數(shù)。相較于 JavaScript 中的if語句,Egg 的if形式更類似于 JavaScript 中的?:運算符。這是一條表達式,而非語句,它會產(chǎn)生一個值,即第二個或第三個參數(shù)的結(jié)果。
Egg 和 JavaScript 在處理條件值時也有些差異。Egg 不會將 0 或空字符串作為假,只有當(dāng)值確實為false時,測試結(jié)果才為假。
我們之所以需要將if表達為特殊形式,而非普通函數(shù),是因為函數(shù)的所有參數(shù)需要在函數(shù)調(diào)用前求值完畢,而if則只應(yīng)該根據(jù)第一個參數(shù)的值,確定求解第二個還是第三個參數(shù)。while的形式也是類似的。
specialForms.while = (args, scope) => { if (args.length != 2) { throw new SyntaxError("Wrong number of args to while"); } while (evaluate(args[0], scope) !== false) { evaluate(args[1], scope); } // Since undefined does not exist in Egg, we return false, // for lack of a meaningful result. return false; };
另一個基本的積木是do,會自頂向下執(zhí)行其所有參數(shù)。整個do表達式的值是最后一個參數(shù)的值。
specialForms.do = (args, scope) => { let value = false; for (let arg of args) { value = evaluate(arg, scope); } };
我們還需要創(chuàng)建名為define的形式,來創(chuàng)建綁定對綁定賦值。define的第一個參數(shù)是一個單詞,第二個參數(shù)是一個會產(chǎn)生值的表達式,并將第二個參數(shù)的計算結(jié)果賦值給第一個參數(shù)。由于define也是個表達式,因此必須返回一個值。我們則規(guī)定define應(yīng)該將我們賦予綁定的值返回(就像 JavaScript 中的=運算符一樣)。
specialForms.define = (args, scope) => { if (args.length != 2 || args[0].type != "word") { throw new SyntaxError("Incorrect use of define"); } let value = evaluate(args[1], scope); scope[args[0].name] = value; return value; };環(huán)境
evaluate所接受的作用域是一個對象,它的名稱對應(yīng)綁定名稱,它的值對應(yīng)這些綁定所綁定的值。 我們定義一個對象來表示全局作用域。
我們需要先定義布爾綁定才能使用之前定義的if語句。由于只有兩個布爾值,因此我們不需要為其定義特殊語法。我們簡單地將true、false兩個名稱與其值綁定即可。
const topEnv = Object.create(null); topScope.true = true; topScope.false = false;
我們現(xiàn)在可以求解一個簡單的表達式來對布爾值求反。
let prog = parse(`if(true, false, true)`); console.log(evaluate(prog, topScope)); // → false
為了提供基本的算術(shù)和比較運算符,我們也添加一些函數(shù)值到作用域中。為了確保代碼短小,我們在循環(huán)中使用 Function來合成一批運算符,而不是分別定義所有運算符。
for (let op of ["+", "-", "*", "/", "==", "<", ">"]) { topScope[op] = Function("a, b", `return a ${op} b;`); }
輸出也是一個實用的功能,因此我們將console.log包裝在一個函數(shù)中,并稱之為print。
topScope.print = value => { console.log(value); return value; };
這樣一來我們就有足夠的基本工具來編寫簡單的程序了。下面的函數(shù)提供了一個便利的方式來編寫并運行程序。它創(chuàng)建一個新的環(huán)境對象,并解析執(zhí)行我們賦予它的單個程序。
function run(program) { return evaluate(parse(program), Object.create(topScope)); }
我們將使用對象原型鏈來表示嵌套的作用域,以便程序可以在不改變頂級作用域的情況下,向其局部作用域添加綁定。
run(` do(define(total, 0), define(count, 1), while(<(count, 11), do(define(total, +(total, count)), define(count, +(count, 1)))), print(total)) `); // → 55
我們之前已經(jīng)多次看到過這個程序,該程序計算數(shù)字 1 到 10 的和,只不過這里使用 Egg 語言表達。很明顯,相較于實現(xiàn)同樣功能的 JavaScript 代碼,這個程序并不優(yōu)雅,但對于一個不足 150 行代碼的程序來說已經(jīng)很不錯了。
函數(shù)每個功能強大的編程語言都應(yīng)該具有函數(shù)這個特性。
幸運的是我們可以很容易地添加一個fun語言構(gòu)造,fun將最后一個參數(shù)當(dāng)作函數(shù)體,將之前的所有名稱用作函數(shù)參數(shù)。
specialForms.fun = (args, scope) => { if (!args.length) { throw new SyntaxError("Functions need a body"); let body = args[args.length - 1]; let params = args.slice(0, args.length - 1).map(expr => { if (expr.type != "word") { throw new SyntaxError("Parameter names must be words"); } return expr.name; }); return function() { if (arguments.length != argNames.length) { throw new TypeError("Wrong number of arguments"); } let localScope = Object.create(scope); for (let i = 0; i < arguments.length; i++) { localScope[params[i]] = arguments[i]; } return evaluate(body, localScope); }; };
Egg 中的函數(shù)可以獲得它們自己的局部作用域。 fun形式產(chǎn)生的函數(shù)創(chuàng)建這個局部作用域,并將參數(shù)綁定添加到它。 然后求解此范圍內(nèi)的函數(shù)體并返回結(jié)果。
run(` do(define(plusOne, fun(a, +(a, 1))), print(plusOne(10))) `); // → 11 run(` do(define(pow, fun(base, exp, if(==(exp, 0), 1, *(base, pow(base, -(exp, 1)))))), print(pow(2, 10))) `); // → 1024編譯
我們構(gòu)建的是一個解釋器。在求值期間,解釋器直接作用域由解析器產(chǎn)生的程序的表示。
編譯是在解析和運行程序之間添加的另一個步驟:通過事先完成盡可能多的工作,將程序轉(zhuǎn)換成一些可以高效求值的東西。例如,在設(shè)計良好的編程語言中,使用每個綁定時綁定引用的內(nèi)存地址都是明確的,而不需要在程序運行時進行動態(tài)計算。這樣可以省去每次訪問綁定時搜索綁定的時間,只需要直接去預(yù)先定義好的內(nèi)存位置獲取綁定即可。
一般情況下,編譯會將程序轉(zhuǎn)換成機器碼(計算機處理可以執(zhí)行的原始格式)。但一些將程序轉(zhuǎn)換成不同表現(xiàn)形式的過程也被認(rèn)為是編譯。
我們可以為 Egg 編寫一個可供選擇的求值策略,首先使用Function,調(diào)用 JavaScript 編譯器編譯代碼,將 Egg 程序轉(zhuǎn)換成 JavaScript 程序,接著執(zhí)行編譯結(jié)果。若能正確實現(xiàn)該功能,可以使得 Egg 運行的非常快,而且實現(xiàn)這種編譯器確實非常簡單。
如果讀者對該話題感興趣,愿意花費一些時間在這上面,建議你嘗試實現(xiàn)一個編譯器作為練習(xí)。
站在別人的肩膀上我們定義if和while的時候,你可能注意到他們封裝得或多或少和 JavaScript 自身的if、while有點像。同樣的,Egg 中的值也就是 JavaScript 中的值。
如果讀者比較一下兩種 Egg 的實現(xiàn)方式,一種是基于 JavaScrip t之上,另一種是直接使用機器提供的功能構(gòu)建程序設(shè)計語言,會發(fā)現(xiàn)第二種方案需要大量工作才能完成,而且非常復(fù)雜。不管怎么說,本章的內(nèi)容就是想讓讀者對編程語言的運行方式有一個基本的了解。
當(dāng)需要完成一些任務(wù)時,相比于自己完成所有工作,借助于別人提供的功能是一種更高效的方式。雖然在本章中我們編寫的語言就像玩具一樣,十分簡單,而且無論在什么情況下這門語言都無法與 JavaScript 相提并論。但在某些應(yīng)用場景中,編寫一門微型語言可以幫助我們更好地完成工作。
這些語言不需要像傳統(tǒng)的程序設(shè)計語言。例如,若 JavaScript 沒有正則表達式,你可以為正則表達式編寫自己的解析器和求值器。
或者想象一下你在構(gòu)建一個巨大的機械恐龍,需要編程實現(xiàn)恐龍的行為。JavaScript 可能不是實現(xiàn)該功能的最高效方式,你可以選擇一種語言作為替代,如下所示:
behavior walk perform when destination ahead actions move left-foot move right-foot behavior attack perform when Godzilla in-view actions fire laser-eyes launch arm-rockets
這通常被稱為領(lǐng)域特定語言(Domain-specific Language),一種為表達極為有限的知識領(lǐng)域而量身定制的語言。它可以準(zhǔn)確描述其領(lǐng)域中需要表達的事物,而沒有多余元素。這種語言比通用語言更具表現(xiàn)力。
習(xí)題 數(shù)組在 Egg 中支持?jǐn)?shù)組需要將以下三個函數(shù)添加到頂級作用域:array(...values)用于構(gòu)造一個包含參數(shù)值的數(shù)組,length(array)用于獲取數(shù)組長度,element(array, n)用于獲取數(shù)組中的第n個元素。
// Modify these definitions... topEnv.array = "..."; topEnv.length = "..."; topEnv.element = "..."; run(` do(define(sum, fun(array, do(define(i, 0), define(sum, 0), while(<(i, length(array)), do(define(sum, +(sum, element(array, i))), define(i, +(i, 1)))), sum))), print(sum(array(1, 2, 3)))) `); // → 6閉包
我們定義fun的方式允許函數(shù)引用其周圍環(huán)境,就像 JavaScript 函數(shù)一樣,函數(shù)體可以使用在定義該函數(shù)時可以訪問的所有局部綁定。
下面的程序展示了該特性:函數(shù)f返回一個函數(shù),該函數(shù)將其參數(shù)和f的參數(shù)相加,這意味著為了使用綁定a,該函數(shù)需要能夠訪問f中的局部作用域。
run(` do(define(f, fun(a, fun(b, +(a, b)))), print(f(4)(5))) `); // → 9
回顧一下fun形式的定義,解釋一下該機制的工作原理。
注釋如果我們可以在 Egg 中編寫注釋就太好了。例如,無論何時,只要出現(xiàn)了井號(#),我們都將該行剩余部分當(dāng)成注釋,并忽略之,就類似于 JavaScript 中的//。
解析器并不需要為支持該特性進行大幅修改。我們只需要修改skipSpace來像跳過空白符號一樣跳過注釋即可,此時調(diào)用skipSpace時不僅會跳過空白符號,還會跳過注釋。修改代碼,實現(xiàn)這樣的功能。
// This is the old skipSpace. Modify it... function skipSpace(string) { let first = string.search(/S/); if (first == -1) return ""; return string.slice(first); } console.log(parse("# hello x")); // → {type: "word", name: "x"} console.log(parse("a # one # two ()")); // → {type: "apply", // operator: {type: "word", name: "a"}, // args: []}修復(fù)作用域
目前綁定賦值的唯一方法是define。該語言構(gòu)造可以同時實現(xiàn)定義綁定和將一個新的值賦予已存在的綁定。
這種歧義性引發(fā)了一個問題。當(dāng)你嘗試為一個非局部綁定賦予新值時,你最后會定義一個局部綁定并替換掉原來的同名綁定。一些語言的工作方式正和這種設(shè)計一樣,但是我總是認(rèn)為這是一種笨拙的作用域處理方式。
添加一個類似于define的特殊形式set,該語句會賦予一個綁定新值,若綁定不存在于內(nèi)部作用域,則更新其外部作用域相應(yīng)綁定的值。若綁定沒有定義,則拋出ReferenceError(另一個標(biāo)準(zhǔn)錯誤類型)。
我們目前采取的技術(shù)是使用簡單的對象來表示作用域?qū)ο螅幚砟壳暗娜蝿?wù)非常方便,此時我們需要更進一步。你可以使用Object.getPrototypeOf函數(shù)來獲取對象原型。同時也要記住,我們的作用域?qū)ο蟛⑽蠢^承Object.prototype,因此若想調(diào)用hasOwnProperty,需要使用下面這個略顯復(fù)雜的表達式。
Object.prototype.hasOwnProperty.call(scope, name);
specialForms.set = function(args, env) { // Your code here. }; run(` do(define(x, 4), define(setx, fun(val, set(x, val))), setx(50), print(x)) `); // → 50 run(`set(quux, true)`); // → Some kind of ReferenceError
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/105051.html
摘要:來源編程精解中文第三版翻譯項目原文譯者飛龍協(xié)議自豪地采用谷歌翻譯部分參考了編程精解第版,這是一本關(guān)于指導(dǎo)電腦的書。在可控的范圍內(nèi)編寫程序是編程過程中首要解決的問題。我們可以用中文來描述這些指令將數(shù)字存儲在內(nèi)存地址中的位置。 來源:ApacheCN『JavaScript 編程精解 中文第三版』翻譯項目原文:Introduction 譯者:飛龍 協(xié)議:CC BY-NC-SA 4.0 自豪地...
摘要:為了運行包裹的程序,可以將這些值應(yīng)用于它們。在瀏覽器中,輸出出現(xiàn)在控制臺中。在英文版頁面上運行示例或自己的代碼時,會在示例之后顯示輸出,而不是在瀏覽器的控制臺中顯示。這被稱為條件執(zhí)行。 來源:ApacheCN『JavaScript 編程精解 中文第三版』翻譯項目原文:Program Structure 譯者:飛龍 協(xié)議:CC BY-NC-SA 4.0 自豪地采用谷歌翻譯 部分參考了《J...
摘要:在本例中,使用屬性指定鏈接的目標(biāo),其中表示超文本鏈接。您應(yīng)該認(rèn)為和元數(shù)據(jù)隱式出現(xiàn)在示例中,即使它們沒有實際顯示在文本中。 來源:ApacheCN『JavaScript 編程精解 中文第三版』翻譯項目原文:JavaScript and the Browser 譯者:飛龍 協(xié)議:CC BY-NC-SA 4.0 自豪地采用谷歌翻譯 部分參考了《JavaScript 編程精解(第 2 版)》 ...
摘要:相反,當(dāng)響應(yīng)指針事件時,它會調(diào)用創(chuàng)建它的代碼提供的回調(diào)函數(shù),該函數(shù)將處理應(yīng)用的特定部分。回調(diào)函數(shù)可能會返回另一個回調(diào)函數(shù),以便在按下按鈕并且將指針移動到另一個像素時得到通知。它們?yōu)榻M件構(gòu)造器的數(shù)組而提供。 來源:ApacheCN『JavaScript 編程精解 中文第三版』翻譯項目原文:Project: A Pixel Art Editor 譯者:飛龍 協(xié)議:CC BY-NC-SA 4...
摘要:來源編程精解中文第三版翻譯項目原文譯者飛龍協(xié)議自豪地采用谷歌翻譯置疑計算機能不能思考就相當(dāng)于置疑潛艇能不能游泳。這張圖將成為我們的機器人在其中移動的世界。機器人在收到包裹時拾取包裹,并在抵達目的地時將其送達。這個機器人已經(jīng)快了很多。 來源:ApacheCN『JavaScript 編程精解 中文第三版』翻譯項目原文:Project: A Robot 譯者:飛龍 協(xié)議:CC BY-NC-S...
閱讀 3464·2021-09-08 10:46
閱讀 1187·2019-08-30 13:17
閱讀 2366·2019-08-30 13:05
閱讀 1209·2019-08-29 15:29
閱讀 2887·2019-08-29 11:31
閱讀 541·2019-08-26 12:13
閱讀 1535·2019-08-26 11:42
閱讀 1838·2019-08-23 18:37