SpringCloud学习笔记(一、SpringCloud 基础)
目录:
- 概述
- 观察者模式
- 代理模式
概述:
spring系列中使用了大量的设计模式,而最常见的便是这观察者、代理模式,所以在讲解SpringCloud之前我们先学习下这两个最常见的设计模式。
观察者模式:
java实现了自己的观察者模式 >>> java.util.Observable;
1、public synchronized void addObserver(Observer o);添加一个观察者
2、public synchronized void deleteObserver(Observer o);删除一个观察者
3、protected synchronized void setChanged();修改通知标记,使得观察者观察的对象发生变化时能记得得知
4、public void notifyObservers(Object arg) ;通知观察者,被观察这已发生变化
public class ObserverDemo extends Observable {
/**
* 通知观察者被观察者已发生变化
*/
public void notifyObserver() {
// 设置变化状态为true >>> 被观察者已发生变化
this.setChanged();
// 将这一变化通知观察者
this.notifyObservers();
}
public static void main(String[] args) {
testAddObserver();
testDeleteObserver();
}
private static void testAddObserver() {
ObserverDemo observerDemo = new ObserverDemo();
observerDemo.addObserver((o, arg) -> System.err.println("服务列表发生变化1"));
observerDemo.notifyObserver();
}
private static void testDeleteObserver() {
ObserverDemo observerDemo = new ObserverDemo();
Observer observer = (o, arg) -> System.err.println("服务列表发生变化2");
observerDemo.addObserver(observer);
observerDemo.notifyObserver();
observerDemo.deleteObserver(observer);
observerDemo.notifyObserver();
}
}
代理模式:
1、静态代理
代理就是将除了自己本身需要做的核心业务外的任务交给代理人来做,如明星将接广告等事情交给经纪人。
public interface Agent {
/**
* 拍广告
*/
void advertising();
}
public class Celebrity implements Agent {
@Override
public void advertising() {
System.err.println("Celebrity拍广告");
}
}
public class Celebrity2 implements Agent {
@Override
public void advertising() {
System.err.println("Celebrity2拍广告");
}
}
public class ProxyDemo implements Agent {
private Agent agent;
public ProxyDemo(Agent agent) {
this.agent = agent;
}
@Override
public void advertising() {
before();
agent.advertising();
after();
}
private void after() {
System.err.println("after");
}
private void before() {
System.err.println("before");
}
}
class ProxyClient {
public static void main(String[] args) {
Agent agent = new ProxyDemo(new Celebrity());
agent.advertising();
System.out.println("----------------------------");
Agent agent2 = new ProxyDemo(new Celebrity2());
agent2.advertising();
}
}
静态代理的缺点是如果代理对象新代理了一个行为,那么委托对象也需要将新的代理行为重实现一遍,这样非常繁琐且重复。
为什么说静态代理繁琐呢,我们来增加一个行为。
public interface Agent {
/**
* 拍广告
*/
void advertising();
/**
* 集资
*/
void collectMoney();
}
然后我们修改一下Celebrity、Celebrity2
public class Celebrity implements Agent {
@Override
public void advertising() {
System.err.println("Celebrity拍广告");
}
@Override
public void collectMoney() {
System.err.println("Celebrity集资");
}
}
public class Celebrity2 implements Agent {
@Override
public void advertising() {
System.err.println("Celebrity2拍广告");
}
@Override
public void collectMoney() {
System.err.println("Celebrity2集资");
}
}
然后我们看看修改后的ProxyDemo
public class ProxyDemo implements Agent {
private Agent agent;
public ProxyDemo(Agent agent) {
this.agent = agent;
}
@Override
public void advertising() {
before();
agent.advertising();
after();
}
@Override
public void collectMoney() {
before();
agent.collectMoney();
after();
}
private void after() {
System.err.println("after");
}
private void before() {
System.err.println("before");
}
}
class ProxyClient {
public static void main(String[] args) {
Agent agent = new ProxyDemo(new Celebrity());
agent.advertising();
agent.collectMoney();
System.out.println("----------------------------");
Agent agent2 = new ProxyDemo(new Celebrity2());
agent2.advertising();
agent2.collectMoney();
}
}
我们发现仅仅增加一个行为委托类也要实现collectMoney,实现时不仅需要调用agent.collectMoney,还需要调用before和after是不是很麻烦,而接下来要讲的动态代理正是为了解决这一问题。
2、动态代理
动态代理的例子还是和静态代理一样,我们首先将委托类写好。
public class DynamicProxyDemo implements InvocationHandler {
private Object target;
public DynamicProxyDemo(Object target) {
this.target = target;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
after();
Object invoke = method.invoke(target, args);
before();
return invoke;
}
private void after() {
System.err.println("after");
}
private void before() {
System.err.println("before");
}
public Object getProxy() {
return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this);
}
}
其中最重要的是InvocationHandler接口和getProxy方法。
此时我们再往Agent增加一个行为,Celebrity、Celebrity2省略
public interface Agent {
/**
* 拍广告
*/
void advertising();
/**
* 集资
*/
void collectMoney();
/**
* 安排行程
*/
void travelArrangements();
}
然后我们发现,DynamicProxyDemo不需要再实现新的行为了,这都归功于InvocationHandler !!!∑(゚Д゚ノ)ノ
class DynamicProxyClient {
public static void main(String[] args) {
DynamicProxyDemo dynamic = new DynamicProxyDemo(new Celebrity());
Agent agent = (Agent) dynamic.getProxy();
agent.advertising();
agent.collectMoney();
agent.travelArrangements();
}
}
SpringCloud学习笔记(一、SpringCloud 基础)的更多相关文章
- SpringCloud学习笔记:SpringCloud简介(1)
1. 微服务 微服务具有的特点: ◊ 按照业务划分服务 ◊ 每个微服务都有独立的基础组件,如:数据库.缓存等,且运行在独立的进程中: ◊ 微服务之间的通讯通过HTTP协议或者消息组件,具有容错能力: ...
- SpringCloud学习笔记(2):使用Ribbon负载均衡
简介 Spring Cloud Ribbon是基于Netflix Ribbon实现的一套客户端负载均衡工具,在注册中心对Ribbon客户端进行注册后,Ribbon可以基于某种负载均衡算法,如轮询(默认 ...
- SpringCloud学习笔记(3):使用Feign实现声明式服务调用
简介 Feign是一个声明式的Web Service客户端,它简化了Web服务客户端的编写操作,相对于Ribbon+RestTemplate的方式,开发者只需通过简单的接口和注解来调用HTTP API ...
- SpringCloud学习笔记(4):Hystrix容错机制
简介 在微服务架构中,微服务之间的依赖关系错综复杂,难免的某些服务会出现故障,导致服务调用方出现远程调度的线程阻塞.在高负载的场景下,如果不做任何处理,可能会引起级联故障,导致服务调用方的资源耗尽甚至 ...
- SpringCloud学习笔记(5):Hystrix Dashboard可视化监控数据
简介 上篇文章中讲了使用Hystrix实现容错,除此之外,Hystrix还提供了近乎实时的监控.本文将介绍如何进行服务监控以及使用Hystrix Dashboard来让监控数据图形化. 项目介绍 sc ...
- SpringCloud学习笔记(6):使用Zuul构建服务网关
简介 Zuul是Netflix提供的一个开源的API网关服务器,SpringCloud对Zuul进行了整合和增强.服务网关Zuul聚合了所有微服务接口,并统一对外暴露,外部客户端只需与服务网关交互即可 ...
- SpringCloud学习笔记(7):使用Spring Cloud Config配置中心
简介 Spring Cloud Config为分布式系统中的外部化配置提供了服务器端和客户端支持,服务器端统一管理所有配置文件,客户端在启动时从服务端获取配置信息.服务器端有多种配置方式,如将配置文件 ...
- SpringCloud学习笔记:服务支撑组件
SpringCloud学习笔记:服务支撑组件 服务支撑组件 在微服务的演进过程中,为了最大化利用微服务的优势,保障系统的高可用性,需要通过一些服务支撑组件来协助服务间有效的协作.各个服务支撑组件的原理 ...
- js学习笔记:webpack基础入门(一)
之前听说过webpack,今天想正式的接触一下,先跟着webpack的官方用户指南走: 在这里有: 如何安装webpack 如何使用webpack 如何使用loader 如何使用webpack的开发者 ...
- Java学习笔记:语言基础
Java学习笔记:语言基础 2014-1-31 最近开始学习Java,目的倒不在于想深入的掌握Java开发,而是想了解Java的基本语法,可以阅读Java源代码,从而拓展一些知识面.同时为学习An ...
随机推荐
- Mysql—添加用户并授权
查询所有用户 -- 方式1 mysql> select host, user, password from mysql.user; -- 5.7版本之前的 mysql> select ho ...
- mysql-installer-community-8.0.17.0.msi安装教程
1.官网 https://dev.mysql.com/downloads/file/?id=488055 我选择自定义安装 注意这里是可以设置路径的,否则是默认地址 然后一直下一步就好 也是一路下一步 ...
- 关于华为模拟器(eNSP)添加路由器启动后一直打印#号的原因
操作系统为win10 专业版 1903:1.首先打开控制面板,找到windows defender防火墙,在“允许应用通过防火墙”中把和espn相关的两个的专网和公网都勾选上,最后确定:然后重启华为模 ...
- a迭代中的燃尽图统计
简单分析:项目一直在跟进,进展较缓,临近迭代结束,项目进度突飞猛进.
- promise 和 setTimeout 在任务队列的执行顺序
setTimeout(() => { console.log() }); const a = new Promise((resolve,reject)=>{ console.log(); ...
- webpack打包优化实践
事情缘由 近段时间在做基于scratch3.0的改造项目.基于scratch-gui改造,项目本身已经很大了,然后里面还要用到scratch-blocks,scratch-vm,scratch-ren ...
- Tarjan在图论中的应用(三)——用Tarjan来求解2-SAT
前言 \(2-SAT\)的解法不止一种(例如暴搜?),但最高效的应该还是\(Tarjan\). 说来其实我早就写过用\(Tarjan\)求解\(2-SAT\)的题目了(就是这道题:[2019.8.14 ...
- 洛谷 P2657 (数位DP)
### 洛谷 P2657 题目链接 ### 题目大意:给你一个数的范围 [A,B] ,问你这段区间内,有几个数满足如下条件: 1.两个相邻数位上的数的差值至少为 2 . 2.不包含前导零. 很简单的数 ...
- go语言使用go-sciter创建桌面应用(八) 窗口显示时,自动加载后端数据。
有些时候我们需要在窗口创建并显示时,加载一些后端的配置,这就需要用到view提供的几个事件. https://sciter.com/docs/content/sciter/View.htm state ...
- three.js实现世界3d地图
概况如下: 1.THREE.Shape绘制世界地图平面地图: 2.THREE.ExtrudeGeometry将绘制的平面沿着Z轴拉伸,实现3d效果: 效果图如下: 预览地址:three.js实现世界3 ...