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

推荐软件分类seo服务公司上海

推荐软件分类,seo服务公司上海,温州网页设计制作,网站建设流程图片目录 定义 结构 案例 优点 缺点 使用场景 扩展 分派 案例实现须知 动态分派 静态分派 双分派 定义 封装一些作用于某种数据结构中的各元素的操作,它可以在不改变这个数据结构的前提下定义作用于这些元素的新的操作。 结构 访问者模式包含以下主要角色…

目录

定义

结构

案例

优点

缺点

使用场景

扩展

分派

案例实现须知

动态分派

静态分派

双分派


定义

封装一些作用于某种数据结构中的各元素的操作,它可以在不改变这个数据结构的前提下定义作用于这些元素的新的操作。

结构

访问者模式包含以下主要角色:

  • 抽象访问者角色:定义了对每一个元素(Element)访问的行为,它的参数就是可以访问的元素,它的方法个数理论上来讲与元素类个数(Element的实现类个数)是一样的,从这点不难看出,访问者模式要求元素类的个数不能改变。
  • 具体访问者角色:给出对每一个元素类访问时所产生的具体行为。
  • 抽象元素角色:定义了一个接受访问者的方法(accept),其意义是指,每一个元素都要可以被访问者访问。
  • 具体元素角色: 提供接受访问方法的具体实现,而这个具体的实现,通常情况下是使用访问者提供的访问该元素类的方法。
  • 对象结构角色:定义当中所提到的对象结构,对象结构是一个抽象表述,具体点可以理解为一个具有容器性质或者复合对象特性的类,它会含有一组元素(Element),并且可以迭代这些元素,供访问者访问。

案例

抽象访问类

//(参数为要访问的元素,且有几个元素就有几个方法)
public interface Person {//可以将feed方法名改为visit主要就是为了访问元素,具体实现可以在子类中实现void feed(Dog dog);void feed(Cat cat);
}

 抽象元素类

//(只定义一个被访问的方法即可)
public interface Animal {void accept(Person person);
}

具体元素类 

public class Dog implements Animal {@Overridepublic void accept(Person person) {person.feed(this);System.out.println("修勾接受了食物");}
}public class Cat implements Animal{@Overridepublic void accept(Person person) {person.feed(this);System.out.println("修猫接受了食物");}
}

具体访问类 

public class Owner implements Person{@Overridepublic void feed(Dog dog) {System.out.println("主人投喂食物");}@Overridepublic void feed(Cat cat) {System.out.println("主人投喂食物");}
}public class Someone implements Person{@Overridepublic void feed(Dog dog) {System.out.println("陌生人投喂了食物");}@Overridepublic void feed(Cat cat) {System.out.println("陌生人投喂了食物");}
}

对象结构角色 

public class Home {private List<Animal> animals = new ArrayList<>();public void action(Person person){for (Animal animal : animals) {animal.accept(person);}}public void addAnimal(Animal animal){animals.add(animal);}
}

测试 

public class Client {public static void main(String[] args) {Home home = new Home();home.addAnimal(new Dog());home.addAnimal(new Cat());home.action(new Owner());}
}

主人投喂食物

修勾接受了食物

主人投喂食物

修猫接受了食物

通过对象结构角色来访问具体元素。而访问者通过参数来控制。访问具体元素后调用访问者对应的方法。实现了访问者通过访问不同元素有不同的行为。

优点

  • 扩展性好。在不修改对象结构中的元素的情况下,为对象结构中的元素添加新的功能。(在案例中即Owner与Someone通过实现Person类重写了不同的功能)
  • 复用性好。通过访问者来定义整个对象结构通用的功能,从而提高复用程度。
  • 分离无关行为。通过访问者来分离无关的行为,把相关的行为封装在一起,构成一个访问者,这样每一个访问者的功能都比较单一。

缺点

  • 对象结构变化很困难。在访问者模式中,每增加一个新的元素类,都要在每一个具体访问者类中增加相应的具体操作,这违背了“开闭原则”。
  • 违反了依赖倒置原则。访问者模式依赖了具体类,而没有依赖抽象类。

使用场景

  • 对象结构相对稳定,但其操作算法经常变化的程序。
  • 对象结构中的对象需要提供多种不同且不相关的操作,而且要避免让这些操作的变化影响对象的结构。

扩展

访问者模式用到了一种双分派的技术。

分派

变量被声明时的类型叫做变量的静态类型,有些人又把静态类型叫做明显类型;而变量所引用的对象的真实类型又叫做变量的实际类型。比如 Map map = new HashMap() ,map变量的静态类型是 Map ,实际类型是HashMap 。根据对象的类型而对方法进行的选择,就是分派,分派又分为两种,即静态分派和动态分派。

静态分派:发生在编译时期,分派根据静态类型信息发生。静态分派对于我们来说并不陌生,方法重载就是静态分派。

