手记

Springboot微服务项目实战入门教程

概述

本文介绍了Spring Boot微服务项目实战,从快速搭建Spring Boot项目到创建微服务模块,涵盖了服务注册与发现、服务间通信与负载均衡以及微服务的部署与测试等内容,帮助开发者掌握Springboot微服务项目实战的全流程。

Spring Boot基础入门

Spring Boot简介

Spring Boot 是一个开源框架,旨在简化Spring应用的初始搭建及开发过程。它通过使用约定优于配置的方式来减少项目配置的工作量,并提供了开箱即用的特性,使得开发者可以快速构建独立运行的、基于Spring的应用程序。Spring Boot 也支持嵌入式Web服务器(如Tomcat, Jetty或Undertow)和数据库,从而使得应用程序可以在任何Java虚拟机上运行。

快速搭建Spring Boot项目

使用Spring Initializr快速搭建一个Spring Boot项目。

  1. 访问Spring Initializr网站(https://start.spring.io/)。
  2. 配置项目属性:
    • Project:Maven Project
    • Language:Java
    • Spring Boot:选择最新版本
    • Project Metadata:
      • Group:com.example
      • Artifact:demo
      • Name:demo
      • Description:Spring Boot Demo
      • Packaging:Jar
      • Java:11
  3. Dependencies:选择所需的依赖,例如Spring Web, Spring Data JPA, Thymeleaf等。

生成项目后,解压文件并导入IDE中开始开发。

项目配置文件详解

Spring Boot项目主要使用application.propertiesapplication.yml进行配置,以下是一些常见的配置项:

# application.properties 示例
spring.application.name=DemoApp
spring.datasource.url=jdbc:mysql://localhost:3306/demo
spring.datasource.username=root
spring.datasource.password=root
spring.datasource.driver-class-name=com.mysql.jdbc.Driver
server.port=8080
# application.yml 示例
spring:
  application:
  name: DemoApp
  datasource:
  url: jdbc:mysql://localhost:3306/demo
  username: root
  password: root
  driver-class-name: com.mysql.jdbc.Driver
server:
  port: 8080

自动配置原则

Spring Boot 使用约定优于配置的原则,通过@EnableAutoConfiguration注解来自动配置项目。例如:

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;

@SpringBootApplication
public class DemoApplication {
    public static void main(String[] args) {
        SpringApplication.run(DemoApplication.class, args);
    }
}

@SpringBootApplication注解等同于@Configuration@EnableAutoConfiguration@ComponentScan的组合,可以自动扫描并配置项目中的组件。

微服务概念与架构

微服务架构简介

微服务架构是一种将应用程序构建为一组小型服务的架构风格,每个服务运行在自己的进程中,并通过轻量级的通信机制(通常是HTTP协议)进行互操作。微服务架构的目标是快速交付软件,提高部署频率,加快软件创新。

微服务与传统单体应用的区别

  • 模块化:微服务架构将应用程序分解为多个小型、独立的服务。
  • 部署:每个服务独立部署,降低部署复杂度。
  • 扩展性:每个服务可以独立扩展,提高资源利用率。
  • 维护:由于每个服务都是独立的,因此维护和更新更加灵活。

微服务优点与挑战

优点

  • 快速部署:由于每个服务是独立的,因此可以快速部署和更新。
  • 灵活性:每个服务可以使用不同的技术栈,提高灵活性。
  • 可扩展性:可以独立扩展每个服务。

挑战

  • 复杂性:服务间通信复杂,需要设计良好的API和通信机制。
  • 监控:需要更复杂的监控和日志记录系统来跟踪每个服务。
  • 安全性:需要确保服务间通信的安全性。

微服务设计原则

  • 单一职责原则:每个服务负责一个特定的业务功能。
  • 松耦合:服务之间尽量减少依赖。
  • 独立部署:每个服务可以独立部署和扩展。
  • 可重用性:服务设计应尽可能通用,可以被其他服务重用。
  • 幂等性:服务调用应该是幂等的,确保重复调用不会产生副作用。

微服务设计示例

假设我们有一个简单的用户服务,它负责用户管理功能。

import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;

@Entity
public class User {
    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    private Long id;
    private String name;
    private String email;

    // 构造函数、getter和setter方法
    public User() {}

    public User(String name, String email) {
        this.name = name;
        this.email = email;
    }

    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;
    }
}
import org.springframework.data.jpa.repository.JpaRepository;

import java.util.List;

public interface UserRepository extends JpaRepository<User, Long> {
    User findByName(String name);
    User findByEmail(String email);
}
import org.springframework.web.bind.annotation.*;

import java.util.List;

@RestController
@RequestMapping("/users")
public class UserController {
    private final UserRepository userRepository;

