1.1值类型和引用类型

1.1.1 值类型

使用值类型之前需要对值类型的所有元素初始化(普通值类型和结构体)。

结构还有一个特性:调用结构上的方法前,需要对其所有的字段进行赋值,为了避免对结构体中所有字段专门赋值,可以通过隐式声明的构造函数去创建一个结构类型变量(new)。(P5)

1.1.2 引用类型

而当使用new操作符时:rPoint1= new RefPoint(1);
则会完成下面几件事:
❑在应用程序堆(Heap)上创建一个引用类型(ReferenceType)对象的实例,并为它分配内存地址。
❑自动传递该实例的引用给构造函数。(正因为如此,才可以在构造函数中使用this来访问这个实例。)
❑调用该类型的构造函数。
❑返回该实例的引用(内存地址),赋值给rPoint1变量。

1.1.3 简单类型

对于自定义的值类型,比如结构,就不能用“==”来判断它们是否相等,而需要在变量上调用Equals()方法来完成。

1.1.4 装箱和拆箱

简单来说,装箱就是将一个值类型转换成等价的引用类型。它的过程分为这样几步:

1)在堆上为新生成的对象实例分配内存。该对象实例包含数据,但它没有名称。

2)将栈上值类型变量的值复制到堆上的对象中。

3)将堆上创建的对象的地址返回给引用类型变量。

需要注意的是:拆箱操作需要显示声明拆箱后转换的类型。它分为两步来完成:

1)获取已装箱的对象的地址。

2)将值从堆上的对象中复制到堆栈上的值变量中。

1.2 对象判等

1.2.1引用类型判等

实例方法Equals(Object obj),静态方法Equals(Object objA,Object objB),静态方法ReferenceEquals(Object objA,Object objB)

ReferenceEquals()这个方法名就可以看出,它判断两个引用变量是不是指向了同一个变量,如果是,那么就返回true。这种相等叫做引用相等(rPoint1==rPoint2等效于ReferenceEquals)。因为它们指向的是同一个对象,所以对rPoint1的操作将会影响rPoint2。(P8)

对于引用类型,即使类型的实例(对象)包含的值相等,如果变量指向的是不同的对象,那么也不相等。(P8)

1.2.2简单值类型判等

值类型都会隐式地继承自System.ValueType类型,而ValueType类型覆盖了基类System.Object类型的Equals()方法,在值类型上调用Equals()方法,会调用ValueType的Equals()。(P9)

结构体变量不能直接用“==”去判断,编译会报错(P10)

1.2.3 复杂值类型判断

对堆上对象的成员(字段)进行一对一的比较,而成员又分为两种类型,一种是值类型,一种是引用类型。
对于引用类型,去判断是否引用相等;对于值类型,如果是简单值类型,那么同前一节讲述的一样去判断;如果是复杂类型,那么当然是递归调用了;最终确定要么是引用类型要么是简单值类型。(P11)

1.3 对象复制
1.3.1 浅度复制
当对对象进行一个浅度复制的时候,对于值类型成员,会复制其本身(值类型变量本身包含了所有数据,复制时进行按位复制);
对于引用类型成员(注意它实际只是一个对象引用,指向了堆上的对象实例),仅仅复制引用,而不在堆上重新创建对象。因此,浅度复制结果就是:新对象的引用成员和复制对象的引用成员指向了同一个对象。(P12)

当复制一个结构类型成员的时候,直接创建一个新的结构类型变量,然后对它赋值,就相当于进行了一个浅度复制,也可以认为结构类型隐式地实现了浅度复制。(P12)

对于引用类型,采用Clone()实现浅度复制,复制后的对象和原先对象成了“连体婴”,它们的引用成员字段依然引用堆上的同一个对象。P(13)

1.3.2 深度复制
可以利用序列化/反序列化来对对象进行深度复制:先把对象序列化(Serialize)到内存中,然后再进行反序列化,通过这种方式来进行对象的深度复制。如果想将对象进行序列化,那么对象本身,及其所有的自定义成员(类、结构),都必须使用Serializable特性进行标记。P(13)

1.4 不可变类型
1.4.1 从类型设计谈起,Class还是Struct
在数据较小的情况下,传值的效率更高一些;而在数据较大的时候,传引用占据更小的内存空间。(P15)
1.4.2 数据不一致的问题
1.4.3 常量性和原子性
❑对象的原子性:对象的状态是一个整体,如果一个字段改变,其他字段也要同时做出相应改变。
❑对象的常量性:对象的状态一旦确定,就不能再次更改了。如果想再次更改,需要重新构造一个对象。

对于原子性,实施的办法是添加一个构造函数,在这个构造函数中为对象的所有字段赋值。而为了实施常量性,不允许在为对象赋值以后还能对对象状态进行修改,所以将属性中的set访问器删除,同时将字段声明为readonly。(P16)

