当前位置: 首页 > news >正文

【微服务】微服务万字实战,带你了解工程原理

微服务实战

  • 1、前期准备
    • 1.1 技术选型
    • 1.2 模块设计
    • 1.3 微服务调用
  • 2、创建父工程
  • 3、创建基础模块
    • 3.1 导入依赖
    • 3.2 创建实体类
  • 4、创建用户微服务
    • 4.1 创建shop-user模块
    • 4.2 用户微服务启动类
    • 4.3 创建配置文件
  • 5、创建商品微服务
    • 5.1 创建shop_product模块
    • 5.2 商品微服务启动类
    • 5.3 创建配置文件
    • 5.4 创建ProductDao接口
    • 5.5 创建ProductService接口和实现类
    • 5.6 创建Controller
    • 5.7 启动商品微服务工程
    • 5.8 测试
  • 6、创建订单微服务
    • 6.1 创建shop-order模块
    • 6.2 订单微服务启动类
    • 6.3 创建配置文件
    • 6.4 创建OrderDao接口
    • 6.5 创建OrderService接口和实现类
    • 6.6 创建RestTemplate
    • 6.7 创建Controller
    • 测试
  • 7、总结

1、前期准备

前面讲了微服务的许多概念和简单性对阶段代码。我们本次就系统化的实现一个小型微服务项目,使用电商项目中的商品、订单、用户为案例来实现。

1.1 技术选型

  • maven:3.3.9
  • 数据库:MySQL 5.7
  • 持久层: SpingData Jpa
  • 其他: SpringCloud Alibaba 技术栈

1.2 模块设计

  • springcloud-alibaba :父工程
  • shop-common :公共模块【实体类】
  • shop-user :用户微服务 【端口: 807x】
  • shop-product :商品微服务 【端口: 808x】
  • shop-order :订单微服务 【端口: 809x】

在这里插入图片描述

1.3 微服务调用

在微服务架构中,最常见的场景就是微服务之间的相互调用。我们以电商系统中常见的用户下单为例来 演示微服务的调用:

客户向订单微服务发起一个下单的请求,在进行保存订单之前需要调用商品微服务查询商品的信息。

我们一般把服务的主动调用方称为服务消费者,把服务的被调用方称为服务提供者。在这里插入图片描述
在这种场景下,订单微服务就是一个服务消费者, 商品微服务就是一个服务提供者。

2、创建父工程

创建一个maven工程,然后在pom.xml文件中添加下面内容。

<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
http://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.1.3.RELEASE</version>
	</parent>

	<groupId>com.itheima</groupId>
	<artifactId>springcloud-alibaba</artifactId>
	<version>1.0-SNAPSHOT</version>
	<packaging>pom</packaging>

	<properties>
		<java.version>1.8</java.version>
		<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
		<project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
		<spring-cloud.version>Greenwich.RELEASE</spring-cloud.version>
		<spring-cloud-alibaba.version>2.1.0.RELEASE</spring-cloud-alibaba.version>
	</properties>
	
	<dependencyManagement>
		<dependencies>
			<dependency>
				<groupId>org.springframework.cloud</groupId>
				<artifactId>spring-cloud-dependencies</artifactId>
				<version>${spring-cloud.version}</version>
				<type>pom</type>
				<scope>import</scope>
			</dependency>
	    </dependencies>
	</dependencyManagement>

3、创建基础模块

3.1 导入依赖

创建 shop-common 模块,在pom.xml中添加依赖:

<?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
http://maven.apache.org/xsd/maven-4.0.0.xsd">
	<parent>
		<artifactId>springcloud-alibaba</artifactId>
		<groupId>com.itheima</groupId>
		<version>1.0-SNAPSHOT</version>
	</parent>
	
	<modelVersion>4.0.0</modelVersion>
	<artifactId>shop-common</artifactId>
	
<dependencies>
	<dependency>
		<groupId>org.springframework.boot</groupId>
		<artifactId>spring-boot-starter-data-jpa</artifactId>
	</dependency>
	
	<dependency>
		<groupId>org.projectlombok</groupId>
		<artifactId>lombok</artifactId>
	</dependency>
	
	<dependency>
		<groupId>com.alibaba</groupId>
		<artifactId>fastjson</artifactId><version>1.2.56</version>
	</dependency>
	
	<dependency>
		<groupId>mysql</groupId>
		<artifactId>mysql-connector-java</artifactId>
		<version>5.1.6</version>
	</dependency>
</dependencies>
</project>

3.2 创建实体类

用户实体类:

//用户
@Entity(name = "shop_user")
@Data
public class User {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Integer uid;//主键
    private String username;//用户名
    private String password;//密码
    private String telephone;//手机号
}

商品实体类:

//商品
@Entity(name = "shop_product")
@Data
public class Product {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Integer pid;//主键
    private String pname;//商品名称
    private Double pprice;//商品价格
    private Integer stock;//库存
}

订单实体类:


//订单
@Entity(name = "shop_order")
@Data
public class Order {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long oid;//订单id
    private Integer uid;//用户id
    private String username;//用户名
}

