c#笔记

c#入门

===

以下几点值得注意:

  • C# 是大小写敏感的。
  • 所有的语句和表达式必须以分号(;)结尾。
  • 程序的执行从 Main 方法开始。
  • 与 Java 不同的是,文件名可以不同于类的名称。

零. 数据契约

[DataContract]:

服务契约定义了远程访问对象和可供调用的方法,数据契约则是服务端和客户端之间要传送的自定义数据类型。

[DataMember]:

来标记被序列化的字段

一. using 关键字

  • using 关键字表明程序使用的是给定命名空间中的名称。例如,我们在程序中使用 System 命名空间,其中定义了类 Console。我们可以只写:
Console.WriteLine ("Hello there");
  • 我们可以写完全限定名称,如下:
System.Console.WriteLine("Hello there");

二. 命名空间

  • 一个 namespace 是一系列的类。HelloWorldApplication 命名空间包含了类 HelloWorld。

2.1 定义命名空间

  • 命名空间的设计目的是提供一种让一组名称与其他名称分隔开的方式。在一个命名空间中声明的类的名称与另一个命名空间中声明的相同的类的名称不冲突。

  • 命名空间的定义是以关键字 namespace 开始,后跟命名空间的名称,如下所示:

namespace namespace_name
{
   // 代码声明
}
  • 下面的程序演示了命名空间的用法:
using System;
namespace first_space
{
   class namespace_cl
   {
      public void func()
      {
         Console.WriteLine("Inside first_space");
      }
   }
}
namespace second_space
{
   class namespace_cl
   {
      public void func()
      {
         Console.WriteLine("Inside second_space");
      }
   }
}   
class TestClass
{
   static void Main(string[] args)
   {
      first_space.namespace_cl fc = new first_space.namespace_cl();
      second_space.namespace_cl sc = new second_space.namespace_cl();
      fc.func();
      sc.func();
      Console.ReadKey();
   }
}
  • 当上面的代码被编译和执行时,它会产生下列结果:
Inside first_space
Inside second_space

在同一个命名空间里的类实例化不需要写命名空间名
  • 您也可以使用 using 命名空间指令,这样在使用的时候就不用在前面加上命名空间名称。该指令告诉编译器随后的代码使用了指定命名空间中的名称。下面的代码演示了命名空间的应用。让我们使用 using 指定重写上面的实例:
using System;
using first_space;
using second_space;

namespace first_space
{
   class abc
   {
      public void func()
      {
         Console.WriteLine("Inside first_space");
      }
   }
}
namespace second_space
{
   class efg
   {
      public void func()
      {
         Console.WriteLine("Inside second_space");
      }
   }
}   
class TestClass
{
   static void Main(string[] args)
   {
      abc fc = new abc();
      efg sc = new efg();
      fc.func();
      sc.func();
      Console.ReadKey();
   }
}

2.2 嵌套命名空间

  • 命名空间可以被嵌套,即您可以在一个命名空间内定义另一个命名空间,如下所示:
namespace namespace_name1 
{
   // 代码声明
   namespace namespace_name2 
   {
     // 代码声明
   }
}
  • 您可以使用点(.)运算符访问嵌套的命名空间的成员,如下所示:
using System;
using SomeNameSpace;
using SomeNameSpace.Nested;

namespace SomeNameSpace
{
    public class MyClass 
    {
        static void Main() 
        {
            Console.WriteLine("In SomeNameSpace");
            Nested.NestedNameSpaceClass.SayHello();
        }
    }

    // 内嵌命名空间
    namespace Nested   
    {
        public class NestedNameSpaceClass 
        {
            public static void SayHello() 
            {
                Console.WriteLine("In Nested");
            }
        }
    }
}
  • 当上面的代码被编译和执行时,它会产生下列结果:
In SomeNameSpace
In Nested

2.3 命名空间System.IO.Compression里的方法ZipFile.ExtractToDirectory

