码迷,mamicode.com
首页 > 编程语言 > 详细

全栈必备 JavaScript基础

时间:2018-03-28 22:07:02      阅读:240      评论:0      收藏:0      [点我收藏+]

标签:简单   流程控制   事件队列   模式匹配   pos   alt   data   新特性   安全   

JavaScript 来了


1995年。诞生了JavaScript语言,那一年,我刚刚从大学毕业。在今年RedMonk 推出的2017 年第一季度编程语言排行榜中。JavaScript 排第一,Java 第二,Python 反超 PHP 排第三。PHP 第四,C# 和 C++ 并列第五。RedMonk 排名的主要依然是各种编程语言在 Stack Overflow 和 GitHub 上的表现,比方编程语言在 Stack Overflow 上的讨论数量,在 GitHub 上的代码量等。虽然有一定的片面性。还是说明了JavaScript 应用的广泛性。从全栈的角度看,Javascript 是必备的一种编程语言。

?

技术分享图片

ECMAScript 和 JavaScript 的关系

JavaScript 诞生于Netscape,可是在1996年。微软公布了与JavaScript 兼容的JScript,面对兼容和发展的须要。网景公司的先贤们努力增加了?ECMA International?标准化组织,致力于JavaScript 的标准化,命名为ECMAScript。后来,因为历史的原因, JavaScript标准的开发主体变成了Mozila基金会。

关于ECMAScript 的最新版本号能够參阅 https://tc39.github.io/ecma262/。

简单地,ECMAScript 是JavaScript语言的标准规范。就像C++的标准相对于C++语言那样。

JavaScript 是怎样的语言

在mozilla 开发人员站点上是这样描写叙述JavaScript的:

JavaScript (JS) is a lightweight interpreted or JIT-compiled programming language with first-class functions.?

意思是说JavaScript 是一个轻量级解释或即时编译的函数式语言。里面有非常多的概念,轻量、解释、编译、即时编译、函数式。

在老码农看来,简单起见,理解为扩展语言较为方便。

一般的编程语言都有着自己相对独立的运行环境。可是JavaScript的运行环境依赖在宿主环境中。宿主环境尤其是client的宿主环境提供了很多其它统一的环境变量,比方浏览器中的window。document等。实际上,JavaScript 和DOM 是可分的,对于不同的运行环境,有着不同的内置宿主对象。

JavaScript作为扩展语言在内置的宿主环境中运行,全局对象在程序启动前就已经存在了。

JavaScript的时空基础

从空间观的角度看。JavaScript包含数据结构,操作符,语句与表达式。函数。从时间的角度看,包含作用域,处理方式,模块与库。关于技术系统的时空观,能够參见《面向全栈的技术管理》一文。

数据结构

JavaScript 中包含的六种基本类型:

  • Boolean

  • Null

  • Undefined

  • Number

  • String

  • Symbol (ECMAScript 6)

其它全是对象。

值是有类型的。变量是没有类型的,类型定义了值的行为特征,变量在没有持有值的时候是undefined。 JavaScript对值和引用的赋值/传递在语法上没有差别,全然依据值的类型来判定。

对于对象的属性和方法而言,全局变量和全局函数是全局对象的属性。全局对象相当于宿主对象的根对象。须要注意是属性的属性中那些不可变对象的实现方式:?

  1. 对象常量: 结合writable和configurable:false 能够创建一个真正的常量属性

  2. 禁止扩张:Object.preventExtensions(..)来禁止一个对象增加新属性并保留已有属性

  3. 密封: 在 Object.seal(..) 后不能增,删,改 该属性

  4. 冻结: Object.freeze(..) 会禁止对于对象本身及随意直接属性的改动

数据类型的判定能够通过 contructor,instanceof。 isPrototypeOf等方法实现,对于鸭子类型的判定还能够使用 in 的相关操作。符号并非对象,而是一种简单标量基本类型。

?JavaScript 中的强制类型转换总是返回基本类型值,将对象强制转换为String 是通过ToPrimitive抽象操作完毕的。而toJSON()是返回一个能够被字符串化的安全的JSON值。

操作符

操作符是空间元素连接的纽带之中的一个,JavaScript操作符包含算术,连接,相等。比較。逻辑,位。类型推断。条件。new,delete, void,"。", ".", "[]"等。

