一、问题

最近在写代码,有两个属性很相近的类,其中80%的属性(字段)都是一样的,剩下的才是各自不一样的,在设计的时候,采用了继承的方式,抽象除了一个父类,大概如下,

有FirstChild和SecondChild两个类,因为其中的属性name、code等是相同的,为此抽出了一个父类BaseDO,如下

  1. package com.example.day01;
  2.  
  3. public class BaseDO {
  4. private String name;
  5. private String code;
  6. private String field1;
  7. private String field2;
  8.  
  9. public String getName() {
  10. return name;
  11. }
  12.  
  13. public void setName(String name) {
  14. this.name = name;
  15. }
  16.  
  17. public String getCode() {
  18. return code;
  19. }
  20.  
  21. public void setCode(String code) {
  22. this.code = code;
  23. }
  24.  
  25. public String getField1() {
  26. return field1;
  27. }
  28.  
  29. public void setField1(String field1) {
  30. this.field1 = field1;
  31. }
  32.  
  33. public String getField2() {
  34. return field2;
  35. }
  36.  
  37. public void setField2(String field2) {
  38. this.field2 = field2;
  39. }
  40.  
  41. @Override
  42. public String toString() {
  43. return "BaseDO{" +
  44. "name='" + name + '\'' +
  45. ", code='" + code + '\'' +
  46. ", field1='" + field1 + '\'' +
  47. ", field2='" + field2 + '\'' +
  48. '}';
  49. }
  50. }

FirstChild

  1. package com.example.day01;
  2.  
  3. public class FirstChild extends BaseDO{
  4. private String myField1;
  5. private String myFiled2;
  6.  
  7. public String getMyField1() {
  8. return myField1;
  9. }
  10.  
  11. public void setMyField1(String myField1) {
  12. this.myField1 = myField1;
  13. }
  14.  
  15. public String getMyFiled2() {
  16. return myFiled2;
  17. }
  18.  
  19. public void setMyFiled2(String myFiled2) {
  20. this.myFiled2 = myFiled2;
  21. }
  22.  
  23. @Override
  24. public String toString() {
  25. return "FirstChild{" +
  26. "myField1='" + myField1 + '\'' +
  27. ", myFiled2='" + myFiled2 + '\'' +
  28. "} " + super.toString();
  29. }
  30. }

SecondChild

  1. package com.example.day01;
  2.  
  3. public class SecondChild extends BaseDO{
  4. private String secondField1;
  5. private String secondField2;
  6.  
  7. public String getSecondField1() {
  8. return secondField1;
  9. }
  10.  
  11. public void setSecondField1(String secondField1) {
  12. this.secondField1 = secondField1;
  13. }
  14.  
  15. public String getSecondField2() {
  16. return secondField2;
  17. }
  18.  
  19. public void setSecondField2(String secondField2) {
  20. this.secondField2 = secondField2;
  21. }
  22.  
  23. @Override
  24. public String toString() {
  25. return "SecondChild{" +
  26. "secondField1='" + secondField1 + '\'' +
  27. ", secondField2='" + secondField2 + '\'' +
  28. "} " + super.toString();
  29. }
  30. }

从上面可以看到两个子类除了含有父类的属性外还有自己各自的属性,现在有个需求是这样的,要实例化这两个子类。

二、如何解决

2.1、分别初始化

何为分别初始化,所谓分别初始化就是各自初始化自己的,为每个子类分别实现初始化其属性的方法,如下

  1. FirstChild fillFirstField(FirstChild firstChild){
  2.  
  3. firstChild.setName("apple");
  4. firstChild.setCode("apple");
  5. firstChild.setMyField1("first Child");
  6.  
  7. return firstChild;
  8. }
  1. SecondChild fillSecondField(SecondChild secondChild){
  2.  
  3. secondChild.setName("apple");
  4. secondChild.setCode("apple");
  5. secondChild.setSecondField1("second Child");
  6. return secondChild;
  7. }

这里作为演示对属性没有全部赋值,如果两个子类相同的属性比较多,那么赋值起来会比较麻烦,而且两个方法的代码重复度会很高。

2.2、抽象出一个公共的方法

既然,已经为两个子类抽象出了公共的属性,那么顺着这个思路下去,也可以抽象出一个公共的方法为这些公共的属性赋值,即为父类填充属性,

  1. BaseDO fillField(BaseDO baseDO){
  2. baseDO.setName("apple");
  3. baseDO.setCode("apple");
  4. return baseDO;
  5. }

好了,在进行子类初始化的时候已经有一个方法可以初始化其公共属性,那么接下来的事情,就是初始化其自己的属性即可,

下面就实例化FirstChild,然后初始化其公有属性

  1. FirstChild firstChild=new FirstChild();
  2. fillField(firstChild);
  3. firstChild.setMyField1("first Child");
  4. System.out.println(firstChild);

打印出firstChild的结果如下

可以看到已经把公共属性name、code和特意属性myField1进行赋值,完成了上面的需求。

可能有些同学会使用下面的写法,

可以看的该种写法存在错误,提示我们需要的类型是FirstChild,但是我们提供的BaseDO,我们知道fillField方法返回的BaseDO类型,一个父类型的实例不可赋值给子类型的引用(相反一个子类型的实例可以赋值给父类型,典型的多态),这怎么办那,向下类型转换,没错,如下

细心的小伙伴会问,那为什么fillField方法可以接受FirstChild的实例那,哈哈哈,前边红字提到了多态。

怎么样,是不是对多态又有了更深的理解。

延申一点