    public UserController(UserRepository userRepository) {
        this.userRepository = userRepository;
    }

    @GetMapping
    public List<User> getAllUsers() {
        return userRepository.findAll();
    }

    @PostMapping
    public User createUser(@RequestBody User user) {
        return userRepository.save(user);
    }

    @GetMapping("/{id}")
    public User getUserById(@PathVariable Long id) {
        return userRepository.findById(id).orElse(null);
    }

    @PutMapping("/{id}")
    public User updateUser(@PathVariable Long id, @RequestBody User user) {
        user.setId(id);
        return userRepository.save(user);
    }

    @DeleteMapping("/{id}")
    public void deleteUser(@PathVariable Long id) {
        userRepository.deleteById(id);
    }
}
Spring Boot微服务实战

创建微服务模块

创建一个简单的Spring Boot微服务模块,用于提供用户管理功能。

  1. 创建一个新的Spring Boot项目,添加Spring Web依赖。
  2. 创建User实体类。
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;

@Entity
public class User {
    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    private Long id;
    private String name;
    private String email;

    // 构造函数、getter和setter方法
    public User() {}

    public User(String name, String email) {
        this.name = name;
        this.email = email;
    }

    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;
    }
}
  1. 创建一个UserRepository接口,继承JpaRepository
import org.springframework.data.jpa.repository.JpaRepository;

import java.util.List;

public interface UserRepository extends JpaRepository<User, Long> {
    User findByName(String name);
    User findByEmail(String email);
}
  1. 创建一个UserController,提供REST API接口。
import org.springframework.web.bind.annotation.*;

import java.util.List;

@RestController
@RequestMapping("/users")
public class UserController {
    private final UserRepository userRepository;

    public UserController(UserRepository userRepository) {
        this.userRepository = userRepository;
    }

    @GetMapping
    public List<User> getAllUsers() {
        return userRepository.findAll();
    }

    @PostMapping
    public User createUser(@RequestBody User user) {
        return userRepository.save(user);
    }

    @GetMapping("/{id}")
    public User getUserById(@PathVariable Long id) {
        return userRepository.findById(id).orElse(null);
    }

    @PutMapping("/{id}")
    public User updateUser(@PathVariable Long id, @RequestBody User user) {
        user.setId(id);
        return userRepository.save(user);
    }

    @DeleteMapping("/{id}")
    public void deleteUser(@PathVariable Long id) {
        userRepository.deleteById(id);
    }
}

服务启动与配置

启动项目并配置Spring Boot应用。

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;

@SpringBootApplication
public class DemoApplication {
    public static void main(String[] args) {
        SpringApplication.run(DemoApplication.class, args);
    }
}

配置文件application.properties

spring.datasource.url=jdbc:mysql://localhost:3306/demo
spring.datasource.username=root
spring.datasource.password=root
spring.datasource.driver-class-name=com.mysql.jdbc.Driver
server.port=8080

使用Spring Boot Actuator监控服务

Spring Boot Actuator提供了生产就绪特性,包括健康检查、审计事件、应用程序上下文、HTTP跟踪等。

  1. 在Spring Boot项目中添加spring-boot-starter-actuator依赖。
  2. 配置安全机制(可选)。
management.endpoints.web.exposure.include=*
management.endpoint.health.show-details=always
  1. 启动服务后,访问http://localhost:8080/actuator查看所有可用的Actuator端点。

实战案例:构建简单的REST API服务

创建一个简单的REST API服务,提供用户管理功能。

  1. 创建并配置User实体类。
  2. 创建并配置UserRepository接口。
  3. 创建并配置UserController控制器。
  4. 启动服务并测试API。

测试API示例:

$ curl http://localhost:8080/users
$ curl -X POST -H "Content-Type: application/json" -d '{"name":"John Doe", "email":"john.doe@example.com"}' http://localhost:8080/users
服务注册与发现

服务注册中心介绍

服务注册中心用于管理和维护服务的地址列表,服务可以通过注册中心发现和调用其他服务。常见的服务注册中心有Eureka、Consul等。

使用Eureka实现服务注册与发现

  1. 在Spring Boot项目中添加spring-cloud-starter-netflix-eureka-server依赖。
<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-netflix-eureka-server</artifactId>
</dependency>
  1. 配置Eureka Server。
spring.application.name=eureka-server
server.port=8761
eureka.instance.hostname=localhost
eureka.client.register-with-eureka=false
eureka.client.fetch-registry=false
  1. 启动服务并访问http://localhost:8761/

使用Consul实现服务注册与发现

  1. 在Spring Boot项目中添加spring-cloud-starter-consul-discovery依赖。
<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-consul-discovery</artifactId>
</dependency>
  1. 配置Consul Server。