在JavaScript中以操作符进行操作往往都附带着类型转换。

一元运算符+ 是显式强制类型转换,而~是先转换为32位数字,然后按位反转。

|| 和&& 更应该算是选择器运算符。其返回值不一定是布尔值,而是两个操作数当中的一个值。一般先对第一个操作数进行toBoolean强制类型转换,然后再运行条件推断。比如:a||b 理解成a?a:b 更通俗。对&& 而言,假设第一个是真值。则把第二个作为返回值。a&&b 理解成a?

b:a 。

== 和=== 都会对操作数进行类型检查。并运行隐性类型转换,须要注意的是:?

  • 假设两边的值中有true或false,千万不要使用==

  • 假设两边有[],””或者0。尽量不要使用==

这里是Github上关于各种相等性的矩阵:?

技术分享图片

语句与表达式

操作符与变量/常量等连接形成了语句和表达式,比如表达式a+1中的null 被强制转换为0。

语句包含声明与块,控制语句有推断。循环。break。continue,return,异常等。每一个语句都有一个结果值。哪怕是undefined。

正則表達式是非常重要的一类表达式。主要使用RegExp类,运行方法test效率高,exec 会得到一个结果对象的数组。

逗号运算符能够把多个独立的表达式串联成一个语句,{ }在不同情况下的意思不尽相同,作为语句块,{ ..} 和for/while循环以及if条件语句中代码块的作用基本相同。

{a,b} 实际上是{a:a,b:b}的简化版本号。

try..catch..finally 中,假设finally中抛出异常。函数会在此处终止。须要注意的是,假设此前try中已经有return设置了返回值,则该值会被丢弃。finally中的return也会覆盖try和catch中的return的返回值。

函数与作用域

函数就是具有运算逻辑的对象,匿名函数不利于调试,回调函数是一种控制反转。全部的函数(对象)都具有名为prototype的属性,prototype属性引用的对象是prototype对象;全部的对象都含有一个隐式链接,用以指向在对象生成过程中所使用的构造函数的prototype对象。

匿名函数没有name 标识符,具有例如以下缺陷:?

  1. 代码更难理解

  2. 调试栈更难追踪

  3. 自我引用(递归,事件(解除)绑定,等)更难

假设function是声明的第一个词,那就是函数声明,否则就是函数表达式。马上运行函数表达式形如:(function …)( )

时空密不可分,作用域是时空连接的纽带之中的一个。作用域包含全局,函数。块级作用域。作用域是依据名称查找变量的一套规则。遍历嵌套作用域链的规则简单:引擎从当前运行作用域逐级向上查找。闭包能够理解为具有状态的函数。

函数作用域指属于这个函数的全部变量都能够在整个函数的范围内使用或复用。块作用域形如 with。 try/catch。 ES6 引入了let,const等。

动态作用域并不关心函数和作用域是怎样声明以及在何处声明的,仅仅关心它们从何处调用的。词法作用域是定义在词法分析阶段的作用域。词法作用域查找会在第一个匹配的标识符时停止。

作用域链是基于调用栈的。而不是代码中的作用域嵌套。ReferenceError 是与作用域判别失败相关。而TypeError则是作用域判别成功,可是对结果的操作非法或不合理。

this 提供了一种优雅方式来隐式“传递”一个对象引用。 this 即没有指向函数的自身,也没有指向函数的作用域。是在函数被调用时发生的绑定,它指向什么全然取决于函数在哪里被调用。假设分析this绑定的话,能够使用调试工具得到调用栈。然后找到栈中的第二个元素,就是真正的调用位置。

this 的绑定规则有:

  1. 默认绑定:独立的函数调用,严格模式不能将全局对象用于默认绑定

  2. 隐式绑定:把函数调用中的this 绑定到函数引用中的上下文对象

  3. 显式绑定:通过call()和apply()方法能够直接指定this的绑定对象。

    当中,硬绑定是一种显式的强制绑定,ES5中提供了内置方法Function.prototype.bind, API中调用的上下文和bind的作用一样。

  4. new 绑定,构造函数仅仅是一些使用new操作符调用的函, 使用new 来调用函数的操作过程大致例如以下:

  • 创建一个全新的对象

  • 这个新对象会被运行[[Prototype]]链接

  • 这个新对象会绑定到函数调用的this

  • 假设函数没有返回其它对象,那么new表达式中的函数调用会自己主动返回这个新对象

