C#面试常考得几个问题-程序员宅基地

技术标签: c#  面试  数据结构与算法  

>什么是.NET?什么是CLI?什么是CLRIL是什么?JIT是什么,它是如何工作的?GC是什么,简述一下GC的工作方式?

CLI是规范; CLR是对CLI的实现; .NET是基于CLR构建的一套框架;

开发人员需要通过ILCLR进行交流,虽然IL本身支持一些面向对象的概念,但是对于开发人员来讲还是过于复杂低效,于是C#应运而生,程序员只需编写C#代码, csc编译器会将其翻译成IL;

虽然CLR理解IL,但是CPU只认识二进制指令,所以CLR需要JIT的帮助,IL翻译成CPU指令. JIT按需工作,当一个.NET方法即将被执行时, JIT会介入,把该方法(IL指令)编译成CPU指令,并保存以供重用.

GC被用来回收当前进程中已无人使用的垃圾对象; GC会被某些事件触发(比如, 0代对象满,内存压力大, appdomain被卸载),随后遍历GC堆上的对象,并通过该对象是否被root直接/间接引用(更进一步, “一个对象是否可以被回收还会依赖于F-Reachable QueueGC Handle Table”)来判断一个对象是否需要被回收.

>类(class)和结构(struct)的区别是什么?它们对性能有影响吗?.NET BCL里有哪些是类(结构),为什么它们不是结构(类)?在自定义类型时,您如何选择是类还是结构?

简单说来,类是引用类型,结构是值类型;

值类型对象直接分配在当前线程的栈上,引用类型对象位于GC堆上,所以值类型对象无法在多个方法中传递,而引用类型会承担更多的任务,比如,用于线程同步(Monitor.Enter(…))当一个引用类型的对象被创建时,会需要4+4=8byte的额外空间开销(32bit OS),同时过度的使用引用类型对象会增加GC堆的压力,频繁的GC对程序的性能还是有一些影响的.

BCL中的Byte类型是一个结构体,至于为什么它不是一个类,我觉得可能是设计者认为Byte不应该被继承,或者8byte的额外开销无法承受吧.

什么时候使用值类型/引用类型还是应当具体情况具体分析, 对于引用类型,能用sealedsealed,在方法调用时开销会小一些.

>.NET程序运行过程中,什么是堆,什么是栈?什么情况下会在堆(栈)上分配数据?它们有性能上的区别吗?结构对象可能分配在堆上吗?什么情况下会发生,有什么需要注意的?

.NET进程被创建时就会有一个堆随之被创建,用来保存该进程在运行中需要使用的对象/数据;当一个线程被创建时,会有一个栈被创建,用来保存方法调用参数,局部变量等轻量型数据.

当一个类里面包含一个结构体类型的变量时,该结构体类型会被分配在堆上.(不知道有什么需要注意的…)

>泛型的作用是什么?它有什么优势?它对性能有影响吗?它在执行时的行为是什么?.NET BCL中有哪些泛型类型?举例说明平时编程中您定义的泛型类型

泛型有利于算法重用.

.NET进程地址空间中,对象和类型是分开存放的,当我们实例化一个泛型的时候(比如List<int> list = new List<int>();),会有一个新的类型对象被创建(该对象并不位于GC堆上),当我们在使用这个实例化泛型去创建新的对象时,才会有一个对象(GC堆上)被创建.所以性能上会有些许的损失.当我们使用一个值类型作为参数,去调用一个接收引用类型参数的方法是,会有装箱发生,这时我们可以考虑实现一个泛型,并在运行时确定方法的参数类型.

>异常的作用是什么?.NET BCL中有哪些常见的异常?在代码中您是如何捕获/处理异常的?在“catch (ex)”中,“throw”和“throw ex”有什么区别?您会如何设计异常的结构,什么情况下您会抛出异常?

,异常可以通知我们程序出错,比如ArgumentException, NullReferenceException…

异常的发生会导致一次stack walk,去寻找对应的exception handler,在这个过程中, stack trace的信息会被一层层的收集, throw ex会清空之前收集的stack trace的信息,相当于抛出了一个新的异常,throw不会,所以throw ex不利于找出问题所在.