从fillField方法来看,我们知道该方法可以不设返回值,为什么可以不设返回值,因为引用类型,不是传值,可以理解为引用,哈哈,所以叫引用类型,在fillField方法中对其引用类型的参数进行了修改,那么在这个方法执行完了之后,引用这个参数的其他引用同样可以感知到其修改,下面的写法就是很好的佐证,

是不是又加深了引用和引用的对象间的关系,多说一句引用在jvm的内存模型中是在哪个区,引用指向的对象那?

三、总结

本文分析了在开发过程中,遇到公有属性很多的多个实体类时的设计思路,抽出公有的父类,由父类承担公有属性。并且在进行属性填充的时候,如果公有属性的值是一样的,那么可以抽出公共的方法进行属性填充,这里又提到了多态。

1、抽出公有属性;

2、对多态的理解;

3、向下类型转换;

4、引用类型的传递;

最后,广大读者们,对于类似的需求,你们有更好的设计思路吗,欢迎踊跃讨论。

在java的继承中你是否有这样的疑惑?的更多相关文章

  1. Java关于继承中的内存分配

    1.定义         super:当前对象的父类对象         this   :当前对象,谁调用this所在的方法,this就是哪一个对象.   2.内存分析 另一个例子: public s ...

  2. java之继承中的构造方法

    继承中的构造方法  1.子类的构造过程中必须调用其基类的构造方法. 2.子类可以在自己的构造方法中使用super(argument_list)调用基类的构造方法. 2.1.使用this(argumen ...

  3. java的继承中构造方法

    构造方法在创建对象的时候是被自动调用的,然后在继承中,是先调用父类的构造方法,然后在调用子类的构造方法, 当构造方法重写之后,在super中添加对应你想要调用构造方法的参数 例:父类 package ...

  4. java之继承中的静态变量

    package Test; /** * Created by wangbin10 on 2018/7/9. * 我们知道静态变量属于类级别变量,对应每个类只有一份,类的所有实例共有一份,而成员变量则分 ...

  5. java在继承中父类的成员变量是否会被子类所覆盖

    假如 父类 int num =7:子类 int num =9:父类是否会被子类所覆盖? 给你看两个例子: 第一个例子: 第二个例子: 这两个例子的区别只有一句话   由此证明了子类从父类继承的时候   ...

  6. JAVA基础--继承中的构造方法

    1. 子类的构造方法必须调用父类的构造方法 2. 子类在自己的构造方法中使用super(argument_list)调用父类的构造方法, 使用this(argument_list)调用自己的其他的构造 ...

  7. c++  与  java  中的 继承

    C++ 代码: #include <iostream> #include <string> using namespace std; class Parent { public ...

  8. Java多态机制和继承中重写重载

    关于Java中多态机制 http://www.cnblogs.com/chenssy/p/3372798.html 这篇博文讲的很透彻 大体意思是 多态定义: 多态就是指程序中定义的引用变量所指向的具 ...

  9. [Java] 继承中,父类被覆盖的成员变量、方法的可访问性

    在 Java 的继承机制里,在子类内部,可以访问父类被覆盖的变量和方法:在子类外部,可以访问父类的被覆盖变量,但是不能访问父类的被覆盖方法. 父类中被覆盖的方法不能在外部被方法,这是出于封装的考虑. ...

随机推荐

  1. 解决mybatis拦截器无法注入spring bean的问题

    公司要整合rabbitmq与mybatis拦截器做一个数据同步功能. 整合过程中大部分环节都没什么问题,就是遇到了mybatis拦截器 @Intercepts(@Signature(type = Ex ...

  2. CF1399F Yet Another Segments Subset

    首先注意一下题面要求,使得选出的线段两两要么包含要么不相交,也就是说一条线段可能会出现不相交的几条线段,而这些线段上面也可能继续这样包含线段.然后我们可以发现我们要做的实际上是在这条线段上选取几条线段 ...

  3. java run()方法无法throws 异常

    感谢大佬:https://blog.csdn.net/z_ssyy/article/details/95345205 其实原因很简单,run()方法是我们调用start()方法后,jvm再去调用run ...

  4. PHP操作Mysql疑问?

    1.Mysql控制台乱码 set character_set_results = 'utf8';

  5. 使用Docker安装ElasticSearch和可视化界面Kibana【图文教学】

    一.前言 Elasticsearch是一个基于Lucene的搜索服务器.它提供了一个分布式多用户能力的全文搜索引擎,基于RESTful web接口.Elasticsearch是用Java语言开发的,并 ...

  6. 强化学习中REIINFORCE算法和AC算法在算法理论和实际代码设计中的区别

    背景就不介绍了,REINFORCE算法和AC算法是强化学习中基于策略这类的基础算法,这两个算法的算法描述(伪代码)参见Sutton的reinforcement introduction(2nd). A ...

  7. Scala中的运算符

    Scala和Java中的运算符用法基本一致. 一.区别 1."=="和"equals"的用法 Java: String str1 = "abc&quo ...

  8. Python面向对象之 - 继承

    情况一:  子类完全继承父类所有的属性和方法, 自己没有一点更改. class Person(): def __init__(self, name, age): self.name = name se ...

  9. 记录一次dns劫持及其解决办法

    发现问题 偶然发现家里的私人云盘不能用了,最开始以为是云盘出现了问题,各种修复重启后发现云盘并没有问题.然后又发现电脑无法使用浏览器访问网页(或者加载异常缓慢),但是各种软件又可以正常使用,win+R ...

  10. [题解]Mail.Ru Cup 2018 Round 1 - A. Elevator or Stairs?

    [题目] A. Elevator or Stairs? [描述] Masha要从第x层楼去第y层楼找Egor,可以选择爬楼梯或者坐直升电梯.已知爬楼梯每层需要时间t1:坐直升电梯每层需要时间t2,直升 ...