4、创建用户微服务

步骤:

  1. 创建模块 导入依赖
  2. 创建SpringBoot主类
  3. 加入配置文件
  4. 创建必要的接口和实现类

新建一个 shop-user 模块,然后进行下面操作:

4.1 创建shop-user模块

依赖于 shop-cmmon模块

<?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
http://maven.apache.org/xsd/maven-4.0.0.xsd">
	<parent>
		<artifactId>springcloud-alibaba</artifactId>
		<groupId>com.itheima</groupId>
		<version>1.0-SNAPSHOT</version>
	</parent>
	
	<modelVersion>4.0.0</modelVersion>
	<artifactId>shop-user</artifactId>
	
	<dependencies>
		<dependency>
			<groupId>com.itheima</groupId>
			<artifactId>shop-common</artifactId>
			<version>1.0-SNAPSHOT</version>
		</dependency>
	</dependencies>
</project>

4.2 用户微服务启动类

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

4.3 创建配置文件

server:
  port: 8071
spring:
  application:
    name: service-product
    datasource:
      driver-class-name: com.mysql.jdbc.Driver
        url: jdbc:mysql:///shop?serverTimezone=UTC&useUnicode=true&characterEncoding=utf-8&useSSL=true
          username: root
          password: root
            jpa:
              properties:
                hibernate:
                  hbm2ddl:
                    auto: update
                      dialect: org.hibernate.dialect.MySQL5InnoDBDialect

5、创建商品微服务

5.1 创建shop_product模块

创建一个名为 shop_product 的模块,并添加springboot依赖,同样依赖于shop-common

<?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
http://maven.apache.org/xsd/maven-4.0.0.xsd">
	<parent>
		<artifactId>springcloud-alibaba</artifactId>
		<groupId>com.itheima</groupId>
		<version>1.0-SNAPSHOT</version>
	</parent>
	
	<modelVersion>4.0.0</modelVersion>
	<artifactId>shop-product</artifactId>
	
	<dependencies>
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-web</artifactId>
		</dependency>
		
	    <dependency>
			<groupId>com.itheima</groupId>
			<artifactId>shop-common</artifactId>
			<version>1.0-SNAPSHOT</version>
		</dependency>
	</dependencies>
</project>

5.2 商品微服务启动类

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

5.3 创建配置文件

server:
port: 8081
spring:
application:
name: service-product
datasource:
driver-class-name: com.mysql.jdbc.Driver
url: jdbc:mysql:///shop?
serverTimezone=UTC&useUnicode=true&characterEncoding=utf-8&useSSL=true
username: root
password: root
jpa:
properties:
hibernate:
hbm2ddl:
auto: update
dialect: org.hibernate.dialect.MySQL5InnoDBDialect

5.4 创建ProductDao接口

package com.itheima.dao;
public interface ProductDao extends JpaRepository<Product,Integer> {
}

5.5 创建ProductService接口和实现类

package com.itheima.service.impl;
@Service
public class ProductServiceImpl implements ProductService {
@Autowired
private ProductDao productDao;
@Override
public Product findByPid(Integer pid) {
return productDao.findById(pid).get();
}
}

5.6 创建Controller

@RestController
@Slf4j
public class ProductController {
@Autowired
private ProductService productService;
@GetMapping("/product/{pid}")
public Product product(@PathVariable("pid") Integer pid) {
Product product = productService.findByPid(pid); log.info("查询到商品:" + JSON.toJSONString(product)); return product;
}
}

5.7 启动商品微服务工程

启动商品微服务工程,等到数据库表创建完毕之后,加入测试数据:

INSERT INTO shop_product VALUE(NULL,'小米','1000','5000');
INSERT INTO shop_product VALUE(NULL,'华为','2000','5000'); 
INSERT INTO shop_product VALUE(NULL,'苹果','3000','5000');
INSERT INTO shop_product VALUE(NULL,'OPPO','4000','5000');

5.8 测试

通过浏览器访问服务:
在这里插入图片描述
表明,商品微服务已完成,可用。

6、创建订单微服务

6.1 创建shop-order模块

创建一个名为 shop-order 的模块,并添加springboot依赖:

<?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
http://maven.apache.org/xsd/maven-4.0.0.xsd">
	<parent>
		<artifactId>springcloud-alibaba</artifactId>
		<groupId>com.itheima</groupId>
		<version>1.0-SNAPSHOT</version>
	</parent>
	
	<modelVersion>4.0.0</modelVersion>
	<artifactId>shop-order</artifactId>
	
	<dependencies>
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-web</artifactId>
		</dependency>
			
		<dependency>
			<groupId>com.itheima</groupId>
			<artifactId>shop-common</artifactId>
			<version>1.0-SNAPSHOT</version>
		</dependency>
	</dependencies>
</project>

6.2 订单微服务启动类

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

6.3 创建配置文件

server:
port: 8091
spring:
application:
name: service-product
datasource:
driver-class-name: com.mysql.jdbc.Driver
url: jdbc:mysql:///shop?
serverTimezone=UTC&useUnicode=true&characterEncoding=utf-8&useSSL=true
username: root
password: root
jpa:
properties:
hibernate:
hbm2ddl:
auto: update
dialect: org.hibernate.dialect.MySQL5InnoDBDialect

