写点什么

用 Spring Boot 打包你的 React 应用

  • 2020-03-31
  • 本文字数:3612 字

    阅读完需:约 12 分钟

用Spring Boot打包你的React应用

AI 大模型超全落地场景&金融应用实践,8 月 16 - 19 日 FCon x AICon 大会联诀来袭、干货翻倍!

先讲一讲这篇文章的背景故事。之前我的团队需要在我们需求的基础架构上节省一些资金,并且由于我们要构建的这个应用程序中,大部分负载都会在客户端而非服务端上,所以我们决定试验一下能否将一个 Spring 应用程序与一个 React 应用结合起来,并打包成一个 war 文件。


这篇文章会告诉你如何将 Create React App 与 Spring Boot 结合使用,从而为你打包出单个 war 文件。


Spring Boot 和 Create React App 的基本功能介绍


  • Create React App 可帮助你非常快速地启动 React 项目。它为你提供了启动并尽快运行项目所需的全部基本功能。

  • Spring Boot 可以帮助你快速而轻松地启动和维护 Spring 应用程序。

步骤

1.目标

  • 在单一 war 文件中包含前端和后端,具有优化的生产构建

  • 保留 Create React App 所提供的好处,如热重载等

2.设置


附注:我选择的 IDE 是 IntelliJ。当使用 React 代码时,我通常会切换到 VS Code。你可以随意使用自己习惯的方法


  • 在 Github 上创建一个空的仓库,并添加自述文件、gitignore 和许可证等。

  • 转到(https://start.spring.io)创建你的 Spring 应用程序,并下载到本地。Group 和 Artifact 也可以随意设置。



GroupId:e.the.awesome


Artifact:spring-react-combo-app

3.将下载的项目解压缩到你的 git 目录中并提交。

你的 SpringReactComboAppApplication 看起来应该像这样。


package e.the.awesome.springreactcomboapp;
import org.springframework.boot.SpringApplication;import org.springframework.boot.autoconfigure.SpringBootApplication;import org.springframework.boot.web.servlet.support.SpringBootServletInitializer;
@SpringBootApplicationpublic class SpringReactComboAppApplication extends SpringBootServletInitializer{
public static void main(String[] args) { SpringApplication.run(SpringReactComboAppApplication.class, args); }
}
复制代码

4.现在创建一个基本服务。

我们将其称为 DadJokesController。这应该在与 SpringReactComboAppApplication 文件所在的文件夹中创建。我知道这不是正确的 Rest API 格式,但现在暂时忽略它。


package e.the.awesome.springreactcomboapp;import org.springframework.web.bind.annotation.GetMapping;import org.springframework.web.bind.annotation.RestController;@RestControllerpublic class DadJokesController {    @GetMapping("/api/dadjokes")    public String dadJokes() {        return "Justice is a dish best served cold, if it were served warm it would be just water.";    }}
复制代码

5.在你的终端运行

mvn spring-boot:run
复制代码


然后在浏览器中检查http://localhost:8080/api/dadjokes。你应该会看到我们添加到控制器中的 dad joke。

6.要创建你的 React 应用,只需在根目录中运行

npx create-react-app basic-frontend-app
复制代码


你可以随心所欲地调用它,我这里只调用我的 basic-frontend-app

7.要运行前端应用程序:

cd basic-frontend-app<br>npm start
复制代码

8.解决代理问题

由于我们要将 Dad Jokes 服务集成到前端,因此首先我们要解决代理问题。你可能已经注意到了,你的服务从 localhost:8080 开始,而前端从 localhost:3000 开始。如果我们尝试从前端调用服务,具体取决于你的浏览器,你可能会收到一个 CORS 错误。



解决此问题的最简单方法是让前端代理处理从端口 3000 到 8080 的所有请求。此更改将在 package.json 文件中进行:


{  "name": "basic-frontend-app",  "version": "0.1.0",  "private": true,  "dependencies": {    "react": "^16.3.1",    "react-dom": "^16.3.1",    "react-scripts": "1.1.4"  },  "scripts": {    "start": "react-scripts start",    "build": "react-scripts build",    "test": "react-scripts test --env=jsdom",    "eject": "react-scripts eject"  },  "proxy": {    "/api": {      "target": "http://localhost:8080",      "ws": true    }  }}
复制代码


将以下内容添加到你的前端 App.js 文件中


import React, {Component} from 'react';import logo from './logo.svg';import './App.css';
class App extends Component {
state = {};
componentDidMount() { this.dadJokes() }
dadJokes = () => { fetch('/api/dadjokes') .then(response => response.text()) .then(message => { this.setState({message: message}); }); };
render() { return ( <div className="App"> <header className="App-header"> <img src={logo} className="App-logo" alt="logo"/> <h3 className="App-title">{this.state.message}</h3> </header> <p className="App-intro"> To get started, edit <code>src/App.js</code> and save to reload. </p> </div> ); }}
export default App;
复制代码


如果你也遇到了下图的这个错误,我的做法是:首先删除了 package-lock.json 文件;其次在 node_modules 文件夹中重新安装了 npm 包并再次运行;然后重新启动前端,问题就解决了。


9.你的应用程序现在应该看起来像这样。

你可以看到 dad jokes API 调用的结果。


10.创建生产版本

现在我们的基本前端和后端已经完成,该创建生产版本和单个 war 文件了。


< dependencies>下添加


<!-- https://mvnrepository.com/artifact/com.github.eirslett/frontend-maven-plugin --><dependency><groupId>com.github.eirslett</groupId><artifactId>frontend-maven-plugin</artifactId><version>1.6</version></dependency>
复制代码


在 pom 文件的< plugins>部分下,我们将添加以下命令,这些命令将在运行 mvn clean install 时执行以下操作。


  • npm 安装指定版本的 node

  • 运行我们前端的生产构建

  • 存放生产构建


<plugin>   <groupId>com.github.eirslett</groupId>   <artifactId>frontend-maven-plugin</artifactId>   <version>1.6</version>   <configuration>      <workingDirectory>basic-frontend-app</workingDirectory>      <installDirectory>target</installDirectory>   </configuration>   <executions>      <execution>         <id>install node and npm</id>         <goals>            <goal>install-node-and-npm</goal>         </goals>         <configuration>            <nodeVersion>v8.9.4</nodeVersion>            <npmVersion>5.6.0</npmVersion>         </configuration>      </execution>      <execution>         <id>npm install</id>         <goals>            <goal>npm</goal>         </goals>         <configuration>            <arguments>install</arguments>         </configuration>      </execution>      <execution>         <id>npm run build</id>         <goals>            <goal>npm</goal>         </goals>         <configuration>            <arguments>run build</arguments>         </configuration>      </execution>   </executions></plugin><plugin>   <artifactId>maven-antrun-plugin</artifactId>   <executions>      <execution>         <phase>generate-resources</phase>         <configuration>            <target>               <copy todir="${project.build.directory}/classes/public">                  <fileset dir="${project.basedir}/basic-frontend-app/build"/>               </copy>            </target>         </configuration>         <goals>            <goal>run</goal>         </goals>      </execution>   </executions></plugin>
复制代码


附注:对于你的插件来说,顺序正确是很重要的,因此请确保在复制构建文件执行之前执行 node/npm 安装

11.运行 mvn clean install

添加此命令后,运行 mvn clean install,并验证 target/classes 目录同时包含了前端文件和 Java 文件。这里你应该是一路畅通的。


最后看一下我的 pom 文件:


https://github.com/Emmanuella-Aninye/Spring-Boot-ReactJS-Starter/blob/master/pom.xml


这就是我的方法。如果你想看看 repo 或使用它,可以在我的 Github 上找到它:


https://github.com/Emmanuella-Aninye/Spring-Boot-ReactJS-Starter


延伸阅读:


https://hackernoon.com/package-your-react-app-with-spring-boot-a-how-to-guide-cdfm329w


2020-03-31 09:522962

评论

发布
暂无评论
发现更多内容

云桌面跟PC相比能有哪些不一样的体验?

青椒云云电脑

桌面云 云桌面

什么是云电脑?

青椒云云电脑

云电脑

如何实现虚拟云桌面?

青椒云云电脑

桌面云 云桌面

作业

大肚皮狒狒

Go 函数

小万哥

Go 程序员 后端 开发 Google

WorkPlus Meet白板和文档共享功能上线,私有化视频会议全新升级

WorkPlus

云桌面系统的运用优势有哪些?

青椒云云电脑

云桌面

云电脑多久能替代普通 PC?

青椒云云电脑

云电脑

不同构架云桌面的部署风险

青椒云云电脑

云桌面 青椒云云桌面

业务喜报丨九科信息成功签约四川中烟工业有限责任公司成都卷烟厂RPA项目

九科Ninetech

RPA RPAxAI

云桌面GPU技术方案

青椒云云电脑

图形工作站

虚拟云桌面和共享云桌面有啥区别

青椒云云电脑

云桌面

企业选择云桌面系统的主要原因是什么?

青椒云云电脑

云桌面

NodeServe:构建高效静态文件服务器的完美指南

凌览

node.js

祝贺!Databend Cloud 入驻 AWS 云市场

Databend

腾讯云TDSQL- C Serverless 2.0版发布,多项核心技术首次公开解析

Geek_2d6073

WorkPlus——高效私有化办公平台,实现即时协作与信息安全的完美结合

WorkPlus

使用云电脑9条注意事项

青椒云云电脑

云电脑

虚拟云桌面在实验教学中的应用与实践

青椒云云电脑

桌面云 云桌面

五点告诉我们云教室比传统机房好

青椒云云电脑

云教室

云桌面应用下的数据防护新思路

青椒云云电脑

云桌面

WorkPlus打造统一用户管理平台,实现企业用户管理的一体化

WorkPlus

私有化部署即时通讯平台,完美替代飞书和钉钉的SaaS系统

WorkPlus

云桌面在教学中的应用

青椒云云电脑

云桌面

影响云桌面性能的三个重要因素是什么?

青椒云云电脑

云桌面

LeetCode题解:7. 整数反转,迭代,JavaScript,详细注释

Lee Chen

JavaScript LeetCode

不同构架云桌面的部署风险

青椒云云电脑

云桌面

无障碍测试解读

QE_LAB

无障碍 测试技术干货 测试技术

为什么要使用虚拟云桌面?

青椒云云电脑

云桌面 青椒云云桌面

【效率提升】手把手教你如何使用免费的 Amazon Code Whisperer 提升开发效率堪比 GitHub Copilot 平替

亚马逊云科技 (Amazon Web Services)

用Spring Boot打包你的React应用_大前端_Emmanuella_InfoQ精选文章