假设同一时候存在多种绑定。那么绑定的优先级大致例如以下:

  1. 由new调用绑定到新创建的对象

  2. 由call 或者apply(或bind)调用绑定到指定的对象

  3. 由上下文对象调用绑定到那个上下文对象

  4. 默认在在严格模式下绑定到undefined,否则绑定到全局对象

更安全地使用this 绑定的做法是传入一个特殊的对象,把this 绑定到这个对象。须要注意的是。箭头函数不使用this的4种规则,而是依据外层(函数或全局)作用域来决定this。

还要注意一点,eval 和 with 会导致作用域变化而引起性能下降,尽量不要使用。

eval() 函数中的字符串是代码,用来运行动态创建的代码,严格模式有自己的作用域,还存在安全隐患;with 是反复引用一个对象中的多个属性的快捷方式,通过将一个对象的引用当作作用域来处理,会改变作用域范围。

处理和运行方式

JavaScript引擎本身没有时间概念。仅仅是一个按需运行随意代码片段的环境,事件调度总是由包含它的宿主环境来运行。一旦有事件须要运行,事件循环队列就会运行。直到队列清空,用户交互、IO和定时器等事件源会向事件队列增加事件。

因为JavaScript的单线程特性,非常多函数的代码具有原子性。

回调函数封装了程序的延续性,常见设计是分离回调(一个用于成功通知,一个用于出错通知)。还有一种回调模式是“error-first”,可能受到防御式编程的影响,NodeJS API 採用了此类的风格。假设成功的话,这个參数就会被清空。须要注意的是,回调函数的嵌套往往称为回调地狱。

Deferred是一种将异步处理串联书写并运行的机制,Deferred对象是一种具有unresolved,resolved,rejected 中某一种状态的对象。

Deferred内部机制是先注冊回调函数,Deferred对象状态发生变化时运行该函数。是一种提高代码可读性的机制。

Deferred对象的状态迁移仅仅能发生一次,以then(),done(),fail(),always(),pipe()指定兴许函数的方法。通过when()来并行处理,将Deferred 对象中的一部分方法删除后得到是Promise对象,对状态的管理由最初创建该Deferred对象的全部者来运行。

Promise 封装了依赖于时间的状态,从而使得本身与时间无关。Promise 能够依照可预測的方式进行,而不用关心时序或底层的结果。一旦Promise决议完毕。就成为了不变值,能够安全地吧这个值传递给第三方,并确保不会改变。

Promise 是一种在异步任务中作为两个或很多其它步骤的流程控制机制,时序上的this-then-that。 不仅表达了多步异步序列的流程控制。还是一个从一个步骤到下一个步骤传递消息的消息通道。事件监听对象能够当成是对promise 的一种模拟,对控制反转的恢复实现了更好的关注点分离。

推断是否是Promise 值的演示样例代码例如以下:

if(
 ? ?p !==null &&
 ? ?( typeof p ===“object” || typeof p ===“function”) && typeof p.then===“function”)
 ? ?{
 ? ? ? ?console.log(“thenable”);
 ? ?}
else{
 ? ?console.log(“not thenable”);
}

生成器是一类特殊的函数。能够一次或多次启动和停止,并不非的一定要完毕,生成器把while true 带回了Javascript的世界。

当中,yield 托付的主要目的是代码组织,以达到与普通函数调用的对称。从生成器yield出一个Promise, 并且让这个Promise 通过一个辅助函数恢复这个生成器。这是通过生成器管理异步的好方法之中的一个。?

须要注意的是,假设在Promise.all([..]) 中传入空数组,会马上完毕, 而Promise.race([..]) 则会挂住。 在各种Promise库中,finally ( .. ) 还是会创建并返回一个新Promise的。

模块与库

模块和库是JavaScript 时空中的还有一纽带。提高了代码的复用性和开发效率。

