基于 Express+JWT + Vue 的前后端分离架构

基于 Express、JWT 和 Vue 的前后端分离架构是一种常见的现代 Web 开发架构。这种架构将前端和后端分开部署,前端使用 Vue.js 框架构建用户界面,后端使用 Express.js 框架处理业务逻辑和提供 API 接口,JWT(JSON Web Token)用于用户的身份验证和授权。

以下是实现这种架构的基本步骤:

一、后端(Express + JWT)

初始化项目

bash 复制代码
mkdir backend
cd backend
npm init -y
npm install express jsonwebtoken body-parser cors mongoose

设置 Express 服务器

javascript 复制代码
// server.js
const express = require('express');
const bodyParser = require('body-parser');
const cors = require('cors');
const jwt = require('jsonwebtoken');

const app = express();
const PORT = process.env.PORT || 3000;

// Middleware
app.use(bodyParser.json());
app.use(cors());

// Dummy user data (in a real application, you should use a database)
const users = [
    { id: 1, username: 'testuser', password: 'testpass' },
];

// Middleware to authenticate JWT
const authenticateJWT = (req, res, next) => {
    const token = req.header('Authorization').replace('Bearer ', '');
    if (!token) return res.sendStatus(401);

    jwt.verify(token, 'your_jwt_secret', (err, user) => {
        if (err) return res.sendStatus(403);
        req.user = user;
        next();
    });
};

// User login route
app.post('/login', (req, res) => {
    const { username, password } = req.body;
    if (users[0].username!=username) return res.sendStatus(401);

    const token = jwt.sign({ id: user.id }, 'your_jwt_secret');
    res.json({ token });
});

// Protected route
app.get('/protected', authenticateJWT, (req, res) => {
    res.json({ message: 'This is a protected route', user: req.user });
});

app.listen(PORT, () => {
    console.log(`Server running on port ${PORT}`);
});

二、前端(Vue)

初始化项目

bash 复制代码
mkdir frontend
cd frontend
npm install -g @vue/cli
vue create my-project
cd my-project
npm install axios

配置 Vue 项目

javascript 复制代码
// src/main.js
import Vue from 'vue';
import App from './App.vue';
import axios from 'axios';

Vue.config.productionTip = false;

Vue.prototype.$axios = axios;

new Vue({
  render: h => h(App),
}).$mount('#app');

创建登录组件

javascript 复制代码
<!-- src/components/Login.vue -->
<template>
  <div>
    <h2>Login</h2>
    <form @submit.prevent="login">
      <div>
        <label for="username">Username:</label>
        <input type="text" v-model="username" id="username" />
      </div>
      <div>
        <label for="password">Password:</label>
        <input type="password" v-model="password" id="password" />
      </div>
      <button type="submit">Login</button>
    </form>
  </div>
</template>

<script>
export default {
  data() {
    return {
      username: '',
      password: '',
    };
  },
  methods: {
    async login() {
      try {
        const response = await this.$axios.post('http://localhost:3000/login', {
          username: this.username,
          password: this.password,
        });
        const token = response.data.token;
        localStorage.setItem('jwt', token);
        this.$router.push('/protected');
      } catch (error) {
        console.error(error);
      }
    },
  },
};
</script>

创建受保护组件

javascript 复制代码
<!-- src/components/Protected.vue -->
<template>
  <div>
    <h2>Protected Route</h2>
    <div v-if="message">{{ message }}</div>
    <div v-else>Loading...</div>
  </div>
</template>

<script>
export default {
  data() {
    return {
      message: '',
    };
  },
  created() {
    this.fetchProtectedData();
  },
  methods: {
    async fetchProtectedData() {
      const token = localStorage.getItem('jwt');
      if (!token) {
        this.$router.push('/login');
        return;
      }

      try {
        const response = await this.$axios.get('http://localhost:3000/protected', {
          headers: { Authorization: `Bearer ${token}` },
        });
        this.message = response.data.message;
      } catch (error) {
        console.error(error);
        localStorage.removeItem('jwt');
        this.$router.push('/login');
      }
    },
  },
};
</script>

配置路由

javascript 复制代码
// src/router/index.js
import Vue from 'vue';
import Router from 'vue-router';
import Login from '@/components/Login.vue';
import Protected from '@/components/Protected.vue';

Vue.use(Router);

export default new Router({
  routes: [
    {
      path: '/',
      redirect: '/login',
    },
    {
      path: '/login',
      name: 'Login',
      component: Login,
    },
    {
      path: '/protected',
      name: 'Protected',
      component: Protected,
    },
  ],
});

三、运行项目

启动后端服务器

bash 复制代码
cd backend
node server.js

启动前端应用

bash 复制代码
cd frontend/my-project
npm run serve
复制代码
相关推荐
CloudWeGo1 小时前
企业级落地案例:抖音搜索核心链路基于 Kitex 流式改造的技术实践
人工智能·架构·开源
绝无仅有1 小时前
大厂面试题MySQL解析:MVCC、Redolog、Undolog与Binlog的区别
后端·面试·架构
绝无仅有1 小时前
MySQL面试题解析:MySQL读写分离与主从同步
后端·面试·架构
火星数据-Tina2 小时前
让电竞数据实时跳动:Spring Boot 后端 + Vue 前端的完美融合实践
前端·vue.js·spring boot
九年义务漏网鲨鱼2 小时前
【多模态大模型面经】现代大模型架构(一): 组注意力机制(GQA)和 RMSNorm
人工智能·深度学习·算法·架构·大模型·强化学习
Lethehong2 小时前
华为CANN异构计算架构技术分析报告:架构、优势与应用实践
人工智能·华为·架构
嘻嘻哈哈猿人2 小时前
从 0 到 1 实现一个支持 @ 提及用户的输入框组件(Vue3 实战)
前端·vue.js
云枫晖2 小时前
Vue3 响应式原理:手写实现 ref 函数
前端·vue.js
哔哩哔哩技术2 小时前
B站消息新架构升级
架构
故渊ZY2 小时前
深入解析JVM:核心架构与调优实战
java·jvm·架构