`
庄表伟
  • 浏览: 1153088 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

敲响OO时代的丧钟——继承、多态与泛型冲突的一个例子

阅读更多

  写技术文章,例子其实很难举,特别是找到有杀伤力的,决定性的例子,相当困难。昨天我接着看《对象揭密》,总算被我找到一个,当然,它那上面的解说,实在是比较模糊,因此我决定用自己的话重新叙述一遍,代码示例用Java的泛型语法,但是要表达的意思,确实所有的具有泛型的OO语言都需要面对的。


java代码: 

public class X {
        protected int i;
        public void add(int a){
                i=i+a;
        }
}

public class Y1 extends X {
        public void add(int a){
                i=i+a*2;
        }
}


public class Y2 extends X {
        public void add(int a){
                i=i+a*3;
        }
}


  这是三个最简单的类,Y1和Y2都继承了X,并且重写了add函数。当然,这只是举例,实际上这三个add中,有两个是不合理的。


java代码: 

ArrayList<X> listx=new ArrayList<X>();
ArrayList<Y1> listy1=new ArrayList<Y1>();
ArrayList<Y2> listy2=new ArrayList<Y2>();
               
listx.add(new X());
listx.add(new Y1());
listx.add(new Y2());

listy1.add(new Y1());
listy2.add(new Y2());


  这几行代码都非常简单,只是为了说明一个道理,在ArrayList<Y1>和ArrayList<Y2>中,能够放的就只有Y1和Y2了,而在以X为泛型的ArrayList中,就可以放X、Y1、Y2了。而当然了,这样的用法,只怕是不合泛型的目标的,本来就是希望能有一个类型的自动检查与转换,都放在ArrayList<X>中,几乎就等于都放在ArrayList<Object>中了。

 

  现在我们有这样一个需求,对于得到的ArryaList,能够一一调用里面的对象的add(int a)方法,当然了,只要这个ArrayList里的对象都是X或者X的子类就行了。我们可以写出这样的代码:


java代码: 

public void addListX(ArrayList<X> listx){
        for(int i=0;i<listx.size();i++){
                X x=listx.get(i);
                x.add(1);
        }
}


  是不是很简单?且慢,这个addListX函数,我们能够把listx传递给它,但是能不能把listy1和listy2

也传递给它呢?如果我们能够把listy1和listy2传递给它,就相当于执行了如下的类型转换代码:


java代码: 

ArrayList<Y1> listy1=new ArrayList<Y1>();
ArrayList<X> listx=listy1;


  这样做行不行呢?在Java和C++中,是不行的。也就是说,如果我们要想只写一遍addListX这样的函数,而不用再多写两遍addListY1();addListY2();这样的函数,就需要把所有的X,Y1,Y2这样的类型都放到ArrayList<X>这样的容器里,否则,addListX函数,是不接受ArrayList<Y1>和ArrayList<Y2>类型的。即使Y1和Y2是X的子类型,ArrayList<X>与ArrayList<Y1>也毫不相干。不能相互转换。

 

  有人也许会说,为什么这么限制严格呢?Eiffel就没有这么这么严格的限制,他允许ArrayList<Y1>自动转型为ArrayList<X>,这样是好事情吗?如果listy能够被转型为ArrayList<X>,那么就可以往里面添加Y2类型的对象了,这又是原来的泛型ArrayList<Y1>不允许的。也就是说:除非addListX能够保证只对listy1做只读操作,否则,类型安全性这个泛型原本要追求的目标就不能实现了。而如果要追求绝对的类型安全性,像C++和Java那样,那么代码要么就得写三遍,要么X、Y1、Y2类型的对象就得都放到ArrayList<X>这样的泛型容器里去。

 

  注意看这其中的左右为难的状况,继承、多态、泛型,并不是真正正交的、互不干扰的,而是在一个相当普通的目标面前,他们就起了冲突了。

 

(未完待续)

分享到:
评论
1 楼 xiaomin98 2009-04-18  
在这个例子当中。你说的论据是不成立的。
ArrayList<Y1>和ArrayList<Y2>本来就不是同一个类型,这是一点疑问都没有的。不光是泛型。同样Y1数组和Y2数组也不是一个类型。这里不存在泛型与继承矛盾的问题。
解决方法非常简答,把addListX方法也写成泛型方法addListX<T>(List<T>)就可以了。
C#示例代码:
using System;
using System.Collections.Generic;
using System.Text;

namespace ConsoleApplication1
{
    class Program
    {
        static void Main(string[] args)
        {
            List<BaseClass> baseList = new List<BaseClass>();
            baseList.Add(new BaseClass());
           
            List<SubClassA> aList = new List<SubClassA>();
            aList.Add(new SubClassA());
            List<SubClassB> bList = new List<SubClassB>();
            CallList<BaseClass>(baseList);
            CallList<SubClassA>(aList);
            Console.Read();
        }
        public static void CallList<T>(List<T> list) where T:BaseClass
        {
            foreach (T var in list)
            {
                var.Show();
            }
        }
    }
  
    public class  BaseClass
    {
        public virtual void Show()
        {
            Console.WriteLine("parent");
        }
    }
    public class SubClassA : BaseClass
    {
        public override void Show()
        {
            Console.WriteLine("child a");
        }
    }
    public class SubClassB : BaseClass
    {
        public override void Show()
        {
            Console.WriteLine("child b");
        }
    }
}

相关推荐

    java 多态的泛型实现

    在这个例子中,`&lt;T&gt;`表示类型参数,`swap`方法可以接受任何类型的数组并交换指定索引处的元素。这既利用了多态的灵活性,也利用了泛型的类型安全性。 此外,泛型还提供了类型擦除的机制。由于Java是静态类型的语言...

    第8章动物类派生——继承与多态.ppt

    继承与多态 继承是面向对象编程(Object-Oriented Programming)中的一种机制,允许一个类继承另一个类的属性和行为。通过继承,子类可以重用父类的代码,减少代码的冗余和重复。同时,继承也可以实现多态性,即...

    C++继承与多态例子

    在这个例子中,`Dog`类继承了`Animal`类,并添加了一个新的成员函数`bark()`。由于继承,`Dog`类的对象可以调用`eat()`方法,这是从`Animal`类继承而来的。 接下来,我们来谈谈多态。多态有两种类型:静态多态...

    C#封装继承多态

    这是关于C#封装继承多态的一个简单的例子,对于深入学习C#中封装继承多态将很有帮助!

    C++经典实例——体现继承和多态

    在提供的压缩包“总实验”中,可能包含了多个C++源文件,每个文件代表一个特定的继承和多态的例子。通过分析这些实例,你可以看到如何在实践中应用这些概念,如何构建类层次结构,以及如何利用多态性提高代码的可...

    QtC++回合制小游戏——继承与多态的练习

    在本项目"QtC++回合制小游戏——继承与多态的练习"中,我们将深入探讨C++编程语言中的两个核心概念:继承与多态。这是一个非常实用的学习方式,通过一个可交互的游戏场景来实践这些理论知识,使得学习过程更加生动...

    封装继承多态总结

    封装继承多态总结

    c++ 的封装继承多态例子

    在C++编程语言中,封装、继承和多态是面向对象编程(OOP)的三大核心概念。下面将深入解析这三个关键概念,并结合"cppgood"这个可能包含示例代码的压缩包,来阐述它们在实际编程中的应用。 **封装** 是面向对象编程...

    易语言的继承多态演示

    压缩包中的"易语言的继承多态演示源码"文件可能包含了一个实际的程序示例,展示了如何在易语言中应用继承和多态。源码通常包括类定义、继承关系的声明、方法的重写以及可能的接口实现。通过分析源码,我们可以看到...

    Java实验报告——类的继承、多态的应用

    在本实验报告中,我们将深入探讨Java编程中的两个核心概念:类的继承和多态的应用。实验主要目标是加深对抽象类和抽象方法的理解,并掌握如何在实际编码中实现类的继承以及多态性。 首先,让我们理解抽象类和抽象...

    关于C#继承多态的应用项目

    在这个例子中,`ChildClass`继承了`ParentClass`,并重写了`Display`方法,这就是多态的基础。 接下来,我们讨论“多态”。多态是指同一种行为在不同的对象上表现出不同的形式。在C#中,这主要通过虚方法和重写来...

    CPP.rar_多态 .cpp_继承与多态_继承与多态C++

    标题中的"CPP.rar_多态 .cpp_继承与多态_继承与多态C++"表明这个压缩包文件包含了关于C++编程语言中多态性(Polymorphism)和继承(Inheritance)的概念示例。多态性和继承是面向对象编程(Object-Oriented ...

    java继承与多态

    下面是一个简单的例子来展示如何在Java中使用继承和多态: ```java // 定义一个基类 Person public class Person { private String name; private int age; public Person(String name, int age) { this.name ...

    易语言的继承多态演示源码

    ' 继承:使得一个类A能直接使用另一类B的属性和方法的途径,类A可以有自己的属性和方法。' 多态:引用同一个类型,使用不同实例而不同操作。' 如下:运行时根据需要生产产品,实现多态。' 封装之接口:抽取方法集合成基类,...

    java中继承与多态的题目

    Java 中继承与多态的题目 本资源摘要信息是关于 Java 中继承和多态的题目,涵盖了面向对象编程的基本概念和继承机制的应用。 继承的概念 继承是面向对象编程的一种机制,允许一个类(子类)继承另一个类(父类)...

    c#类与对象的继承与多态——图形间的继承关系

    在这个例子中,多态性使得我们可以对形状列表中的每一个对象调用相同的方法,即使它们的类型不同。这就是面向对象编程中的“鸭子类型”原则:如果它走起来像鸭子,叫起来也像鸭子,那它就是鸭子。 最后,考虑到"类...

    java基础继承封装多态

    继承是 Java 中的一种机制,允许一个类继承另一个类的属性和方法。继承的语法是使用 `extends` 关键字,例如 `public class Child extends Parent`。在 Java 中,只支持单一继承,也就是说一个类只能有一个直接父类...

    Educoder题目:Java面向对象 - 封装、继承和多态答案解析.md

    Educoder题目:Java面向对象 - 封装、继承和多态答案解析

    教学视频-继承&多态

    在编程领域,继承和多态是面向对象编程(OOP)中的两个核心概念,尤其是在Java这样的面向对象语言中。这两个概念极大地增强了代码的重用性和灵活性,使得软件设计更加高效和模块化。以下是对这两个概念的详细解释: ...

    C#第六章 初始继承和多态.pdf

    C#第六章 初始继承和多态.pdfC#第六章 初始继承和多态.pdfC#第六章 初始继承和多态.pdf C#第六章 初始继承和多态.pdfC#第六章 初始继承和多态.pdf

Global site tag (gtag.js) - Google Analytics