不同的layer应该catch不同exception,最上层处理最generalexception,底层去处理一些detailexception.

> List<T>T[]的区别是什么,平时你如何进行选择?Dictionary<TKey, TValue>是做什么的?.NET BCL中还有哪些常用的容器?它们分别是如何实现的(哪种数据结构)?分别是适用于哪些场景?

T[]继承自Array,List<T>仅仅是对T[]的封装;相比于T[], List<T>size是动态变化的.

Dictionary<TKey, TValue>可以用来存储键/值对.其他的比如HashTable, SortedList.

>抽象类和接口有什么区别?使用时有什么需要注意的吗?如何选择是定义一个完全抽象的抽象类,还是接口?什么是接口的显式实现?为什么说它很重要?

抽象类定义了一个类及其子类是什么,而接口更多的表现出一个类可以做什么.

当一个类实现连个不同的接口,而这两个接口中包含一些相同签名的方法时需要用到显示实现.

>字符串是引用类型类型还是结构类型?它和普通的引用类型相比有什么特别的地方吗?使用字符串时有什么需要注意的地方?为什么说StringBuilder比较高效?在连接多个字符串时,它无论何时都比直接相加更高效吗?

String是引用类型,其特殊之处在于一个string是不可变的,当我们对两个string使用连接操作时,会生成一个新的string对象,而原来的两个string保持不变.

在和native代码做interop,对于传出参数(char* outStr),应当选择使用stringbuilder而非string.

StringBuilder内部维护着一个char[]数组,在做连接字符的操作时会动态增加其大小,但是,当原有的数组不够用时, StringBuilder会重新创建一个新的char[]数组,值得注意的是原来的数组不会被抛弃,新创建的数组只会用作存储新添加的字符.

如果说StringBuilder不高效的话,应该就是在原有数组空间用尽的情况下吧.

>如何高效地进行数组复制?二维数组数组的数组有什么区别?在使用双重循环遍历一个二维数组时,如何选择内外层的遍历顺序?

不清楚,因为数组的元素在内存中的分布是连续的,我能想到的方法是直接使用内存拷贝API.

二维数组是二维的,数组的数组是一维的.

根据内存局部性原理, CPU在读取二维数组的第一个元素是,第一行的数据也会被一起读入cache,所以应当先遍历行,随后遍历列.

>什么是.NET?什么是CLI?什么是CLRIL是什么?JIT是什么,它是如何工作的?GC是什么,简述一下GC的工作方式?

CLI是规范; CLR是对CLI的实现; .NET是基于CLR构建的一套框架;

开发人员需要通过ILCLR进行交流,虽然IL本身支持一些面向对象的概念,但是对于开发人员来讲还是过于复杂低效,于是C#应运而生,程序员只需编写C#代码, csc编译器会将其翻译成IL;

虽然CLR理解IL,但是CPU只认识二进制指令,所以CLR需要JIT的帮助,IL翻译成CPU指令. JIT按需工作,当一个.NET方法即将被执行时, JIT会介入,把该方法(IL指令)编译成CPU指令,并保存以供重用.

GC被用来回收当前进程中已无人使用的垃圾对象; GC会被某些事件触发(比如, 0代对象满,内存压力大, appdomain被卸载),随后遍历GC堆上的对象,并通过该对象是否被root直接/间接引用(更进一步, “一个对象是否可以被回收还会依赖于F-Reachable QueueGC Handle Table”)来判断一个对象是否需要被回收.具体细节还这真不是半个小时能讲完的J

>类(class)和结构(struct)的区别是什么?它们对性能有影响吗?.NET BCL里有哪些是类(结构),为什么它们不是结构(类)?在自定义类型时,您如何选择是类还是结构?

简单说来,类是引用类型,结构是值类型;

值类型对象直接分配在当前线程的栈上,引用类型对象位于GC堆上,所以值类型对象无法在多个方法中传递,而引用类型会承担更多的任务,比如,用于线程同步(Monitor.Enter(…))当一个引用类型的对象被创建时,会需要4+4=8byte的额外空间开销(32bit OS),同时过度的使用引用类型对象会增加GC堆的压力,频繁的GC对程序的性能还是有一些影响的.

