Unity3D 笔试题

part1

选择题目:4*12 = 48

1、关于C#中System.String 与 System.Text.StringBuilder 区别,描述错误的是:C
A、String是可不变字符串
B、StringBuilder存放了一个可变的字符串,并提供修改字符串的方法。
C、String的字符串拼接操作(+运算)会创建新对象,但String是值类型不会创建堆内存
D、StringBuilder的Append操作不会创建新对象

2、在C#中,下面描述错误的是: B
A、 const字段只能在该字段的声明中初始化
B、 const和static一起修饰字段就是静态常量
C、 readonly字段可以在声明或构造函数中初始化
D、 const 可以用于声明方法内变量,readonly不可以。

3、关于虚方法、抽象方法描述不正确的是:D
A、虚方法可以被子类继承并重写方法。
B、抽象方法不可以带实现。
C、抽象方法非抽象子类必须实现。
D、抽象方法可以被子类实现成虚方法

4、下面代码在运行中会产生几个临时对象:D

var a = new string("abc");
a = (a.ToUpper() + "123").Substring(0,2);

A、1
B、5
C、4
D、代码错误

5、以下哪个组件是任何GameObject必备的组件:B
A、MeshRenderer
B、Transform
C、GameObject
D、Camera

6、关于MonoBehavior.LateUpdate函数描述错误的是:B
A、当MonoBehavior类型应用后,每帧调用一次
B、常被用于处理RigidBody的更新
C、在所有Update函数执行后才能被调用
D、常被用于实现跟随相机效果,且目标物体的位置已经在Update函数中被更新

7、Unity脚本中的函数执行顺序,哪个是对的?:A
A、Awake() OnEnable() Start() Update();
B、OnEnable() Awake() Start() Update();
C、Start() Awake(); OnEnable() Update();
D、Awake() Start() Update() OnEnable();

8、下列哪个函数不属于碰撞事件:C
A、OnCollisionEnter
B、OnCollisionExit
C、OnCollisionUpdate
D、OnCollisionStay

9、如何通过脚本来删除其自身对应的Gameobject:A
A、Destroy(gameObject)
B、this.Destroy()
C、Destroy(this)
D、其他三项都可以

10、下列选项中有关Animator的说法错误的是:D
A、Animator是Unity引擎内置组件
B、任何一个具有动画状态机功能的GameObject都需要一个Animator组件
C、它主要用于角色行为的设置,包括StateMachines, 混合Blend trees以及通过脚本控制的事件
D、Animator同Animation组件的用法是相同的

11、Unity引擎使用的是左手坐标系还是右手坐标系:A
A、左手坐标系
B、右手坐标系
C、可以通过ProjectSetting切换右手坐标系
D、.可以通过Reference切换左手坐标系

12、什么是导航网格(NavMesh):B
A、一种用于描述相机轨迹的网格
B、一种用于实现自动寻址的网格
C、一种被优化过的物体网格
D、一种用于物理碰撞的网格

多选题 3*5 = 15
1、下面关于C# 的 set和get说明正确的有:ABC
A、get访问器,必须返回属性类型的值
B、get和set可以根据实际需要而出现其中一个,不一定两个都要出现
C、对象属性赋值的时候,调用的是set访问器
D、set相当于选取字段的值

2、关于Vector3 的API,以下说法正确的是:BC
A、Vector3.normalize 可以获取一个三维向量的法线向量
B、Vector3.magnitude 可以获取一个三维向量的长度
C、Vector3.forward 与 Vector3(0,0,1)是一样的意思
D、Vector3.Dot(向量A,向量B)是用来计算向量A与向量B的叉乘

3、关于GameObject的销毁接口,以下哪些说法正确:BC
A、Destroy是当帧立即执行
B、DestroyImmediate是当帧立即执行
C、Destroy是下一帧执行
D、DestroyImmediate只能在非运行时调用

part2