模块充分利用了闭包的强大能力。从模块中返回一个实际的对象并非必须的。也能够直接返回一个内部函数,比如:jQauery 和 $标识符就是jQuery 模块的公共API。

模块有两个必要条件:?

  1. 必须有外部的封闭函数。该函数必须至少被调用一次?

  2. 封闭函数必须返回至少一个内部函数,这样内部函数才干在私有作用域中形成闭包,并且能够訪问或改动私有的状态

import 能够将一个模块的一个或多个API导入到当前作用域中。并分别绑定在一个变量上;module 则将整个模块的API 导入并绑定到一个变量上, export 将当前模块的一个标识符导出为公共API。

大多数模块所依赖的加载器/管理器本质上是将这种模块定义封装进一个API。基于函数的模块并非一个能被静态识别的模式(编译器),API定义仅仅有在运行时考虑进来。

可是ES6 模块的API 是静态的,必须被定义在独立的文件里。

JavaScript 中的库浩如烟海,这里仅对JQuery做简要说明。JQuery压缩后大约31k,轻巧灵活,通过链式语法实现逻辑功能。通过CSS3选择器及自己定义选择器获取元素,支持插件。可扩展性高。

JQuery中 的特色函数——$ ,能够抽取与选择器匹配的元素,或者创建新的DOM元素,将已有的DOM元素转换为jQuery对象,对DOM构造完毕后的事件监听器进行设定等等。JQuery 对DOM,样式,AJAX 均可有效处理。

通过扩展JQuery.fn 就能够创建JQuery的插件,code.google.com/apis/libraries 给出了非常多JQuery 的插件信息。

利用JavaScript 的时空观,能够对这一语言有一些主要的梳理。就语言本身而言。keyword是不能回避的,对JavaScript keyword,在StackOverFlow中有人给出了例如以下诗一样的总结:

Let this long package float,
Goto private class if short。
While protected with debug case,
Continue volatile interface。

Instanceof super synchronized throw。 Extends final export throws. Try import double enum?

-False, boolean, abstract function. Implements typeof transient break! Void static,default do, Switch int native new, else, delete null public var, In return for const, true, char, …… finally catch byte.

client应用

一门语言所被使用的广泛程度取决于使用的场景,正如PHP被广泛採用那样,互联网应用不仅是JavaScript 的家乡,并且是它大展身手的最重要场所,没有JavaScript 的Web应用差点儿绝迹了。

web应用中使用JavaScript有拖拽操作。异步读取。键盘訪问 和动画效果等基本功能。对于清晰地使用JavaScript实现Web应用而言。理解浏览器网页处理过程是必要的。

一般地,浏览器先分析HTML。然后构造DOM树。再加载外部Javascript 文件以及CSS文件,接下来加载图像文件等外部资源。最后在分析Javascript后開始运行至全部完毕。

在HTML中加载JavaScript的方式有多种:

  • <script> 标签。在body 结束标签前写

  • 读取外部JavaScript 文件,读取完就開始运行,浏览器能够缓存

  • onload 事件加载

  • DOMContentLoaded是在完毕HTML解析后发生的事件,也能够用于加载JavaScript

  • 动态加载。这样JS在加载时不会阻断其它操作,如

    var script = document.createElement(‘script’);

    script.src = ‘my-javascript.js’;

    document.getElementsByTagName(‘head’)[0].appendChild(script)

window对象是JavaScript所能操作的最高层对象,当中的属性包含navigator,location,history,screen,frames。document,parent,top。self 等。

DOM 是一种API。完毕对HTML/XML 的树形结构訪问。如标签,元素。节点等。节点能够通过ID。标签名。名称和类名进行检索。比如:

var element = document.getElementById(“abel”)
var allelements = document.getElementByTagName(‘*’) 

因为返回的是NodeList对象,性能较差。能够通过?var array = Array.prototye.slice.call(allelements)转换为array 后处理。

节点的訪问能够通过XPath 进行灵活的訪问,当然,Selector API 比XPath更简单且相同灵活。比如:

