设计模式——依赖倒转原则

news/2024/7/5 6:13:29

文章目录

  • 基本介绍
  • 应用实例
  • 依赖关系传递的三种方式和应用案例
    • 1, 接口传递,应用案例代码
    • 2, 构造方法传递,应用案例代码
    • 3, setter 方式传递,应用案例代码
  • 依赖倒转原则的注意事项和细节

基本介绍

依赖倒转原则(Dependence Inversion Principle)是指:

  1. 高层模块不应该依赖低层模块,二者都应该依赖其抽象
  2. 抽象不应该依赖细节,细节应该依赖抽象
  3. 依赖倒转(倒置)的中心思想是面向接口编程
  4. 依赖倒转原则是基于这样的设计理念:相对于细节的多变性,抽象的东西要稳定的多。以抽象为基础搭建的架构比以细节为基础的架构要稳定的多。在 java 中,抽象指的是接口或抽象类,细节就是具体的实现类
  5. 使用接口或抽象类的目的是制定好规范,而不涉及任何具体的操作,把展现细节的任务交给他们的实现类去完成

应用实例

请编程完成 Person 接收消息 的功能。

  1. 实现方案 1 + 分析说明
public class DependecyInversion {
	public static void main(String[] args) {
		
		Person person = new Person();
		person.receive(new Email());
		
		//微信
		person.receive(new Weixin());
	}
}

class Email{
	public String getInfo() {
		return "电子邮件信息:hello,world";
	}
}


class Weixin{
	public String getInfo() {
		return "微信信息:hello,world";
	}
}

//完成Person接受消息的功能
//方式1分析
//1.简单,比较容易想到
//2.如果我们获取的对象微信,短信等,则新增类,同时Person也要增加相应的接受方法
//3.解决思路:引入一个抽象类的接口IReceiver,表示接收者,这样Person类与IReceiver接口发生依赖
// 因为 Email,微信 等等 都属于接受的范围,他们各自实现IReceiver,接口就ok,就符合依赖倒转原则
class Person{
	
	public void receive(Email email) {
		System.out.println(email.getInfo());
	}
	
	//增加一个微信的重载方法
	public void receive(Weixin weixin) {
		System.out.println(weixin.getInfo());
	}
}

  1. 实现方案 2(依赖倒转) + 分析说明
public class DependecyInversion {
	public static void main(String[] args) {
		
		//客户端无需改变
		Person person = new Person();
		person.receive(new Email());
		
		person.receive(new Weixin());
		
	}
}


//定义一个接口
interface Ireceiver{
	String getInfo();
}


class Email implements Ireceiver{
	
	@Override
	public String getInfo() {
		return "电子邮件信息:hello,world";
	}
	
}

//增加微信接收
//只需要实现下接口即可
class Weixin implements Ireceiver{
	@Override
	public String getInfo() {
		return "微信信息:hello,world";
	}
}



//方式2
class Person{
	
	//对接口的一个依赖,接口不能实例化,只需要实例化对应的实现即可
	public void receive(Ireceiver receiver) {
		System.out.println(receiver.getInfo());
	}
}

依赖关系传递的三种方式和应用案例

1, 接口传递,应用案例代码

public class Demo1 {

	public static void main(String[] args) {
		// TODO Auto-generated method stub
		ChangHong changHong = new ChangHong();
		OpenAndClose openAndClose = new OpenAndClose();
		openAndClose.open(changHong);
	}
}

// 方式1: 通过接口传递实现依赖
// 开关的接口
interface IOpenAndClose {
	public void open(ITV tv); // 抽象方法,接收接口
}

interface ITV { // ITV接口
	public void play();
}

class ChangHong implements ITV {

	@Override
	public void play() {
		// TODO Auto-generated method stub
		System.out.println("长虹电视机,打开");
	}

}

// 实现接口
class OpenAndClose implements IOpenAndClose {
	public void open(ITV tv) {
		tv.play();
	}
}

2, 构造方法传递,应用案例代码

public class Demo2 {
	public static void main(String[] args) {
		ChangHong changHong = new ChangHong();

		// 通过构造器进行依赖传递
		OpenAndClose openAndClose = new OpenAndClose(changHong);
		openAndClose.open();
	}
}

//方式2: 通过构造方法依赖传递
interface IOpenAndClose {
	public void open(); // 抽象方法
}

interface ITV { // ITV接口
	public void play();
}

class OpenAndClose implements IOpenAndClose {
	public ITV tv; // 成员