一:什么是协同程序?
在主线程运行的同时开启另一段逻辑处理,来协助当前程序的执行,协程很像多线程,但是不是多线程,Unity的协程实在每帧结束之后去检测yield的条件是否满足。
二:Unity3d中的碰撞器和触发器的区别?
碰撞器是触发器的载体,而触发器只是碰撞器身上的一个属性。当Is Trigger=false时,碰撞器根据物理引擎引发碰撞,产生碰撞的效果,可以调用OnCollisionEnter/Stay/Exit函数;当Is Trigger=true时,碰撞器被物理引擎所忽略,没有碰撞效果,可以调用OnTriggerEnter/Stay/Exit函数。如果既要检测到物体的接触又不想让碰撞检测影响物体移动或要检测一个物件是否经过空间中的某个区域这时就可以用到触发器
三:物体发生碰撞的必要条件?
两个物体都必须带有碰撞器(Collider),其中一个物体还必须带有Rigidbody刚体,而且必须是运动的物体带有Rigidbody脚本才能检测到碰撞。
四:请简述ArrayList和List的主要区别?
ArrayList存在不安全类型(ArrayList会把所有插入其中的数据都当做Object来处理),装箱拆箱的操作(费时),List是泛型类,功能跟ArrayList相似,但不存在ArrayList所说的问题。
五:如何安全的在不同工程间安全地迁移asset数据?三种方法
1.将Assets目录和Library目录一起迁移
2.导出包,export Package
3.用unity自带的assets Server功能
六:OnEnable、Awake、Start运行时的发生顺序?哪些可能在同一个对象周期中反复的发生
Awake –>OnEnable->Start,OnEnable在同一周期中可以反复地发生。
七:MeshRender中material和sharedmaterial的区别?
修改sharedMaterial将改变所有物体使用这个材质的外观,并且也改变储存在工程里的材质设置。不推荐修改由sharedMaterial返回的材质。如果你想修改渲染器的材质,使用material替代。
八:Unity提供了几种光源,分别是什么
四种。
平行光:Directional Light
点光源:Point Light
聚光灯:Spot Light
区域光源:Area Light
九:简述一下对象池,你觉得在FPS里哪些东西适合使用对象池
对象池就存放需要被反复调用资源的一个空间,当一个对象回大量生成的时候如果每次都销毁创建会很费时间,通过对象池把暂时不用的对象放到一个池中(也就是一个集合),当下次要重新生成这个对象的时候先去池中查找一下是否有可用的对象,如果有的话就直接拿出来使用,不需要再创建,如果池中没有可用的对象,才需要重新创建,利用空间换时间来达到游戏的高速运行效果,在FPS游戏中要常被大量复制的对象包括子弹,敌人,粒子等
十:CharacterController和Rigidbody的区别
Rigidbody具有完全真实物理的特性,Unity中物理系统最基本的一个组件,包含了常用的物理特性,而CharacterController可以说是受限的的Rigidbody,具有一定的物理效果但不是完全真实的,是Unity为了使开发者能方便的开发第一人称视角的游戏而封装的一个组件
十一:简述prefab的用处
在游戏运行时实例化,prefab相当于一个模板,对你已经有的素材、脚本、参数做一个默认的配置,以便于以后的修改,同时prefab打包的内容简化了导出的操作,便于团队的交流。
十二:请简述sealed关键字用在类声明时与函数声明时的作用
sealed修饰的类为密封类,类声明时可防止其他类继承此类,在方法中声明则可防止派生类重写此方法。
十三:请简述private,public,protected,internal的区别
public:对任何类和成员都公开,无限制访问
private:仅对该类公开
protected:对该类和其派生类公开
internal:只能在包含该类的程序集中访问该类
十四:使用Unity3d实现2d游戏,有几种方式?
使用本身的GUI,在Unity4.6以后出现的UGUI
2.把摄像机的Projection(投影)值调为Orthographic(正交投影),不考虑z轴;
LPR8">3.使用2d插件,如:2DToolKit,和NGUI
十五:在物体发生碰撞的整个过程中,有几个阶段,分别列出对应的函数
三个阶段,1.OnCollisionEnter 2.OnCollisionStay 3.OnCollisionExit
十六:Unity3d的物理引擎中,有几种施加力的方式,分别描述出来
rigidbody.AddForce/AddForceAtPosition,都在rigidbody系列函数中。大家可以自己去查看一下rigidbody的API
十七:什么叫做链条关节?
Hinge Joint,可以模拟两个物体间用一根链条连接在一起的情况,能保持两个物体在一个固定距离内部相互移动而不产生作用力,但是达到固定距离后就会产生拉力。
十八:物体自身旋转使用的函数?
Transform.Rotate()
十九:Unity3d提供了一个用于保存和读取数据的类(PlayerPrefs),请列出保存和读取整形数据的函数
PlayerPrefs.SetInt() PlayerPrefs.GetInt()
二十:Unity3d脚本从唤醒到销毁有着一套比较完整的生命周期,请列出系统自带的几个重要的方法。
Awake——>OnEnable–>Start——>Update——>FixedUpdate——>LateUpdate——>OnGUI——>OnDisable——>OnDestroy
二十一:物理更新一般放在哪个系统函数里?
FixedUpdate,固定时间间隔执行 可以在edit->project setting->time设置 update 是在渲染帧执行,和Update不同的是FixedUpdate是渲染帧执行,如果你的渲染效率低下的时候FixedUpdate调用次数就会跟着下降。FixedUpdate比较适用于物理引擎的计算,因为是跟每帧渲染有关。Update就比较适合做控制。
二十二:在场景中放置多个Camera并同时处于活动状态会发生什么?
游戏界面可以看到很多摄像机的混合。
二十三:如何销毁一个UnityEngine.Object及其子类?
使用Destroy()方法;
二十四:请描述为什么Unity3d中会发生在组件上出现数据丢失的情况
一般是组件上绑定的物体对象被删除了
二十五:LOD是什么,优缺点是什么?
LOD(Level of detail)多层次细节,是最常用的游戏优化技术。它按照模型的位置和重要程度决定物体渲染的资源分配,降低非重要物体的面数和细节度,从而获得高效率的渲染运算。缺点是增加了内存。
二十六:MipMap是什么,作用?
MipMapping:在三维计算机图形的贴图渲染中有常用的技术,为加快渲染进度和减少图像锯齿,贴图被处理成由一系列被预先计算和优化过的图片组成的文件,这样的贴图被称为MipMap。
二十七:请描述Interface与抽象类之间的不同
抽象类表示该类中可能已经有一些方法的具体定义,但接口就是公公只能定义各个方法的界面 ,不能具体的实现代码在成员方法中。类是子类用来继承的,当父类已经有实际功能的方法时该方法在子类中可以不必实现,直接引用父类的方法,子类也可以重写该父类的方法。实现接口的时候必须要实现接口中所有的方法,不能遗漏任何一个。
二十八:.Net与Mono的关系?
mono是.net的一个开源跨平台工具,就类似java虚拟机,java本身不是跨平台语言,但运行在虚拟机上就能够实现了跨平台。.net只能在windows下运行,mono可以实现跨平台跑,可以运行于linux,Unix,Mac OS等。
二十九:简述Unity3D支持的作为脚本的语言的名称
Unity的脚本语言基于Mono的.Net平台上运行,可以使用.NET库,这也为XML、数据库、正则表达式等问题提供了很好的解决方案。Unity里的脚本都会经过编译,他们的运行速度也很快。这三种语言实际上的功能和运行速度是一样的,区别主要体现在语言特性上。JavaScript、 C#、Boo
三十:U3D中用于记录节点空间几何信息的组件名称,及其父类名称
Transform 父类是 Component
三十一:向量的点乘、叉乘以及归一化的意义?
1.点乘描述了两个向量的相似程度,结果越大两向量越相似,还可表示投影
2.叉乘得到的向量垂直于原来的两个向量
3.标准化向量:用在只关系方向,不关心大小的时候
三十二:为何大家都在移动设备上寻求U3D原生GUI的替代方案
不美观,OnGUI很耗费时间,效率不高,使用不方便
三十三:请简述如何在不同分辨率下保持UI的一致性
NGUI很好的解决了这一点,屏幕分辨率的自适应性,原理就是计算出屏幕的宽高比跟原来的预设的屏幕分辨率求出一个对比值,然后修改摄像机的size。UGUI通过锚点和中心点和分辨率也解决这个问题
三十四:什么是LightMap?
LightMap:就是指在三维软件里实现打好光,然后渲染把场景各表面的光照输出到贴图上,最后又通过引擎贴到场景上,这样就使物体有了光照的感觉。
三十五:Unity和cocos2d的区别
Unity3D支持C#、javascript等,cocos2d-x 支持c++、Html5、Lua等。
cocos2d 开源 并且免费
Unity3D支持iOS、Android、Flash、Windows、Mac、Wii等平台的游戏开发,cocos2d-x支持iOS、Android、WP等。
三十六:C#和C++的区别?
简单的说:C# 与C++ 比较的话,最重要的特性就是C# 是一种完全面向对象的语言,而C++ 不是,另外C# 是基于IL 中间语言和.NET Framework CLR 的,在可移植性,可维护性和强壮性都比C++ 有很大的改进。C# 的设计目标是用来开发快速稳定可扩展的应用程序,当然也可以通过Interop 和Pinvoke 完成一些底层操作。更详细的区别大家可以参考这里
三十七:结构体和类有何区别?
结构体是一种值类型,而类是引用类型。(值类型、引用类型是根据数据存储的角度来分的)就是值类型用于存储数据的值,引用类型用于存储对实际数据的引用。那么结构体就是当成值来使用的,类则通过引用来对实际数据操作
三十八:ref参数和out参数是什么?有什么区别?
ref和out参数的效果一样,都是通过关键字找到定义在主函数里面的变量的内存地址,并通过方法体内的语法改变它的大小。不同点就是输出参数必须对参数进行初始化。ref必须初始化,out 参数必须在函数里赋值。ref参数是引用,out参数为输出参数。
三十九:C#的委托是什么?有何用处?
委托类似于一种安全的指针引用,在使用它时是当做类来看待而不是一个方法,相当于对一组方法的列表的引用。用处:使用委托使程序员可以将方法引用封装在委托对象内。然后可以将该委托对象传递给可调用所引用方法的代码,而不必在编译时知道将调用哪个方法。与C或C++中的函数指针不同,委托是面向对象,而且是类型安全的。
四十:C#中的排序方式有哪些?
选择排序,冒泡排序,快速排序,插入排序,希尔排序,归并排序
四十一:射线检测碰撞物的原理是?
射线是3D世界中一个点向一个方向发射的一条无终点的线,在发射轨迹中与其他物体发生碰撞时,它将停止发射 。
四十二:Unity中,照相机的Clipping Planes的作用是什么?调整Near、Fare两个值时,应该注意什么?
剪裁平面 。从相机到开始渲染和停止渲染之间的距离。
四十三:如何让已经存在的GameObject在LoadLevel后不被卸载掉?
void Awake() { DontDestroyOnLoad(transform.gameObject); }
四十四:请简述GC(垃圾回收)产生的原因,并描述如何避免?
GC回收堆上的内存
避免:1.减少new产生对象的次数
2.使用公用的对象(静态成员)
3.将String换为StringBuilder
四十五:反射的实现原理?
审查元数据并收集关于它的类型信息的能力。实现原理:在运行时根据程序集及其中的类型得到元数据。下面是实现步骤:
导入using System.Reflection;
Assembly.Load(“程序集”)加载程序集,返回类型是一个Assembly
得到程序集中所有类的名称
foreach (Type type in assembly.GetTypes()) { string t = type.Name; }
4. Type type = assembly.GetType(“程序集.类名”);获取当前类的类型
Activator.CreateInstance(type); 创建此类型实例
MethodInfo mInfo = type.GetMethod(“方法名”);获取当前方法
m.Info.Invoke(null,方法参数);
四十六:简述四元数的作用,四元数对欧拉角的优点?
四元数用于表示旋转
相对欧拉角的优点:
1.能进行增量旋转
2.避免万向锁
3.给定方位的表达方式有两种,互为负(欧拉角有无数种表达方式)
四十七:移动相机动作在哪个函数里,为什么在这个函数里?
LateUpdate,是在所有的update结束后才调用,比较适合用于命令脚本的执行。官网上例子是摄像机的跟随,都是所有的update操作完才进行摄像机的跟进,不然就有可能出现摄像机已经推进了,但是视角里还未有角色的空帧出现。
四十八:GPU的工作原理
简而言之,GPU的图形(处理)流水线完成如下的工作:(并不一定是按照如下顺序) 顶点处理:这阶段GPU读取描述3D图形外观的顶点数据并根据顶点数据确定3D图形的形状及位置关系,建立起3D图形的骨架。在支持DX8和DX9规格的GPU中,这些工作由硬件实现的Vertex Shader(定点着色器)完成。 光栅化计算:显示器实际显示的图像是由像素组成的,我们需要将上面生成的图形上的点和线通过一定的算法转换到相应的像素点。把一个矢量图形转换为一系列像素点的过程就称为光栅化。例如,一条数学表示的斜线段,最终被转化成阶梯状的连续像素点。 纹理帖图:顶点单元生成的多边形只构成了3D物体的轮廓,而纹理映射(texture mapping)工作完成对多变形表面的帖图,通俗的说,就是将多边形的表面贴上相应的图片,从而生成“真实”的图形。TMU(Texture mapping unit)即是用来完成此项工作。 像素处理:这阶段(在对每个像素进行光栅化处理期间)GPU完成对像素的计算和处理,从而确定每个像素的最终属性。在支持DX8和DX9规格的GPU中,这些工作由硬件实现的Pixel Shader(像素着色器)完成。 最终输出:由ROP(光栅化引擎)最终完成像素的输出,1帧渲染完毕后,被送到显存帧缓冲区。
总结:GPU的工作通俗的来说就是完成3D图形的生成,将图形映射到相应的像素点上,对每个像素进行计算确定最终颜色并完成输出。
四十九:什么是渲染管道?
是指在显示器上为了显示出图像而经过的一系列必要操作。 渲染管道中的很多步骤,都要将几何物体从一个坐标系中变换到另一个坐标系中去。主要步骤有:
本地坐标->视图坐标->背面裁剪->光照->裁剪->投影->视图变换->光栅化
五十:如何优化内存?
有很多种方式,例如
1.压缩自带类库;
2.将暂时不用的以后还需要使用的物体隐藏起来而不是直接Destroy掉;
3.释放AssetBundle占用的资源;
4.降低模型的片面数,降低模型的骨骼数量,降低贴图的大小;
5.使用光照贴图,使用多层次细节(LOD),使用着色器(Shader),使用预设(Prefab)。
6.代码中少产生临时变量
五十一:动态加载资源的方式?他们之间的区别
1.Resources.Load();
2.AssetBundle
区别参考
五十二:请描述游戏动画有哪几种,以及其原理?
主要有关节动画、骨骼动画、单一网格模型动画(关键帧动画)。
关节动画:把角色分成若干独立部分,一个部分对应一个网格模型,部分的动画连接成一个整体的动画,角色比较灵活,Quake2中使用这种动画;
骨骼动画,广泛应用的动画方式,集成了以上两个方式的优点,骨骼按角色特点组成一定的层次结构,有关节相连,可做相对运动,皮肤作为单一网格蒙在骨骼之外,决定角色的外观;
单一网格模型动画由一个完整的网格模型构成,在动画序列的关键帧里记录各个顶点的原位置及其改变量,然后插值运算实现动画效果,角色动画较真实。
五十三:alpha blend工作原理
Alpha Blend 实现透明效果,不过只能针对某块区域进行alpha操作,透明度可设。
五十四:写出光照计算中的diffuse的计算公式
diffuse = Kd x colorLight x max(N*L,0);Kd 漫反射系数、colorLight 光的颜色、N 单位法线向量、L 由点指向光源的单位向量、其中N与L点乘,如果结果小于等于0,则漫反射为0。
五十五:两种阴影判断的方法、工作原理。
本影和半影:参考本影和半影
本影:景物表面上那些没有被光源直接照射的区域(全黑的轮廓分明的区域)。
半影:景物表面上那些被某些特定光源直接照射但并非被所有特定光源直接照射的区域(半明半暗区域)
工作原理:从光源处向物体的所有可见面投射光线,将这些面投影到场景中得到投影面,再将这些投影面与场景中的其他平面求交得出阴影多边形,保存这些阴影多边形信息,然后再按视点位置对场景进行相应处理得到所要求的视图(利用空间换时间,每次只需依据视点位置进行一次阴影计算即可,省去了一次消隐过程)
五十六:Vertex Shader是什么,怎么计算?
顶点着色器是一段执行在GPU上的程序,用来取代fixed pipeline中的transformation和lighting,Vertex Shader主要操作顶点。
Vertex Shader对输入顶点完成了从local space到homogeneous space(齐次空间)的变换过程,homogeneous space即projection space的下一个space。在这其间共有world transformation, view transformation和projection transformation及lighting几个过程。
五十七:下列代码在运行中会产生几个临时对象?
string a = new string(“abc”); a = (a.ToUpper() + “123”).Substring(0, 2);
在C#中第一行是会报错的(Java中倒是可行)。
应该这样初始化:
string b = new string(new char[]{‘a’,’b’,’c’});
答案为:5个临时对象
五十八:下列代码在运行中会发生什么问题?如何避免?
List ls = new List(new int[] { 1, 2, 3, 4, 5 }); foreach (int item in ls) { Console.WriteLine(item * item); ls.Remove(item); }
产生运行时错误,在 ls.Remove(item)这行,因为foreach是只读的。不能一边遍历一边修改。
五十九:Unity3D是否支持写成多线程程序?如果支持的话需要注意什么?
仅能从主线程中访问Unity3D的组件,对象和Unity3D系统调用
支持:如果同时你要处理很多事情或者与Unity的对象互动小可以用thread,否则使用coroutine。
注意:C#中有lock这个关键字,以确保只有一个线程可以在特定时间内访问特定的对象
六十:Unity3D的协程和C#线程之间的区别是什么?
多线程程序同时运行多个线程 ,而在任一指定时刻只有一个协程在运行,并且这个正在运行的协同程序只在必要时才被挂起。除主线程之外的线程无法访问Unity3D的对象、组件、方法。
Unity3d没有多线程的概念,不过unity也给我们提供了StartCoroutine(协同程序)和LoadLevelAsync(异步加载关卡)后台加载场景的方法。 StartCoroutine为什么叫协同程序呢,所谓协同,就是当你在StartCoroutine的函数体里处理一段代码时,利用yield语句等待执行结果,这期间不影响主程序的继续执行,可以协同工作。
六十一:矩阵相乘的意义及注意点
用于表示线性变换:旋转、缩放、投影、平移、仿射
注意矩阵的蠕变:误差的积累
六十二:为什么dynamic font在unicode环境下优于static font
Unicode是国际组织制定的可以容纳世界上所有文字和符号的字符编码方案。
使用动态字体时,Unity将不会预先生成一个与所有字体的字符纹理。当需要支持亚洲语言或者较大的字体的时候,若使用正常纹理,则字体的纹理将非常大。
六十三:当一个细小的高速物体撞向另一个较大的物体时,会出现什么情况?如何避免?
穿透(碰撞检测失败)
六十四:请简述OnBecameVisible及OnBecameInvisible的发生时机,以及这一对回调函数的意义?
当物体是否可见切换之时。可以用于只需要在物体可见时才进行的计算。
六十五:什么叫动态合批?跟静态合批有什么区别?
如果动态物体共用着相同的材质,那么Unity会自动对这些物体进行批处理。动态批处理操作是自动完成的,并不需要你进行额外的操作。
区别:动态批处理一切都是自动的,不需要做任何操作,而且物体是可以移动的,但是限制很多。静态批处理:自由度很高,限制很少,缺点可能会占用更多的内存,而且经过静态批处理后的所有物体都不可以再移动了。
参考
六十六:简述StringBuilder和String的区别?
String是字符串常量。
StringBuffer是字符串变量 ,线程安全。
StringBuilder是字符串变量,线程不安全。
String类型是个不可变的对象,当每次对String进行改变时都需要生成一个新的String对象,然后将指针指向一个新的对象,如果在一个循环里面,不断的改变一个对象,就要不断的生成新的对象,所以效率很低,建议在不断更改String对象的地方不要使用String类型。
StringBuilder对象在做字符串连接操作时是在原来的字符串上进行修改,改善了性能。这一点我们平时使用中也许都知道,连接操作频繁的时候,使用StringBuilder对象。
六十七:Unity3D Shader分哪几种,有什么区别?
表面着色器的抽象层次比较高,它可以轻松地以简洁方式实现复杂着色。表面着色器可同时在前向渲染及延迟渲染模式下正常工作。
顶点片段着色器可以非常灵活地实现需要的效果,但是需要编写更多的代码,并且很难与Unity的渲染管线完美集成。
固定功能管线着色器可以作为前两种着色器的备用选择,当硬件无法运行那些酷炫Shader的时,还可以通过固定功能管线着色器来绘制出一些基本的内容。
六十八:已知strcpy函数的原型是:char * strcpy(char * strDest,const char * strSrc); 1.不调用库函数,实现strcpy函数。2.解释为什么要返回char *

