摘要:不使用框架創(chuàng)建簡(jiǎn)單的應(yīng)用需求創(chuàng)建一個(gè)可以上傳圖片的應(yīng)用。自定義模塊剛才我們定義了一個(gè)簡(jiǎn)單的服務(wù)。處理數(shù)據(jù)存儲(chǔ)本地安裝服務(wù)安裝客戶端使用命令行新建文件修改待續(xù)如何使用創(chuàng)建一個(gè)代理服務(wù)器的適用場(chǎng)景
不使用框架創(chuàng)建簡(jiǎn)單的Node.js web應(yīng)用
需求: 創(chuàng)建一個(gè)可以上傳圖片的web應(yīng)用。用戶可以瀏覽應(yīng)用,有一個(gè)文件上傳的表單。選擇圖片上傳,上傳完成之后可以預(yù)覽圖片。上傳的圖片信息需要入庫(mysql)。一個(gè)簡(jiǎn)單的http服務(wù)
const http = require("http"); // 創(chuàng)建一個(gè)服務(wù) const server = http.createServer((req, res) => { res.writeHead(200, { "Content-Type": "text/plain" }); res.end("Hello World"); }); // 服務(wù)錯(cuò)誤監(jiān)聽 server.on("clientError", (err, socket) => { socket.end("HTTP/1.1 400 Bad Request "); }); // 當(dāng)前服務(wù)監(jiān)聽的端口 server.listen(8888); console.log("Server has started.");
涉及到的api:
http.createServer
res.writeHead
res.end
當(dāng)我們?cè)诰帉憂ode服務(wù)的時(shí)候,如果服務(wù)器有異常,那node服務(wù)就直接掛掉了。那如何保證我們的node服務(wù)不會(huì)關(guān)閉,并且會(huì)自動(dòng)重啟呢?
或者是我們修改了代碼,如何實(shí)現(xiàn)修改的代碼直接生效而不用重新手動(dòng)重啟node服務(wù)呢?
npm install -g nodemon
在生產(chǎn)環(huán)境我一般使用pm2來管理node服務(wù)。
自定義模塊剛才我們定義了一個(gè)簡(jiǎn)單的http服務(wù)。其中http是一個(gè)內(nèi)置的模塊。那其實(shí)我們的服務(wù)都會(huì)有一個(gè)入口文件,在這里我們定義為index.js。那我們?nèi)绾蜗褚胔ttp內(nèi)置模塊一樣,在index.js里使用server.js呢?
exports 與 module.exports 的區(qū)別:
exports是module.exports的簡(jiǎn)寫。如果修改了exports的引用,也就是重新給exports賦值,則exports只是在當(dāng)前文件級(jí)作用域內(nèi)可用,并且exports的修改不會(huì)影響到module.exports的值。
server.js
const http = require("http"); function start() { const server = http.createServer((req, res) => { res.writeHead(200, { "Content-Type": "text/plain" }); res.end("Hello World"); }); server.on("clientError", (err, socket) => { socket.end("HTTP/1.1 400 Bad Request "); }); server.listen(8888); console.log("Server has started."); } // 通過給exports添加【屬性值】的方式導(dǎo)出方法 // 或者通過給module.exports添加屬性值 exports.start = start;
index.js
const server = require("./server"); server.start();
node index.js路由處理
我們知道,訪問一個(gè)web網(wǎng)站會(huì)有不同的頁面或者會(huì)調(diào)用不同的接口,那這些就對(duì)應(yīng)這不同的請(qǐng)求路徑,同時(shí)這些請(qǐng)求還會(huì)對(duì)應(yīng)不同的請(qǐng)求方法(GET, POST等)。那node如何針對(duì)這些不同的請(qǐng)求路徑去匹配對(duì)應(yīng)的處理函數(shù)呢?
為了處理http請(qǐng)求的參數(shù),我們需要獲取到http請(qǐng)求的request,從中獲取到請(qǐng)求方式以及請(qǐng)求路徑。在這里會(huì)依賴 url內(nèi)置模塊。
首先建立routes文件夾存放路由處理
routes/index.js
module.exports = (req, res) => { res.writeHead(200, { "Content-Type": "text/plain" }); res.end("Hello World"); }
routes/upload.js
module.exports = (req, res) => { res.writeHead(200, { "Content-Type": "text/plain" }); res.end("upload file"); }
新建route.js文件處理路由
function route(handle, pathname, req, res) { if (typeof handle[pathname] === "function") { handle[pathname](req, res); } else { console.log("No request handler found for " + pathname); res.end("404 Not found"); } } exports.route = route;
修改server.js
const http = require("http"); const url = require("url"); const routes = {}; function use(path, routeHandler) { routes[path] = routeHandler; } function start(route) { function handleRequest(req, res) { const pathname = url.parse(req.url).pathname; route(routes, pathname, req, res) } const server = http.createServer(handleRequest); server.on("clientError", (err, socket) => { socket.end("HTTP/1.1 400 Bad Request "); }); server.listen(8888); console.log("Server has started."); } module.exports = { start, use };
修改index.js
const server = require("./server"); const index = require("./routes/index"); const upload = require("./routes/upload"); const router = require("./route"); server.use("/", index); server.use("/upload", upload); server.start(router.route);處理POST請(qǐng)求
我們顯示一個(gè)文本區(qū)(textarea)供用戶輸入內(nèi)容,然后通過POST請(qǐng)求提交給服務(wù)器。最后,服務(wù)器接受到請(qǐng)求,通過處理程序?qū)⑤斎氲膬?nèi)容展示到瀏覽器中。
給request注冊(cè)監(jiān)聽事件
request.addListener("data", function(chunk) { // called when a new chunk of data was received }); request.addListener("end", function() { // called when all chunks of data have been received });
querystring登場(chǎng),解析上傳數(shù)據(jù)
修改server.js里的handleRequest方法
function handleRequest(req, res) { const pathname = url.parse(req.url).pathname; let postData = ""; // 設(shè)置編碼 req.setEncoding("utf8"); req.addListener("data", function(postDataChunk) { postData += postDataChunk; console.log("Received POST data chunk ""+ postDataChunk + ""."); }); req.addListener("end", function() { route(routes, pathname, req, res, postData); }); }
route.js多加一個(gè)參數(shù)postData
function route(handle, pathname, req, res, postData) { if (typeof handle[pathname] === "function") { handle[pathname](req, res, postData); } else { console.log("No request handler found for " + pathname); res.end("404 Not found"); } } exports.route = route;
index.js
const exec = require("child_process").exec; module.exports = (req, res, postData) => { // 可以使用node模板 const body = ""+ ""+ ""+ ""+ ""+ ""+ ""+ ""; res.end(body); }
upload.js 修改
const querystring = require("querystring"); module.exports = (req, res, postData) => { const content = querystring.parse(postData).text; res.writeHead(200, { "Content-Type": "text/plain; charset=utf-8"}); res.end(content || "Empty"); }處理文件上傳
npm install formidable
server.js修改
function handleRequest(req, res) { const pathname = url.parse(req.url).pathname; route(routes, pathname, req, res); }
routes/index.js
const exec = require("child_process").exec; module.exports = (req, res) => { // 可以使用node模板 const body = ""+ ""+ ""+ ""+ ""+ ""+ ""+ ""; res.writeHead(200, {"Content-Type": "text/html; charset=utf-8"}); res.end(body); }
showFile.js獲取磁盤圖片信息
const fs = require("fs"); const path = require("path"); function show(req, res) { const rootPath = path.resolve(); fs.readFile(`${rootPath}/tmp/test.png`, "binary", function(error, file) { if(error) { res.writeHead(500, {"Content-Type": "text/plain"}); res.write(error + " "); res.end(); } else { res.writeHead(200, {"Content-Type": "image/png"}); res.write(file, "binary"); res.end(); } }); } module.exports = show;
routes/uoload.js
const fs = require("fs"); const formidable = require("formidable"); const path = require("path"); module.exports = (req, res) => { const form = new formidable.IncomingForm(); const rootPath = path.resolve(); form.parse(req, function(error, fields, files) { console.log("parsing done"); fs.renameSync(files.upload.path, `${rootPath}/tmp/test.png`); res.writeHead(200, {"Content-Type": "text/html; charset=utf-8"}); res.write("received image:
"); res.write(""); res.end(); }); }
同時(shí)在index.js中添加相應(yīng)的路由。
處理數(shù)據(jù)存儲(chǔ)本地安裝mysql服務(wù)
安裝mysql客戶端/使用命令行
npm install mysql
CREATE SCHEMA `nodestart` ; CREATE TABLE `nodestart`.`file` ( `id` INT NOT NULL AUTO_INCREMENT, `filename` VARCHAR(300) NULL, `path` VARCHAR(500) NULL, `size` VARCHAR(45) NULL, `type` VARCHAR(45) NULL, `uploadtime` VARCHAR(45) NULL, PRIMARY KEY (`id`));
新建db.js文件
const mysql = require("mysql"); const connection = mysql.createConnection({ host : "localhost", user : "root", password : "123456", database : "nodestart" }); function query(sql, params) { return new Promise((resolve, reject) => { connection.connect(); connection.query(sql, params, function (error, results, fields) { if (error) reject(error); console.log("The solution is: ", results); resolve(fields); }); connection.end(); }); } exports.query = query;
修改routes/upload.js
const fs = require("fs"); const formidable = require("formidable"); const path = require("path"); const db = require("../db"); module.exports = (req, res) => { const form = new formidable.IncomingForm(); const rootPath = path.resolve(); form.parse(req, function(error, fields, files) { console.log("parsing done"); fs.renameSync(files.upload.path, `${rootPath}/tmp/test.png`); const fileObj = { size: files.upload.size, path: `${rootPath}/tmp/test.png`, filename: files.upload.name, type: files.upload.type, uploadtime: Date.now() } db.query("insert into nodestart.file set ?", fileObj).then((res) => { console.log(res) }).catch((err) => { console.log(err); }) res.writeHead(200, {"Content-Type": "text/html; charset=utf-8"}); res.write("received image:
"); res.write(""); res.end(); }); }
待續(xù):
如何使用Node創(chuàng)建一個(gè)代理服務(wù)器 Node 的適用場(chǎng)景文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/105738.html
摘要:在年成為最大贏家,贏得了實(shí)現(xiàn)的風(fēng)暴之戰(zhàn)。和他的競(jìng)爭(zhēng)者位列第二沒有前端開發(fā)者可以忽視和它的生態(tài)系統(tǒng)。他的殺手級(jí)特性是探測(cè)功能,通過檢查任何用戶的功能,以直觀的方式讓開發(fā)人員檢查所有端點(diǎn)。 2016 JavaScript 后起之秀 本文轉(zhuǎn)載自:眾成翻譯譯者:zxhycxq鏈接:http://www.zcfy.cc/article/2410原文:https://risingstars2016...
摘要:在,是當(dāng)之無愧的王者,贏得了與之間的戰(zhàn)爭(zhēng),攻陷了的城池。于月發(fā)布了版本,這一版本為了更好的表現(xiàn)加入了渲染方式。前端框架這個(gè)前端框架清單可能是年疲勞的元兇之一。的創(chuàng)建者,目前在工作為尋找構(gòu)建簡(jiǎn)單性和自主配置性之間的平衡做了很大的貢獻(xiàn)。 春節(jié)后的第一篇就從這個(gè)開始吧~本文已在前端早讀課公眾號(hào)上首發(fā) 原文鏈接 JavasScript社區(qū)在創(chuàng)新的道路上開足了馬力,曾經(jīng)流行過的也許一個(gè)月之后就過...
摘要:它不僅從前端移動(dòng)到后端,我們也開始看到它用于機(jī)器學(xué)習(xí)和增強(qiáng)現(xiàn)實(shí),簡(jiǎn)稱。由于其高使用率,年的現(xiàn)狀調(diào)查將其稱為采用的安全技術(shù)。機(jī)器學(xué)習(xí)框架在年的開發(fā)者峰會(huì)上,宣布了他們的機(jī)器學(xué)習(xí)框架的實(shí)現(xiàn),稱為。更高級(jí)別的用于在之上構(gòu)建機(jī)器學(xué)習(xí)模型。 2019,開發(fā)者應(yīng)該學(xué)習(xí)的16個(gè)JavaScript框架 showImg(https://segmentfault.com/img/remote/14600...
摘要:轉(zhuǎn)載來源包管理器管理著庫,并提供讀取和打包它們的工具。能構(gòu)建更好應(yīng)用的客戶端包管理器。一個(gè)整合和的最佳思想,使開發(fā)者能快速方便地組織和編寫前端代碼的下一代包管理器。很棒的組件集合。隱秘地使用和用戶數(shù)據(jù)。 轉(zhuǎn)載來源:https://github.com/jobbole/aw... 包管理器管理著 javascript 庫,并提供讀取和打包它們的工具。?npm – npm 是 javasc...
摘要:轉(zhuǎn)載來源包管理器管理著庫,并提供讀取和打包它們的工具。能構(gòu)建更好應(yīng)用的客戶端包管理器。一個(gè)整合和的最佳思想,使開發(fā)者能快速方便地組織和編寫前端代碼的下一代包管理器。很棒的組件集合。隱秘地使用和用戶數(shù)據(jù)。 轉(zhuǎn)載來源:https://github.com/jobbole/aw... 包管理器管理著 javascript 庫,并提供讀取和打包它們的工具。?npm – npm 是 javasc...
摘要:一個(gè)專注于瀏覽器端和兼容的包管理器。一個(gè)整合和的最佳思想,使開發(fā)者能快速方便地組織和編寫前端代碼的下一代包管理器。完全插件化的工具,能在中識(shí)別和記錄模式。健壯的優(yōu)雅且功能豐富的模板引擎。完整的經(jīng)過充分測(cè)試和記錄數(shù)據(jù)結(jié)構(gòu)的庫。 【導(dǎo)讀】:GitHub 上有一個(gè) Awesome – XXX 系列的資源整理。awesome-javascript 是 sorrycc 發(fā)起維護(hù)的 JS 資源列表...
閱讀 2678·2021-11-25 09:43
閱讀 2590·2021-11-22 09:34
閱讀 2861·2021-11-12 10:34
閱讀 1447·2021-10-20 13:46
閱讀 2308·2019-08-30 13:21
閱讀 938·2019-08-30 11:21
閱讀 492·2019-08-30 11:20
閱讀 2196·2019-08-29 17:20