1.4.4 避免外部类型对类型内部的访问

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Runtime.Serialization.Formatters.Binary;
using System.IO;
using System.Text.RegularExpressions; namespace _.net之美
{
class Program
{
static void Main(string[] args)
{
string[] phones = { "", "" };
Address a = new Address("四川", "成都", "", phones);
try
{
string[] b = a.Phones;
b[] = "";
//a = new Address("海南", "海口", "62055");
}
catch
{ }
phones[] = "";
Console.WriteLine(a.ToString());
Console.Read();
}
public struct Address
{
private readonly string province;
private readonly string city;
private readonly string zip;
private readonly string[] phones;
public Address(string province, string city, string zip,string[] phones)
{
this.province = province;
this.city = city;
this.zip = zip;
this.phones = new string[phones.Length];
phones.CopyTo(this.phones, );
CheckZip(zip); // 验证格式
} public string Province
{
get { return province; }
} public string City
{
get { return city; }
} public string Zip
{
get { return zip; }
} public string[] Phones
{
get
{
string[] rtn = new string[phones.Length];
phones.CopyTo(rtn, );
return rtn;
}
} private void CheckZip(string value)
{
string pattern = @"\d{6}";
if (!Regex.IsMatch(value, pattern))
throw new Exception("inValid!");
} public override string ToString()
{
return string.Format("P:{0},C:{1},Z:{2},P:{3}", province, city, zip, phones[]);
}
}
} }

第1章 C#类型基础的更多相关文章

  1. 《C#从现象到本质》读书笔记(三)第3章C#类型基础(下)

    <C#从现象到本质>读书笔记第3章C#类型基础(下) 常量以关键字const修饰.C#支持静态字段(类型字段)和实例字段. 无参属性的get方法不支持参数,而有参属性的get方法支持传入一 ...

  2. 《C#从现象到本质》读书笔记(二)第2章 C#类型基础(上)

    <C#从现象到本质>读书笔记第二篇 第2章 C#类型基础(上) 类型指的是集合{类,结构,接口,枚举,委托}中的任意一个成员.任何拥有某类型的值(value)称为某类型的一个实例(inst ...

  3. 《Programming WPF》翻译 第9章 1.自定义控件基础

    原文:<Programming WPF>翻译 第9章 1.自定义控件基础 在写一个自定义控件之前,你需要问的第一个问题是,我真的需要一个自定义控件吗?一个写自定义控件的主要原因是为了用户界 ...

  4. 《Programming WPF》翻译 第8章 1.动画基础

    原文:<Programming WPF>翻译 第8章 1.动画基础 动画包括在一段时间内改变用户界面的某些可见的特征,如它的大小.位置或颜色.你可以做到这一点,非常困难的通过创建一个tim ...

  5. 《Programming WPF》翻译 第7章 1.图形基础

    原文:<Programming WPF>翻译 第7章 1.图形基础 WPF使得在你的应用程序中使用图形很容易,以及更容易开发你的显卡的能力.这有很多图形构架的方面来达到这个目标.其中最重要 ...

  6. 第一章 oracle数据库基础

    第一章   oracle数据库基础 1.oracle简介-->数据库管理系统    1.1:数据库    1.2:全局数据库名    1.3:数据库实例    1.4:表空间    1.5:数据 ...

  7. 【RL-TCPnet网络教程】第27章 DNS域名系统基础知识

    第27章      DNS域名系统基础知识 本章节为大家讲解DNS(Domain Name System,域名系统),通过前面章节对TCP和UDP的学习,需要大家对DNS也有个基础的认识. (本章的知 ...

  8. 【RL-TCPnet网络教程】第18章 BSD Sockets基础知识

    第18章      BSD Sockets基础知识 本章节为大家讲解BSD Sockets,需要大家对BSD Sockets有个基础的认识,方便后面章节Socket实战操作. (本章的知识点主要整理自 ...

  9. 零基础学Python--------第2章 Python语言基础

    第2章  Python语言基础 2.1 Python语法特点 2.11注释 在Python中,通常包括3种类型的注释,分别是单行注释.多行注释和中文编码声明注释. 1.单行注释 在Python中,使用 ...

随机推荐

  1. 双节点(nginx+keepalived)为两台apache服务器提供负载均衡

    说明:本实验为双节点nginx为两台apache服务器提供负载均衡,本文不是做lvs,所以realserver不是配置在keepalived.conf而是在nginx的配置文件中upstream.此架 ...

  2. SQL Server 2012 创建数据库快照

    不是所有的MSSQL数据库版本都支持数据库快照,只有Enterprise版本的才支持. 在其他版本上,以Business Intelligence Edition版本为例,创建快照时,会报如下错误 消 ...

  3. windows下用一台机器配置分布式redis(主从服务器)

    目录1.Replication的工作原理2.如何配置Redis主从复制 1.Replication的工作原理在Slave启动并连接到Master之后,它将主动发送一条SYNC命令.此后Master将启 ...

  4. 黄聪:基于jQuery+JSON的省市区三级地区联动

    查看演示:http://www.helloweba.com/demo/cityselect/ 源码下载:http://files.cnblogs.com/files/huangcong/citysel ...

  5. MapReduce单表关联学习~

    首先考虑表的自连接,其次是列的设置,最后是结果的整理. 文件内容: import org.apache.hadoop.conf.Configuration; import org.apache.had ...

  6. ADF_Database Develop系列3_设计数据库表之Reconcile Database/Reverse Objects

    2013-05-01 Created By BaoXinjian

  7. 中大bbs

    发现个好像很好玩的东东, 中大bbs: luit -encoding gbk telnet bbs.sysu.edu.cn

  8. nginx/Windows-1.9.3启动脚本

    启动nginx.bat @echo off D: cd D:\Program Files\nginx-1.9.3 tasklist | findstr /i "nginx.exe" ...

  9. win7访问ubuntu所在分区

    用ext2explore就可以了

  10. 三层VS控制器

    三层中的Model 是数据模型 与数据库字段一一对应而mvc中的model是 显示模型,指在页面的显示字段的列一一对应,不一定在数据库中有这样的一个表,如当页 面上要显示多个表的综合信息时. 当在Co ...