# 第一個Node程序
在前面的所有章節中,我們編寫的JavaScript代碼都是在瀏覽器中運行的,因此,我們可以直接在瀏覽器中敲代碼,然后直接運行。
從本章開始,我們編寫的JavaScript代碼將_不能_在瀏覽器環境中執行了,而是在Node環境中執行,因此,JavaScript代碼將直接在你的計算機上以命令行的方式運行,所以,我們要先選擇一個文本編輯器來編寫JavaScript代碼,并且把它保存到本地硬盤的某個目錄,才能夠執行。
那么問題來了:文本編輯器到底哪家強?
推薦兩款文本編輯器:
一個是[Sublime Text](http://www.sublimetext.com/),免費使用,但是不付費會彈出提示框:

一個是[Notepad++](http://notepad-plus-plus.org/),免費使用,有中文界面:

請注意,用哪個都行,但是絕對不能用Word和寫字板,Windows自帶的記事本也強烈不推薦使用。Word和寫字板保存的不是純文本文件,而記事本會自作聰明地在文件開始的地方加上幾個特殊字符(UTF-8 BOM),結果經常會導致程序運行出現莫名其妙的錯誤。
安裝好文本編輯器后,輸入以下代碼:
```
'use strict';
console.log('Hello, world.');
```
第一行總是寫上`'use strict';`是因為我們總是以嚴格模式運行JavaScript代碼,避免各種潛在陷阱。
然后,選擇一個目錄,例如`C:\Workspace`,把文件保存為`hello.js`,就可以打開命令行窗口,把當前目錄切換到`hello.js`所在目錄,然后輸入以下命令運行這個程序了:
```
C:\Workspace>node hello.js
Hello, world.
```
也可以保存為別的名字,比如`first.js`,但是必須要以`.js`結尾。此外,文件名只能是英文字母、數字和下劃線的組合。
如果當前目錄下沒有`hello.js`這個文件,運行`node hello.js`就會報錯:
```
C:\Workspace>node hello.js
module.js:338
throw err;
^
Error: Cannot find module 'C:\Workspace\hello.js'
at Function.Module._resolveFilename
at Function.Module._load
at Function.Module.runMain
at startup
at node.js
```
報錯的意思就是,沒有找到`hello.js`這個文件,因為文件不存在。這個時候,就要檢查一下當前目錄下是否有這個文件了。
## 命令行模式和Node交互模式
請注意區分命令行模式和Node交互模式。
看到類似`C:\>`是在Windows提供的命令行模式:

在命令行模式下,可以執行`node`進入Node交互式環境,也可以執行`node hello.js`運行一個`.js`文件。
看到`>`是在Node交互式環境下:

在Node交互式環境下,我們可以輸入JavaScript代碼并立刻執行。
此外,在命令行模式運行`.js`文件和在Node交互式環境下直接運行JavaScript代碼有所不同。Node交互式環境會把每一行JavaScript代碼的結果自動打印出來,但是,直接運行JavaScript文件卻不會。
例如,在Node交互式環境下,輸入:
```
> 100 + 200 + 300;
600
```
直接可以看到結果`600`。
但是,寫一個`calc.js`的文件,內容如下:
```
100 + 200 + 300;
```
然后在命令行模式下執行:
```
C:\Workspace>node calc.js
```
發現什么輸出都沒有。
這是正常的。想要輸出結果,必須自己用`console.log()`打印出來。把`calc.js`改造一下:
```
console.log(100 + 200 + 300);
```
再執行,就可以看到結果:
```
C:\Workspace>node calc.js
600
```
## 小結
用文本編輯器寫JavaScript程序,然后保存為后綴為`.js`的文件,就可以用node直接運行這個程序了。
Node的交互模式和直接運行`.js`文件有什么區別呢?
直接輸入`node`進入交互模式,相當于啟動了Node解釋器,但是等待你一行一行地輸入源代碼,每輸入一行就執行一行。
直接運行`node hello.js`文件相當于啟動了Node解釋器,然后一次性把`hello.js`文件的源代碼給執行了,你是沒有機會以交互的方式輸入源代碼的。
在編寫JavaScript代碼的時候,完全可以一邊在文本編輯器里寫代碼,一邊開一個Node交互式命令窗口,在寫代碼的過程中,把部分代碼粘到命令行去驗證,事半功倍!前提是得有個27'的超大顯示器!
## 參考源碼
[hello.js](https://github.com/michaelliao/learn-javascript/blob/master/samples/node/basic/hello.js)
- JavaScript教程
- JavaScript簡介
- 快速入門
- 基本語法
- 數據類型和變量
- 字符串
- 數組
- 對象
- 條件判斷
- 循環
- Map和Set
- iterable
- 函數
- 函數定義和調用
- 變量作用域
- 方法
- 高階函數
- map/reduce
- filter
- sort
- 閉包
- 箭頭函數
- generator
- 標準對象
- Date
- RegExp
- JSON
- 面向對象編程
- 創建對象
- 原型繼承
- 瀏覽器
- 瀏覽器對象
- 操作DOM
- 更新DOM
- 插入DOM
- 刪除DOM
- 操作表單
- 操作文件
- AJAX
- Promise
- Canvas
- jQuery
- 選擇器
- 層級選擇器
- 查找和過濾
- 操作DOM
- 修改DOM結構
- 事件
- 動畫
- 擴展
- underscore
- Collections
- Arrays
- Functions
- Objects
- Chaining
- Node.js
- 安裝Node.js和npm
- 第一個Node程序
- 模塊
- 基本模塊
- fs
- stream
- http
- buffer
- Web開發
- koa
- mysql
- swig
- 自動化工具
- 期末總結
- Python 2.7教程
- Python簡介
- 安裝Python
- Python解釋器
- 第一個Python程序
- 使用文本編輯器
- 輸入和輸出
- Python基礎
- 數據類型和變量
- 字符串和編碼
- 使用list和tuple
- 條件判斷和循環
- 使用dict和set
- 函數
- 調用函數
- 定義函數
- 函數的參數
- 遞歸函數
- 高級特性
- 切片
- 迭代
- 列表生成式
- 生成器
- 函數式編程
- 高階函數
- map/reduce
- filter
- sorted
- 返回函數
- 匿名函數
- 裝飾器
- 偏函數
- 模塊
- 使用模塊
- 安裝第三方模塊
- 使用__future__
- 面向對象編程
- 類和實例
- 訪問限制
- 繼承和多態
- 獲取對象信息
- 面向對象高級編程
- 使用__slots__
- 使用@property
- 多重繼承
- 定制類
- 使用元類
- 錯誤、調試和測試
- 錯誤處理
- 調試
- 單元測試
- 文檔測試
- IO編程
- 文件讀寫
- 操作文件和目錄
- 序列化
- 進程和線程
- 多進程
- 多線程
- ThreadLocal
- 進程 vs. 線程
- 分布式進程
- 正則表達式
- 常用內建模塊
- collections
- base64
- struct
- hashlib
- itertools
- XML
- HTMLParser
- 常用第三方模塊
- PIL
- 圖形界面
- 網絡編程
- TCP/IP簡介
- TCP編程
- UDP編程
- 電子郵件
- SMTP發送郵件
- POP3收取郵件
- 訪問數據庫
- 使用SQLite
- 使用MySQL
- 使用SQLAlchemy
- Web開發
- HTTP協議簡介
- HTML簡介
- WSGI接口
- 使用Web框架
- 使用模板
- 協程
- gevent
- 實戰
- Day 1 - 搭建開發環境
- Day 2 - 編寫數據庫模塊
- Day 3 - 編寫ORM
- Day 4 - 編寫Model
- Day 5 - 編寫Web框架
- Day 6 - 添加配置文件
- Day 7 - 編寫MVC
- Day 8 - 構建前端
- Day 9 - 編寫API
- Day 10 - 用戶注冊和登錄
- Day 11 - 編寫日志創建頁
- Day 12 - 編寫日志列表頁
- Day 13 - 提升開發效率
- Day 14 - 完成Web App
- Day 15 - 部署Web App
- Day 16 - 編寫移動App
- 期末總結
- Python3教程
- Python簡介
- 安裝Python
- Python解釋器
- 第一個Python程序
- 使用文本編輯器
- Python代碼運行助手
- 輸入和輸出
- Python基礎
- 數據類型和變量
- 字符串和編碼
- 使用list和tuple
- 條件判斷
- 循環
- 使用dict和set
- 函數
- 調用函數
- 定義函數
- 函數的參數
- 遞歸函數
- 高級特性
- 切片
- 迭代
- 列表生成式
- 生成器
- 迭代器
- 函數式編程
- 高階函數
- map/reduce
- filter
- sorted
- 返回函數
- 匿名函數
- 裝飾器
- 偏函數
- 模塊
- 使用模塊
- 安裝第三方模塊
- 面向對象編程
- 類和實例
- 訪問限制
- 繼承和多態
- 獲取對象信息
- 實例屬性和類屬性
- 面向對象高級編程
- 使用__slots__
- 使用@property
- 多重繼承
- 定制類
- 使用枚舉類
- 使用元類
- 錯誤、調試和測試
- 錯誤處理
- 調試
- 單元測試
- 文檔測試
- IO編程
- 文件讀寫
- StringIO和BytesIO
- 操作文件和目錄
- 序列化
- 進程和線程
- 多進程
- 多線程
- ThreadLocal
- 進程 vs. 線程
- 分布式進程
- 正則表達式
- 常用內建模塊
- datetime
- collections
- base64
- struct
- hashlib
- itertools
- XML
- HTMLParser
- urllib
- 常用第三方模塊
- PIL
- virtualenv
- 圖形界面
- 網絡編程
- TCP/IP簡介
- TCP編程
- UDP編程
- 電子郵件
- SMTP發送郵件
- POP3收取郵件
- 訪問數據庫
- 使用SQLite
- 使用MySQL
- 使用SQLAlchemy
- Web開發
- HTTP協議簡介
- HTML簡介
- WSGI接口
- 使用Web框架
- 使用模板
- 異步IO
- 協程
- asyncio
- async/await
- aiohttp
- 實戰
- Day 1 - 搭建開發環境
- Day 2 - 編寫Web App骨架
- Day 3 - 編寫ORM
- Day 4 - 編寫Model
- Day 5 - 編寫Web框架
- Day 6 - 編寫配置文件
- Day 7 - 編寫MVC
- Day 8 - 構建前端
- Day 9 - 編寫API
- Day 10 - 用戶注冊和登錄
- Day 11 - 編寫日志創建頁
- Day 12 - 編寫日志列表頁
- Day 13 - 提升開發效率
- Day 14 - 完成Web App
- Day 15 - 部署Web App
- Day 16 - 編寫移動App
- FAQ
- 期末總結
- Git教程
- Git簡介
- Git的誕生
- 集中式vs分布式
- 安裝Git
- 創建版本庫
- 時光機穿梭
- 版本回退
- 工作區和暫存區
- 管理修改
- 撤銷修改
- 刪除文件
- 遠程倉庫
- 添加遠程庫
- 從遠程庫克隆
- 分支管理
- 創建與合并分支
- 解決沖突
- 分支管理策略
- Bug分支
- Feature分支
- 多人協作
- 標簽管理
- 創建標簽
- 操作標簽
- 使用GitHub
- 自定義Git
- 忽略特殊文件
- 配置別名
- 搭建Git服務器
- 期末總結