解决node.js中前端使用POST请求,后端end过早,前端拿不到数据返回空白的问题

文章描述了一种前端向本地后端发送POST请求时遇到的问题,即后端在数据未完全接收时就发送了响应。为了解决这个问题,作者提出了使用回调函数的方法,在数据收集和处理完毕后才执行响应,确保前端能接收到完整数据。
摘要由CSDN通过智能技术生成

前端结构如下:

 

点击'登录-post'按钮将用户名和密码返回给后端,后端接收到后再返回数据给前端

       在server.js中将use()方法和serverStart()方法暴露出去,use()方法用于不断将新的接口合并到已有接口中。

const http = require('http');

const router = {};
function use(obj) {
    Object.assign(router, obj)
}

const serverStart = () => {
    http.createServer((req, res) => {
        const curUrl = new URL(req.url, 'http://127.0.0.1:3000');
        //   console.log(curUrl.pathname);
        // router(curUrl.pathname, res)
        try {
            router[curUrl.pathname](req, res, (data) => {
                res.write(data)
                res.end()
            })
        } catch (error) {
            router['/404'](req, res)
        }
    }).listen(3000, () => {
        console.log('server start');
    })
}

exports.serverStart = serverStart
exports.use = use

使用use()方法:

index.js

const server = require('./server.js');
const route = require('./router.js');
const api = require('./api.js');

server.use(route)
server.use(api)

server.serverStart()

       我在使用POST请求时遇到了一个问题,前端点击按钮将用户名和密码传给本地自己写的'/api/loginpost'接口:

<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
</head>

<body>
    <div>
        用户名:<input type="text" id="userName">
    </div>
    <div>
        密码:<input type="text" id="password">
    </div>
    <div>
        <button id="login">登录-get</button>
        <button id="login-post">登录-post</button>
    </div>

    <script>
        let dom = {
            getUserName: document.querySelector('#userName'),
            getPassword: document.querySelector('#password'),
            getLogin: document.querySelector('#login'),
            getLoginPost: document.querySelector('#login-post')
        }

        dom.getLogin.onclick = function () {
            fetch(`/api/login?username=${dom.getUserName.value}&password=${dom.getPassword.value}`)
                .then((res) => { return res.text(); })
                .then((res) => { console.log(res); });
        }
        dom.getLoginPost.onclick = function () {
            fetch(`/api/loginpost`, {
                method: 'POST',
                body: JSON.stringify({
                    userName: dom.getUserName.value,
                    password: dom.getPassword.value
                }),
                headers: {
                    'Content-Type': 'application/json'
                }
            }).then((res) => { return res.text(); })
                .then((res) => { console.log(res); });
        }
    </script>
</body>

</html>

在'/api/loginpost'接口中接收数据并处理返回数据给前端:

'/api/loginpost': (req, res, cb) => {
        let post = '';
        req.on('data', (chunk) => {
            post += chunk;
        })
        req.on('end', () => {
            post = JSON.parse(post);
            console.log(post);
            if (post.userName === 'fanfan' && post.password === '666') {
                render(res, `{'fanfan': 666}`, cb)
            } else {
                render(res, `{'fanfan': 0}`, cb)
            }
        })
}


const render = (res, data, cb, type = 'application/json') => {
    res.writeHead(200, {
        'Content-Type': type + `;charset=utf-8`
    })
    cb(data)
}

       最开始的时候在req上监听on事件一点一点接收数据,数据收集完毕时调用render()方法处理数据并返回给前端,然后在server.js中的服务器方法中执行res.end()响应。

       这里有一个问题,由于在接口中监听的on事件是异步执行的,在server.js中调用接口函数时,数据还没收集完,render()方法还没有渲染完数据返回给前端,服务器就过早执行end响应了,导致页面上后端返回的是空白。

在这里我想到了使用回调函数来解决这个问题

具体做法:

在服务器调用接口方法时传入一个回调函数

 

在接口方法中接收,数据收集完毕后执行render()方法将这个回调函数传进去

 

在render()方法中接收回调函数

       等数据收集完毕并处理好后再执行回调函数,在回调函数里面执行end响应就能避免end响应过早的问题 ,前端就能正常拿到后端返回的数据

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值