成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

不使用框架創(chuàng)建簡(jiǎn)單的Node.js web應(yīng)用

LeanCloud / 2222人閱讀

摘要:不使用框架創(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

相關(guān)文章

  • (譯 & 轉(zhuǎn)載) 2016 JavaScript 后起之秀

    摘要:在年成為最大贏家,贏得了實(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...

    darry 評(píng)論0 收藏0
  • 2016-JavaScript之星

    摘要:在,是當(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è)月之后就過...

    Binguner 評(píng)論0 收藏0
  • 2019,開發(fā)者應(yīng)該學(xué)習(xí)16個(gè)JavaScript框架

    摘要:它不僅從前端移動(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...

    Harpsichord1207 評(píng)論0 收藏0
  • javascript功能插件大集合 前端常用插件 js常用插件

    摘要:轉(zhuǎn)載來源包管理器管理著庫,并提供讀取和打包它們的工具。能構(gòu)建更好應(yīng)用的客戶端包管理器。一個(gè)整合和的最佳思想,使開發(fā)者能快速方便地組織和編寫前端代碼的下一代包管理器。很棒的組件集合。隱秘地使用和用戶數(shù)據(jù)。 轉(zhuǎn)載來源:https://github.com/jobbole/aw... 包管理器管理著 javascript 庫,并提供讀取和打包它們的工具。?npm – npm 是 javasc...

    netmou 評(píng)論0 收藏0
  • javascript功能插件大集合 前端常用插件 js常用插件

    摘要:轉(zhuǎn)載來源包管理器管理著庫,并提供讀取和打包它們的工具。能構(gòu)建更好應(yīng)用的客戶端包管理器。一個(gè)整合和的最佳思想,使開發(fā)者能快速方便地組織和編寫前端代碼的下一代包管理器。很棒的組件集合。隱秘地使用和用戶數(shù)據(jù)。 轉(zhuǎn)載來源:https://github.com/jobbole/aw... 包管理器管理著 javascript 庫,并提供讀取和打包它們的工具。?npm – npm 是 javasc...

    Hydrogen 評(píng)論0 收藏0
  • javascript功能插件大集合,寫前端親們記得收藏

    摘要:一個(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 資源列表...

    cfanr 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<