面向对象学习之二:继承
1、成员的可访问性:
访问修饰符 | 意义 |
public | 成员定义范围的外部和内部都是完全可见的,即对公共成员的访问不受限制。 |
protected | 成员仅对定义类及其派生类可见。 |
internal | 成员在包含它的程序集内部的任何地方可见,这包括定义在类以及程序集内定义类之外的任何范围。 |
private | 成员只在定义类内可见,这是最严格的访问形式,是类成员默认的访问级别。 |
2、成员隐藏:
继承可以扩展功能,但不能移除功能,例如,基类里可用的公共方法,在派生类及派生类的派生类的实例里都是可用的。不能从派生类中移除这些功能。请看下面代码:
namespace 隐藏成员
{
public class A
{
public void DoSomething()
{
Console.WriteLine("A.DoSomething");
}
}
public class B : A
{
public new void DoSomething()
{
Console.WriteLine("B.DoSomething");
}
public void DoSomethingElse()
{
Console.WriteLine("B.DoSomethingElse");
}
}
class Program
{
static void Main(string[] args)
{
B b = new B();
b.DoSomething();
b.DoSomethingElse();
A a = b;
a.DoSomething();
Console.ReadKey();
}
}
}
分析:
程序运行结果为:
虽然类B隐藏了类A的DoSomething实现,但并没有移除它,只是在通过B引用来调用的时候隐藏了。在Main方法中可以看到,可以轻松的绕过这一点。通过隐式转换把B实例引用转换成A实例引用,就可以通过A实例引用来调用A.dosomething的实现。因此A.dosomething并没有丢掉,而是隐藏了,需要多做一点工作来找到它。
3、override和new方法:
在派生类中重写方法,必须用override修饰符来标记方法。否则,编译器会提示警告,要求在派生方法中提供new修饰符或override。而且,编译器默认使用new修饰符,看看下面的代码:
namespace 继承和虚方法
{
public class A
{
public virtual void SomeMethod()
{
Console.WriteLine("A.SomeMethod");
}
}
public class B : A
{
public void SomeMethod()
{
Console.WriteLine("B.SomeMethod");
}
}
class Program
{
static void Main(string[] args)
{
B b = new B();
A a = b;
a.SomeMethod();
Console.ReadLine();
}
}
}
分析:
运行结果是:调用A.SomeMethod。
New关键字打破了这个层次结构中的虚函数链。当一个虚方法通过对象引用调用的时候,调用的方法取决于运行时的方法表,如果方法是虚的,运行时会搜索层次结构寻找方法在继承体系中最底层的派生版本,然后调用它(这里是类B)。但是,如果在搜索过程中遇到一个标有new修饰符的方法,它会退回继承体系中的上一级类(这里是类A),并使用这个类中的方法。由于C#在new和override都不存在的情况下默认使用new修饰符,这就是为什么A.SomeMethod被调用的原因。
如果B.SomeMethod标记为override,这段代码就会调用B.SomeMethod。
4、Base关键字:base允许访问一个实例的基类实现,看看下面代码:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace 继承
{
public class A
{
private int x;
public A(int var)
{
this.x = var;
}
public virtual void DoSomething()
{
Console.WriteLine("A.DoSomething");
}
}
public class B : A
{
public B():base(123)
{
}
public override void DoSomething()
{
Console.WriteLine("B.DoSomething");
base.DoSomething();
}
}
class Program
{
static void Main(string[] args)
{
B b = new B();
b.DoSomething();
Console.ReadKey();
}
}
}
分析:
第2个地方是B.DoSomething实现,在类B的实现中,实现B.DoSomething的时候借用类A中的A.DoSomething实现,因此,可以在B.DoSomething实现内通过base关键字直接调用A.DoSomething。
通常,调用实例的虚方法会调用虚方法最底层的实现,也就是B.DoSomething。但是,如果通过base关键字调用,就会调用基类最底层的的派生方法。因为A是基类,A.DoSomething是DoSomething相对于类A的基类方法,因此base.DoSomething就会调用A.DoSomething。这样可以实现一个重写方法,同时借用基类的实现。
亿速云「云服务器」,即开即用、新一代英特尔至强铂金CPU、三副本存储NVMe SSD云盘,价格低至29元/月。点击查看>>
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。