《CLR via C#》读书笔记(5)基元类型、引用类型和值类型
编译器直接支持的数据类型称为基元类型(primitive type)。
int a = 0; //最方便的语法
System.Int32 b = 0; //方便的语法
int c = new int(); //不方便的语法
System.Int32 d = new System.Int32(); //最不方便的语法
C#中的基元类型 |
FCL类型 |
是否与CLS兼容 |
描述 |
sbyte |
System.SByte |
N |
有符号8位值 |
byte |
System.Byte |
Y |
无符号8位值 |
short |
System.Int16 |
Y |
有符号16位值 |
ushort |
System.UInt16 |
N |
无符号16位值 |
int |
System.Int32 |
Y |
有符号32位值 |
uint |
System.UInt32 |
N |
无符号32位值 |
long |
System.Int64 |
Y |
有符号64位值 |
ulong |
System.UInt64 |
N |
无符号64位值 |
char |
System.Char |
Y |
16位Unicode字符(不像非托管C++中那样,char表示的是一个8位值) |
float |
System.Single |
Y |
IEEE32位浮点数 |
double |
System.Double |
Y |
IEEE 64位浮点数 |
bool |
System.Boolean |
Y |
一个True或者Flase值 |
decimal |
System.Decimal |
Y |
128位高精度浮点值,通常用于不容许有摄入误差的金融计算场合。在这128位中,1位表示浮点值的符号,96位表示浮点值本身(一个整数值,小数点位置由下面8个位来确定),8位表示用96位值除以浮点值所得结果的10的幂次(0~28)。其余的位尚未使用 |
string |
System.String |
Y |
一个字符数组 |
object |
System.Object |
Y |
所有类型的基类型 |
dynamic |
System.Object |
Y |
对于CLR, dynamic 和 object 完全一致。然而,C#编译器允许使用一个简单的语法,让dynamic变量参与动态调度。 |
- 尽量使用有符号的数值类型(比如 Int32 和 Int64),而不要使用无符号的数值类型(比如 UInt32 和 UInt64),这允许编译器检测更多的上溢/下溢错误。
- 如果代码发生不希望的溢出(可能因为无效的输入数据发生的),就把这些代码放在 checked 块中。同时还应捕捉 OverflowException。
- 将允许发生溢出的代码显示的放在一个 unchecked 块中,比如一个计算校验和(checksum)的时候。
- 引用类型总是从托管堆上分配
- 对象中的其他字节总设为零
- 从托管堆上分配一个对象时,可能强制执行一次垃圾收集操作
- 值类型一般在线程栈上分配
- 值类型并不包含一个指向实例的指针,相反,它包含了实例本身的字段,
- 值类型的实例不受垃圾回收站的控制
- 所有结构都是 System.ValueType 直接派生类,所有枚举都是从 System.Enum 的抽象类派生,后者又是从 System.ValueType 派生
1装箱
struct Point
{
public Int32 x, y;
}
public sealed class Program1
{
public static void Main()
{
ArrayList a = new ArrayList();
Point p;
for (int i = 0; i < 10; i++)
{
p.x = p.y = i;
a.Add(p); //对值类型进行装箱,并将引用添加到 ArrayList 中
}
Console.Read();
}
}
1.在托管堆中分配内存。分配的内存量是值类型的各个字段需要的内存量加上托管堆的所有对象都有的两个额外成员(类型对象指针和同步索引块)需要的内存量。
2.值类型的字段赋值到新分配的堆内存。
3.返回对象的地址。值类型现在是一个引用类型。
2拆箱
Int32 x = 5;
Object o = x;
Int16 y = (Int16)o; //抛出一个“InvalidCastException”类型的未经处理的异常
下面代码演示了涉及装箱和拆箱的几种情形
class CLass2
{
static void Main()
{
Point p1 = new Point(10, 10);
Point p2 = new Point(20, 20);
//调用ToString(一个虚方法)时,不会对p1进行装箱
Console.WriteLine(p1.ToString());
//调用GetType(一个非虚方法)时,要对p进行装箱
Console.WriteLine(p1.GetType());
//第一次调用CompareTo时,不会对p1进行装箱
//由于调用的是CompareTo(Point),所以对p2不会装箱
Console.WriteLine(p1.CompareTo(p2));
//p1要进行装箱,将引用放到c中
IComparable c = p1;
Console.WriteLine(c.GetType());
//第二次调用CompareTo时,p1不需要装箱
//由于向CompareTo传递的不是一个Point变量,所以调用CompareTo(object),要求获取一个已装箱的 Point 的引用
//c不会装箱,因为它引用的本来就是一个已经装箱的 Point
Console.WriteLine(p1.CompareTo(c));
//第上次调用CompareTo时,C不需要装箱,因为它引用的本来就是一个已经装箱的Point
//p2会被装箱,因为调用的是CompareTo(object)
Console.WriteLine(c.CompareTo(p2));
//对 c 进行拆箱,字段复制到 p2 中
p2 = (Point)c;
//证明字段已经复制到p2中
Console.WriteLine(p2.ToString());
Console.Read();
}
}
internal struct Point : IComparable
{
private Int32 m_x, m_y;
public Point(Int32 x, Int32 y)
{
m_x = x;
m_y = y;
}
public override string ToString()
{
return string.Format("{0},{1}", m_x, m_y);
}
//实现类型安全的 CompareTo方法
public Int32 CompareTo(Point other)
{
//使用勾股定力来计算哪个point的距离原点(0,0)更远
return Math.Sign(
Math.Sqrt(m_x * m_x + m_y * m_y) - Math.Sqrt(other.m_x * other.m_x + other.m_y * other.m_y)
);
}
//实现IComparable的CompareTo方法
public int CompareTo(object obj)
{
if (GetType() != obj.GetType())
{
throw new ArgumentException("obj is not a Point");
}
//调用类型安全的CompareTo方法
return CompareTo((Point)obj);
}
}
internal class DynamicDemo
{
static void Main()
{
for (int demo = 0; demo < 2; demo++)
{
dynamic arg = (demo == 0) ? (dynamic)5 : (dynamic)"A";
dynamic result = Plus(arg);
M(result);
//M(Int32):10
//M(String):AA
}
Console.Read();
}
private static dynamic Plus(dynamic arg) { return arg + arg; }
private static void M(int n) { Console.WriteLine("M(Int32):" + n); }
private static void M(string s) { Console.WriteLine("M(String):" + s); }
}
《CLR via C#》读书笔记(5)基元类型、引用类型和值类型的更多相关文章
- [Clr via C#读书笔记]Cp5基元类型引用类型值类型
Cp5基元类型引用类型值类型 基元类型 编译器直接支持的类型,基元类型直接映射到FCL中存在的类型. 作者希望使用FCL类型名称而避免使用关键字.他的理由是为了更加的清晰的知道自己写的类型是哪种.但是 ...
- CLR via C#(02)-基元类型、引用类型、值类型
http://www.cnblogs.com/qq0827/p/3281150.html 一. 基元类型 编译器能够直接支持的数据类型叫做基元类型.例如int, string等.基元类型和.NET框架 ...
- [CLR via C#]引用类型和值类型
一.引用类型与值类型的区别 CLR支持两种类型:引用类型和值类型.引用类型总是从托管堆上分配的,C#的new操作符会返回对象的内存地址.使用引用类型时,必须注意到一些性能问题. 1)内存必须从托管堆上 ...
- 《CLR via C#》读书笔记--基元类型、引用类型和值类型
编程语言的基元类型 编译器直接支持的数据类型称为基元类型.基元类型直接映射到Framework类库中存在的类型.例如:C#中的int直接映射到System.Int32类型.下表给出了C#基元类型与对应 ...
- CLR via C#深解笔记三 - 基元类型、引用类型和值类型 | 类型和成员基础 | 常量和字段
编程语言的基元类型 某些数据类型如此常用,以至于许多编译器允许代码以简化的语法来操纵它们. System.Int32 a = new System.Int32(); // a = 0 a = 1 ...
- <NET CLR via c# 第4版>笔记 第5章 基元类型、引用类型和值类型
5.1 编程语言的基元类型 c#不管在什么操作系统上运行,int始终映射到System.Int32; long始终映射到System.Int64 可以通过checked/unchecked操作符/语句 ...
- 【CLR Via C#】第5章 基元类型、引用类型、值类型
第二遍看这本书,决定记录一下加深印象. 1,基元类型 什么事基元类型?基元类型是直接映射到FrameWork类库(FCL)中存在的类型,编译器直接支持的数据类型.比如int直接映射到System.In ...
- 重温CLR(四)基元类型、引用类型、值类型
编程语言的基元类型 编译器直接支持的数据类型称为基元类型(primitive type).基元类型直接映射到framework类型(fcl)中存在的类型. 下表列出fcl类型 从另一个角度,可以认为C ...
- CLR VIA C#: 基元类型、 引用类型 和 值类型
一.基元类型 . 引用类型 和 值类型的区别: 1.基元类型(primitive type):编译器直接支持的数据类型: 基元类型 直接映射到 FCL 中存在的类型. C# 小写是基元类型,例如:st ...
随机推荐
- iOS项目目录结构
一. 目前最为流行的目录结构是: <先根据模块后根据功能> 的文件目录结构 优点: 模块分明, 并且开发和维护时方便查阅各个功能 缺点: 可能会出现模块内随意建立文件夹, 导致局部逻辑紊乱 ...
- php之CI框架多语言的用法
public function index() { // 加载语言包,可以加载多个 $this->lang->load('email'); echo $this->lang-> ...
- [转]Aptana Studio 3配置Python开发环境图文教程
转载URL:http://www.cr173.com/html/49260_1.html 一.安装Aptana Studio 3 安装完运行时建议将相关默认工作目录设定在英文的某个目录下.避免可能出现 ...
- Redis系列-配置文件小结
如果不指定配置文件,Redis也可以启动,此时,redis使用默认的内置配置.不过在正式环境,常常通过配置文件[通常叫redis.conf]来配置redis. redis.conf配置格式如下: ke ...
- FlexPaper使用小结
FlexPaper相关介绍及后台swf生成,参见 FlexPaper实现文档在线浏览(附源码) 前台swf在flash中的预览: 1.下载相关文档 FlexPaper Classic 2.将下载的文件 ...
- SQL Server 流程控制
流程控制语句: BEGIN ... END WAITFOR GOTO WHILE IF ... ELSE BREAK RETURN CONTINURE 1.BEGIN ... END BEGIN ...
- Bootstrap datepicker可配置网址
http://eternicode.github.io/bootstrap-datepicker/?markup=input&format=&weekStart=&startD ...
- hdu2457
AC自动机+DP #include <cstdio> #include <queue> #include <cstring> using namespace std ...
- cxGrid的使用方法
来源 http://www.cnblogs.com/djcsch2001/archive/2010/07/19/1780573.html 1. 去掉GroupBy栏 cxGrid1DBTable ...
- SAP SMARTFORMS-基于内存方式在report程序中传递数据到smartforms显示
一.准备工作 1.新建include程序 1> include程序名字:ZDD_INC_0104 2> ZDD_INC_0104 程序中的内容为 2.使用T-CODE :SE11新建两个 ...