char * strcpy(char * strDest,const char * strSrc) 
{ 
	if ((strDest==NULL)||(strSrc==NULL)) 
	throwInvalid argument(s); 
	char * strDestCopy=strDest; 
	while ((*strDest++=*strSrc++)!=’\0); 
	return strDestCopy; 
} 

六十九:C#中四种访问修饰符是哪些?各有什么区别?
1.属性修饰符 2.存取修饰符 3.类修饰符 4.成员修饰符。
属性修饰符:
Serializable:按值将对象封送到远程服务器。
STATread:是单线程套间的意思,是一种线程模型。
MATAThread:是多线程套间的意思,也是一种线程模型。
存取修饰符:
public:存取不受限制。
private:只有包含该成员的类可以存取。
internal:只有当前工程可以存取。
protected:只有包含该成员的类以及派生类可以存取。
类修饰符:
abstract:抽象类。指示一个类只能作为其它类的基类。
sealed:密封类。指示一个类不能被继承。理所当然,密封类不能同时又是抽象类,因为抽象总是希望被继承的。
成员修饰符:
abstract:指示该方法或属性没有实现。
sealed:密封方法。可以防止在派生类中对该方法的override(重载)。不是类的每个成员方法都可以作为密封方法密封方法,必须对基类的虚方法进行重载,提供具体的实现方法。所以,在方法的声明中,sealed修饰符总是和override修饰符同时使用。
delegate:委托。用来定义一个函数指针。C#中的事件驱动是基于delegate + event的。
const:指定该成员的值只读不允许修改。
event:声明一个事件。
extern:指示方法在外部实现。
override:重写。对由基类继承成员的新实现。
readonly:指示一个域只能在声明时以及相同类的内部被赋值。
static:指示一个成员属于类型本身,而不是属于特定的对象。即在定义后可不经实例化,就可使用。
virtual:指示一个方法或存取器的实现可以在继承类中被覆盖。
new:在派生类中隐藏指定的基类成员,从而实现重写的功能。 若要隐藏继承类的成员,请使用相同名称在派生类中声明该成员,并用 new 修饰符修饰它。
七十:Heap与Stack有何区别?
1.heap是堆,stack是栈。
2.stack的空间由操作系统自动分配和释放,heap的空间是手动申请和释放的,heap常用new关键字来分配。
3.stack空间有限,heap的空间是很大的自由区。
七十一:值类型和引用类型有何区别?
1.值类型的数据存储在内存的栈中;引用类型的数据存储在内存的堆中,而内存单元中只存放堆中对象的地址。
2.值类型存取速度快,引用类型存取速度慢。
3.值类型表示实际数据,引用类型表示指向存储在内存堆中的数据的指针或引用
4.值类型继承自System.ValueType,引用类型继承自System.Object
5.栈的内存分配是自动释放;而堆在.NET中会有GC来释放
6.值类型的变量直接存放实际的数据,而引用类型的变量存放的则是数据的地址,即对象的引用。
七十二:请写出求斐波那契数列任意一位的值得算法
递归实现:

