Spring Cloud微服务学习笔记(三)

案例搭建

使用微服务架构的分布式系统,微服务之间通过网络通信。我们通过服务提供者与服务消费者来描述微服务间的调用关系。

  • 服务提供者:服务的被调用方,提供调用接口的一方
  • 服务消费者:服务的调用方,依赖于其他服务的一方

我们以电商系统中常见的用户下单为例,用户向订单微服务发起一个购买的请求。在进行保存订单之前需要调用商品微服务查询当前商品库存,单价等信息。在这种场景下,订单微服务就是一个服务消费者,商品微服务就是一个服务提供者。

数据库表

用户表

CREATE TABLE `tb_user` (
	`id` int(11) NOT NULL AUTO_INCREMENT,
	`username` varchar(40) DEFAULT NULL COMMENT '用户名',
	`password` varchar(40) DEFAULT NULL COMMENT '密码',
	`age` int(3) DEFAULT NULL COMMENT '年龄',
	`balance` decimal(10,2) DEFAULT NULL COMMENT '余额',
	`address` varchar(80) DEFAULT NULL COMMENT '地址',
	PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8

商品表

CREATE TABLE `tb_product` (
	 `id` int(11) NOT NULL AUTO_INCREMENT,
	 `product_name` varchar(40) DEFAULT NULL COMMENT '名称',
	 `status` int(2) DEFAULT NULL COMMENT '状态',
	 `price` decimal(10,2) DEFAULT NULL COMMENT '单价',
	 `product_desc` varchar(255) DEFAULT NULL COMMENT '描述',
	 `caption` varchar(255) DEFAULT NULL COMMENT '标题',
	 `inventory` int(11) DEFAULT NULL COMMENT '库存',
	 PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8

订单表

CREATE TABLE `tb_order` (
	 `id` int(11) NOT NULL AUTO_INCREMENT,
	 `user_id` int(11) DEFAULT NULL COMMENT '用户id',
	 `product_id` int(11) DEFAULT NULL COMMENT '商品id',
	 `number` int(11) DEFAULT NULL COMMENT '数量',
	 `price` decimal(10,2) DEFAULT NULL COMMENT '单价',
	 `amount` decimal(10,2) DEFAULT NULL COMMENT '总额',
	 `product_name` varchar(40) DEFAULT NULL COMMENT '商品名',
	 `username` varchar(40) DEFAULT NULL COMMENT '用户名',
	 PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8

搭建环境

(1)创建父工程shop_parent
在IDEA中创建父工程 shop_parent 并引入坐标

<parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.1.6.RELEASE</version>
    </parent>
    <properties>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
        <java.version>1.8</java.version>
    </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-logging</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <version>1.18.4</version>
            <scope>provided</scope>
        </dependency>
    </dependencies>
    <dependencyManagement>
        <dependencies>
            <dependency>
                <groupId>org.springframework.cloud</groupId>
                <artifactId>spring-cloud-dependencies</artifactId>
                <version>Greenwich.RELEASE</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>
        </dependencies>
    </dependencyManagement>
    <repositories>
        <repository>
            <id>spring-snapshots</id>
            <name>Spring Snapshots</name>
            <url>http://repo.spring.io/libs-snapshot-local</url>
            <snapshots>
                <enabled>true</enabled>
            </snapshots>
        </repository>
        <repository>
            <id>spring-milestones</id>
            <name>Spring Milestones</name>
            <url>http://repo.spring.io/libs-milestone-local</url>
            <snapshots>
                <enabled>false</enabled>
            </snapshots>
        </repository>
        <repository>
            <id>spring-releases</id>
            <name>Spring Releases</name>
            <url>http://repo.spring.io/libs-release-local</url>
            <snapshots>
                <enabled>false</enabled>
            </snapshots>
        </repository>
    </repositories>
    <pluginRepositories>
        <pluginRepository>
            <id>spring-snapshots</id>
            <name>Spring Snapshots</name>
            <url>http://repo.spring.io/libs-snapshot-local</url>
            <snapshots>
                <enabled>true</enabled>
            </snapshots>
        </pluginRepository>
        <pluginRepository>
            <id>spring-milestones</id>
            <name>Spring Milestones</name>
            <url>http://repo.spring.io/libs-milestone-local</url>
            <snapshots>
                <enabled>false</enabled>
            </snapshots>
        </pluginRepository>
    </pluginRepositories>
    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
    </build>

(2)创建微服务工程模块

  • 创建公共模块 shop_service_common ,用于存放公共的实体类和工具类
  • 创建订单微服务模块 shop_service_order
  • 创建商品微服务模块 shop_service_product
  • 创建用户微服务模块 shop_service_user

(3)设置依赖
工程的依赖结构如图

搭建商品微服务

编写实体类

在shop_service_common中创建 cn.itcast.entity.Product 实体类,并配置

@Data
@Entity
@Table(name="tb_product")
public class Product {
	 @Id
	 private Long id;
	 private String productName;
	 private Integer status;
	 private BigDecimal price;
	 private String productDesc;
	 private String caption;
}

编写dao接口

在shop_service_product中创建 cn.itcast.product.dao.ProductDao 接口

public interface ProductDao extends JpaRepository<Product,Long> ,     
JpaSpecificationExecutor<Product> {}

编写service层

在shop_service_product中创建 cn.itcast.product.service.ProductService

public interface ProductService {
	 //根据id查询
	 Product findById(Long id);
	 //查询全部
	 List findAll();
	 //保存
	 void save(Product product);
	 //更新
	 void update(Product product);
	 //删除
	 void delete(Long id);
}

在shop_service_product中创建 cn.itcast.product.service.ProductServiceImpl 实现类

@Service
public class ProductServiceImpl implements ProductService {
	 @Autowired
	 private ProductDao productDao;
	 @Override
	 public Product findById(Long id) {
		 return productDao.findById(id).get();
	 }
	 @Override
	 public List findAll() {
		 return productDao.findAll();
	 }
	 @Override
	 public void save(Product product) {
		 productDao.save(product);
	 }
	  @Override
	 public void update(Product product) {
		 productDao.save(product);
	 }
	 @Override
	 public void delete(Long id) {
		 productDao.deleteById(id);
	}
}

编写web层

在shop_service_product中创建 cn.itcast.product.controller.ProductController

@RestController
@RequestMapping("/product")
public class ProductController {
 	@Autowired
 	private ProductService productService;
 	@GetMapping
 	public List findAll() {
 		return productService.findAll();
 	}
	 @GetMapping("/{id}")
	 public Product findById(@PathVariable Long id) {
		 return productService.findById(id);
 	}
	 @PostMapping
 	public String save(@RequestBody Product product) {
 		productService.save(product);
	 	return "保存成功";
 	}
	 @PutMapping("/{id}")
	 public String update(@RequestBody Product product) {
	 	productService.update(product);
	 	return "修改成功";
 	}
	 @DeleteMapping("/{id}")
	 public String delete(Long id) {
	 	productService.delete(id);
	 return "删除成功";
	}
}

controller中使用的@GetMapping是一个组合注解,相当与@RequestMapping(method=“get”)。类似的注解还有@PostMapping,@PutMapping,@DeleteMapping

配置启动类

在shop_service_product中创建 cn.itcast.product.ProductApplication 启动类

@SpringBootApplication(scanBasePackages="cn.itcast.product")
@EntityScan("cn.itcast.entity")
public class ProductApplication {
 	public static void main(String[] args) {
 		SpringApplication.run(ProductApplication.class, args);
 	}
}

配置yml文件

server:
 port: 9002
spring:
 application:
   name: shop-service-product
 datasource:
   driver-class-name: com.mysql.jdbc.Driver
   url: jdbc:mysql://localhost:3306/shop?useUnicode=true&characterEncoding=utf8
   username: root
   password: 111111
 jpa:
   database: MySQL
   show-sql: true
   open-in-view: true

其他微服务

同理创建出用户微服务和订单微服务

服务调用

前文已经编写了三个基础的微服务,在用户下单时需要调用商品微服务获取商品数据。那应该怎么做呢?总人皆知商品微服务提供了供人调用的HTTP接口。所以可以再下定单的时候使用http请求的相关工具类完成,如常见的HttpClient,OkHttp,当然也可以使用Spring提供的RestTemplate

RestTemplate介绍

Spring框架提供的RestTemplate类可用于在应用中调用rest服务,它简化了与http服务的通信方式,统一了RESTful的标准,封装了http链接, 我们只需要传入url及返回值类型即可。相较于之前常用的HttpClient,RestTemplate是一种更优雅的调用RESTful服务的方式。
在Spring应用程序中访问第三方REST服务与使用Spring RestTemplate类有关。RestTemplate类的设计原则与许多其他Spring 模板类(例如JdbcTemplate、JmsTemplate)相同,为执行复杂任务提供了一种具有默认行为的简化方法。
RestTemplate默认依赖JDK提供http连接的能力(HttpURLConnection),如果有需要的话也可以通过setRequestFactory方法替换为例如 Apache HttpComponents、Netty或OkHttp等其它HTTP library。
考虑到RestTemplate类是为调用REST服务而设计的,因此它的主要方法与REST的基础紧密相连就不足为奇了,后者是HTTP协议的方法:HEAD、GET、POST、PUT、DELETE和OPTIONS。例如,RestTemplate类具有headForHeaders()、getForObject()、postForObject()、put()和delete()等方法。

RestTemplate方法介绍

通过RestTemplate调用微服务

(1)在 shop_service_order工程中ProductApplication启动类 中配置RestTemplate

//配置RestTemplate交给spring管理
 @Bean
 public RestTemplate getRestTemplate() {
 	return new RestTemplate();
 }

(2) 编写下订单方法

@PostMapping("/{id}")
 public String order(Integer num) {
 	//通过restTemplate调用商品微服务
 	Product object = restTemplate.getForObject("http://127.0.0.1:9002/product/1", Product.class);
 	System.out.println(object);
 	return "操作成功";
 }

硬编码存在的问题

至此已经可以通过RestTemplate调用商品微服务的RESTFul API接口。但是我们把提供者的网络地址(ip,端口)等硬编码到了代码中,这种做法存在许多问题:

  • 应用场景有局限
  • 无法动态调整

那么应该怎么解决呢,就需要通过注册中心动态的对服务注册和服务发现

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值