java程序员的NodeJS初识篇
摘要
作为一个一直用java来写后端的程序员用NodeJS来写后台,实在不是很爽。这里记下这两个月的NodeJS学习所遇之坑,与java转NodeJS的同仁共勉。学习时间不长,若有理解错误,望指正。
一.JS基本
exports,module.exports
- exports 就是module.exports的引用
- 在module 被计算之前,会将module.exports的值赋给exports
- 当module.exports赋值之后,再对exports改值,不会影响module.exports的值,而外部
require module的时候,如果module.exports有定义,调用的是module.exports的值。
eg1.
module.exports.hello = true; // Exported from require of module
exports = { hello: false }; // Not exported, only available in the module
eg2.
module.js
module.exports = 'a';
exports.B = 'b';
call.js
var module = require('module');
console.log(module.B);
//undefined
- exports 就是传统的module实例,可以exports 变量,也可以exports 方法,调用的时候都一样,通过
instance.xxx来调用
eg. module.js
exports.A = 'a'
exports.add = function add(a, b){
return a+b;
}
call.js
var module = require('module');
module.A;
module.add(xx,xx);
- module.exports可以将module,exports成任何合法的js 类型,boolean,JSON,function都可以。
从这点上说,就不能将javascript中的module类比成java中的类/实例。
module.exports = function (a, b) {
return a + b;
}
import 某个module的属性
与直接import不同
import { test } from ‘xxx’
issue
class 与instance
js中并没有class,一切皆为对象。面向对象的实现是通过prototype来实现的。
例如一般在某个文件中调用某个js文件,就是在文件开始部分require,然后在文件中
各处引用。而这点与java不同,java是首先import,然后具体用的时候需要new 一个instance再使用
js中require 以后直接使用。不需要new instance。所以各处用的都是同一个instance。
如果想new一个object,创建新的instance,则需要使用原型。这样每个new出来的对象就有对应的
原型方法了。
eg.
test.js
// 类似于构造函数
function Test() {
}
Test.prototype.add = function (a,b) {
return a+b;
};
module.exports = Test;
call.js
require Test from 'test';
function sub(){
var test1 = new Test();
test1.add();
}
作用域
对于方法而言,js没有类似于java的那么强的域控制public,protected,private
就分内部与外部,只在函数内部使用的就不要export出去。
二.异步回调
js 代码写起来和java代码最大的不同就是回调了。
java代码基本消灭了随意跳转的goto。阅读代码块或者写代码块时基本就是按照从上
到下的顺序即可。因为java代码都是同步执行的。而JS很多都是异步执行,所以如果你想
你的逻辑是顺序执行的话,必须等待异步执行返回结果后,再去执行下面的代码。因为js
方法大多是非阻塞。
—-2017.3.14更新———-
其实这代表了两种不同的并发处理方式,一种是java的,基于线程的并发,一个task一个线程。
写起来也是顺序执行的。但是task增多,多个线程之间切换代价昂贵,可能会导致吞吐量下降。
而nodejs,则是基于事件的并发,单线程处理事件,每个并发流实现为一个有限状态机。所以需要回调。应用直接控制。但是当并发负载增加的时候,吞吐量饱和响应时间线性增长
还有一种的话就是之前介绍的cassandra实现的SEDA模型
eg.实现查mongo数据库
java code
MongoClient client = new MongoClient(url);
Collection coll = client.getCollection('test');
ResultSet<String> rs = coll.find({});
而js则是
MongoClient.connect(url, function(err, db) {
//回调,等待连接成功,才能执行下一步。
if (err) {
callback(err);
}
var coll = db.collection('test');
coll.find({name:"mike"}).toArray(function(err, results) {
console.log(results);
db.close();
});
});
三.内存溢出
NodeJs使用google V8来管理内存,V8会将js代码编译为本地代码,然后执行它。
V8会按需进行内存的分配和释放。和JVM差不多了。将内存区域分区,
- 代码区域
- 栈
值类型的数据,内部变量,控制程序的指针。 - 堆
保存引用类型(对象,字符串,闭包)
在做mongodb数据大量插入的时候遇到过一次内存溢出的问题,所以需要分析溢出原因,
java中一般是dump处文件,然后用其他工具分析对象。NodeJs也类似。
var heapdump = require('heapdump');
heapdump.writeSnapshot();
然后在chrome中的Profile工具来分析溢出对象。但是实际中此效果不好,原因是chrome的内存不够大(可能需要调整浏览器内存大小),另外结果不是很直观。还可以使用util包中输出内存占用
var util = require('util');
console.log(util.inspect(process.memoryUsage()));
内存溢出原因:
在上面提到过在js中基本都是回调函数,mongo插入同样也是。使用mongo.insertMany(array)来批量插入提高性能。同时充分利用异步特点,使用async.each来控制,模拟多线程并发。但是这边就存在一个问题,有经验的老手就能看出来了,插入的数据array占用的内存什么时候释放。mongo.insertMany调用开始,到真正插入到db中需要一定时间。插入上千万条数据的,内存回收不及时的话肯定是要溢出的。所以要加一层并发控制,比如说以10万条数据为内层并发,这些数据的插入是并发操作的,无序的,等这一批数据插入成功后,再进行下一批数据插入,释放内存。
四.打包
java中依赖其他包,使用jar包。build工具可以用maven,gradle。
nodejs中依赖其他包,使用module.使用npm来构建
在package.json中的 files属性中定义要打包的文件
“files”: [
“src/publish”,
],
在main属性中定义main文件
“main”: “src/index.js”,
使用npm publish命令到repository。如果是本地的调用,使用npm pack 打包
在其他project中使用 npm install -s $PATH/publish-service-0.0.1.tgz添加引用
五.参考
http://www.hacksparrow.com/node-js-exports-vs-module-exports.html
https://cnodejs.org/topic/55accdeab4ab1d7d02bf0d8c
http://wwsun.github.io/posts/understanding-nodejs-gc.html
java程序员的NodeJS初识篇的更多相关文章
- 为何我建议1-3年的Java程序员仔细看看这篇文章
此文的目的是为了督促自己去不断学习,让自己有更明确的方向去提升自己.以技能树为基础,以面试要点为大纲,我觉得比抓住什么看什么要更有目的,更能坚持下去.世界瞬息万变,我们要时刻准备着.时刻提高着自己,才 ...
- Java程序员金三银四精心准备的面试题及答案(基础篇)
1.面向对象的特征有哪些方面? [基础] 答:面向对象的特征主要有以下几个方面: 1)抽象:抽象就是忽略一个主题中与当前目标无关的那些方面,以便更充分地注意与当前目标有关的方面.抽象并不打算了解全部问 ...
- Java 程序员们值得一看的好书推荐
"学习的最好途径就是看书",这是我自己学习并且小有了一定的积累之后的第一体会.个人认为看书有两点好处: 能出版出来的书一定是经过反复的思考.雕琢和审核的,因此从专业性的角度来说,一 ...
- 分享下对JAVA程序员成长之路的总结<转>
我也搞了几年JAVA了,由于一向懒惰,没有成为大牛,只是一普通程序猿,手痒来给新人分享下从新手成长为老鸟的已见. 首先初识语法的阶段,必须要学会怎么操作对象,操作if和for,操作list set ...
- Java 程序员们值得一看的好书推荐[转载]
“学习的最好途径就是看书“,这是我自己学习并且小有了一定的积累之后的第一体会.个人认为看书有两点好处: 能出版出来的书一定是经过反复的思考.雕琢和审核的,因此从专业性的角度来说,一本好书的价值远超其他 ...
- Java教程-Java 程序员们值得一看的好书推荐
学习的最好途径就是看书“,这是我自己学习并且小有了一定的积累之后的第一体会.个人认为看书有两点好处: 能出版出来的书一定是经过反复的思考.雕琢和审核的,因此从专业性的角度来说,一本好书的价值远超其他资 ...
- 【转载】分享下多年积累的对JAVA程序员成长之路的总结
注:该文是从百度贴吧转载过来,之前看到觉得写得还不错,对Java开发学习者来说很有意义的,可以看看. 我也搞了几年JAVA了,由于一向懒惰,没有成为大牛,只是一普通程序猿,不爱玩社交网站,不爱玩微博, ...
- 分享下多年积累的对JAVA程序员成长之路的总结
http://blog.csdn.net/zhongzelin/article/details/8643269我也搞了几年JAVA了,由于一向懒惰,没有成为大牛,只是一普通程序猿,不爱玩社交网站,不爱 ...
- 前后端分离时代,Java 程序员的变与不变!
事情的起因是这样的,有个星球的小伙伴向邀请松哥在知乎上回答一个问题,原题是: 前后端分离的时代,Java后台程序员的技术建议? 松哥认真看了下这个问题,感觉对于初次接触前后端分离的小伙伴来说,可能都会 ...
随机推荐
- JAVA-MyBaits对应XML的两种使用方式
概述 在使用XML写SQL语句的时候,遇到参数传递的两种方式,也就是Mapper里面带@Param注解和不带的情况,容易混淆,对应XML的写法也不相同,使用的时候要注意对照代码比对(备注XML里面的关 ...
- Web程序-----批量生成二维码并形成一张图片
需求场景:客户根据前台界面列表所选择的数据,根据需要的信息批量生成二维码并形成一张图片,并且每张图片显示的二维码数量是固定的,需要分页(即总共生成的二维码图片超出每页显示的需另起一页生成),并下载到客 ...
- pycharm的list的应用
li = [11,22,22,33,44] v = li.count(22) print (v) #输出结果2 #计算元素的次数 count的应用 li = [11,22,33,22,44] li.e ...
- mstsc的事 随笔
当个备份吧, 记不得了,就翻一下自己的博客. MSTSC 设置, 平台:Windows 10 企业版 Windows 10 企业版,功能最全.
- PBRT笔记(10)——体积散射
体散射处理过程 3个影响参与介质在环境中的辐射度分布的主要因素: 吸收:减少光能,并将其转化为别的能量,例如热量. 发光:由光子发射光能至环境中. 散射:由于粒子碰撞,使得一个方向的辐射度散射至其他方 ...
- Handler Timer TimerTask ScheduledExecutor 循环任务解析
使用Handler执行循环任务 private Handler handler = new Handler(); private int mDelayTime = 1000; private Runn ...
- JUC
1.Java JUC简介 在Java5.0提供了java.util.concurrent(简称JUC)包,在此包中增加了在并发编程中很常用的实用工具类,用于定义类似于线程的自定义子系统,包括线程池.异 ...
- Ubuntu wpa 代替network-manager
1. Ubuntu启动时,如果出现60秒等待:Waiting up to 60 seconds for network configuration 解决方法: a. /etc/init ,打开fail ...
- python d:\test.py File "<stdin>", line 1 python d:\test.py ^ SyntaxError: invalid syntax
pyhton出错: python d:\test.py File "<stdin>", line 1 python d:\test.py ^SyntaxError: i ...
- 让数字变化炫酷起来,数字滚动Text组件[Unity]
让数字滚动起来 上周我的策划又提了样需求,当玩家评分发生变动时,屏幕出现人物评分浮层UI,播放评分数字滚动动画.这类数字滚动需求非常常见,我就按一般思路,将startvalue与endvalue每隔一 ...