public static void ExtractToDirectory(
	string sourceArchiveFileName,
	string destinationDirectoryName,
	Encoding entryNameEncoding

2.4 Convert类常用的类型转换方法Convert.ToInt32()

2.5 System.Text命名空间包含与字符串处理和编码相关的类型

2.6 System.Collections.Generic命名空间包含用于处理集合的泛型类型

2.7 System.Linq命名空间,ToList()扩展方法,ToDictionary(),ToArray()

2.8 System.IO里Path类

  • 对包含文件或目录路径信息的 String 实例执行操作。 这些操作是以跨平台的方式执行的。
  • 包含的方法:
  1. Combine(String,String,String):将几个字符组合成一个路径

0001.png

  1. HasExtension()方法:确定路径是否包括文件扩展名。
  2. IsPathRooted()方法:获取一个值,该值指示指定的路径字符串是否包含根。
  3. GetFullPat()方法:返回指定路径字符串的绝对路径。
  4. GetTempPath()方法:返回当前用户的临时文件夹的路径。
  5. GetTempFileName():在磁盘上创建磁唯一命名的零字节的临时文件并返回该文件的完整路径
  6. GetFileNameWithoutExtension(): 返回不具有扩展名的指定路径字符串的文件名。

2.9 AppContext 类

  • 属性:
  1. BaseDirectory:获取程序集解析程序用于探测程序集的基目录的路径名。
  2. TargetFrameworkName:获取当前应用程序所针对的框架版本的名称。
  • 参数:
  1. sourceArchiveFileName
    Type: System.String
    要解压缩存档的路径。
  2. destinationDirectoryName Type: System.String 放置解压缩文件的目录的路径,指定为相对或绝对路径。 相对路径是指相对于当前工作目录的路径。
  3. entryNameEncoding Type: System.Text.Encoding
    在存档中读取或写入项名时使用的编码。 仅当需要针对具有不支持项名的 UTF-8 编码的 zip 归档工具和库的互操作性进行编码时,为此参数指定一个值。

2.10 Directory类

  • 是用于文件夹操作

包含的方法:

  1. CreateDirectory():创建一个文件夹
  2. Delete():删除一个文件夹
  3. Move():移动文件夹的位置
  4. Exists():判断文件夹是否存在,返回布尔类型
  5. GetFiles():获取目录下的所有文件的路径,返回到字符串数组
  6. GetDirectories():获取目录下所有文件夹的路径,返回字符串数组

三. class 关键字

  • class 关键字用于声明一个类。

3.1 成员变量

  • 变量是类的属性或数据成员,用于存储数据

3.2 成员函数

  • 函数是一系列执行指定任务的语句。类的成员函数是在类内声明的。

3.3 实例化一个类

 abc fc = new abc();
efg sc = new efg();

Console

=== 1

Console.Write("Hello World!");

Console.WriteLine("Hello World!");
  • 区别:前者输出后不换行,后者输出后自动换行

=== 2

Console.ReadKey();
  • 这个函数是为了在控制台窗口停留一下,直到敲击键盘为止。
    不然运行时,"Hello World!" 这句话会在控制台窗口一闪而过,没法查看。

四. c#方法

  • 一个方法是把一些相关的语句组织在一起,用来执行一个任务的语句块。每一个 C# 程序至少有一个带有 Main 方法的类。要使用一个方法,包含定义方法调用方法

4.1 定义方法

  • 当定义一个方法时,从根本上说是在声明它的结构的元素。在 C# 中,定义方法的语法如下:
<Access Specifier> <Return Type> <Method Name>(Parameter List)
{
   Method Body
}

下面是方法的各个元素:

  • Access Specifier:访问修饰符,这个决定了变量或方法对于另一个类的可见性。
  • Return type:返回类型,一个方法可以返回一个值。返回类型是方法返回的值的数据类型。如果方法不返回任何值,则返回类型为 void。
  • Method name:方法名称,是一个唯一的标识符,且是大小写敏感的。它不能与类中声明的其他标识符相同。
  • Parameter list:参数列表,使用圆括号括起来,该参数是用来传递和接收方法的数据。参数列表是指方法的参数类型、顺序和数量。参数是可选的,也就是说,一个方法可能不包含参数。

4.2 调用方法

  • 您可以使用方法名调用方法。
  • 您也可以使用类的实例从另一个类中调用其他类的公有方法。

带public关键字的方法可以使用类的实例从类的外部进行访问

如:方法 FindMax 属于 NumberManipulator 类,您可以从另一个类 Test 中调用它。

using System;

namespace CalculatorApplication
{
    class NumberManipulator
    {
        public int FindMax(int num1, int num2)
        {
            /* 局部变量声明 */
            int result;

            if (num1 > num2)
                result = num1;
            else
                result = num2;

            return result;
        }
    }
    class Test
    {
        static void Main(string[] args)
        {
            /* 局部变量定义 */
            int a = 100;
            int b = 200;
            int ret;
            NumberManipulator n = new NumberManipulator();
            //调用 FindMax 方法
            ret = n.FindMax(a, b);
            Console.WriteLine("最大值是: {0}", ret );
            Console.ReadLine();

        }
    }
}
  • 递归方法调用:一个方法可以自我调用。这就是所谓的 递归。下面的实例使用递归函数计算一个数的阶乘:

4.3 参数传递

  • 当调用带有参数的方法时,您需要向方法传递参数。在 C# 中,有三种向方法传递参数的方式:
  1. 值参数: 这种方式复制参数的实际值给函数的形式参数,实参和形参使用的是两个不同内存中的值。在这种情况下,当形参的值发生改变时,不会影响实参的值,从而保证了实参数据的安全。
  • 这是参数传递的默认方式。在这种方式下,当调用一个方法时,会为每个值参数创建一个新的存储位置。
    实际参数的值会复制给形参,实参和形参使用的是两个不同内存中的值。所以,当形参的值发生改变时,不会影响实参的值,从而保证了实参数据的安全
  1. 引用参数: 这种方式复制参数的内存位置的引用给形式参数。这意味着,当形参的值发生改变时,同时也改变实参的值。
  • 引用参数是一个对变量的内存位置的引用。当按引用传递参数时,与值参数不同的是,它不会为这些参数创建一个新的存储位置。引用参数表示与提供给方法的实际参数具有相同的内存位置。ref 关键字声明引用参数
using System;
namespace CalculatorApplication
{
   class NumberManipulator
   {
      public void swap(ref int x, ref int y)
      {
         int temp;

         temp = x; /* 保存 x 的值 */
         x = y;    /* 把 y 赋值给 x */
         y = temp; /* 把 temp 赋值给 y */
       }
   
      static void Main(string[] args)
      {
         NumberManipulator n = new NumberManipulator();
         /* 局部变量定义 */
         int a = 100;
         int b = 200;

         Console.WriteLine("在交换之前,a 的值: {0}", a);
         Console.WriteLine("在交换之前,b 的值: {0}", b);

         /* 调用函数来交换值 */
         n.swap(ref a, ref b);

         Console.WriteLine("在交换之后,a 的值: {0}", a);
         Console.WriteLine("在交换之后,b 的值: {0}", b);
 
         Console.ReadLine();

      }
   }
}
  1. 输出参数: 这种方式可以返回多个值。
  • return 语句可用于只从函数中返回一个值。但是,可以使用 输出参数 来从函数中返回两个值。输出参数会把方法输出的数据赋给自己,其他方面与引用参数相似。
using System;

namespace CalculatorApplication
{
   class NumberManipulator
   {
      public void getValue(out int x )
      {
         int temp = 5;
         x = temp;
      }
   
      static void Main(string[] args)
      {
         NumberManipulator n = new NumberManipulator();
         /* 局部变量定义 */
         int a = 100;
         
         Console.WriteLine("在方法调用之前,a 的值: {0}", a);
         
         /* 调用函数来获取值 */
         n.getValue(out a);

         Console.WriteLine("在方法调用之后,a 的值: {0}", a);
         Console.ReadLine();

      }
   }
}

五. 封装

  • 封装:被定义为"把一个或多个项目封闭在一个物理的或者逻辑的包中"。在面向对象程序设计方法论中,封装是为了防止对实现细节的访问。

  • C# 封装根据具体的需要,设置使用者的访问权限,并通过 访问修饰符 来实现。

  • 一个 访问修饰符 定义了一个类成员的范围和可见性。C# 支持的访问修饰符如下所示:

  1. public:所有对象都可以访问;
  • Public 访问修饰符允许一个类将其成员变量和成员函数暴露给其他的函数和对象。任何公有成员可以被外部的类访问。
  1. private:对象本身在对象内部可以访问;
  • Private 访问修饰符允许一个类将其成员变量和成员函数对其他的函数和对象进行隐藏。只有同一个类中的函数可以访问它的私有成员。即使是类的实例也不能访问它的私有成员。
  1. protected:只有该类对象及其子类对象可以访问
  • Protected 访问修饰符允许子类访问它的基类的成员变量和成员函数。这样有助于实现继承。我们将在继承的章节详细讨论这个。更详细地讨论这个。
  1. internal:同一个程序集的对象可以访问;
  2. protected internal:访问限于当前程序集或派生自包含类的类型。

六. 可空类型(Nullable)

  • C# 提供了一个特殊的数据类型,nullable 类型(可空类型),可空类型可以表示其基础值类型正常范围内的值,再加上一个 null 值。

例如:

  1. Nullable< Int32 >,读作"可空的 Int32",可以被赋值为 -2,147,483,648 到 2,147,483,647 之间的任意值,也可以被赋值为 null 值。
  2. 类似的,Nullable< bool > 变量可以被赋值为 true 或 false 或 null。

C# 单问号 ? 与 双问号 ??

  • ? : 单问号用于对 int,double,bool 等无法直接赋值为 null 的数据类型进行 null 的赋值,意思是这个数据类型是 NullAble 类型的。
  • ?? : 双问号 可用于判断一个变量在为 null 时返回一个指定的值。
    接下来我们详细说明。

语法

  • 声明一个 nullable 类型(可空类型)的语法如下:
< data_type> ? <variable_name> = null;

Null 合并运算符( ?? )

  • Null 合并运算符用于定义可空类型和引用类型的默认值。Null 合并运算符为类型转换定义了一个预设值,以防可空类型的值为 Null。Null 合并运算符把操作数类型隐式转换为另一个可空(或不可空)的值类型的操作数的类型。
    如果第一个操作数的值为 null,则运算符返回第二个操作数的值,否则返回第一个操作数的值。
using System;
namespace CalculatorApplication
{
   class NullablesAtShow
   {
         
      static void Main(string[] args)
      {
         
         double? num1 = null;
         double? num2 = 3.14157;
         double num3;
         num3 = num1 ?? 5.34;      // num1 如果为空值则返回 5.34
         Console.WriteLine("num3 的值: {0}", num3);
         num3 = num2 ?? 5.34;
         Console.WriteLine("num3 的值: {0}", num3);
         Console.ReadLine();

      }
   }
}

输出结果为:

num3 的值: 5.34
num3 的值: 3.14157

七. 数组(Array)

  • 数组是一个存储相同类型元素的固定大小的顺序集合。数组是用来存储数据的集合,通常认为数组是一个同一类型变量的集合。

7.1 数组声明

datatype[] arrayName;

其中,

  • datatype 用于指定被存储在数组中的元素的类型。
  • [ ] 指定数组的秩(维度)。秩指定数组的大小
  • arrayName 指定数组的名称。

如:

double[] balance;

7.2 初始化数组

  • 一个数组不会在内存中初始化数组。当初始化数组变量时,您可以赋值给数组。
  • 数组是一个引用类型,所以您需要使用 new 关键字来创建数组的实例。

例如:

double[] balance = new double[10];

7.3 赋值给数组

  1. 可以通过使用索引号赋值给一个单独的数组元素
double[] balance = new double[10];
balance[0] = 4500.0;
  1. 可以在声明数组的同时给数组赋值
double[] balance = { 2340.0, 4523.69, 3421.0};
  1. 您也可以创建并初始化一个数组
int [] marks = new int[5]  { 99,  98, 92, 97, 95};
  • 当您创建一个数组时,C# 编译器会根据数组类型隐式初始化每个数组元素为一个默认值。例如,int 数组的所有元素都会被初始化为 0。

7.4 访问数组元素

  • 元素是通过带索引的数组名称来访问的。这是通过把元素的索引放置在数组名称后的方括号中来实现的。

使用 foreach 循环

八. 类(Class)

8.1 类的定义

  • 类的定义是以关键字 class 开始,后跟类的名称。类的主体,包含在一对花括号内。

8.2 成员函数和封装

  • 类的成员函数是一个在类定义中有它的定义或原型的函数,就像其他变量一样。作为类的一个成员,它能在类的任何对象上操作,且能访问该对象的类的所有成员。
    成员变量是对象的属性(从设计角度),且它们保持私有来实现封装。这些变量只能使用公共成员函数来访问。

8.3 构造函数

  • 类的构造函数是类的一个特殊的成员函数,当创建类的新对象时执行。
    构造函数的名称与类的名称完全相同,它没有任何返回类型。
using System;
namespace LineApplication
{
   class Line
   {
      private double length;   // 线条的长度
      public Line()
      {
         Console.WriteLine("对象已创建");
      }

      public void setLength( double len )
      {
         length = len;
      }
      public double getLength()
      {
         return length;
      }

      static void Main(string[] args)
      {
         Line line = new Line();    
         // 设置线条长度
         line.setLength(6.0);
         Console.WriteLine("线条的长度: {0}", line.getLength());
         Console.ReadKey();
      }
   }
}

结果如下:

对象已创建
线条的长度: 6

8.4 参数化构造函数

  • 默认的构造函数没有任何参数。但是如果你需要一个带有参数的构造函数可以有参数,这种构造函数叫做参数化构造函数。这种技术可以帮助你在创建对象的同时给对象

8.5 析构函数

  • 类的析构函数是类的一个特殊的成员函数,当类的对象超出范围时执行。

  • 析构函数的名称是在类的名称前加上一个波浪****形(~)作为前缀,它不返回值,也不带任何参数。

  • 析构函数用于在结束程序(比如关闭文件、释放内存等)之前释放资源。析构函数不能继承或重载。

8.6 静态成员

  • 我们可以使用 static 关键字把类成员定义为静态的。当我们声明一个类成员为静态时,意味着无论有多少个类的对象被创建,只会有一个该静态成员的副本。

  • 关键字 static 意味着类中只有一个该成员的实例

  • 静态变量用于定义常量,因为它们的值可以通过直接调用类而不需要创建类的实例来获取

  • 静态变量可在成员函数或类的定义外部进行初始化。你也可以在类的定义内部初始化静态变量。

  • 你也可以把一个成员函数声明为 static。
    +** 这样的函数只能访问静态变量**。静态函数在对象被创建之前就已经存在.

九. 泛型

  • 泛型(Generic) 允许您延迟编写类或方法中的编程元素的数据类型的规范,直到实际在程序中使用它的时候。换句话说,泛型允许您编写一个可以与任何数据类型一起工作的类或方法。

例 :

using System;
using System.Collections.Generic;

namespace GenericApplication
{
    public class MyGenericArray<T>
    {
        private T[] array;
        public MyGenericArray(int size)
        {
            array = new T[size + 1];
        }
        public T getItem(int index)
        {
            return array[index];
        }
        public void setItem(int index, T value)
        {
            array[index] = value;
        }
    }
           
    class Tester
    {
        static void Main(string[] args)
        {
            // 声明一个整型数组
            MyGenericArray<int> intArray = new MyGenericArray<int>(5);
            // 设置值
            for (int c = 0; c < 5; c++)
            {
                intArray.setItem(c, c*5);
            }
            // 获取值
            for (int c = 0; c < 5; c++)
            {
                Console.Write(intArray.getItem(c) + " ");
            }
            Console.WriteLine();
            // 声明一个字符数组
            MyGenericArray<char> charArray = new MyGenericArray<char>(5);
            // 设置值
            for (int c = 0; c < 5; c++)
            {
                charArray.setItem(c, (char)(c+97));
            }
            // 获取值
            for (int c = 0; c < 5; c++)
            {
                Console.Write(charArray.getItem(c) + " ");
            }
            Console.WriteLine();
            Console.ReadKey();
        }
    }
}

输出结果为:

0 5 10 15 20
a b c d e

泛型(Generic)的特性

===

使用泛型是一种增强程序功能的技术,具体表现在以下几个方面:

  • 它有助于您最大限度地重用代码、保护类型的安全以及提高性能。
  • 您可以创建泛型集合类。.NET 框架类库在 System.Collections.Generic 命名空间中包含了一些新的泛型集合类。您可以使用这些泛型集合类来替代 System.Collections 中的集合类。
  • 您可以创建自己的泛型接口、泛型类、泛型方法、泛型事件和泛型委托。
  • 您可以对泛型类进行约束以访问特定数据类型的方法。
  • 关于泛型数据类型中使用的类型的信息可在运行时通过使用反射获取。

泛型(Generic)方法

  • 通过类型参数声明泛型方法

十. List

  • List类是ArrayList类的泛型等效类。它的大部分用法都与ArrayList相似,因为List类也继承了IList接口。最关键的区别在于,在声明List集合时,我们同时需要为其声明List集合内数据的对象类型。

10.1 List的基础、常用方法:

  1. List mList = new List();

例:

List<string> mList = new List<string>();
  1. List testList =new List (IEnumerable collection);

以一个集合作为参数创建List:

string[] temArr = { "Ha", "Hunter", "Tom", "Lily", "Jay", "Jim", "Kuku", "Locu"};
List<string> testList = new List<string>(temArr);

  1. List的进阶、强大方法:
  • List. Add(T item)添加一个元素
  • List. AddRange(IEnumerable collection)添加一组元素
  • Insert(intindex, T item);在index位置添加一个元素a
原文地址:https://www.cnblogs.com/monkey-moon/p/9073965.html