vue前端与Java后端进行跨域交互

Posted Coding With you.....

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了vue前端与Java后端进行跨域交互相关的知识,希望对你有一定的参考价值。

1.后端的几种解决方法

1.在Controller上面加上@CrossOrigin

2.写一个配置文件并且在Controller层加上注解@CORSConfig

package com.wolwo.langyage.base.util;
 
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.CorsRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurerAdapter;
/**
 * CORS configuration
 */
@Configuration
public class CORSConfig 
 
    @Bean
    public WebMvcConfigurer corsConfigurer() 
        return new WebMvcConfigurerAdapter() 
            @Override
            public void addCorsMappings(CorsRegistry registry) 
                registry.addMapping("/**")
                        .allowedOrigins("http://127.0.0.1:8080,http://localhost:8080")
                        .allowedMethods("POST,GET, OPTIONS,DELETE,PUT")
                        .allowedHeaders("Content-Type,ContentType,Access-Control-Allow-Headers,Access-Control-Allow-Origin, Authorization, X-Requested-With")
                        .allowCredentials(true);
            
        ;
    
 

3.在springmvc配置文件里加入

4.写一个过滤器类并且在web.xml中注册过滤器

package com.pmsapi.filter;
import java.io.IOException;
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

public class CrossingFilter implements Filter 


private boolean isCross = false;

@Override
public void destroy() 
	isCross = false;


@Override
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
		throws IOException, ServletException 
	if (isCross) 
		HttpServletRequest httpServletRequest = (HttpServletRequest) request;
		HttpServletResponse httpServletResponse = (HttpServletResponse) response;
		System.out.println("拦截请求: " + httpServletRequest.getServletPath());
		httpServletResponse.setHeader("Access-Control-Allow-Origin", "*");
		// httpServletResponse.setHeader("Access-Control-Allow-Methods", "*"); // 表示所有请求都有效
		httpServletResponse.setHeader("Access-Control-Allow-Methods", "POST, GET, OPTIONS, DELETE");
		httpServletResponse.setHeader("Access-Control-Max-Age", "0");
		httpServletResponse.setHeader("Access-Control-Allow-Headers",
				"Origin, No-Cache, X-Requested-With, If-Modified-Since, Pragma, Last-Modified, Cache-Control, Expires, Content-Type, X-E4M-With,userId,token");
		httpServletResponse.setHeader("Access-Control-Allow-Credentials", "true");
		httpServletResponse.setHeader("XDomainRequestAllowed", "1");
	
	chain.doFilter(request, response);


@Override
public void init(FilterConfig filterConfig) throws ServletException 
	String isCrossStr = filterConfig.getInitParameter("IsCross");
	isCross = isCrossStr.equals("true") ? true : false;
	System.out.println("跨域开启状态:" + isCrossStr);


<!-- 配置跨域过滤器 -->
<filter>
	<filter-name>CrossingFilter</filter-name>
	<filter-class>com.pmsapi.filter.CrossingFilter</filter-class>
	<init-param>
		<param-name>IsCross</param-name>
		<param-value>true</param-value>
	</init-param>