var a_label = document.querySelector(‘#abel’)
var b_all = document.querySelectorAll(‘div’)

假设先改动DocumentFragment。再对实际的document对象操作。DOM 的操作性能会较高一些。

事件侦听器的设定能够制定HTML元素的属性。也能够指定DOM元素的属性。还能够通过EventTarget.addEventListenser()进行指定。事件的处理包含捕获。目标处理和事件冒泡三个阶段。捕获的过程是:

 window -> document -> html -> body -> div -> button ?

然后处理器运行,冒泡向上传播的过程是遍历DOM树,须要注意的是 focus 不会冒泡。

DOM2中的标准事件有HTMLEvent,MouseEvent。UIEvent和MutationEvent。DOM3 中的事件很多其它:UIEvent,FocusEvent,MouseEvent, WheelEvent, TextEvent,KeyboardEvent 和compositionEvent等,还能够通document.createEvent来自己定义事件。

通过JavaScript 对CSS样式变更的方法有通过className 属性变更class名,通过classList属性更改class名(当中classList 是H5对DOM TokenList接口的实现),还能够更改Style 属性或者直接更改样式表。通过JavaScript能够对屏幕位置(screenX。screenY),窗体位置(clientX,clientY)。文档坐标(pageX。pageY。由浏览器自行实现的),特定元素内的相对位置(layerX,layerY 或offsetX offsetY)进行改动。

通过JavaScript能够对表单中的元素,控件和内容进行验证,可用于验证的事件有submit,focus。blur,change,keydown/up/press。input。使用表单而不产生页面跳转的方式能够是指向到一个 (0,0 )的空iframe。

对于动画而言。css的动画性能一般要更好一些。

AJAX 在Web应用中是不可或缺的。简单地说,是一种不发生页面跳转就能异步加载内容并改写页面内容的技术。主要通过 XMLHttpRequest 对象的创建,实现通/异步通信,处理超时和响应。AJAX有着跨源限制,实现跨源通信的方式有:JSONP。iframe hack,window.postMessage() 以及 XMLHttpRequest Level 2。

HTML5+CSS3+JavaScript的综合使用才可能成就一个Web应用。

H5中的 History API 使用了window属性的history对象监听popstate事件。用于恢复页面状态的处理。ApplicationCache 在html标签的manifest 属性中指定了缓存清单文件的路径,必须通过text/cache-manifest 这一MIME type 来公布缓存清单文件。注意清单中的CACHE,NETWORK,和FALLBACK 的区分。

通过navigator.onLine 能够获知网络状态,还能够通过online/offline事件来侦听连接状态的切换时机。

online/offline事件是document.body 触发的,并传给document对象和window对象。

<p> network is : <span id = “indicator”> (state unknown) </span> </p>
<script>
{
 ? ?function updateIndicator = document.getElementById(‘indicator’);
 ? ?indicator.textContext = navigator.online?’online’:’offline’;
}
document.body.onload = updateIndicator;
document.body.ononline= updateIndicator;
document.body.onoffline = updateIndicator;
</script>

DataTransfer 是Drag Drop API 的核心,在全部拖拽事件的事件对象中,都有该属性,主要是接收数据。

拖拽文件从浏览器保存到桌面:event.dataTransfer.setData(‘DownloadURL’,’MIMETYPE: 文件url’)比如:

 ? ?<a href=“http://a.b.c/abel.pdf” 
 ? ?data-downloadurl = “application/pdf:abel.pdf:http://a.b.c/abel.pdf”
 ? ?class=“dragout” draggable = “true”>download </a>
 ? ?<script>
 ? ?var files = document.querySelectorAll(‘.dragout’);
 ? ?for (var i = 0,file; file =files[i];i++) {
 ? ? ? ?file.addEventListener(‘dragstart’,function(event){
 ? ?event.dataTransfer.setData(“DownloadURL”,this.dataset.downloadurl);
 ? ?},false);
 ? ?}
 ? ?</script>

FileAPI 通过FileReader 读取文件,也能够读取dataURL。FileReaderSync 用于同步读取文件内容,能够在Web Worker 中使用。

Web Storage 为全部源共享5M空间。localStorage 和sessionStorage 的差别在于数据的生命周期。cookie 最大4k,发请求时一起发送,保存会话等重要信息。

indexedDB 能够归为文档型数据库, 作为client存储又一选择。

var indexdb = window.indexDB||window.webkitIndexedDB||window.mozIndexedDB;

Web worker 是H5 的新特性,是宿主环境(浏览器)的功能,JavaScript 本身是不支持多线程的。专用的worker 与创建它的程序之间是一对一的关系。

Web worker 能在另外的线程中创建新的Javascript 运行环境,使JavaScripts能够在后台处理。主线程和工作线程分离。无法使用对方环境的变量。工作线程无法引用document对象,须要通过消息收发完毕数据传递。 在主线程创建工作线程。大约向var worker = new Worker(‘work.js’)这样?在主线程中停止worker的方式是worker.terminate();?worker 自身停止的方式是?self.close();worker 中 能够通个 importScripts 方法,在工作线程内读取外部的文件。

了解了这些基础方式和方法,仅仅是Web应用中JavaScript开发的第一步吧。

服务端应用

技术系统总是又着向超系统进化的趋势。JavaScript 也不例外。

JavaScript 应用于服务端的开发源于2009年初出现的CommonJS,后来成为为了server端javaScript的规范。基于JavaScript没有模块系统、标准库较少、缺乏包管理工具等现状,CommonJS规范希望JavaScript能够在不论什么地方运行,以达到Java、C#、PHP这些后台语言具备开发大型应用的能力。

CommonJS是一种思想,它的终极目标是使应用程序开发人员依据CommonJS API编写的JavaScript应用能够在不同的JavaScript解析器和HOST环境上运行。比如编写服务端应用。命令行工具。基于GUI的桌面应用和混合应用编程等,详情參加 www.commonjs.org 。

NodeJS能够理解成CommonJS规范的一种实现。并且是部分实现。NodeJS以V8作为JavaScript的实现引擎,通用的异步处理事件循环,提供了一系列非堵塞函数库来支持实践循环特性。同一时候。NodeJS提供了高度优化的应用库。来提高server效率,比如其http 模块是为高速非堵塞式http服务而用C语言重写的。另外,NodeJS还有shell的命令行工具,通过包系统实现扩展,扩展列表能够详情參见: GitHub.com/node/wiki/modules。

JavaScript 中的主要实现引擎包含:IE採用的JScript,Firefox採用的SpiderMoneky。Chrome 採用的V8,Safari採用的webkit中的 javacriptcore灯。假设要对引擎有进一步的了解,能够研读一下javascriptcore等相关的源码。

V8 是NodeJS 中的核心引擎。NodeJS的系统架构大致例如以下:

技术分享图片

与浏览器相相应。Node 中的全局变量能够通过?Object.keys(global);?获得。 看一看NodeJS中的 “hello world” 程序:

var http = require(‘http‘);
http.createServer(function (req,res){
 ? ?res.writeHead(200,{‘Content-type‘:‘text/plain‘});
 ? ?res.end(‘Hello Node.js \n‘);
}).listen(1234,"127.0.0.1");
console.log(‘Server running on http://127.0.0.1:1234/‘);

几行代码就实现一个简单web server。 使Pythoner 们联想到了 Tornado, 它们都走在单线程异步IO的路上。

NodeJS 提供了对https 的支持,能够通过openssl 生成证书的方式大致是:

openssl req ?-new -x509 -keyout key.pen -out cert.perm

使用证书的示比例如以下:

var fs ?=require(‘fs’);
var options = {
 ? ?key: fs.readFileSync(‘key.perm’);
 ? ?cert:fs.readFileSync(‘cert.perm’);
}

NodeJS支持socket 和文件处理,配合系统扩展能够使用各种模版语言。

基于NodeJS的实际在业界非常广泛,比方面向websocket的IM系统,各种web应用站点等等。

鉴于微服务架构的兴起。也诞生了基于Node的微服务架构——Seneca。它使用完备的模式匹配接口来连接各个服务,从代码中将传输数据抽象出来。使编写具有高扩展性的软件变得相当easy。

Seneca 没有使用依赖注入,可是在处理控制反转上相当灵活,没有keyword和强制的字段。仅仅需一组键值对。用于模式匹配的引擎中。详细參考实现,能够參考《node.js 微服务》一书。

基于JavaScript的全栈

假设在整个应用系统中主要使用JavaScript编程语言作为技术栈,那么也能够成为基于JavaScript 的全栈,关于全栈的论述能够參加《全栈的技术栈设想》和《再谈< 全栈架构师>》两篇文字。

比如MEAN架构。即MongoDB + Express + Angular + Node,MEAN 技术栈代表着一种全然现代的 Web 开发方法:一种语言运行在应用程序的全部层次上,从client到server,再到持久层。

借助JavaScript的測试框架。比方MochaJSJasmineJS?和?KarmaJS,能够为自己的 MEAN 应用程序编写深入而又全面的測试套件,说MEAN有代替LAMP/LNMP的的趋势。但还需保持慎重。

引擎的差异

正像Java 那样。虽然又着虚拟机规范。但各个JVM的实现还是有着些许的不同,JavaScript 也是如此。JavaScript各引擎中相同存在着少量的限制,比如:

  • 字符串常量中同意的最大字符数

  • 作为參数传递到函数中的数据大小(栈大小)

  • 函数声明中的參数个数

  • 函数调用链的最大长度

  • 以堵塞方式在浏览器中运行的最大时间

  • 变量名的最大长度

虽然如此,JavaScript 在浏览器中的表现还是基本上可信的。

从软件到硬件

实际上,JavaScript已经嵌入到了从机器人到各种家电等各种各样的设备中。这里隆重推荐我非常敬佩的好友——周爱民老师,他在Ruff(南潮信息科技)做的事情就是JavaScript 在物联网上的进一步应用。

Ruff 是一个能够让开发人员实现敏捷开发智能硬件的系统平台。它包含了Ruff SDK、Ruff OS。Rap Registry等。从技术上讲,Ruff 是一个 JavaScript 运行时,专为硬件开发而设计。

Ruff 对硬件进行了抽象,使用了基于事件驱动、异步 I/O 的模型,使硬件开发变得轻量并且高效。硬件抽象层,使得操作硬件宛如普通程序库,减少了硬件领域进入门槛。

Ruff 为开发人员提供了完好的开发服务。

从项目生产、软件包管理、应用管理、外设管理到固件管理等一系列现代软件开发方式。PC 端完毕开发,无需烧板子。提升开发人员的开发效率。

Ruff 还提供了完好的測试框架,支持 assert、test、mock 等模块,在开发机上測试逻辑,硬件測试也能 TDD。

官网(ruff.io) 上给出的示比例如以下:

$.ready(function() {
 ? $(‘#led-0‘).turnOn();
});

打开电路板上的一个LED 灯,就是如此的简单。

眼下,一个 Ruff 硬件同一时候仅仅能运行一款 Ruff 应用,它将拥有自己独立的进程。着可能也受到JavaScript自身的限制吧。

关注性能

性能是全栈关注的一个重要维度。那句“过早优化是万恶之源”实际上是我们对高德纳先生的断章取义,原文是这种:

我们应该在比如97%的时间里,忘掉小处的效率;
过早优化是万恶之源。


但我们不应该错过关键的3%中的机会。

实际上是非关键路径上的优化是万恶之源,问题在于怎样确定我们的代码是否在关键路径上。

不论节省的时间多么少,花费在关键路径上的性能优化都是值得的。

对于性能优化工具,用于JavaScript源码压缩有 google Closure complier。packer。YUI compressor,JSmin等。页面的性能优化工具有YSlow 和Page Speed等。

实际上。不论什么有意义且可靠的性能測试都是基于统计学上的合理实践。 就JavaScript 代码本身的性能而言,benchmarkjs 是一个非常好的工具,而jsperf.com 提供了对JavaScript 运行环境的性能測试。

总之,JavaScript 是一个具有强大生命力的语言,前端框架更是日新月异。从Angular。Vue。到React, 乃至React Native,给人以目不暇接的感觉,可是,老码农认为基础认识还是非常必要的,勿在浮沙筑高塔。

參考阅读

https://developer.mozilla.org/en-US/docs/Web/JavaScript?

https://tc39.github.io/ecma262/#sec-global-object


技术分享图片

微信扫一扫
关注该公众号

全栈必备 JavaScript基础

标签:简单   流程控制   事件队列   模式匹配   pos   alt   data   新特性   安全   

原文地址:https://www.cnblogs.com/llguanli/p/8666349.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!