6.4 创建OrderDao接口

package com.itheima.dao;
public interface OrderDao extends JpaRepository<Order,Long> {
}

6.5 创建OrderService接口和实现类

@Service
public class OrderServiceImpl implements OrderService {
@Autowired
private OrderDao orderDao;
@Override
public void save(Order order) {
orderDao.save(order);
}
}

6.6 创建RestTemplate

@SpringBootApplication
public class OrderApplication {
public static void main(String[] args) {
SpringApplication.run(OrderApplication.class, args);
}
@Bean
public RestTemplate getRestTemplate() {
return new RestTemplate();
}
}

6.7 创建Controller

package com.itheima.controller;
@RestController
@Slf4j
public class OrderController {
	@Autowired
	private RestTemplate restTemplate;
	@Autowired
	private OrderService orderService;
	//准备买1件商品
	@GetMapping("/order/prod/{pid}")
	public Order order(@PathVariable("pid") Integer pid) { 
	log.info(">>客户下单,这时候要调用商品微服务查询商品信息"); 
	//通过restTemplate调用商品微服务
	Product product = restTemplate.getForObject("http://localhost:8081/product/" + pid, Product.class);
}

测试

在这里插入图片描述

7、总结

至此,用户微服务、商品微服务、订单微服务都已经简单的搭好了,也能互相调用。但相信大家发现其中的一个点,就是我们把服务提供者的网络地址(ip,端 口)等硬编码到了代码中

例如:

在这里插入图片描述

通过RestTemplate,把服务提供者的网络地址(ip,端 口)等硬编码到了代码中,这样会有什么问题呢?该怎么进行改进呢?大家可以想一下,我们在下篇文章中会解决这个问题。

从这篇文章中,大家可以简单的了解到微服务的架构是怎样的,其中的工程原理以及服务之间的调用等。

相关文章:

  • MySQL面试50题【mysql】
  • 【消息中间件】RocketMQ设计浅析
  • C语言学习完后,C++与Java我应该怎么选择
  • SpringCloud Bus消息总线
  • 图形学-反走样/抗锯齿
  • 2.1.5操作系统之线程概念与多线程模型
  • Spring中的AOP概念介绍使用、AOP相关术语、切入点表达式(面向切面编程上篇)
  • heic图片转换
  • 数据分析 | Pandas 200道练习题,每日10道题,学完必成大神(2)
  • 2017年某高校848数据结构真题复习
  • python正态分布中的normal函数
  • 场景金融持续引发行业关注,4.0时代打造金融服务新生态
  • 大数据面试重点之mysql篇
  • 【网络安全篇】JavaSript基础内容大全
  • 【数据结构与算法】时间复杂度和空间复杂度
  • Angular Elements 及其运作原理
  • Create React App 使用
  • HTTP那些事
  • JS变量作用域
  • js如何打印object对象
  • Promise初体验
  • Python socket服务器端、客户端传送信息
  • 初识 beanstalkd
  • 多线程 start 和 run 方法到底有什么区别?
  • 深入浅出webpack学习(1)--核心概念
  • 世界上最简单的无等待算法(getAndIncrement)
  • 微信支付JSAPI,实测!终极方案
  • 再次简单明了总结flex布局,一看就懂...
  • Python 之网络式编程
  • scrapy中间件源码分析及常用中间件大全
  • 蚂蚁金服CTO程立:真正的技术革命才刚刚开始
  • 容器镜像
  • $(function(){})与(function($){....})(jQuery)的区别
  • (2015)JS ES6 必知的十个 特性
  • (day6) 319. 灯泡开关
  • (Redis使用系列) Springboot 整合Redisson 实现分布式锁 七
  • (二)WCF的Binding模型
  • (附源码)spring boot基于小程序酒店疫情系统 毕业设计 091931
  • (论文阅读22/100)Learning a Deep Compact Image Representation for Visual Tracking
  • (轉貼)《OOD启思录》:61条面向对象设计的经验原则 (OO)
  • .NET 6 Mysql Canal (CDC 增量同步,捕获变更数据) 案例版
  • .net core开源商城系统源码,支持可视化布局小程序
  • .net6+aspose.words导出word并转pdf
  • .NET大文件上传知识整理
  • .NET开发不可不知、不可不用的辅助类(三)(报表导出---终结版)
  • .NET是什么
  • .net知识和学习方法系列(二十一)CLR-枚举
  • @Transactional 竟也能解决分布式事务?
  • [ solr入门 ] - 利用solrJ进行检索
  • [android] 手机卫士黑名单功能(ListView优化)
  • [C/C++] C/C++中数字与字符串之间的转换
  • [CISCN2019 华北赛区 Day1 Web5]CyberPunk --不会编程的崽
  • [Deep Learning] 神经网络基础
  • [hive] 窗口函数 ROW_NUMBER()
  • [Java]快速入门优先队列(堆)手撕相关面试题