int Fib1(int index) 
{ 
	if(index<1) 
{ 
return -1; 
} 
if(index==1|| index==2) 
{ 
return 1; 
} 
return Fib1(index-1)+Fib1(index-2); 
} 
迭代实现:
int Fib5(int index) 
{ 
if(index<1) 
{ 
return -1; 
} 
int a1 - 1, a2 = 1, a3 = 1; 
for(int i = 0; i < index - 2; i++) 
{ 
	a3=a1+a2; 
	a1=a2; 
	a2=a3; 
} 
return a3; 
} 

参看更多实现方法
七十三:协同程序的执行代码是什么?有何用处,有何缺点?

function Start() { 
	// 协同程序WaitAndPrint在Start函数内执行,可以视同于它与Start函数同步执行. 
	StartCoroutine(WaitAndPrint(2.0)); 
	print (“Before WaitAndPrint Finishes ” + Time.time ); 
} 
function WaitAndPrint (waitTime : float) { 
	// 暂停执行waitTime秒 
	yield WaitForSeconds (waitTime); 
	print (“WaitAndPrint “+ Time.time ); 
}

作用:一个协同程序在执行过程中,可以在任意位置使用yield语句。yield的返回值控制何时恢复协同程序向下执行。协同程序在对象自有帧执行过程中堪称优秀。协同程序在性能上没有更多的开销。
缺点:协同程序并非真线程,可能会发生堵塞。
七十四:什么是里氏代换元则?
里氏替换原则(Liskov Substitution Principle LSP)面向对象设计的基本原则之一。通俗点:就是子类对象可以赋值给基类对象,基类对象不能赋值给子类对象
参考
七十五:Mock和Stub有何区别?
Mock与Stub的区别:Mock:关注行为验证。细粒度的测试,即代码的逻辑,多数情况下用于单元测试。Stub:关注状态验证。粗粒度的测试,在某个依赖系统不存在或者还没实现或者难以测试的情况下使用,例如访问文件系统,数据库连接,远程协议等。
七十六:概述序列化:
序列化简单理解成把对象转换为容易传输的格式的过程。比如,可以序列化一个对象,然后使用HTTP通过Internet在客户端和服务器端之间传输该对象
七十七:堆和栈的区别?
栈通常保存着我们代码执行的步骤,如在代码段1中 AddFive()方法,int pValue变量,int result变量等等。而堆上存放的则多是对象,数据等。我们可以把栈想象成一个接着一个叠放在一起的盒子。当我们使用的时候,每次从最顶部取走一个盒子。栈也是如此,当一个方法(或类型)被调用完成的时候,就从栈顶取走,接着下一个。堆则不然,像是一个仓库,储存着我们使用的各种对象等信息,跟栈不同的是他们被调用完毕不会立即被清理掉。
七十八:概述c#中代理和事件?
代理就是用来定义指向方法的引用。
C#事件本质就是对消息的封装,用作对象之间的通信;发送方叫事件发送器,接收方叫事件接收器
七十九:客户端与服务器交互方式有几种?
socket通常也称作”套接字”,实现服务器和客户端之间的物理连接,并进行数据传输,主要有UDP和TCP两个协议。Socket处于网络协议的传输层。
http协议传输的主要有http协议 和基于http协议的Soap协议(web service),常见的方式是 http 的post 和get 请求,web 服务。
八十:Unity和Android与iOS如何交互?
Unity可以到处Android和iOS的工程,然后通过安卓或者iOS的类去给Unity发消息,调用Unity中的方法
八十一:如何在Unity3D中查看场景的面试,顶点数和Draw Call数?如何降低Draw Call数?
在Game视图右上角点击Stats。降低Draw Call 的技术是Draw Call Batching
这个在5.0以后在window-》Profiler下面,快捷键是cmd + 7(ctl + 7
八十二:请问alpha test在何时使用?能达到什么效果?
Alpha Test ,中文就是透明度测试。简而言之就是V&F shader中最后fragment函数输出的该点颜色值(即上一讲frag的输出half4)的alpha值与固定值进行比较。AlphaTest语句通常于Pass{}中的起始位置。Alpha Test产生的效果也很极端,要么完全透明,即看不到,要么完全不透明。
八十三:UNITY3d在移动设备上的一些优化资源的方法
1.使用assetbundle,实现资源分离和共享,将内存控制到200m之内,同时也可以实现资源的在线更新
2.顶点数对渲染无论是cpu还是gpu都是压力最大的贡献者,降低顶点数到8万以下,fps稳定到了30帧左右
3.只使用一盏动态光,不是用阴影,不使用光照探头
粒子系统是cpu上的大头
4.剪裁粒子系统
5.合并同时出现的粒子系统
6.自己实现轻量级的粒子系统
animator也是一个效率奇差的地方
7.把不需要跟骨骼动画和动作过渡的地方全部使用animation,控制骨骼数量在30根以下
8.animator出视野不更新
9.删除无意义的animator
10.animator的初始化很耗时(粒子上能不能尽量不用animator)
11.除主角外都不要跟骨骼运动apply root motion
12.绝对禁止掉那些不带刚体带包围盒的物体(static collider )运动
NUGI的代码效率很差,基本上runtime的时候对cpu的贡献和render不相上下
13每帧递归的计算finalalpha改为只有初始化和变动时计算
14去掉法线计算
15不要每帧计算viewsize 和windowsize
16filldrawcall时构建顶点缓存使用array.copy
17.代码剪裁:使用strip level ,使用.net2.0 subset
18.尽量减少smooth group
19.给美术定一个严格的经过科学验证的美术标准,并在U3D里面配以相应的检查工具
八十四:四元数有什么作用?
对旋转角度进行计算时用到四元数
八十五:将Camera组件的ClearFlags选项选成Depth only是什么意思?有何用处?
如果把摄像机的ClearFlags勾选为Deapth Only,那么摄像机就会只渲染看得见的对象,把背景会完全透明,这种情况一般用在两个摄像机以上的场景中
八十六:在编辑场景时将GameObject设置为Static有何作用?
设置游戏对象为Static时,这些部分被静态物体挡住而不可见时,将会剔除(或禁用)网格对象。因此,在你的场景中的所有不会动的物体都应该标记为Static。
八十七:有A和B两组物体,有什么办法能够保证A组物体永远比B组物体先渲染?
把A组物体的渲染对列大于B物体的渲染队列,通过shader里面的渲染队列来渲染
八十八:将图片的TextureType选项分别选为““Texture”和“Sprite”有什么区别
Sprite作为UI精灵使用,Texture作用模型贴图使用。Sprite需要2的整次幂,打包图片省资源
八十九:问一个Terrain,分别贴3张,4张,5张地表贴图,渲染速度有什么区别?为什么?
没有区别,因为不管几张贴图只渲染一次。
九十:什么是DrawCall?DrawCall高了又什么影响?如何降低DrawCall?
Unity中,每次引擎准备数据并通知GPU的过程称为一次Draw Call。DrawCall越高对显卡的消耗就越大。降低DrawCall的方法:
Dynamic Batching
Static Batching
高级特性Shader降级为统一的低级特性的Shader。
九十一:实时点光源的优缺点是什么?
可以有cookies – 带有 alpha通道的立方图(Cubemap )纹理。点光源是最耗费资源的。
九十二:Unity的Shader中,Blend SrcAlpha OneMinusSrcAlpha这句话是什么意思?
作用就是Alpha混合。公式:最终颜色 = 源颜色 x 源透明值 + 目标颜色 x(1 - 源透明值)
九十三:简述水面倒影的渲染原理
原理就是对水面的贴图纹理进行扰动,以产生波光玲玲的效果。用shader可以通过GPU在像素级别作扰动,效果细腻,需要的顶点少,速度快
九十四:简述NGUI中Grid和Table的作用?
对Grid和Table下的子物体进行排序和定位
九十五:请简述NGUI中Panel和Anchor的作用
只要提供一个half-pixel偏移量,它可以让一个控件的位置在Windows系统上精确的显示出来(只有这个Anchor的子控件会受到影响)
如果挂载到一个对象上,那么他可以将这个对象依附到屏幕的角落或者边缘
3.UIPanel用来收集和管理它下面所有widget的组件。通过widget的geometry创建实际的draw call。没有panel所有东西都不能够被渲染出来,你可以把UIPanel当做Renderer
九十六:能用foreach遍历访问的对象需要实现接口或声明_方法的类型

part3

单选题 40
1、关于C#的关键词下面描述错误的是:A
A、private 修饰私有成员时,内部类无法使用该成员。
B、protocted 修饰的无参数构造函数,子类会默认继承
C、public 修饰的成员,完全公开没有访问限制。
D、internal 修饰的类,在同一命名空间访问。

2、在C#中,下面变量命名不合法的是:B
A、A2
B、int
C、from
D、name123

3、下面C#代码运行后输出:C

int a = 1;
int b = a++;
b = (++a)+b;

Console.WriteLine($"{a}{b}");

A、2、3
B、3、3
C、3、4
D、2、4

4、下面类型中不是值类型的是: C
A、float
B、char
C、string
D、int

5、关于Unity内存优化下面操作没有作用的是:D
A、暂时不需要的物体隐藏起来,可以减少重复加载造成的GC压力。
B、释放Assetbundle占用的资源。
C、降低模型面数、降低骨骼数量
D、使用LOD,可以降低内存占用

6、关于MonoBehaviour生命周期,下面描述不正确的是:D
A、MonoBehaviour 是 Component的子类
B、不建议在MonoBehaviour子类构造函数中初始化成员
C、Awake方法早于Start方法
D、Update方法 和 FixedUpdate 方法没有区别,都是每帧调用

7、下面描述错误的是:A
A、欧拉角可以很好的处理旋转插值,任何情况下都可以使用。
B、Unity使用4X4矩阵来处理空间变换
C、Unity使用的左手坐标系,正方向为(0,0,1)
D、可以通过两个向量来获得一个LookRotation,分别是当前的up向量和方向向量

8、关于Lua语言下面描述错误的是:D
A、lua中使用userdata来处理与C#的类型转义
B、lua变量查找顺序是先function内,再upval最后再查找_G表
C、lua中文件local变量是一直存在
D、xlua无法卸载lua虚拟机

多选题:10
9、下面关于Unity优化描述正确的是: ABCD
A、使用对象池可以减少内存分配次数,降低GC造成的卡顿
B、删除没有代码的MonoBehaviour Update方法能提升代码运行速度
C、使用 StringBuilder 的 Append 替换string的加法运行能降低GC开销
D、使用协程可以将算力分散,处理复杂计算时能提升刷新帧率

10、关于C#使用描述错误的是: CD
A、可以使用C#语言来进行惹更新,但需要ilruntime第三方库
B、可以通过C#反射,获取到类的方法进行调用
C、C#中struct类型不能使用类作为属性
D、Unity中不能开启新的C#线程

编程题: 50

单向链表倒转: 10

比如输入:
[1,2,4,5,9]

输出:
[9,5,4,2,1]

时间限制:1秒
内存限制:256M
语言限制:[Kotlin, Typescript, Python, C++, Groovy, Rust, C#, Java, Go, C, Scala, Javascript, Ruby, Swift, Php, Python 3]

20

Unity3D 笔试题_第1张图片

part4

Lua基础数据类型
有 8 个基本类型分别为:nil、boolean、number、string、userdata、function、thread 和 table。
Lua垃圾回收机制以及内存泄漏检测原理
[Lua]Lua内存泄露检测原理_ouyangshima的博客-CSDN博客_lua内存泄漏​

blog.csdn.net/shimazhuge/article/details/43794347
Unity3D 笔试题_第2张图片

Lua如何实现面向对象
面向对象的基本原则:封装,继承,多态,抽象
https://www.cnblogs.com/Fflyqaq/p/13292388.html#11lua%E7%B1%BB%E7%9A%84%E5%88%9B%E5%BB%BA%E5%92%8C%E5%AE%9E%E4%BE%8B%E5%8C%96​

www.cnblogs.com/Fflyqaq/p/13292388.html#11lua%E7%B1%BB%E7%9A%84%E5%88%9B%E5%BB%BA%E5%92%8C%E5%AE%9E%E4%BE%8B%E5%8C%96
Lua的sort排序用的是什么排序方法
快速排序

public void QuickSort(int[] arr,int low,int high){
    if (low>high)
        return;
    int pivot = partition(arr,0,arr.Length()-1);
    QuickSort(arr,low,pivot-1);
    QuickSort(arr,pivot+1,high);
}
public int partition(int[] arr,int low,int high){
    int pivot = arr[low];
    while(low<high){
        while( low < high && arr[high] >= pivot) high--;
        arr[low] = arr[high];
        while(low > high && arr[low] <= pivot) low++;
        arr[high] = arr[low];
    }
    arr[low] = pivot;
    return low;
}

Lua字符串如何拼接?如果字符串很多用什么?
https://blog.csdn.net/fengshenyun/article/details/89952494​

blog.csdn.net/fengshenyun/article/details/89952494
Lua和C#的交互原理
C# -->Lua:由C#先将数据放入栈中,由lua去栈中获取数据,然后返回结果数据到栈顶,再由栈顶返回至C#。
Lua -->Wrap -->C#:先生成C#源文件所对应的Wrap文件或者编写C#源文件所对应的C模块,然后将源文件内容通过Wrap文件或者C模块注册到Lua虚拟机中,然后由Lua去调用这个模块的函数
C#与Lua交互原理_Ye Hui的博客-CSDN博客_c#与lua交互​

blog.csdn.net/UnityHUI/article/details/79752296
Unity3D 笔试题_第3张图片

Lua闭包
看这样一段代码

function func()
    local a = 10
    return function ()
        print(a)
        a = a + 1
    end
end
f = func()
f()--10
f()--11

由此可以看出函数的局部变量是可以保存在函数内部的,通过调用该函数内嵌的函数可以获取并修改局部变量的值,该函数的局部变量(upvalue)和内嵌函数的组合使用,形成了闭包。
李斌:谈一谈Lua中的闭包9 赞同 · 0 评论文章
浅拷贝和深拷贝
浅拷贝只复制指向某个对象的指针,而不复制对象本身,新旧对象还是共享同一块内存。但深拷贝会另外创造一个一模一样的对象,新对象跟原对象不共享内存,修改新对象不会改到原对象。

lua用“=”进行浅拷贝时有两种情况:
1、如果是值类型,会进行复制,创建一个新的对象,拷贝出来的对象和原来的对象互不影响。
2、如果是引用类型,例如:table。拷贝出来的对象和原来的对象实际上是同一个对象,所以修改拷贝出来的对象中的元素的值也会使原来的对象中元素的值发生改变
lua中没有深拷贝的api,只能自己写方法实现。

function CopyTable(tab)
	function _copy(obj)
		if type(obj) ~= "table" then
			return obj
		end
		local new_table = {}
		for k, v in pairs(obj) do
			new_table[_copy(k)] = _copy(v)
		end
		return setmetatable(new_table, getmetatable(obj))
	end
	return _copy(tab)
end

Lua热更原理
这个帖子回答的浅一些,基本上就是深度拷贝替换原来模块中的值
Lua热更新原理及示例-腾讯游戏学堂​

gameinstitute.qq.com/community/detail/120538
Unity3D 笔试题_第4张图片

这个相对深一些,考虑到如果要单独替换某个模块中的函数,upvalue的替换问题
lua热更新学习​

www.jianshu.com/p/30416db4f649
Unity3D 笔试题_第5张图片

part5

单选题 40
1、下面关键字不可以用于定义类的是:D
A、public B、private C、sealed D、virtual

2、关于C#中值类型与引用类型的描述,不正确的是: B
A、值类型用来描述简单类型,数据存储在栈上,超出作用域就删除
B、值类型和引用类型的父类都是System.Object
C、引用类型的释放依靠垃圾回收
D、string是特殊引用类型,修改string值不会改变原内存值

3、foreach 和 for 的区别描述正确的是:B
A、对与list的循环,无论是使用foreach还是for都可以在循环体中使用 list.remove 方法来删除元素
B、foreach的原理是获取对象的 IEnumerator GetEnumerator() 方法,只要有该方法就可以使用foreach
C、推荐使用foreach来替换for循环,这样可以提升代码的性能。
D、在C#的高版本中(C#8)仍需要考虑lambda闭包的问题

4、关于Unity中monobehavior描述错误的是:B
A、MonoBehaviour是 Component 的子类
B、MonoBehaviour可以定义带参数的构造函数,并可以使用该构造函数初始化对象
C、Awake方法是MonoBehaviour生命周期的开始 ,OnDestroy 是生命周期的结束
D、FixedUpdate 和 Update不一样,Unity会优先保证FixedUpdate的执行

5、关于热更新下面描述错误的是: D
A、资源热更新可以使用assetbundle完成
B、Unity不支持逻辑热更新,逻辑热更新需要依赖第三方库来完成
C、Addressable 系统是基于 assetbundle 的管理系统
D、只可以使用lua来完成热更新

6、下面代码将输出:A

using System;
using System.Collections;
using System.Collections.Generic;
namespace TestDemo
{
    class Program
    {     
        class  A 
        {
            protected A()
            {
                Console.WriteLine("I am A");
            }
        }
        class  B : A 
        {
            protected B(string name)
            {
                Console.WriteLine($"I am {name}");   
            }
        }
        class C : B 
        {
            public C() : base(nameof(C))
            {
            }
        }    
        static void Main(string[] args)
        {
            var c = new C();           
        }
    } 
}

A、
I am A
I am C
B、
I am C
C、
I am C
I am A

D、代码编译错误

7、在C#9.0版本后,下面代码中错误的是: B

public interface IA
{
    void Say() => Console.WriteLine("I am A");
    int B = 0;
    static int A = 0;
    int GetA { get; }
}

A、
void Say() => Console.WriteLine(“I am A”);
B、
int B = 0;
C、
static int A = 0;
D、
int GetA { get; }

8、关于C#中的容器类,下面描述正确的是:A
A、插入性能: LinkedList > Dictionary > HashTable > List
B、插入性能:List > LinkedList > Dictionary > HashTable
C、插入性能: Dictionary > LinkedList > HashTable > List
D、插入性能:HashTable>Dictionary > LinkedList > List

编程题填空题 15
补充如下代码,完成输出:
99 9 3 2 1 1 1 -7
代码:

using System;
using System.Collections;
using System.Collections.Generic;
namespace TestDemo
{
    class Program
    {
        public static void BubbleSort(int[] arr)
        {
            for (int i = 0; i < arr.Length; i++)
            {
                for (int j = 0; j < arr.Length; j++)
                {
                    if(i==j) continue;
                    1) _____
                    {
                        2)_____
                    }
                }
            }
        }
        
        static void Main(string[] args)
        {
            var arr = new int[] { 1,2,1,3,9,-7,1,99};
            BubbleSort(arr);
            for (int i = 0; i < arr.Length; i++)
            {
                3) ____
            }
        }
    }
}
  1. 选择: A
    A、
    if (arr[i] > arr[j])
    B、
    if (arr[i] < arr[j])
    C、
    if (arr[i] != arr[j])
    D、
    if (arr[i] == arr[j])

