在nodejs常见的不良做法及其优化解决方案

在nodejs常见的不良做法及其优化解决方案

当涉及到在expressnodejs中开发应用程序时。遵循最佳实践对于确保项目的健壮性、可维护性和安全性至关重要。

在本文中,我们将探索开发人员经常遇到的几种常见的错误做法,并通过代码示例研究优化的最佳做法,以纠正这些错误。

在本文的结尾,我们将清楚地了解如何避免这些错误并采用最佳最佳实践。

嵌套回调

nodejs早期,开发人员使用嵌套回调处理MongoDB数据库和异步操作。虽然这种方法有效,,但它可以很快导致回调地狱,使代码难以读取和维护。

app.get('/users', (req, res) => {
    
    
    User.find({
    
    }, (err, users) => {
    
    
        if (err) {
    
    
            console.log(err);
            return res.status(500).send('Error occurred');
        }
        UserDetails.find({
    
     userId: users[0]._id }, (err, userDetails) => {
    
    
            if (err) {
    
    
                console.log(err);
                return res.status(500).send('Error occurred');
            }
            return res.status(200).json({
    
     users, userDetails });
        });
    });
});

这个代码片段演示了嵌套的回调,用于查询用户和他们的详细信息,这些信息来自于使用MongoDB数据库。可以看到管理多个嵌套的回调会变得很麻烦。

不使用嵌套回调,我们可以利用 promiseasync/await 来提高代码可读性和优雅地管理异步操作。

使用promise和async/await

promise提供了一种更整洁的方式来处理异步操作,并使代码更具可读性。下面是同样的例子,使用的是promiseasync/wawit

app.get('/users', async (req, res) => {
    
    
    try {
    
    
        const users = await User.find({
    
    });
        const userDetails = await UserDetails.find({
    
     userId: users[0]._id });
        return res.status(200).json({
    
     users, userDetails });
    } catch (err) {
    
    
        console.log(err);
        return res.status(500).send('Error occurred');
    }
});

使用async/await消除嵌套结构,大大提高了代码可读性。它提高了代码的可维护性,使错误处理更加简单。

缺乏错误处理

不能正确处理错误会导致意外的崩溃和潜在的安全漏洞。许多开发人员忽略了错误处理,这可能会导致他们的应用程序在使用MongoDB时表现出不可预测的行为。

app.get('/user/:id', (req, res) => {
    
    
    User.findById(req.params.id, (err, user) => {
    
    
        if (err) {
    
    
            console.log(err);
            // What if an error occurs here? It will be unhandled!
        }
        return res.status(200).json(user);
    });
});

在上面的示例中,如果在对MongoDB数据库查询中发生错误,将不会处理错误,,能导致意外行为或崩溃。为了解决这个问题,必须实施适当的错误处理。

实现错误处理

处理错误的方法建议是使用try/catch代码块:

扫描二维码关注公众号,回复: 16957185 查看本文章
app.get('/user/:id', async (req, res) => {
    
    
    try {
    
    
        const user = await User.findById(req.params.id);
        if (!user) {
    
    
            return res.status(404).json({
    
     message: 'User not found' });
        }
        return res.status(200).json(user);
    } catch (err) {
    
    
        console.log(err);
        return res.status(500).send('Error occurred');
    }
});

通过使用try/catch块,我们可以捕捉在执行MongoDB查询期间发生的任何错误。此外,我们可以根据特定的错误类型定制错误响应,以向客户端提供信息反馈。

不使用中间件

中间件通过允许开发人员拦截和修改传入的请求,在使用MongoDB 时中间件发挥着至关重要的作用。忽略使用中间件会导致路由膨胀和代码重复。

app.get('/users', (req, res) => {
    
    
    // 一些逻辑
    // 缺乏中间件使模块化代码和处理通用功能变得困难
    // 将在不同的路线重复使用更多代码,导致维护问题。
});

使用中间件

中间件允许开发人员集中处理重复任务,提高代码模块化和可维护性。这里有一个使用中间件操作MongoDB 数据库的例子:

const authenticateUser = async (req, res, next) => {
    
    
    try {
    
    
        const user = await User.findById(req.userId);
        if (!user) {
    
    
            return res.status(401).send('Unauthorized');
        }
        req.user = user;
        return next();
    } catch (err) {
    
    
        console.log(err);
        return res.status(500).send('Error occurred');
    }
};

app.get('/users', authenticateUser, (req, res) => {
    
    
    // 现在,这个路由处理程序可以假定用户是经过身份验证的。
    // 身份验证逻辑已抽象成中间件。
    // 其他路由也可以重用"认证用户"中间件。
    return res.status(200).json({
    
     user: req.user });
});

通过利用中间件,我们保持我们的路径清洁,并专注于他们的具体任务。中间件提高了代码的可重用性,简化了调试,并使应用程序更易于管理。

不管理环境变量

在使用 MongoDB数据的应用程序中的敏感信息,如数据库凭证和API键时,不应该在代码栏中硬编码。

const databasePassword = 'mysecretpassword';
const apiKey = 'myapikey';
// 直接在代码中使用敏感信息是不安全的。
// 这些信息最终可能会进入版本控制系统,从而带来安全风险。

将敏感信息直接存储在代码中会使其面临潜在的安全漏洞,并使其难以在不同环境中管理配置。

确保环境变量的安全

为了安全地管理特定环境的配置和敏感数据,我们应该使用类似 dotenv库进行处理。

低效的数据库查询

MongoDB数据库查询效率低下,会导致响应时间慢,并对应用程序性能产生负面影响。通常,开发人员可能不会优化他们的查询,从而导致性能问题。

app.get('/products', (req, res) => {
    
    
    Product.find({
    
    }, (err, products) => {
    
    
        if (err) {
    
    
            console.log(err);
            return res.status(500).send('Error occurred');
        }
        return res.status(200).json(products);
    });
});

在本例中,查询使用MongoDB数据库检索所有产品,对于小型数据集来说,这可能是很好的。然而,随着数据集的增长,这个查询可能变得缓慢且资源密集。

优化数据库查询

为了提高MongoDB数据库查询的性能,开发人员应该专注于查询优化。一种常见的方法是使用select 只检索所需字段的方法:

app.get('/products', async (req, res) => {
    
    
    try {
    
    
        const products = await Product.find({
    
    }).select('name price');
        return res.status(200).json(products);
    } catch (err) {
    
    
        console.log(err);
        return res.status(500).send('Error occurred');
    }
});

只从products 收集数据时,我们减少了数据库和应用程序之间的数据传输量,从而提高了性能。

结论

在本文中,我们探讨了与MongoDB一起使用时的常见不良做法。我们还了解了它们的潜在缺点,并用代码示例探讨了优化的最佳实践,以纠正这些问题。

通过遵循这些最佳实践,可以确保能够带来更加高效、可维护和安全的处理。优先排序代码可读性,优雅地处理错误,并采用中间件更好地模块化。此外,使用dotenv 能更安全地管理敏感数据并优化数据库查询,以提高应用程序性能。

猜你喜欢

转载自blog.csdn.net/qq_42880714/article/details/133442639