zoukankan      html  css  js  c++  java
  • .NET Core CSharp初级篇 1-3面向对象

    .NET Core CSharp初级篇 1-3

    本节内容为面向对象初级教程

    简介

    面向对象是整个C#中最核心最有特色的一个模块了,它很好的诠释了程序与现实世界的联系。

    面向对象的三大特征:继承、多态、封装;继承的含义可以理解为集合中的包含关系,例如人类属于动物类的一个分支,这就是一种继承。多态的理解就可以是人的呼吸用肺,鲤鱼使用鳃,这就是一种同种操作对应不同的实现。封装可以理解为一堆零件可以组成一个手机,这个过程就叫做封装。而将电脑显卡等拆下来组装成另一台电脑,则属于类的拆箱装箱。

    封装一个类的好处在哪里呢?我举一个例子:
    首先,我们考察一个常见的生活实例来进行说明,例如每当发工资的日子小王都来到 ATM 机
    前,用工资卡取走一笔钱为女朋友买礼物,从这个很帅的动作,可以得出以下的结论:

    • 小王和 ATM 机之间,以银行卡进行交互。要取钱,请交卡。
    • 小王并不知道 ATM 机将钱放在什么地方,取款机如何计算钱款,又如何通过银行卡返回小王
      所要数目的钱。对小王来说,ATM 就是一个黑匣子,只能等着取钱;而对银行来说,ATM 机就
      像银行自己的一份子,是安全、可靠、健壮的员工。
    • 小王要想取到自己的钱,必须遵守 ATM 机的对外约定。他的任何违反约定的行为都被视为不
      轨,例如欲以砖头砸开取钱,用公交卡冒名取钱,盗卡取钱都将面临法律风险,所以小王只能
      安分守己地过着月光族的日子。
      那么小王和 ATM 机的故事,能给我们什么样的启示?对应上面的 3 条结论,我们的分析如下:
    • 小王以工资卡和 ATM 机交互信息,ATM 机的入卡口就是 ATM 机提供的对外接口,砖头是塞不
      进去的,公交卡放进去也没有用。
    • ATM 机在内部完成身份验证、余额查询、计算取款等各项服务,具体的操作对用户小王是不
      可见的,对银行来说这种封闭的操作带来了安全性和可靠性保障。
    • 小王和 ATM 机之间遵守了银行规定、国家法律这样的协约。这些协约和法律,就挂在 ATM 机旁边的墙上。

    具体来说,封装隐藏了类内部的具体实现细节,对外则
    提供统一访问接口,来操作内部数据成员。这样实现的好处是实现了 UI 分离,程序员不需要知道
    类内部的具体实现,只需按照接口协议进行控制即可。同时对类内部来说,封装保证了类内部成
    员的安全性和可靠性。在上例中,ATM 机可以看做封装了各种取款操作的类,取款、验证的操作
    对类 ATM 来说,都在内部完成。而 ATM 类还提供了与小王交互的统一接口,并以文档形式——
    法律法规,规定了接口的规范与协定来保证服务的正常运行
    类属于在堆分布的变量,意味着它的大小是不固定的。可以动态的进行调节。

    创建与实例化类

    类的创建非常的简单,实例化也非常的简单,创建类就是把一个具体的事物抽象化,实例化就是将抽象化的类给转换成具象化的对象。例如我们定义一个People类,内含若干个变量;

    //定义类使用class关键字
    class People
    {
        public string Name;
        public int Age;
    }
    //实例化类
    People p = new People();
    

    或许你还看不太懂这些,别急,请继续往下看。

    修饰符

    访问控制修饰符

    • public:对访问没有任何限制,属于最高级别的访问权限
    • private:私有权限,最低级别的访问,只能在声明的代码段(类)中使用。
    • protected:保护权限,只有继承了该类才可以使用。
    • internal:仅包含当前程序集使用
    • protect internal:同一个程序集的类和其派生的类可以使用

    这样说或许过于抽象,我们这样来解释吧,一个程序就类似一个公司,public就好比是董事长、CEO一类的权限,拥有着最高级别的访问;protected你可以理解为部门经理,它的下属就是继承该部门,下属可以访问父类(部门)的资源,但不可以访问其他部门的protected资源,体现为一种纵向的权限控制。Internal类似与考勤部门,无论该部门是否属于考勤部门领导,考勤部门都可以管辖其他部门,体现为一种横向的权限控制。Protected internal则是具有两种属性。

    可选修饰符

    • static(可用于类内成员):静态的,表示只被创建一次,属于所有对象公用的变量
    • sealed:密封类,禁止类被继承
    • abstract:抽象类,要求类被继承,并且不能实例化
    • virtual(不能用于类):表示可以被重写
    • readonly(用于字段):表示该字段只读
    • const(用于字段):表示常量
    • extern(用于函数):表示该函数由外部实现
    • async(用于函数):表示该函数为异步函数

    函数

    函数也被称为方法,是包含一系列语句的代码块。封装了类的行为,提供了类的对外表现。用于将封装的内部细节以公有方法提供对外接口,从而实现与外部的交互与响应。例如,从上面属性的分析我们可知,实际上对属
    性的读写就是通过方法来实现的。因此,对外交互的方法,通常实现为 public。程序通过调用该方法并指定任何所需的方法参数使语句得以执行。 在 C# 中,每个执行的指令均在方法的上下文中执行。

    函数的构成由访问控制关键字+修饰符+返回值+函数名称+函数参数+函数体,如果一个类内函数或者其他成员使用了static关键字,则可以不实例化类对其进行调用,因为使用了static标明的成员,属于全体该类对象共有。例如下面这个例子:

    class Man
    {
        static void GettingOld()
        {
            //life - 1s
        }
        public void Eat()
        {
        }
    }
    //静态方法可以直接调用
    Man.GettingOld();
    Man man = new Man();
    man.Eat();
    

    函数的使用如下,其中x,y称为形参,x1,y1称为实参,通常对形参的操作并不会影响到实参

    public static int Add(int x,int y)
    {
        x++;y++;
        return x+y;
    }
    int x1 =5;
    int y1 =6;
    Add(x1,y1);
    //带有默认参数
    public static int Add(int x,int y=4)
    {
    }
    //不定参数
    public void Add(params object[] a)
    {
    }
    
    

    当然不是所有的方法都被实现为 public,否则类内部的实现岂不是全部暴露在外。必须对对外
    的行为与内部操作行为加以区分。因此,通常将在内部的操作全部以 private 方式来实现,而将需要与外部交互的方法实现为 public,这样既保证了对内部数据的隐藏与保护,又实现了类的对外交互。例如在 ATM 类中,对钱的计算、用户验证这些方法涉及银行的关键数据与安全数据的保护问题,必须以 private 方法来实现,以隐藏对用户不透明的操作,而只提供返回钱款这一 public 方法接口即可。在封装原则中,有效地保护内部数据和有效地暴露外部行为一样关键。

    函数的重载与重写

    重载函数表示使用同一个函数名,通过参数的不同,从而实现使用同一个名称可以选择调用多种函数。例如实现两个数字的相加,传入的有可能是整型参数也有可能是浮点型参数,因此,我们选择使用重载函数去实现。以下是一个重载的例子;

    public static int Add(int x,int y)
    {
        return x+y;
    }
    public static double Add(double x,double y)
    {
        return x+y;
    }
    Add(1,2);//调用第一个Add
    Add(1.1,2.2);//调用第二个函数
    

    通过调用函数时传入的参数不同,就可以很简单的用不同方法实现。

    函数重写则多出现在面向对象的多态性中,这里我不会很详细的讲解,在后面会有一个详细的讲解。重写就可以理解为,人呼吸用肺,大部分鱼类呼吸用鳃,呼吸这个函数就是在两个类中被重写过(即实现方法在不同的类中)。具体的实现我会在后一步进行讲解

    需要注意的是,重载需要在参数上有本质的区别,例如个数、类型不同,重写则需要方法可以被重写,使用override关键字表明重写的函数

    类中重要的两个函数

    构造函数:构造函数是一种特殊的函数,它的签名和类名一致,并且没有返回值。它可以接受任意个参数。当类被实例化的时候,对应的构造函数会被调用。可以说,对象是通过调用类的构造函数进行创建。如果不指定构造函数,C#会自动调用默认的无参构造函数。如果重载了构造函数,并且传入了指定参数,则会调用对应的构造函数。

    析构函数:类似与构造函数,但是调用是在GC(垃圾回收器)回收类对象的时候自动调用,通常无需去重写。例如:

    class A
    {
           //默认无参构造函数
           public A()
           {
           }
           public A(int a)
           {
           } 
           // 析构函数
           ~A()
           {
           }
    }
    A a = new A(1);//调用第二个构造函数
    

    字段和属性

    (此处补充IL代码)

    字段:类中具体实现存储数据的变量,你可以理解为各个零件。通常而言,字段不会对外进行开发访问权限。例如:幼儿园读书的小朋友类,里面有一个Age(年龄)字段。假设一个人实例化,我们给年龄赋值上1000。这符合常理吗?显然是不符合的。那么我们就要使用属性进行控制输入的变量。

    属性:属性不存储数据,通常定义为 public,表示类的对外成员。属性具有可读、可写属性,通过 get 和 set 访问器来实现其读写控制。但是如果你使用默认的属性实现方法,例如public string a {get;set;},C#会自动的为你隐式生成一个私有的字段a。属性本质上是作为外部访问字段的一个媒介、桥梁,也称之为接口。通常来说,我们会将字段定义为私有的,将属性定义为公有的,通过属性去返回和设置其中的值。在这里,我们涉及到了两个从未见过的关键字——get,set。

    get访问器:get访问器本质上是一个返回值为属性类型的函数,你可以使用dnSpy进行反编译查看。你一般需要在get中返回你需要访问的变量。

    set访问器:使用value关键字接受外界传来的参数并且赋值给你的字段,本质上也只是一个函数,当你对属性赋值的时候,就会调用他的set控制块内的代码

    class A
    {
        private int a;
        public int A 
        {
            get{return a;}
            set
            {
                if(value>5)
                {
                    a=value;
                }
            }
        }
        //自动生成一个b字段
        public int B{get;set;}
    }
    

    选看 函数参数修饰符

    对C#了解的人都知道,实参到形参的传递时存在一个数据备份的过程,因此我们对形参的操作本质上是对备份的变量进行操作,并不会影响到实参的值。但是在C#中,可以对形参进行修饰,使传入的变量按内存地址进行传入,这样就可以实现改变实参的值了。

    ref 关键字

    ref关键字的要求有以下几点:

    • 被调用函数的参数和调用时都必须使用ref标记参数
    • 传递到 ref 参数的参数必须初始化,否则程序会报错

    例如:

            static void Main(string[] args)
            {
                int a = 10;
                int b = 20;
                Test(ref a,ref b);
                Console.WriteLine("a:{0},b:{1}", a, b);
            }
            static void Test(ref int a, ref int b) 
            {
                a = a+b;  
                b = 6;
            }
    

    out 关键字

    out关键字的要求有以下几点:

    • 方法定义和调用方法都必须显式使用 out关键字

    • out关键字无法将参数值传递到out参数所在的方法中,只能传递参数的引用,所以out参数的参数值初始化必须在其方法内进行,否则程序会报错

            static void Main(string[] args)
            {
                int a=100;
                int b;
                Test(out a, out b);
                Console.WriteLine("a:{0},b:{1}", a, b);
            }
            static void Test(out int a, out int b)
            {
                a = 1+2;
                b = 1;
            }
    

    in关键字

    in关键字的要求有:

    • 它让形参成为实参的别名,这必须是变量。 换而言之,对形参执行的任何操作都是对实参执行的。
    • in 参数无法通过调用的方法进行修改。 out 参数必须由调用的方法进行修改,这些修改在调用上下文中是可观察的,而 ref 参数是可以修改的
    int readonlyArgument = 44;
    InArgExample(readonlyArgument);
    Console.WriteLine(readonlyArgument);     // value is still 44
    
    void InArgExample(in int number)
    {
        // Uncomment the following line to see error CS8331
        //number = 19;
    }
    

    练习题

    • 请试着创建一个圆类(Circular),要求封装圆周率和半径(或直径),并且定义一个含有一个参数的构造函数,传入的是半径(直径)。并写入计算周长和面积的函数。

    • 定义一个用户类,要求用户名和密码不可以被访问,只允许设置,并且密码小于6位需要输出相应提示并且不进行赋值要求重新赋值。

    前往Github获取更多本节资料(PPT,实例代码)
    如果我的教程帮到了您,希望您动动小手,在GitHub给我一个star

    Github

    BiliBili主页

    WarrenRyan's Blog

    博客园

    Reference

    《你必须知到的.NET》

  • 相关阅读:
    Hadoop入门
    Redis缓存分布式高可用架构原理
    ES分布式搜索引擎架构原理
    Java锁,多线程与高并发,JUC并发包
    图算法--拓扑序列
    数据结构--数组模拟队列
    数据结构--数组模拟栈
    数据结构--数组模拟双链表
    数据结构--数组模拟单链表
    基础算法--双指针
  • 原文地址:https://www.cnblogs.com/WarrenRyan/p/11208495.html
Copyright © 2011-2022 走看看