【游戏开发笔记】编程篇_C#面向对象{上}
@
1.变量和表达式
1.1注释
- 单行注释://
- 多行注释:/**/
- 文档注释:///
VS编译后,会产生一个文本文件,该文件可创建文档
1.2C#控制台程序基本结构
using System;
namespace ConsoleApplication1
{
class Program
{
static void Main(string[] args)
{
Console.WriteLine("The first App in Begining C# Program!");
Console.Readkey();
}
}
}
折叠预处理
#region
……
……
#endregion
(折叠中间部分)
1.3变量(从存储长度来看)
-
转义字符:为了将字面值赋给变量不出错
eg:
string myString = "\"string\" is";
-
整数
sbyte [-128, 127] 8位
byte [0, 255] 8位
short [-32 768, 32767] 16位
ushort [0, 65535] 16位
int [-2 147 483 648, 2 147 483 647] 32位
uint [0, 4 294 967 296] 32位
long [9 223 372 036 854 775 808, 9 223 372 036 854 775 807] 64位
ulong [0, 18 446 744 073 709 551 615] 64位 -
浮点
前两种存储类型:+/- m x 2^e
后一种存储方式:+/- m x 10^e
类型 m-min m-max e-min e-max
float 0 2^24 -149 104
double 0 2^53 -1075 970
decimal 0 2^96 -28 0 -
文本和布尔
char [0, 65 535]
bool ture/false
string 没有上限
1.4变量的命名
1、变量名的第一个字母必须是字母、下划线或@
2、其后可以是字母、下划线或数字
3(注意)变量名不能是关键字
1.5字面值
类型 后缀
int, uint, long, ulong 无
uint、ulong u或U
long、ulong l或L
ulong (u、l 位置关系和大小写不影响)Ul,Lu……均可
float f或F
double 无、d或D
decimal m或M
@
置于字符串最前端,无需转义,即可正确存储和表达字面值
1.6运算符
-
比较运算符
==、!=、<、>、<=、>= -
条件运算符
&&、| | -
赋值运算符
=、+=、-=、*=、/=、%=、&=、|=、^= -
其他
一元:++、--,+(正)、-(负)
三元:逻辑表达式?a:b
逻辑表达式成立,三元表达式值为a,否则为b
二元:+、-、*、/…… -
优先级
(优先级记忆不用担心,因为写代码时一般会使用括号指定优先级,增加可读性)
1、前缀:++,--,一 元:+,-
2、,/,%
3、+,-
4、>>、<<
5、>、<、>=、<=
6、==, !=
7、&
8、^
9、|
10、&&
11、| |
12、=,+=,-=,=,/=、<<=、>>=、&=、|=、^=
后缀:++,--
2流程控制
该部分同c
2.1分支
三元运算符
<test>? <resultIfTrue> : <resultIfFalse>
if语句
if (<test>)
<code executed if <test> is true>;
或
if (<test>)
<code executed if <test> is true>;
else
<code executed if <test> is false>;
[效果同三元运算符]
或
if (varl == 1)
{
// Do something.
}
else if (varl == 2)
{
// Do something else.
}
else
{
// Do something else.
}
switch语句
switch (<testVar>)
{
case <comparisonVal1>:
<code to execute if <testVar> == <comparisonVal1> >
break;
case <comparisonVal2>:
<code to execute if <testVar> == <comparisonVal2> >
break;
……
case <comparisonValN>:
<code to execute if <testVar> == <comparisonValN> >
break;
default:
<code to execute if <testVar> != <comparisonVals> >
break;
}
2.2循环
do循环
do
{
<code to be looped>
}while (<Text>)
while循环
while (<Text>)
{
<code to be looped>
}
for循环
for (<initialization>; <condition>; <operation>)
{
<code to loop>
}
无限循环
while (ture)
{}
for (; ;)
{}
……
3变量知识拓展
3.1类型转换
- 隐式转换
- 显式转换
(<destinationType>) <sourceVar>
3.2枚举
- 定义
enum <typeName>:<underlyingType>
{
<value1> = <actualVal1>
<value2> = <actualVar2>
……
<valueN> = <actualVarN>
}
(数据默认为int类型,每个值根据顺序定义【从0开始】)
声明新类型的变量
<typeName> <varName>
- 赋值:
<varName> = <typeName>.<value>;
3.3结构(体)
- 定义
struct <typeName>
{
<memberDeclarations>
}
- 结构的数据成员
<memberDeclarations>
- 部分包含变量的声明:
<accessibility> <type> <name>
定义结构类型的变量类似枚举
可以通过句点字符访问这个组合变量中的数据成员
3.4数组
数组的索引是从0算起,如果要查找数组中第三个元素,语法为:myArray[2]
- 定义:
<baseType> [ ] <name>
(必须在访问之前初始化)
举例
int [] myIntArray = {1, 2, 3};
或,int [] myIntArray = new int[arraySize];
arraySize可以是常数,也可以是变量
或,int [] myIntArray = new int[10] {2, 5, 5};
foreach循环
遍历数组中每一个元素
foreach (<baseType> <name> in <array>)
{
// can use <name> for each element
}
多维数组
[, ]
与一位数组类似,只是使用的时候需要更多的逗号和更多的大括号,数据在内存中顺序存储
eg:double [, ] hillHight = new double[3, 4] {{2, 3 , 4, 5}, {2, 33, 55.7, 5}, {2, 5}};
数组的数组
声明:int[ ] [ ] jaggedIntArray;
- 初始化:
jaggedIntArray = new int [2] [ ];
jaggedIntArray [0] = new int [3];
jaggedIntArray [0] = new int [2];
//或是下面这样
jaggedIntArray = new int [3] [ ]{
new int [2] {2, 4}, new int [3] {2, 3, 4},
new int [1] {5} };
- 又或声明与初始化放一起
int [ ] [ ] jaggedIntArray = new int [3] [] {
new int [2] {2, 4}, new int [3] {2, 3, 4},
new int [1] {5} };
- foreach实现遍历
foreach (int [ ] sonArray in fatherArray)
{
foreach (int sprit in sonArray)
{
// work;
}
}
3.5字符串的处理
- 思路
- 转为字符数组,(一般)使用封装好的方法操作即可
1、字符串转字符数组:<stringName>. ToCharArray( )
【其后方法使用方式一致】,有返回值
2、获取字符串中字符数:Length( )
,有返回值
3、转为大,小写:ToUpper( )
,ToLower( )
,有返回值
4、删除指定字符:Trim(需要的字符),有返回值,默认删除字符串前后空格
5、删除字符串前后空格:TrimStart( )
,TrimEnd( )
, 有返回值
6、在字符串左边或右边添加指定字符至指定长度:PadLeft(<length>,<char> )
,PadRight(<length> , <char>)
, 有返回值,默认添加空格
4函数
static <returnType> <FunctionName>( <type> <typeName>)
{
// do something.
return <typeVar>
}
如果是void类型,则之间return; 或省略不写
4.1返回值
如果方法有返回值,在方法体内,所有情况下,都必须有返回内容,否则会报错 “并不是所有的处理路径都用返回值”
针对一行代码的方法,可以使用 表达式体方法(expression-bodied method)处理
如:
static double Multiply (double myVal1, double myVal2)
{
return myVal1 * myVal2;
}
可改写为:
static double Multiply (double myVal1, double myVal2) => myVal1 * myVal2;
4.2参数
-
参数匹配
参数类型、个数。顺序 在定义与使用时保持一致 -
参数数组
只能指定一个特殊参数可以传递同类型的多个参数:
static <returnType> <FunctionName> (params <type>[ ] <name>)
{
……
return <returnValue>
}
-
引用参数和值参数
把变量作为一个参数,调用方法时并不影响主函数中变量的值。(实质是重新开辟了一块空间,传参时将变量给新空间复制了一份,在方法执行完毕后,复制的变量就会和方法一起被清理掉,而原来的参数并没有参与方法中的任何操作)
改变原变量的方法:
1、使用返回值
缺点:不直观、只有一个返回值
2、使用ref 限定
static void ShowDouble ( ref int val)
{
val *= 2;
WriteLine($ "val double = {val}");
}
调用(必须加 ref):
int myNumber = 5;
ShowDouble( ref myNumber);
ref的限制:
a、值参数必须是“非常量”变量
b、必须使用初始化过的变量
3、使用全局变量——可读性差
-
输出参数
out
使用方法及作用同ref区别:
1、未赋值的变量可以做out的参数,但不能做ref的参数
2、函数使用out参数时,必须将其看作未赋值的
5.调试和错误处理
5.1 VS中调试
- 非中断模式下调试
1.输出调试信息
Debug.WriteLine( ) //仅在调试模式下运行
Trace.WriteLine( ) //还可用于发布程序
2.跟踪点
-
中断模式下调试
1.断点进入
Debug.Assert( )
Trace.Assert( )第一个参数:布尔值,为false时触发语句
第二个参数:字符串,将信息弹到对话框
第三个参数:字符串,将信息弹到Output窗口
Debug.Assert ( myVar < 10, " my Var is 10 or qreater", "Assertion occurred in Main( ).")
2.语句进入
3.监视变量内容
4.单步执行
5.Immediate 和 Command 窗口
调整变量值,测试表达式
6.Call Stack 窗口
描述程序如何执行到当前位置
5.2 错误处理
- try……catch……finally
try
{
……
}
catch ( <exceptionType> e) when (filterIsTure)
{
<await methodName( e); >
……
}
finally
{
<await method name>
……
}
正常执行顺序:就像字面意思,先尝试执行 “try” 中的代码,如果有错误,跳到 “catch” 抓住异常,执行相应操作,最后“finally”。
catch的补充1:也可以将错误 “catch”后抛出:
catch(Exception)
{
throw;
}
代码可以嵌套,因此如果外层代码仍没有处理抛出的异常,编译器就会迫使程序暂停(不能正常编译)
catch的补充2:如上述代码,catch后的括号里需要跟异常的类名,如果不知道异常的类名,可以选择跟异常之父——“Exception”。只有catch了正确的异常类——即程序异常的类名与catch后括号的类名一致时,才会执行catch中的代码,同理,可以有多个catch并列,但catch(Exception)只能位于最后。
eg:
try
{
int[4] myArray = {1, 2, 3, 4};
int myElem = myArray[ 4];
}
catch (Exception ex)
{
Debug.WriteLine( ex.Message);
}
- 列出和配置异常
6.面向对象编程
6.1 面向对象
-
类
类:类是功能的封装,例:衣服
对象:对象是类中的一个实例,例:第三行第二列的那件衣服C#中可以创建类,类中有成员。
static问题:
静态的即全局的,静态的类就可以通过点句运算符直接访问,参考Console()进行理解。静态构造函数调用时机:
创建含静态函数的类实例
访问含静态构造函数的类的静态成员
1.构造函数
构造函数——数据成员初始化
本质:方法
特点:没有返回值,与类同名例——用new调用默认构造函数,实例化一个对象:
CupOfCoffee myCup = new CupOfCoffee();
与可以是非默认构造函数:
CupOfCoffee myCup = new CupOfCoffee(" Blue Mountain");
构造函可public也可private,因此可以private默认构造函数,强制类的用户使用非默认构造函数
2.构析函数
归还内存
~<ClassName>()
{} //当析构语块结束时,析构函数执行
也可以使用CG
CG.Collect( );
//这个看CG的心情,CG觉得时机不成熟,即使有该语句也不会进行清理
3.方法
解决问题
4.字段
存储数据——变量
5.属性
保护字段读写
- UML
UML即Unified Modeling Language
是一种建模语言
类的表示:
第一行:类名
第二行:成员- 成员名: 成员类型
第三行:方法名( 标识符 参数名: 参数类型):方法类型
标识符:out 和 inout 分别对应 out 和 ref,in 不使用以上两种关键字(默认情况),return 表返回值
(注:二、三行“+、-”号表示public和private,下划线表private)
类似于伪代码,不受编程语言是语法限制,来表述问题解决思路。
6.2 OOP技术
-
接口
把公共实例(非静态)方法和属性组合起来,以封装特定功能的一个集合可理解为将多个有相似特性的类组合在一起的大类。
(区别是:类只能继承一个,接口可以继承多个)可删除对象:使用using关键字后,代码块执行完毕,就会调用
Dispose()
方法,删除代码块
using (<ClassName > <VariableName> = new <ClassName>)
{
……
}
-
继承
任何类都可以从另一个类继承
基类——父类,派生类——子类
注:C#的对象仅能直接派生一个基类可访问性:派生类不能访问基类的私有成员,可访问共有成员,外部代码同。
第三种可访问性:protected。家族类可访问,外部代码不能。继承行为:基类成员可以是虚拟的,成员可以由继承它的类重写。派生类可以提供成员的另一种实现方法,这种实现代码不会删除原来的代码,但外部代码不能访问它们。如果没有提供其他实现方式,通过派生类使用成员的外部代码就自动访问基类中成员的实现代码。
(虚拟成员不能是私有变量)抽象类:不能实例化,使用时必须继承它,并在派生类中重写。可封装功能
类可密封,密封的类不能做基类。
C#中,所有对象有一个共同的基类Object
-
多态
条件:有家族关系、有相同的类
使用方法:调用类的方法需要一个过渡变量
//Cow是Animal的派生类,Cow继承Animal的EatFood方法
//且有一个基类没有的Moo()方法
Cow myCow = new Cow();
Animal myAnimal = myCow;
myAnimal.EatFood;
Cow myNewCow = (cow) myAniaml;
myNewCow.Moo();
使用环境:作用在同一个类的不同对象上执行任务
接口的多态性:条件——有方法、有共同接口
优势:不用依赖于一个公共的基类
唯一的区别:必须提供方法的实现代码
- 对象之间的关系
1.包含关系
类似于继承
*2.集合关系
存储多个同类型的变量,类似于数组(变量即对象,万物皆可对象)
转载请注明出处