</filter>
<filter-mapping>
	<filter-name>CrossingFilter</filter-name>
	<url-pattern>/*</url-pattern>
</filter-mapping>

获取前端传来的数据

public void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException 
            // 1. 从 map 取 前端发过来的 post 数据
//username=[xxx],password=[xxx],hobby=[eat,drink]
            Map<String, String[]> parameterMap = req.getParameterMap();
            // 2. 遍历 map 数据 ----> 字符串
            String namePwd = "";
            if (parameterMap != null) 
                for (Map.Entry<String, String[]> entry : parameterMap.entrySet()) 
                    namePwd = entry.getKey();
                    System.out.println(namePwd);
                
            
            // 3. 字符串 ----> json
            JSONObject params = JSON.parseObject(namePwd);
            String username = params.getString("username");
            int pwd = params.getInteger("pwd");

2.前端的处理   

查看vue版本npm list vue或是在package.json文件里找”dependencies”直接看到

vue2

原理:给url加上了前缀 /api,我们前端发起请求/dev/unit/queryTree访问https://locall/host:8080/dev/unit/queryTree就当于访问了:https://locall/host:8080/api/dev/unit/queryTree

又因为在 index.js 中的 proxyTable 中拦截了 /api ,并把 /api 及其前面的所有替换成了 target 中的内容,因此实际访问 Url 是https://www.wd.com/dev/unit/queryTree。红色是target,绿色是前端发起的具体请求,对应后端的请求

1.vue.main.js  配置访问的url前缀

import Vue from 'vue'
import ElementUI from 'element-ui'
import 'element-ui/lib/theme-chalk/index.css'
import App from './App'
import axios from 'axios'//引入axios,来发送请求
Vue.config.productionTip = false
Vue.use(ElementUI)
Vue.prototype.axios = axios
axios.defaults.timeout = 5000;// 在超时前,所有请求都会等待 5 秒
axios.defaults.headers.post['Content-Type']='application/x-www-form-urlencoded;charset=UTF-8';// 配置请求头
axios.defaults.baseURL = '/api';// 作用是我们每次发送的请求都会带一个/api的前缀。
axios.defaults.withCredentials = false;
new Vue(
    el: '#app',
    components: App,
    template: '<App/>'
    // render: h => h(App)
)

2.修改config文件夹下的index.js文件,在proxyTable中加上如下代码:  代理,如果没有这部分,那么跨域的路径就是baseurl+具体请求

'/api':
    target: "https://www.wd.com/dev",
    changeOrigin:true,
    pathRewrite:
        '^/api':''     //因为本身的请求中没有api,所以去掉,eg:前端请求/api/ser,实际后端:https://www.wd.com/dev/ser
    

3.app.vue

<template>
  <div>
    <el-menu></el-menu>
        <el-table></el-table>
        <br/>
        <el-page></el-page>
  </div>
</template>
<script>
import nav from "./components/NavMenu.vue"
import eletable from "./components/Table.vue"
import pageination from "./components/Pagination.vue"
export default 
  components:
        ElMenu:nav,
        ElTable:eletable,
        ElPage:pageination
    ,
    data()
        return
    ,
    methods:
    ,
    mounted() 
            this.axios(
                method: 'get',
                url: '/test/test',
                params: this.urlParams
            ).then((res) => 
                console.log(res)
                console.log(res.data)
                alert(res.data.msg)
            ).catch((err) => 
                console.log(err)
            )
    

</script>

vue3

在config.js中这样写
devServer: 
    open: true, //是否自动弹出浏览器页面
    host: "localhost",
    port: '8080',
    https: false,
    hotOnly: false,
    proxy: //或者直接写上协议 域名 端口号,在请求的时候加上请求和这个拼接上

        '/api': 
            target: 'https://www.v2ex.com/api', //API服务器的地址
            changeOrigin: true,
            pathRewrite: 
                '^/api': ''
            
        
    ,

3.vue+webpack项目搭建实现前后端交互

1.检查本地环境

  • node -v
  • npm -v
  • vue -V

2.进入目录文件夹创建项目

vue init webpack vue-project-name

3.安装依赖

npm操作: 与cnpm yarn的区别是他们下载源不同,npm会更满=慢些
安装vue-router:
npm install vue-router --save-dev     dev表示生产环境,开发环境不加dev
安装element-ui: 与路由相关的元素上需要加上router,在vue文件中的element元素上,如果表单中某个元素的值是变量,需加上:    ,比如:key="i"
npm i element-ui -S         -S表示plus的意思,不加plus的只支持vue2.几版本,vue3使用plus的
安装SAss加载器:
npm install sass-loader node-sass --save-dev
安装axios:
npm install --save axios vue-axios
注意安装依赖之后要重新执行nmp install

4.启动浏览器进行访问

npm run dev是适用于老版本;新版本使用npm run server

完成后,可在生成的项目中修改基础配置,比如运行监听端口,默认是8080,如果担心有冲突可自行更改,更改文件在config—index.js 。项目基础信息在package.json文件中,有项目名称、作者、运行说明等,可自行查阅,启动项目命令:npm run dev启动完成后回返回请求的地址,复制到浏览器即可请求项目。


5.实现登录

1

  • http是无状态的通过cookie在客户端记录状态通过session在服务器端记录状态通过token方式维持状态
  • 将登录之后的token,保存到客户端的sessionStorage中.项目中除了登录之外的其他API接口,必须在登录之后才能访问;token只应在当前网站打开期间生效,所以将token保存在sessionStorage中

2.Element-UI组件实现布局

  • el-form
  • el-form-item
  • el-input
  • el-button
  • 字体图标

3.Login.vue

<template>
  <div class="login-wrap">
    <el-form class="login-container">
      <h1 class="title">用户登录</h1>
      <el-form-item label="账号">
        <el-input type="password" v-model="uname" placeholder="登录账号" autocomplete="off"></el-input>
      </el-form-item>
      <el-form-item label="登录密码">
        <el-input type="password" v-model="password" placeholder="登录密码" autocomplete="off"></el-input>
      </el-form-item>
 
      <el-form-item>
        <el-button type="primary" style="width: 100%;" @click="dosubmit()">登录</el-button>
      </el-form-item>
    </el-form>
  </div>
</template>
 
<script>
  export default 
    name: 'login',
    data: function() 
      return 
        uname: "张三",
        password: "123"
      
    ,
    methods: 
      dosubmit: function() 
        let params = 
          uname: this.uname,
          password: this.password
        ;
      
      
    
  
</script>
 
<!-- Add "scoped" attribute to limit CSS to this component only -->
<style scoped>
  .login-wrap 
    box-sizing: border-box;
    width: 100%;
    height: 100%;
    padding-top: 10%;
    background-image: url();
    /* background-color: #112346; */
    background-repeat: no-repeat;
    background-position: center right;
    background-size: 100%;
  
 
  .login-container 
    border-radius: 10px;
    margin: 0px auto;
    width: 350px;
    padding: 30px 35px 15px 35px;
    background: #fff;
    border: 1px solid #eaeaea;
    text-align: left;
    box-shadow: 0 0 20px 2px rgba(0, 0, 0, 0.1);
  
 
  .title 
    margin: 0px auto 40px auto;
    text-align: center;
    color: #505458;
  