BCL中的Byte类型是一个结构体,至于为什么它不是一个类,我觉得可能是设计者认为Byte不应该被继承,或者8byte的额外开销无法承受吧.

什么时候使用值类型/引用类型还是应当具体情况具体分析,个人倾向引用类型,因为这样在设计的时候会少费些脑子,不过能用sealedsealed,在方法调用时开销会小一些.

>.NET程序运行过程中,什么是堆,什么是栈?什么情况下会在堆(栈)上分配数据?它们有性能上的区别吗?“结构”对象可能分配在堆上吗?什么情况下会发生,有什么需要注意的吗?

.NET进程被创建时就会有一个堆随之被创建,用来保存该进程在运行中需要使用的对象/数据;当一个线程被创建时,会有一个栈被创建,用来保存方法调用参数,局部变量等轻量型数据.

当一个类里面包含一个结构体类型的变量时,该结构体类型会被分配在堆上.(不知道有什么需要注意的…)

>泛型的作用是什么?它有什么优势?它对性能有影响吗?它在执行时的行为是什么?.NET BCL中有哪些泛型类型?举例说明平时编程中您定义的泛型类型.

泛型有利于算法重用.

.NET进程地址空间中,对象和类型是分开存放的,当我们实例化一个泛型的时候(比如List<int> list = new List<int>();),会有一个新的类型对象被创建(该对象并不位于GC堆上),当我们在使用这个实例化泛型去创建新的对象时,才会有一个对象(GC堆上)被创建.所以性能上会有些许的损失.当我们使用一个值类型作为参数,去调用一个接收引用类型参数的方法是,会有装箱发生,这时我们可以考虑实现一个泛型,并在运行时确定方法的参数类型.

>异常的作用是什么?.NET BCL中有哪些常见的异常?在代码中您是如何捕获/处理异常的?在“catch (ex)”中,“throw”“throw ex”有什么区别?您会如何设计异常的结构,什么情况下您会抛出异常?

,异常可以通知我们程序出错,比如ArgumentException, NullReferenceException…

异常的发生会导致一次stack walk,去寻找对应的exception handler,在这个过程中, stack trace的信息会被一层层的收集, throw ex会清空之前收集的stack trace的信息,相当于抛出了一个新的异常,throw不会,所以throw ex不利于找出问题所在.

不同的layer应该catch不同exception,最上层处理最generalexception,底层去处理一些detailexception.

> List<T>T[]的区别是什么,平时你如何进行选择?Dictionary<TKey, TValue>是做什么的?.NET BCL中还有哪些常用的容器?它们分别是如何实现的(哪种数据结构)?分别是适用于哪些场景?

T[]继承自Array,List<T>仅仅是对T[]的封装;相比于T[], List<T>size是动态变化的.

Dictionary<TKey, TValue>可以用来存储键/值对.其他的比如HashTable, SortedList.

>抽象类和接口有什么区别?使用时有什么需要注意的吗?如何选择是定义一个完全抽象的抽象类,还是接口?什么是接口的显式实现?为什么说它很重要?

抽象类定义了一个类及其子类是什么,而接口更多的表现出一个类可以做什么.

当一个类实现连个不同的接口,而这两个接口中包含一些相同签名的方法时需要用到显示实现.

>字符串是引用类型类型还是结构类型?它和普通的引用类型相比有什么特别的地方吗?使用字符串时有什么需要注意的地方?为什么说StringBuilder比较高效?在连接多个字符串时,它无论何时都比直接相加更高效吗?

String是引用类型,其特殊之处在于一个string是不可变的,当我们对两个string使用连接操作时,会生成一个新的string对象,而原来的两个string保持不变.

在和native代码做interop,对于传出参数(char* outStr),应当选择使用stringbuilder而非string.

StringBuilder内部维护着一个char[]数组,在做连接字符的操作时会动态增加其大小,但是,当原有的数组不够用时, StringBuilder会重新创建一个新的char[]数组,值得注意的是原来的数组不会被抛弃,新创建的数组只会用作存储新添加的字符.

如果说StringBuilder不高效的话,应该就是在原有数组空间用尽的情况下吧.