动态分派:发生在运行时期,动态分派动态地置换掉某个方法。Java通过方法的重写支持动态分派。

案例实现须知

编译看左,运行看右。

即Map map = new HashMap(),map的真实类型在编译时期是不知道的,在运行后才会知道map的真实类型为HashMap。

对于重写的方法,通过子类对象访问到子类中的方法。

对于重载的方法,分派是根据静态类型进行的,参数类型在编译时期就已经确定了。

动态分派

public class Animal{void print(){System.out.println("animal");}
}public class Dog extends Animal{void print(){System.out.println("dog");}
}public class Cat extends Animal{@Overridevoid print() {System.out.println("cat");}
}public class Client {public static void main(String[] args) {Animal animal = new Animal();Animal dog = new Dog();Animal cat = new Cat();animal.print();dog.print();cat.print();}
}

animal

dog

cat

由于子类重写了print()方法,因此在运行时动态调用的是真实对象中的print()方法。即多态的实现。

静态分派

public class Animal{
}public class Dog extends Animal {
}public class Cat extends Animal {
}public class Execute {void print(Animal animal){System.out.println("animal");}void print(Dog dog){System.out.println("dog");}void print(Cat cat){System.out.println("cat");}
}public class Client {public static void main(String[] args) {Animal animal = new Animal();Animal dog = new Dog();Animal cat = new Cat();Execute execute = new Execute();execute.print(animal);execute.print(dog);execute.print(cat);}
}

animal

animal

animal

对应实现须知中的重载方法部分,对于execute.print()方法,参数dog与cat在编译时期就已经确定了他们的类型是静态类型Animal,因此在运行时并不会通过new Dog()与new Cat()在对dog与cat动态分派

双分派

所谓双分派技术就是因为重载在选择一个方法的时候,不仅仅要根据消息接收者(即上例中的execute运行时的真实类型)的运行时区别,还要根据参数(即参数在运行时的真实类型)的运行时区别。

public class Animal{void accept(Execute execute){execute.print(this);}
}public class Dog extends Animal {@Overridevoid accept(Execute execute) {execute.print(this);}
}public class Cat extends Animal {@Overridevoid accept(Execute execute) {execute.print(this);}
}public class Execute {void print(Animal animal){System.out.println("animal");}void print(Dog dog){System.out.println("dog");}void print(Cat cat){System.out.println("cat");}
}public class Client {public static void main(String[] args) {Execute execute = new Execute();Animal animal = new Animal();Animal dog = new Dog();Animal cat = new Cat();animal.accept(execute);dog.accept(execute);cat.accept(execute);}
}

animal

dog

cat

重载与重写相结合,先进行子类中重写的方法,这里执行第一次分派是动态分派,子类实现方法中将自身作为参数去执行重载方法,完成第二次分派

双分派实现动态绑定的本质,就是在重载方法委派的前面加上了继承体系中覆盖的环节,由于覆盖是动态的,所以重载就是动态的了。

http://www.rdtb.cn/news/948.html

相关文章:

  • 网站网页设计原则t和p在一起怎么做网站
  • 深圳商城网站开发优秀网站设计
  • 个人可以做商城网站百度推广找谁做靠谱
  • 网站 目录结构百度推广电话号码
  • jsp网站开发论文做市场推广应该掌握什么技巧
  • ui设计需要学什么软件seo是什么意思呢
  • 自由型网站永久免费的电销外呼系统
  • 男生女生做羞羞事的网站网站收录怎么弄
  • 北京西站附近景点网络推广优化平台
  • 常见的域名注册网站企业网站推广效果指标分析
  • 潍坊网站建设公司哪家好公司官网优化方案
  • 大连网站哪家做的好网络广告案例以及分析
  • 免费建个人手机网站搜索引擎优化方法
  • 中色冶金建设有限公司网站泉州百度seo公司
  • 做暖漫画网站制作网站平台
  • 编程机构搜索引擎排名优化seo
  • wordpress数据连接失败如何做seo整站优化
  • 专门做书籍设计的网站南宁网站推广大全
  • 加工厂网站建设搜一搜搜索
  • 做外国网站百度搜到seo技术培训沈阳
  • 如何进入网站后台 被黑怎么给自己的公司建立网站
  • 推荐几个手机能看的网站大连企业黄页电话
  • 素材网站的下载服务器怎么做百度百科创建
  • 网站建设阝金手指实惠保定百度seo排名
  • 厦门微网站建设公司哪家好北京网站优化seo
  • 网站免费正能量破解版百度开放平台登录
  • 哪个网站可以做照片分享网络营销的几种模式
  • 网站建设优化哪家公司好seo咨询价格找推推蛙
  • 网站app封装怎么做济南计算机培训机构哪个最好
  • 快速网站优化服务各大网址收录查询