2)选择: D
A、
(arr[i], arr[j]) = (arr[i], arr[j]);
B、
int temp = arr[i];
arr[j]=temp;
arr[i]=arr[j];
C、
(arr[i], arr[j]) = (arr[i], arr[j+1]);

D、
(arr[i], arr[j]) = (arr[j], arr[i]);

  1. 选择:B
    A、
    Console.Write(arr[i]);
    Console.Write(“\n” );
    B、
    Console.Write(arr[i]+" “);
    C、
    Console.WriteLine(arr[i]);
    D、
    Console.Write(arr[i]);
    Console.WriteLine(” " );

编程题 45

1、写一个函数 f(N) 返回1到N之间出现1的个数:如:f(11)返回4
输入:11
返回:4

2、给定一个字符串s,返回最长的回文字符串,若字符为null,返回空字符串。s最大长度为1000
如:AACBZC 返回 CBZC

输入: ABBC
返回: BB

3、 一个电话号码盘有09共10个数字,分布着az共26个字母,即0~7上按顺序各3个字母,8上是y,9上是 z。连续按N次(N>0),请输出N次的全部可能的字母组合。
示例1:
输入
[0,1,9]
输出 [“adz”,“aez”,“afz”,“bdz”,“bez”,“bfz”,“cdz”,“cez”,“cfz”]

part6

一、单选题

1、如果一个函数成员需要被子类使用且不需要提供其他类来使用,需要的修饰符(B)
A、private
B、protected
C、internal
D、public

2、阅读以下的C#代码:

class A{
 public A( ) {
  Console.WriteLine("A");
 }
}
class B:A{
 public B(){
  Console.WriteLine("B");
 }
}
class Program{
 public static void Main(){
  var b = new B();
  Console.ReadLine();
 }
}

上述代码运行后,将在控制台窗口输出(C)。
A. A
B. B
C. A B
D.B A

3、针对下面的C#代码:

using System;
delegate void D(int i);
class P
{
   public static void Main()
   {
      V(new D(R));
   }
   public static void R(int t) 
   {
      V(21);
   }
   public static void V(int i)
   {
      Console.WriteLine(i.ToString());
      Console.ReadLine();
   }
}
     以下说法正确的是(B )
A.代码中存在错误,delegate void D(int i);不能定义在名称空间或者类之外
B.代码中存在错误,代码行V(new D(R));使用委托错误
C.程序正常运行,输出为0
D.程序正常运行,输出为21

4、在C#语法中,在派生类中对基类的虚函数进行重写,要求在派生类声明中使用(A)关键字。
A.override
B.new
C.static
D.virtual

5、阅读以下的C#代码:

public class TEApp
{
       public static void ThrowException(){
              throw new Exception();
        }
       public static void Main(){
              try{
                     Console.WriteLine("try");
                     ThrowException();
              }
              catch(Exception e){
                     Console.WriteLine("catch");
              }
              finally{
                     Console.WriteLine("finally");
              }
       }
}

请问代码运行结果是(A )。
A.try catch finally
B.try
C.try catch
D.try finally

6、以下C#代码,试图用来定义一个接口:

public interface IFile
{
      int A;
      int delFile(){
          A = 3;
      }
      void disFile();
}

关于以上的代码,以下描述错误的是 B
A、以上的代码中存在的错误包括:不能在接口中定义变量,所以int A代码行将出现错误;
B、以上的代码中存在的错误包括:接口方法delFile是不允许实现的,所以不能编写具体的实现函数;
C、代码void disFile();声明无错误,接口可以没有返回值;
D、代码void disFile();应该编写为void disFile(){};

7、常用的字符串判空串方法:(D)
A、bool isEmpty = (str.Length0)
B、bool isEmpty = (str
String.Empty)
C、bool isEmpty = (str==“”)
D、都一样

8、& 和&&的区别是(A)
A、&是算术”与”运算符,用于整形和bool;&&是条件”与”运算符,仅用于bool
B、&和&&是算术”与”运算符,用于整形和bool
C、&&是算术”与”运算符,用于整形和bool;&是条件”与”运算符,仅用于bool
D、&和&&是条件”与”运算符,仅用于bool

9、下列关于构造函数的描述正确的是(C)
A、构造函数可以声明返回类型。
B、构造函数不可以用private修饰
C、构造函数必须与类名相同
D、构造函数不能带参数

10、代码

int k1 = 10;
int k2 = 20;
if((++k1>10) || (k2++ > 20)){}

k1=?,k2=? ( B )
A、10,20
B、11,20
C、11,21
D、10,21

二、多选题

11、下面关于Task与协程之间的区别,以下描述正确的是(AC)
A、Task 可以使用 try cath 协程不行
B、Task 与协程都会出现线程同步的问题
C、Task可以返回多个值,协程不能
D、异步方法和协程方法可以使用out作为参数的描述

12、关于lua语言,以下描述正确的是(ACD)
A、lua是解析型语言
B、lua是面向过程的语言,无法实现面向对象设计
C、lua元表可以实现运算符的重载。
D、Unity支持Lua开发需要引入第三方库

13、关于Unity的能力以下描述正确的是(ABC)
A、Unity 中 MonoBehaviour方法的调用顺序是 Awake->Start->Enable->Update->LateUpdate
B、动画状态机(Animator)中trigger类型参数,动画状态切换后就会自动变成false
C、Unity 默认提供三个渲染管线,他们分别是Build-in、URP和HDRP
D、在URP渲染管线中,渲染透明物体的透明测试(Alpha Test) 晚于模版测试 (Stencil Test)

14、关于Unity性能优化,下面方法有作用的是(BCD)
A、把面数高的模型、拆分成多个模型可以降低渲染的消耗
B、内存过高时应该卸载当前不需要的资源
C、可以使用时间换空间的方法来降低对内存的依赖
D、使用硬件压缩格式的纹理可以降低纹理的内存消耗

part7

单选题:40
1、下列关于Unity中批处理的说法中正确的是 A
A、批处理优化主要是针对CPU的
B、静态批处理可以降低内存
C、动态批处理和批的条件是材质必须相同,但材质实例可以不一样
D、GPU instancing 的合批没有顶点限制

2、下列关于unity 的说法中正确的是 D
A、资源加载中,同名Assetbundle资源可以像Resources加载一样,重复加载多次
B、物理Update逻辑运算发生在普通Update逻辑计算之前
C、图片的纹理在内存中展开后大小与图片的物理大小的计算方式一样
D、若不考虑Unity 的着色器编译机制,片元着色器函数实现中,加减运算大部分情况下比乘除运算效率更高

3、下面选项中,哪项全是值类型: D
A、float、int、double、decimal、Float
B、struct、enum、delegate
C、object、string、interface
D、struct、enum、bool

4、下面代码输出结果是: B

using System;
using System.Collections;
using System.Collections.Generic;
namespace TestDemo
{
    class Program
    {
        class  U 
        {
            ~U()
            {
              Console.WriteLine("removed");    
            }
            public U()
            {
                throw new NotImplementedException();
            }
        }
        private static void Foo()
        {
            try
            {
                var u = new U();
                GC.Collect();
            }
            catch
            {
                Console.WriteLine("catch");
                GC.Collect();
            }
            finally
            {
                Console.WriteLine("finally");
            }
        }
        static void Main(string[] args)
        {
            Foo();
            GC.Collect();
        }
    }
}

A、
catch
removed
finally
B、
catch
finally
removed
C、
catch
finally

D、
removed
catch
finally

5、代码编写错误: A
A、
private const string CNo = string.Empty;

B、
private static readonly string SNo = string.Empty;

C、
private readonly string No = string.Empty;

D、
private readonly string DNo = default;

6、对于容器描述正确的是: C
A、List 的 Index 的复杂度是 O(0)
B、LinkedList 无法通过 Index 访问
C、Dictionary 的 Add 操作是 O(1)
D、List 的 Remove 操作是O(1)

7、关于struct 和 class的描述,正确的是:D
A、struct类型可以继承
B、class可以继承自struct
C、struct定义的类型可以继承class类型
D、struct定义的类型可以实现interface

8、关于lua下面描述不正确的是:D
A、lua可以实现继承
B、冒号的函数调用是把当前变量作为函数的第一参数
C、在xlua中C#的对象被描述成 userdata 类型
D、xlua不能调用C#的扩展方法

多选题:10
1、下列说法中正确的是 ABC
A、渲染管线顺序,渲染的应用阶段发生在几何阶段之前
B、应用阶段的主要任务是把 顶点数据、shader、贴图、材质球、灯光以及一些设置等等传入GPU的过程,这个过程叫做 DrawCall
C、顶点着色器的mvp变换中,Model矩阵的作用是实现模型的Transform仿射变换
D、深度测试发生在透明度测试之前

2、下列对于Unity 协程的说法中正确的是 ACD
A、协程运行在主线程
B、协程中如果不想继续执行后续逻辑,yield return null即可
C,在协程中可以开启另一个协程
D,在一个函数中运行协程时,协程程序的执行发生在此函数的调用栈

编程题: 50

回型遍历
给定一组二维整型数组,需要按照从左到右,从上到下,从右到左,从下倒上的顺序不断循环的螺旋(回形)遍历并生成一个一维数组。
比如传入
[
[ 1,2,3,4,5,6],
[18,19,20,21,22, 7],
[17,28,29,30,23,8],
[16,27,26,25,24,9],
[15,14,13,12,11,10]
]
处理完后返回[1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,24,25,26,27,28,29,30]

时间限制:1秒
内存限制:256M
语言限制:[Kotlin, Typescript, Python, C++, Groovy, Rust, C#, Java, Go, C, Scala, Javascript, Ruby, Swift, Php, Python 3]

示例1:
输入
[[1,2,3,4,5,6], [18,19,20,21,22,7], [17,28,29,30,23,8], [16,27,26,25,24,9], [15,14,13,12,11,10]] 输出
[1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,24,25,26,27,28,29,30]

字符串倒置:
给定一串字符将字符根据空格倒置过来:
比如输入:“student a am I”
输出: “I am a student”
时间限制:1秒
内存限制:256M
语言限制:[Kotlin, Typescript, Python, C++, Groovy, Rust, C#, Java, Go, C, Scala, Javascript, Ruby, Swift, Php, Python 3]
输入:
“TAL To Welcome”
输出:
“Welcome To TAL”

你可能感兴趣的:(Unity,笔试面试,c#,unity,开发语言)