>如何高效地进行数组复制?二维数组数组的数组有什么区别?在使用双重循环遍历一个二维数组时,如何选择内外层的遍历顺序?

不清楚,因为数组的元素在内存中的分布是连续的,我能想到的方法是直接使用内存拷贝API.

二维数组是二维的,数组的数组是一维的.

根据内存局部性原理, CPU在读取二维数组的第一个元素是,第一行的数据也会被一起读入cache,所以应当先遍历行,随后遍历列.

>什么是元编程,.NET有哪些元编程的手段和场景?什么是反射?能否举一些反射的常用场景?有人说反射性能较差,您怎么看待这个问题?有什么办法可以提高反射的性能吗

对元编程了解十分有限, .NETCodeDom(或者使用reflection emit动态创建类型)应当是其中的一个场景.

基于元数据,反射帮助我们在运行时动态获取程序集/类型/方法/属性等等的信息,可以用于加载Addin.

有得必有失,反射功能强大,只要不是滥用,其带来的益处远大于性能上的损失.

>委托是什么?匿名方法是什么?在C# 3.0中,Lambda表达式是什么?扩展方法是什么?LINQ是什么?您觉得C# 3.0中还有哪些重要的特性,它们带来了什么优势?BCL中哪些类库和这些特性有关?您平时最常用哪些

委托可以认为是类型安全的函数指针,此处省略741个字

>工作之外您看哪些技术相关的书、网站、社区、项目等等?您还接触哪些.NET以外的技术,能和.NET.NET中有针对性的部分做个对比吗

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/iteye_13633/article/details/82325732

智能推荐

使用nginx解决浏览器跨域问题_nginx不停的xhr-程序员宅基地