spring.application.name=service-discovery
spring.cloud.consul.discovery.hostname=localhost
spring.cloud.consul.discovery.port=8500
  1. 启动服务并访问http://localhost:8500/ui

实战案例:配置服务注册与发现

创建一个简单的服务注册与发现案例。

  1. 创建一个Eureka Server项目。
  2. 创建一个服务注册发现客户端项目。
  3. 配置服务注册发现客户端。
  4. 启动服务并测试注册与发现。

测试示例:

$ curl http://localhost:8761/
$ curl http://localhost:8500/ui
服务间通信与负载均衡

RPC与HTTP通信介绍

服务间通信通常使用远程过程调用(RPC)或HTTP协议。RPC实现过程调用,而HTTP协议实现HTTP请求。

使用Feign实现声明式服务调用

  1. 在项目中添加spring-cloud-starter-openfeign依赖。
<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-openfeign</artifactId>
</dependency>
  1. 配置Feign客户端。
import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;

@FeignClient(name = "service-name", url = "http://localhost:8080")
public interface UserClient {
    @GetMapping("/users/{id}")
    User getUserById(@PathVariable Long id);
}
  1. 在主类中添加@EnableFeignClients注解启动Feign。
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.openfeign.EnableFeignClients;

@SpringBootApplication
@EnableFeignClients
public class DemoApplication {
    public static void main(String[] args) {
        SpringApplication.run(DemoApplication.class, args);
    }
}

使用Ribbon实现客户端负载均衡

  1. 在项目中添加spring-cloud-starter-netflix-ribbon依赖。
<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-netflix-ribbon</artifactId>
</dependency>
  1. 配置服务提供方地址列表。
spring.cloud.loadbalancer.ribbon.listOfServers=http://localhost:8080,http://localhost:8081
  1. 使用@LoadBalanced注解的RestTemplate实现客户端负载均衡。
import org.springframework.cloud.client.loadbalancer.LoadBalanced;
import org.springframework.context.annotation.Bean;
import org.springframework.web.client.RestTemplate;

@SpringBootApplication
public class DemoApplication {
    @Bean
    @LoadBalanced
    public RestTemplate restTemplate() {
        return new RestTemplate();
    }
}

实战案例:构建服务间通信链路

创建一个简单的服务间通信链路案例。

  1. 创建一个服务提供方项目。
  2. 创建一个服务消费方项目。
  3. 配置服务消费方项目。
  4. 启动服务并测试服务调用。

测试示例:

$ curl http://localhost:8080/users/{id}
$ curl http://localhost:8081/users/{id}
微服务部署与测试

微服务部署策略

常见的微服务部署策略包括蓝绿部署、金丝雀部署和滚动部署。

  • 蓝绿部署:同时运行旧版本(蓝)和新版本(绿)服务,当确认新版本稳定后,将所有流量切换到新版本。
  • 金丝雀部署:将新版本服务部署到一个子集的用户中,观察其表现后再全面部署。
  • 滚动部署:逐个替换旧版本服务,逐步过渡到新版本。

使用Docker和Kubernetes部署微服务

  1. 使用Docker构建微服务镜像。
FROM openjdk:11-jre-slim
VOLUME /tmp
COPY target/*.jar app.jar
ENTRYPOINT ["java","-jar","/app.jar"]
  1. 构建并运行Docker镜像。
$ docker build -t <image-name> .
$ docker run -p 8080:8080 <image-name>
  1. 使用Kubernetes部署微服务。
apiVersion: apps/v1
kind: Deployment
metadata:
  name: demo-service
spec:
  replicas: 3
  selector:
    matchLabels:
      app: demo-service
  template:
    metadata:
      labels:
        app: demo-service
    spec:
      containers:
      - name: demo-service
        image: <image-name>
        ports:
        - containerPort: 8080
---
apiVersion: v1
kind: Service
metadata:
  name: demo-service
spec:
  selector:
    app: demo-service
  ports:
  - protocol: TCP
    port: 8080
    targetPort: 8080
  type: LoadBalancer
  1. 应用Kubernetes配置。
$ kubectl apply -f deployment.yaml

测试微服务应用的方法与工具

  • 单元测试:使用JUnit等工具测试单个组件。
  • 集成测试:测试服务间通信。
  • 端到端测试:模拟真实场景测试整个应用。
  • 性能测试:使用JMeter等工具测试性能。

实战案例:微服务的打包与部署

创建一个简单的微服务打包与部署案例。

  1. 创建并配置微服务项目。
  2. 构建并打包微服务应用。
  3. 使用Docker构建并运行微服务镜像。
  4. 使用Kubernetes部署微服务。

测试示例:

$ docker build -t <image-name> .
$ docker run -p 8080:8080 <image-name>
$ kubectl apply -f deployment.yaml
0人推荐
随时随地看视频
慕课网APP