Springboot
Spring Boot 是基于 Spring 框架的快速开发脚手架,它简化了 Spring 应用的创建和部署过程。Spring Boot 提供了自动配置、内嵌服务器、生产就绪的特性监控等功能,让开发者能够快速构建独立的、生产级别的 Spring 应用程序。
本指南介绍如何在 CloudBase HTTP 云函数上部署 Gin 应用程序。
示例源码请参考: cloudrun-springboot
前置条件
在开始之前,请确保您已经:
- 安装了 JDK 8 或更高版本
- 安装了 Maven 3.6+ 或 Gradle
- 拥有腾讯云账号并开通了 CloudBase 服务
- 了解基本的 Java 和 Spring Boot 开发知识
第一步:创建 Spring Boot 应用
💡 提示:如果您已经有一个 Spring Boot 应用,可以跳过此步骤。
使用 Spring Initializr 创建项目
- 访问 start.spring.io
- 选择以下配置:
Project: Maven
Language: Java
Spring Boot: 2.7.18 (或最新稳定版)
Project Metadata:
- Group: com.tencent
- Artifact: cloudrun-springboot
- Name: cloudrun-springboot
- Description: Demo project for Spring Boot
- Package name: com.tencent.cloudrun
- Packaging: Jar
- Java: 8
Dependencies:
- Spring Web
- Spring Boot Actuator (健康检查)
- 点击 GENERATE 下载项目压缩包
- 解压到本地目录
使用 Maven 命令创建(可选)
mvn archetype:generate \
-DgroupId=com.tencent.cloudrun \
-DartifactId=cloudrun-springboot \
-DarchetypeArtifactId=maven-archetype-quickstart \
-DinteractiveMode=false
cd cloudrun-springboot
配置 pom.xml 文件
如果使用 Maven 命令创建项目,需要手动配置 pom.xml 文件以支持 Spring Boot:
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>2.7.18</version>
<relativePath/>
</parent>
<groupId>com.tencent.cloudrun</groupId>
<artifactId>cloudrun-springboot</artifactId>
<version>1.0-SNAPSHOT</version>
<name>cloudrun-springboot</name>
<description>Demo project for Spring Boot</description>
<properties>
<java.version>8</java.version>
<maven.compiler.source>8</maven.compiler.source>
<maven.compiler.target>8</maven.compiler.target>
</properties>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-actuator</artifactId>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>
</project>
创建主应用类
如果使用 Maven 命令创建项目,还需要创建 Spring Boot 主应用类。
在 src/main/java/com/tencent/cloudrun 目录下创建 CloudrunApplication.java:
package com.tencent.cloudrun;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
@SpringBootApplication
public class CloudrunApplication {
public static void main(String[] args) {
SpringApplication.run(CloudrunApplication.class, args);
}
}
本地测试应用
进入项目目录并启动应用:
cd cloudrun-springboot
mvn spring-boot:run
打开浏览器访问 http://localhost:8080,您应该能看到 Spring Boot 默认页面。
第二步:添加 API 接口
让我们创建一些 RESTful API 来演示 Spring Boot 的功能。
创建用户实体类
在 src/main/java/com/tencent/cloudrun/entity 目录下创建 User.java:
package com.tencent.cloudrun.entity;
public class User {
private Long id;
private String name;
private String email;
// 构造函数
public User() {}
public User(Long id, String name, String email) {
this.id = id;
this.name = name;
this.email = email;
}
// Getter 和 Setter
public Long getId() { return id; }
public void setId(Long id) { this.id = id; }
public String getName() { return name; }
public void setName(String name) { this.name = name; }
public String getEmail() { return email; }
public void setEmail(String email) { this.email = email; }
}
创建响应包装类
在 src/main/java/com/tencent/cloudrun/dto 目录下创建 ApiResponse.java:
package com.tencent.cloudrun.dto;
public class ApiResponse<T> {
private boolean success;
private String message;
private T data;
public ApiResponse(boolean success, String message, T data) {
this.success = success;
this.message = message;
this.data = data;
}
public static <T> ApiResponse<T> success(T data) {
return new ApiResponse<>(true, "操作成功", data);
}
public static <T> ApiResponse<T> error(String message) {
return new ApiResponse<>(false, message, null);
}
// Getter 和 Setter
public boolean isSuccess() { return success; }
public void setSuccess(boolean success) { this.success = success; }
public String getMessage() { return message; }
public void setMessage(String message) { this.message = message; }
public T getData() { return data; }
public void setData(T data) { this.data = data; }
}
创建用户控制器
在 src/main/java/com/tencent/cloudrun/controller 目录下创建 UserController.java:
package com.tencent.cloudrun.controller;
import com.tencent.cloudrun.dto.ApiResponse;
import com.tencent.cloudrun.entity.User;
import org.springframework.web.bind.annotation.*;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.atomic.AtomicLong;
@RestController
@RequestMapping("/api/users")
public class UserController {
private final List<User> users = new ArrayList<>();
private final AtomicLong counter = new AtomicLong();
public UserController() {
// 初始化测试数据
users.add(new User(counter.incrementAndGet(), "张三", "zhangsan@example.com"));
users.add(new User(counter.incrementAndGet(), "李四", "lisi@example.com"));
users.add(new User(counter.incrementAndGet(), "王五", "wangwu@example.com"));
}
@GetMapping
public ApiResponse<List<User>> getAllUsers(
@RequestParam(defaultValue = "1") int page,
@RequestParam(defaultValue = "10") int limit) {
int startIndex = (page - 1) * limit;
int endIndex = Math.min(startIndex + limit, users.size());
if (startIndex >= users.size()) {
return ApiResponse.success(new ArrayList<>());
}
List<User> paginatedUsers = users.subList(startIndex, endIndex);
return ApiResponse.success(paginatedUsers);
}
@GetMapping("/{id}")
public ApiResponse<User> getUserById(@PathVariable Long id) {
User user = users.stream()
.filter(u -> u.getId().equals(id))
.findFirst()
.orElse(null);
if (user == null) {
return ApiResponse.error("用户不存在");
}
return ApiResponse.success(user);
}
@PostMapping
public ApiResponse<User> createUser(@RequestBody User user) {
if (user.getName() == null || user.getEmail() == null) {
return ApiResponse.error("姓名和邮箱不能为空");
}
user.setId(counter.incrementAndGet());
users.add(user);
return ApiResponse.success(user);
}
}
创建健康检查控制器
在 src/main/java/com/tencent/cloudrun/controller 目录下创建 HealthController.java:
package com.tencent.cloudrun.controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
import java.time.LocalDateTime;
import java.util.HashMap;
import java.util.Map;
@RestController
public class HealthController {
@GetMapping("/health")
public Map<String, Object> health() {
Map<String, Object> health = new HashMap<>();
health.put("status", "healthy");
health.put("timestamp", LocalDateTime.now());
health.put("framework", "Spring Boot");
health.put("version", getClass().getPackage().getImplementationVersion());
health.put("java_version", System.getProperty("java.version"));
return health;
}
@GetMapping("/")
public Map<String, String> home() {
Map<String, String> response = new HashMap<>();
response.put("message", "欢迎使用 Spring Boot CloudBase 应用!");
response.put("status", "running");
return response;
}
}
配置应用属性
编辑 src/main/resources/application.properties:
# 服务器配置
server.port=${PORT:8080}
server.servlet.context-path=/
# 应用配置
spring.application.name=cloudrun-springboot
management.endpoints.web.exposure.include=health,info
# 日志配置
logging.level.com.tencent.cloudrun=INFO
logging.pattern.console=%d{yyyy-MM-dd HH:mm:ss} - %msg%n
第三步:本地测试
启动应用
mvn spring-boot:run
测试 API 接口
# 测试健康检查
curl http://localhost:8080/health
# 测试首页
curl http://localhost:8080/
# 测试用户列表
curl http://localhost:8080/api/users
# 测试分页
curl "http://localhost:8080/api/users?page=1&limit=2"
# 测试获取单个用户
curl http://localhost:8080/api/users/1
# 测试创建用户
curl -X POST http://localhost:8080/api/users \
-H "Content-Type: application/json" \
-d '{"name":"新用户","email":"newuser@example.com"}'
第四步: 准备部署文件
HTTP 云函数需要 scf_bootstrap 启动脚本和特定的端口配置。
1. 修改应用配置
编辑 src/main/resources/application.properties,确保云函数环境使用 9000 端口:
# 云函数端口配置(默认 8080,云函数环境使用 9000)
server.port=${PORT:8080}
server.servlet.context-path=/
# 应用配置
spring.application.name=cloudrun-springboot
management.endpoints.web.exposure.include=health,info
⚠️ 重要提示:CloudBase HTTP 云函数要求应用监听 9000 端口,通过环境变量
PORT=9000来控制。
2. 创建启动脚本
创建 scf_bootstrap 文件(无扩展名):
#!/bin/bash
export PORT=9000
export JAVA_OPTS="-Xmx512m -Xms256m -XX:+UseG1GC"
java $JAVA_OPTS -jar *.jar
💡 说明:
- 使用
*.jar通配符来匹配当前目录下的 JAR 包- CloudBase 云函数会将上传的文件解压到工作目录,JAR 包直接位于当前目录下
为启动脚本添加执行权限:
chmod +x scf_bootstrap
3. 构建 JAR 包
mvn clean package -DskipTests
4. 项目结构
cloudrun-springboot/
├── src/
│ └── main/
│ ├── java/
│ └── resources/
├── target/
│ └── *.jar # 构建产物(如:cloudrun-springboot-1.0-SNAPSHOT.jar)
├── pom.xml
├── scf_bootstrap # 🔑 云函数启动脚本
└── README.md
💡 说明:
scf_bootstrap是 CloudBase 云函数的启动脚本- 云函数会将上传的文件解压到工作目录,JAR 包直接位于当前目录下
- 设置
PORT=9000环境变量确保应用监听正确端口- 配置 JVM 参数优化内存使用
第五步: 部署应用
通过控制台部署
⚠️ 注意:
- java 服务启动较慢,建议将函数执行超时时间调整为 30s
- 登录 CloudBase 控制台
- 选择您的环境,进入「云函数」页面
- 点击「新建云函数」
- 选择「HTTP 云函数」
- 填写函数名称(如:
springboot-app) - 选择运行时:Java 8(或其他支持的版本)
- 提交方法选择:本地上传代码包
- 上传构建后的 JAR 包和
scf_bootstrap文件 - 自动安装依赖:关闭(Java 应用无需此选项)
- 点击「创建」按钮等待部署完成
通过 CLI 部署(敬请期待)
打包部署
如果需要手动打包:
# 构建应用
mvn clean package -DskipTests
# 创建部署包(使用 -j 参数避免目录结构)
zip -j springboot-app.zip target/*.jar scf_bootstrap
第六步:访问您的应用
部署成功后,您可以参考通过 HTTP 访问云函数设置自定义域名访问 HTTP 云函数。
示例请求
# 健康检查
curl https://your-app-url/health
# 获取用户列表
curl https://your-app-url/api/users
# 分页查询
curl "https://your-app-url/api/users?page=1&limit=2"
# 创建新用户
curl -X POST https://your-app-url/api/users \
-H "Content-Type: application/json" \
-d '{"name":"测试用户","email":"test@example.com"}'
常见问题
Q: 为什么使用 *.jar 而不是具体的 JAR 包名称?
A: 使用通配符有以下优势:
- 避免因项目版本号变化导致的路径错误
- 简化部署流程,无需每次修改启动脚本
- 确保能够找到构建后的 JAR 包,无论其具体名称如何
- 云函数会将文件解压到当前工作目录,使用相对路径即可
Q: 如果当前目录下没有 JAR 包怎么办?
A: 检查以下几点:
- 确保使用
zip -j参数打包,避免目录结构问题 - 验证
mvn clean package命令执行成功 - 检查
target目录下是否生成了 JAR 包 - 确保打包时包含了 JAR 包文件
Q: 为什么使用 zip -j 参数?
A: -j 参数的作用是"junk paths"(忽略路径):
- 使用
zip -j会忽略文件的目录结构,只保存文件本身 - 这样
target/cloudrun-springboot-1.0-SNAPSHOT.jar在解压后会直接变成cloudrun-springboot-1.0-SNAPSHOT.jar - 避免了在云函数中出现
/opt/target/的多层目录问题 - 简化了打包过程,无需手动复制和删除文件
Q: scf_bootstrap 中的 /opt 目录是什么?
A: /opt 是 CloudBase 云函数运行时环境的标准工作目录:
- 当您上传代码包到云函数时,所有文件会被解压到
/opt目录 - JAR 包、配置文件等都会位于
/opt目录下 - 这是云函数平台的标准约定,无需手动创建
- 启动脚本中必须使用绝对路径
/opt/your-jar-file.jar来引用 JAR 包
Q: 为什么 HTTP 云函数必须使用 9000 端口?
A: CloudBase HTTP 云函数要求应用监听 9000 端口,这是平台的标准配置。应用通过环境变量 PORT=9000 来控制端口,本地开发时默认使用 8080 端口。
Q: Spring Boot 应用冷启动时间较长怎么办?
A: 可以通过以下方式优化:
- 启用
spring.main.lazy-initialization=true - 减少自动配置的组件
- 使用 GraalVM 原生镜像(实验性)
- 合理设置 JVM 参数
Q: 运行 mvn spring-boot:run 时提示 "No plugin found for prefix 'spring-boot'" 怎么办?
A: 这是因为项目缺少 Spring Boot Maven 插件配置。解决方案:
- 确保
pom.xml中包含spring-boot-starter-parent作为父项目 - 在
<build><plugins>部分添加spring-boot-maven-plugin - 参考文档中的完整
pom.xml配置示例
Q: 如何处理 JAR 包过大的问题?
A:
- 使用
spring-boot-maven-plugin的repackage目标 - 排除不必要的依赖
- 使用
provided作用域排除运行时不需要的依赖
Q: 云函数中如何查看 Spring Boot 日志?
A: 在 CloudBase 控制台的云函数页面,点击函数名称进入详情页查看运行日志。
最佳实践
1. 环境变量管理
在 application.properties 中使用环境变量:
# 数据库配置
spring.datasource.url=${DB_URL:jdbc:h2:mem:testdb}
spring.datasource.username=${DB_USERNAME:sa}
spring.datasource.password=${DB_PASSWORD:}
# 应用配置
app.jwt.secret=${JWT_SECRET:mySecret}
app.upload.path=${UPLOAD_PATH:/tmp/uploads}
2. 端口配置策略
为了同时支持两种部署方式,建议使用动态端口配置:
# 支持多环境端口配置(默认 8080,云函数环境通过 PORT=9000 控制)
server.port=${PORT:8080}
# 云函数环境检测
spring.profiles.active=${SPRING_PROFILES_ACTIVE:default}
3. 添加全局异常处理
创建 GlobalExceptionHandler.java:
@RestControllerAdvice
public class GlobalExceptionHandler {
@ExceptionHandler(Exception.class)
public ApiResponse<String> handleException(Exception e) {
return ApiResponse.error("系统错误:" + e.getMessage());
}
@ExceptionHandler(IllegalArgumentException.class)
public ApiResponse<String> handleIllegalArgument(IllegalArgumentException e) {
return ApiResponse.error("参数错误:" + e.getMessage());
}
}
4. 配置 CORS 支持
创建 WebConfig.java:
@Configuration
public class WebConfig implements WebMvcConfigurer {
@Override
public void addCorsMappings(CorsRegistry registry) {
registry.addMapping("/api/**")
.allowedOrigins("*")
.allowedMethods("GET", "POST", "PUT", "DELETE")
.allowedHeaders("*");
}
}
5. JVM 参数优化
针对不同部署方式的 JVM 优化:
# HTTP 云函数(内存受限)
JAVA_OPTS="-Xmx512m -Xms256m -XX:+UseG1GC -XX:MaxGCPauseMillis=200"
# 云托管(更多资源)
JAVA_OPTS="-Xmx1g -Xms512m -XX:+UseG1GC"
6. 健康检查增强
自定义健康检查指示器:
@Component
public class CustomHealthIndicator implements HealthIndicator {
@Override
public Health health() {
// 检查应用状态
boolean isHealthy = checkApplicationHealth();
if (isHealthy) {
return Health.up()
.withDetail("status", "应用运行正常")
.withDetail("timestamp", LocalDateTime.now())
.build();
} else {
return Health.down()
.withDetail("status", "应用异常")
.build();
}
}
private boolean checkApplicationHealth() {
// 实现具体的健康检查逻辑
return true;
}
}
7、部署前检查清单
-
scf_bootstrap文件存在且有执行权限 - 端口配置为 9000
- JAR 包构建成功且可执行
- JVM 参数适合云函数环境
- 启用懒加载优化冷启动
- 测试本地启动是否正常
下一步
- 了解更多 HTTP 云函数配置选项
- 学习如何 连接 CloudBase 数据库