文章浏览阅读1k次。通过使用ajax方法跨域请求是浏览器所不允许的,浏览器出于安全考虑是禁止的。警告信息如下:不过jQuery对跨域问题也有解决方案,使用jsonp的方式解决,方法如下:$.ajax({ async:false, url: 'http://www.mysite.com/demo.do', // 跨域URL ty..._nginx不停的xhr

在 Oracle 中配置 extproc 以访问 ST_Geometry-程序员宅基地

文章浏览阅读2k次。关于在 Oracle 中配置 extproc 以访问 ST_Geometry,也就是我们所说的 使用空间SQL 的方法,官方文档链接如下。http://desktop.arcgis.com/zh-cn/arcmap/latest/manage-data/gdbs-in-oracle/configure-oracle-extproc.htm其实简单总结一下,主要就分为以下几个步骤。..._extproc

Linux C++ gbk转为utf-8_linux c++ gbk->utf8-程序员宅基地

文章浏览阅读1.5w次。linux下没有上面的两个函数,需要使用函数 mbstowcs和wcstombsmbstowcs将多字节编码转换为宽字节编码wcstombs将宽字节编码转换为多字节编码这两个函数,转换过程中受到系统编码类型的影响,需要通过设置来设定转换前和转换后的编码类型。通过函数setlocale进行系统编码的设置。linux下输入命名locale -a查看系统支持的编码_linux c++ gbk->utf8

IMP-00009: 导出文件异常结束-程序员宅基地

文章浏览阅读750次。今天准备从生产库向测试库进行数据导入,结果在imp导入的时候遇到“ IMP-00009:导出文件异常结束” 错误,google一下,发现可能有如下原因导致imp的数据太大,没有写buffer和commit两个数据库字符集不同从低版本exp的dmp文件,向高版本imp导出的dmp文件出错传输dmp文件时,文件损坏解决办法:imp时指定..._imp-00009导出文件异常结束

python程序员需要深入掌握的技能_Python用数据说明程序员需要掌握的技能-程序员宅基地

文章浏览阅读143次。当下是一个大数据的时代,各个行业都离不开数据的支持。因此,网络爬虫就应运而生。网络爬虫当下最为火热的是Python,Python开发爬虫相对简单,而且功能库相当完善,力压众多开发语言。本次教程我们爬取前程无忧的招聘信息来分析Python程序员需要掌握那些编程技术。首先在谷歌浏览器打开前程无忧的首页,按F12打开浏览器的开发者工具。浏览器开发者工具是用于捕捉网站的请求信息,通过分析请求信息可以了解请..._初级python程序员能力要求

Spring @Service生成bean名称的规则(当类的名字是以两个或以上的大写字母开头的话,bean的名字会与类名保持一致)_@service beanname-程序员宅基地

文章浏览阅读7.6k次,点赞2次,收藏6次。@Service标注的bean,类名:ABDemoService查看源码后发现,原来是经过一个特殊处理:当类的名字是以两个或以上的大写字母开头的话,bean的名字会与类名保持一致public class AnnotationBeanNameGenerator implements BeanNameGenerator { private static final String C..._@service beanname

随便推点

二叉树的各种创建方法_二叉树的建立-程序员宅基地

文章浏览阅读6.9w次,点赞73次,收藏463次。1.前序创建#include&lt;stdio.h&gt;#include&lt;string.h&gt;#include&lt;stdlib.h&gt;#include&lt;malloc.h&gt;#include&lt;iostream&gt;#include&lt;stack&gt;#include&lt;queue&gt;using namespace std;typed_二叉树的建立

解决asp.net导出excel时中文文件名乱码_asp.net utf8 导出中文字符乱码-程序员宅基地

文章浏览阅读7.1k次。在Asp.net上使用Excel导出功能,如果文件名出现中文,便会以乱码视之。 解决方法: fileName = HttpUtility.UrlEncode(fileName, System.Text.Encoding.UTF8);_asp.net utf8 导出中文字符乱码

笔记-编译原理-实验一-词法分析器设计_对pl/0作以下修改扩充。增加单词-程序员宅基地

文章浏览阅读2.1k次,点赞4次,收藏23次。第一次实验 词法分析实验报告设计思想词法分析的主要任务是根据文法的词汇表以及对应约定的编码进行一定的识别,找出文件中所有的合法的单词,并给出一定的信息作为最后的结果,用于后续语法分析程序的使用;本实验针对 PL/0 语言 的文法、词汇表编写一个词法分析程序,对于每个单词根据词汇表输出: (单词种类, 单词的值) 二元对。词汇表:种别编码单词符号助记符0beginb..._对pl/0作以下修改扩充。增加单词

android adb shell 权限,android adb shell权限被拒绝-程序员宅基地

文章浏览阅读773次。我在使用adb.exe时遇到了麻烦.我想使用与bash相同的adb.exe shell提示符,所以我决定更改默认的bash二进制文件(当然二进制文件是交叉编译的,一切都很完美)更改bash二进制文件遵循以下顺序> adb remount> adb push bash / system / bin /> adb shell> cd / system / bin> chm..._adb shell mv 权限

投影仪-相机标定_相机-投影仪标定-程序员宅基地

文章浏览阅读6.8k次,点赞12次,收藏125次。1. 单目相机标定引言相机标定已经研究多年,标定的算法可以分为基于摄影测量的标定和自标定。其中,应用最为广泛的还是张正友标定法。这是一种简单灵活、高鲁棒性、低成本的相机标定算法。仅需要一台相机和一块平面标定板构建相机标定系统,在标定过程中,相机拍摄多个角度下(至少两个角度,推荐10~20个角度)的标定板图像(相机和标定板都可以移动),即可对相机的内外参数进行标定。下面介绍张氏标定法(以下也这么称呼)的原理。原理相机模型和单应矩阵相机标定,就是对相机的内外参数进行计算的过程,从而得到物体到图像的投影_相机-投影仪标定

Wayland架构、渲染、硬件支持-程序员宅基地

文章浏览阅读2.2k次。文章目录Wayland 架构Wayland 渲染Wayland的 硬件支持简 述: 翻译一篇关于和 wayland 有关的技术文章, 其英文标题为Wayland Architecture .Wayland 架构若是想要更好的理解 Wayland 架构及其与 X (X11 or X Window System) 结构;一种很好的方法是将事件从输入设备就开始跟踪, 查看期间所有的屏幕上出现的变化。这就是我们现在对 X 的理解。 内核是从一个输入设备中获取一个事件,并通过 evdev 输入_wayland

推荐文章

热门文章

相关标签