</style>

4.配置路由

import Vue from 'vue'
import Router from 'vue-router'
import HelloWorld from '@/components/HelloWorld'
import Login from '@/view/Login'
 
Vue.use(Router)
 
export default new Router(
  routes: [
    
      path: '/',
      name: 'Login',
      component: Login
    ,
  ]
)

5.跨域请求

get:

<script>
import axios from 'axios'
import qs from 'qs'
  export default 
    name: 'login',
    data: function() 
      return 
        uname: "张三",
        password: "123"
      
    ,
    methods: 
      dosubmit: function() 
        let params = 
          uname: this.uname,
          password: this.password
        ;
        var url = "http://localhost:8080/ssm02/login";
        axios.get(url,  //注意数据是保存到json对象的params属性
          params: params
        ).then(function(response) 
           console.log(response);
         ).catch(function(error) 
          console.log(error);
        );
        var str=qs.stringify(params);
        axios.post(url,str//注意数据是直接保存到json对象
        ).then(function(response) 
          console.log(response);
        ).catch(function(error) 
          console.log(error);
        );
 
 
      
    
  
</script>

post

var str=qs.stringify(params);
        axios.post(url,str//注意数据是直接保存到json对象
        ).then(function(response) 
          console.log(response);
        ).catch(function(error) 
          console.log(error);
        );
  因为POST提交的参数的格式是Request Payload,这样后台取不到数据的(详情见资料“05 Vue中axios踩坑之路-POST传参 - RainSun - CSDN博客.mht”),
           解决方案:使用qs.js库,将a:'b',c:'d'转换成'a=b&c=d'

以上是关于vue前端与Java后端进行跨域交互的主要内容,如果未能解决你的问题,请参考以下文章

增删改查项目属于前端还是后端

前后端交互数据加解密

SonarQube 9.x与Jenkins进行集成并扫描后端java以及前端vue代码;

前端采用vue+webpack后端采用java全栈开发,怎么实现自动部署

前端和后端如何进行交互流程

前后端数据交互中的特殊字符