	public OpenAndClose(ITV tv) { // 构造器
		this.tv = tv;
	}

	public void open() {
		this.tv.play();
	}
}

class ChangHong implements ITV {

	@Override
	public void play() {
		// TODO Auto-generated method stub
		System.out.println("长虹电视机,打开");
	}

}

3, setter 方式传递,应用案例代码

public class Demo3 {
	public static void main(String[] args) {
		
		ChangHong changHong = new ChangHong();
		// 通过setter方法进行依赖传递
		OpenAndClose openAndClose = new OpenAndClose();
		openAndClose.setTv(changHong);
		openAndClose.open();

	}
}

//方式3 , 通过setter方法传递
interface IOpenAndClose {
	public void open(); // 抽象方法

	public void setTv(ITV tv);
}

interface ITV { // ITV接口
	public void play();
}

class OpenAndClose implements IOpenAndClose {
	private ITV tv;

	public void setTv(ITV tv) {
		this.tv = tv;
	}

	public void open() {
		this.tv.play();
	}
}

class ChangHong implements ITV {

	@Override
	public void play() {
		// TODO Auto-generated method stub
		System.out.println("长虹电视机,打开");
	}
}

依赖倒转原则的注意事项和细节

  1. 低层模块尽量都要有抽象类或接口,或者两者都有,程序稳定性更好.
  2. 变量的声明类型尽量是抽象类或接口, 这样我们的变量引用和实际对象间,就存在一个缓冲层,利于程序扩展和优化
  3. 继承时遵循里氏替换原则

http://lihuaxi.xjx100.cn/news/1465262.html

相关文章

stm32单片机/51单片机蜂鸣器不响(proteus模拟)

蜂鸣器不发生原因就1个:电压不够 所以需要提高蜂鸣器2端的电压:可以采用的方法有: 1提高蜂鸣器电阻,这样根据分压原理,可以提升蜂鸣器2段电压 2更改蜂鸣器的工作电压为更小的值,这个可以通过在proteus内…

Gin+微服务实现抖音视频上传到七牛云

文章目录 安装获取凭证Gin处理微服务处理 如果你对Gin和微服务有一定了解,看本文较容易。 安装 执行命令: go get github.com/qiniu/go-sdk/v7获取凭证 Go SDK 的所有的功能,都需要合法的授权。授权凭证的签算需要七牛账号下的一对有效的A…

2023前端面试笔记 —— HTML5(持续收集更新)

系列文章目录 内容链接2023前端面试笔记HTML5 文章目录 系列文章目录前言一、HTML 文件中的 DOCTYPE 是什么作用二、HTML、XML、XHTML 之间有什么区别三、前缀为 data- 开头的元素属性是什么四、谈谈你对 HTML 语义化的理解五、HTML5 对比 HTML4 有哪些不同之处六、meta 标签有…

【C++随笔01】联合体union —— 一种节省空间的类

【C随笔01】联合体union —— 一种节省空间的类 一、联合体(union)二、定义三、用法1、定义union、访问union成员2、匿名union3、使用类管理union成员4、管理并销毁string 一、联合体(union) 联合体是一种特殊的类。一个union可以有多个数据成员&#x…

茂名 湛江阳江某学校 ibm x3850服务器维修经历

简介:中国广东省阳江市某中学联想 IBM System x3850 x6服务器维修 io板故障处理经历分享: 这一天一位阳江的老师经其他学校老师介绍推荐对接我,说他们学校有一台ibm服务器出问题了,老师大致跟我描述了一下这台服务器发生故障的前…

5.7.webrtc线程的启动与运行

那在上一节课中呢?我向你介绍了web rtc的三大线程,包括了信令线程,工作线程以及网络线程。那同时呢,我们知道了web rtc 3大线程创建的位置以及运行的时机。 对吧,那么今天呢?我们再继续深入了解一下&#…

AUTOSAR规范与ECU软件开发(实践篇)4.4 软件组件代码及描述文件配置生成(下)

目录 4、Simulink-AUTOSAR Mapping配置 (1) Inports/Outports (2) Entry Point Functions (3) Function Callers (4) Data Transfers</

LLM提示词工程和提示词工程师Prompting and prompt engineering

你输入模型的文本被称为提示&#xff0c;生成文本的行为被称为推断&#xff0c;输出文本被称为完成。用于提示的文本或可用的内存的全部量被称为上下文窗口。尽管这里的示例显示模型表现良好&#xff0c;但你经常会遇到模型在第一次尝试时无法产生你想要的结果的情况。你可能需…