一:什么是协同程序? 答:在主线程运行时同时开启另一段逻辑处理,来协助当前程序的执行。换句话说,开启协程就是开启一个可以与程序并行的逻辑。可以用来控制运动、序列以及对象的行为。
二:Unity3D中的碰撞器和触发器的区别?
答:碰撞器是触发器的载体,而触发器只是碰撞器身上的一个属性。
当Is Trigger=false
时,碰撞器根据物理引擎引发碰撞,产生碰撞的效果,可以调用OnCollisionEnter/Stay/Exit
函数;
当Is Trigger=true
时,碰撞器被物理引擎所忽略,没有碰撞效果,可以调用OnTriggerEnter/Stay/Exit
函数。
如果既要检测到物体的接触又不想让碰撞检测影响物体移动或要检测一个物件是否经过空间中的某个区域这时就可以用到触发器
三:物体发生碰撞的必要条件
答:两个物体都必须带有碰撞器Collider
,其中一个物体还必须带有Rigidbody
刚体。
- ArrayList会把所有插入其中的数据都当做Object来处理
- 装箱拆箱的操作
List
是接口,ArrayList
是一个实现了该接口的类,可以被实例化
五:请简述GC(垃圾回收)产生的原因,并描述如何避免?
答:GC回收堆上的内存
避免:
1)减少new产生对象的次数
2)使用公用的对象(静态成员)
3)将String
换为StringBuilder
六:反射的实现原理? 答:审查元数据并收集关于它的类型信息的能力。 实现步骤:
导入using System.Reflection
;
Assembly.Load("程序集");
//加载程序集,返回类型是一个Assembly
得到程序集中所有类的名称
代码语言:javascript复制foreach (Type type in assembly.GetTypes())
{
string t = type.Name;
}
Type type = assembly.GetType("程序集.类名");//获取当前类的类型
Activator.CreateInstance(type); //创建此类型实例
MethodInfo mInfo = type.GetMethod("方法名");//获取当前方法
mInfo.Invoke(null,方法参数);
- 能进行增量旋转
- 避免万向锁
- 给定方位的表达方式有两种,互为负(欧拉角有无数种表达方式)
八:如何安全的在不同工程间安全地迁移asset数据?三种方法
答:
1.将Assets
和Library
一起迁移
2.导出包package
3.用unity自带的assets Server功能
九:OnEnable
、Awake
、Start
运行时的发生顺序?哪些可能在同一个对象周期中反复的发生?
答:Awake
–>OnEnable
->Start
OnEnable
在同一周期中可以反复地发生!
十:MeshRender
中material
和sharedmaterial
的区别?
答:
修改sharedMaterial
将改变所有物体使用这个材质的外观,并且也改变储存在工程里的材质设置。
不推荐修改由sharedMaterial返回的材质。如果你想修改渲染器的材质,使用material替代。
十一:请简述ArrayList和List之间的主要区别。
十二:TCP/IP协议栈各个层次及分别的功能 答:网络接口层:这是协议栈的最低层,对应OSI的物理层和数据链路层,主要完成数据帧的实际发送和接收。 网络层:处理分组在网络中的活动,例如路由选择和转发等,这一层主要包括IP协议、ARP、ICMP协议等。 传输层:主要功能是提供应用程序之间的通信,这一层主要是TCP/UDP协议。 应用层:用来处理特定的应用,针对不同的应用提供了不同的协议,例如进行文件传输时用到的FTP协议,发送email用到的SMTP等。
十四:简述一下对象池,你觉得在FPS里哪些东西适合使用对象池? 对象池就存放需要被反复调用资源的一个空间,比如游戏中要常被大量复制的对象,子弹,敌人,以及任何重复出现的对象。
十六:移动相机动作在哪个函数里,为什么在这个函数里?
LateUpdate
,是在所有的Update
结束后才调用,比较适合用于命令脚本的执行。官网上例子是摄像机的跟随,都是所有的Update
操作完才进行摄像机的跟进,不然就有可能出现摄像机已经推进了,但是视角里还未有角色的空帧出现。
十八:请简述sealed
关键字用在类声明时与函数声明时的作用。
答:类声明时可防止其他类继承此类,在方法中声明则可防止派生类重写此方法。
二十:简述SkinnedMesh的实现原理
二十一: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图形的生成,将图形映射到相应的像素点上,对每个像素进行计算确定最终颜色并完成输出。
二十五:你用过哪些插件?
二十六: 使用Unity3d实现2d游戏,有几种方式?
答:
1.使用本身的GUI
、UGUI
2.把摄像机的Projection(投影)
值调为Orthographic(正交投影)
,不考虑z轴;
3.使用2d插件,如:2DToolKit
、NGUI
二十七:在物体发生碰撞的整个过程中,有几个阶段,分别列出对应的函数 三个阶段 答:
OnCollisionEnter
OnCollisionStay
OnCollisionExit
二十八:Unity3d的物理引擎中,有几种施加力的方式,分别描述出来 答:
rigidbody.AddForce
rigidbody.AddForceAtPosition
二十九:什么叫做链条关节?
答:Hinge Joint
,可以模拟两个物体间用一根链条连接在一起的情况,能保持两个物体在一个固定距离内部相互移动而不产生作用力,但是达到固定距离后就会产生拉力。
三十:物体自身旋转使用的函数?
答: Transform.Rotate()
三十一:Unity3d提供了一个用于保存和读取数据的类(PlayerPrefs),请列出保存和读取整形数据的函数
答:
PlayerPrefs.SetInt()
PlayerPrefs.GetInt()
三十二:Unity3d脚本从唤醒到销毁有着一套比较完整的生命周期,请列出系统自带的几个重要的方法。 答:Awake——>Start——>Update——>FixedUpdate——>LateUpdate——>OnGUI——>Reset——>OnDisable——>OnDestroy
三十三:物理更新一般放在哪个系统函数里?
答:
FixedUpdate
,每固定帧绘制时执行一次,和Update不同的是FixedUpdate是渲染帧执行,如果你的渲染效率低下的时候FixedUpdate调用次数就会跟着下降。
- FixedUpdate比较适用于物理引擎的计算,因为是跟每帧渲染有关。
- Update就比较适合做控制。
三十四:在场景中放置多个Camera
并同时处于活动状态会发生什么?
答:游戏界面可以看到很多摄像机的混合。
三十五:如何销毁一个UnityEngine.Object及其子类?
答: 使用Destroy()
方法;
三十六:请描述游戏动画有哪几种,以及其原理? 答:主要有关节动画、骨骼动画、单一网格模型动画(关键帧动画)。 关节动画:把角色分成若干独立部分,一个部分对应一个网格模型,部分的动画连接成一个整体的动画,角色比较灵活,Quake2中使用这种动画; 骨骼动画,广泛应用的动画方式,集成了以上两个方式的优点,骨骼按角色特点组成一定的层次结构,有关节相连,可做相对运动,皮肤作为单一网格蒙在骨骼之外,决定角色的外观; 单一网格模型动画由一个完整的网格模型构成,在动画序列的关键帧里记录各个顶点的原位置及其改变量,然后插值运算实现动画效果,角色动画较真实。
三十七:请描述为什么Unity3d中会发生在组件上出现数据丢失的情况 答: 一般是组件上绑定的物体对象被删除了
三十八:alpha blend
工作原理
答:Alpha Blend
实现透明效果,不过只能针对某块区域进行alpha操作,透明度可设。
三十九:写出光照计算中的diffuse的计算公式 答:diffuse = Kd x colorLight x max(N*L,0);Kd 漫反射系数、colorLight 光的颜色、N 单位法线向量、L 由点指向光源的单位向量、其中N与L点乘,如果结果小于等于0,则漫反射为0。
四十:LOD是什么,优缺点是什么?
答:LOD(Level of detail)
多层次细节,是最常用的游戏优化技术。它按照模型的位置和重要程度决定物体渲染的资源分配,降低非重要物体的面数和细节度,从而获得高效率的渲染运算。
四十一:两种阴影判断的方法、工作原理。 本影和半影: 本影:景物表面上那些没有被光源直接照射的区域(全黑的轮廓分明的区域)。 半影:景物表面上那些被某些特定光源直接照射但并非被所有特定光源直接照射的区域(半明半暗区域) 工作原理:从光源处向物体的所有可见面投射光线,将这些面投影到场景中得到投影面,再将这些投影面与场景中的其他平面求交得出阴影多边形,保存这些阴影多边形信息,然后再按视点位置对场景进行相应处理得到所要求的视图(利用空间换时间,每次只需依据视点位置进行一次阴影计算即可,省去了一次消隐过程)
四十二: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几个过程。
四十三:MipMap是什么,作用? 答:MipMapping:在三维计算机图形的贴图渲染中有常用的技术,为加快渲染进度和减少图像锯齿,贴图被处理成由一系列被预先计算和优化过的图片组成的文件,这样的贴图被称为MipMap。
四十四:请描述Interface与抽象类之间的不同 答:抽象类表示该类中可能已经有一些方法的具体定义,但接口就是公公只能定义各个方法的界面 ,不能具体的实现代码在成员方法中。 类是子类用来继承的,当父类已经有实际功能的方法时该方法在子类中可以不必实现,直接引用父类的方法,子类也可以重写该父类的方法。 实现接口的时候必须要实现接口中所有的方法,不能遗漏任何一个。
四十五:下列代码在运行中会产生几个临时对象? string a = new string("abc"); a = (a.ToUpper() "123").Substring(0, 2); 答:其实在C#中第一行是会出错的(Java中倒是可行)。应该这样初始化:string b = new string(new char[]{'a','b','c'});
四十六:下列代码在运行中会发生什么问题?如何避免?
代码语言:javascript复制List<int> ls = new List<int>(new int[] { 1, 2, 3, 4, 5 });
foreach (int item in ls)
{
Console.WriteLine(item * item);
ls.Remove(item);
}
答:会产生运行时错误,因为foreach是只读的。不能一边遍历一边修改。
四十七:.Net与Mono的关系? 答:mono是.net的一个开源跨平台工具,就类似java虚拟机,java本身不是跨平台语言,但运行在虚拟机上就能够实现了跨平台。.net只能在windows下运行,mono可以实现跨平台编译运行,可以运行于Linux,Unix,Mac OS等。
四十八:简述Unity3D支持的作为脚本的语言的名称 答:Unity的脚本语言基于Mono的.Net平台上运行,可以使用.NET库,这也为XML、数据库、正则表达式等问题提供了很好的解决方案。 Unity里的脚本都会经过编译,他们的运行速度也很快。这三种语言实际上的功能和运行速度是一样的,区别主要体现在语言特性上。 JavaScript、 C#、Boo
四十九:Unity3D是否支持写成多线程程序?如果支持的话需要注意什么? 答:仅能从主线程中访问Unity3D的组件,对象和Unity3D系统调用 支持:如果同时你要处理很多事情或者与Unity的对象互动小可以用thread,否则使用coroutine。 注意:C#中有lock这个关键字,以确保只有一个线程可以在特定时间内访问特定的对象
五十:Unity3D的协程和C#线程之间的区别是什么?
答:多线程程序同时运行多个线程 ,而在任一指定时刻只有一个协程在运行,并且这个正在运行的协同程序只在必要时才被挂起。
除主线程之外的线程无法访问Unity3D的对象、组件、方法。
Unity3d没有多线程的概念,不过unity也给我们提供了StartCoroutine
(协同程序)和LoadLevelAsync
(异步加载关卡)后台加载场景的方法。 StartCoroutine
为什么叫协同程序呢,所谓协同,就是当你在StartCoroutine
的函数体里处理一段代码时,利用yield
语句等待执行结果,这期间不影响主程序的继续执行,可以协同工作。
五十一:U3D中用于记录节点空间几何信息的组件名称,及其父类名称
答:Transform
父类是 Component
五十二:向量的点乘、叉乘以及归一化的意义? 答: 1)点乘描述了两个向量的相似程度,结果越大两向量越相似,还可表示投影 2)叉乘得到的向量垂直于原来的两个向量 3)标准化向量:用在只关系方向,不关心大小的时候
五十三:矩阵相乘的意义及注意点 答:用于表示线性变换:旋转、缩放、投影、平移、仿射 注意矩阵的蠕变:误差的积累
五十四:为何大家都在移动设备上寻求U3D原生GUI的替代方案 答:不美观,OnGUI很耗费时间,使用不方便
五十五:请简述如何在不同分辨率下保持UI的一致性 答:NGUI很好的解决了这一点,屏幕分辨率的自适应性,原理就是计算出屏幕的宽高比跟原来的预设的屏幕分辨率求出一个对比值,然后修改摄像机的size。
五十六:为什么dynamic font在unicode环境下优于static font 答:Unicode是国际组织制定的可以容纳世界上所有文字和符号的字符编码方案。 使用动态字体时,Unity将不会预先生成一个与所有字体的字符纹理。当需要支持亚洲语言或者较大的字体的时候,若使用正常纹理,则字体的纹理将非常大。
五十七:当一个细小的高速物体撞向另一个较大的物体时,会出现什么情况?如何避免? 答:穿透(碰撞检测失败)
五十八:请简述OnBecameVisible及OnBecameInvisible的发生时机,以及这一对回调函数的意义? 答:当物体是否可见切换之时。可以用于只需要在物体可见时才进行的计算。
五十九:什么叫动态合批?跟静态合批有什么区别? 答:如果动态物体共用着相同的材质,那么Unity会自动对这些物体进行批处理。动态批处理操作是自动完成的,并不需要你进行额外的操作。 区别:动态批处理一切都是自动的,不需要做任何操作,而且物体是可以移动的,但是限制很多。静态批处理:自由度很高,限制很少,缺点可能会占用更多的内存,而且经过静态批处理后的所有物体都不可以再移动了。
六十:简述StringBuilder
和String
的区别?
答:
String
是字符串常量。
StringBuffer
是字符串变量 ,线程安全。
StringBuilder
是字符串变量,线程不安全。
String
类型是个不可变的对象,当每次对String进行改变时都需要生成一个新的String对象,然后将指针指向一个新的对象,如果在一个循环里面,不断的改变一个对象,就要不断的生成新的对象,所以效率很低,建议在不断更改String
对象的地方不要使用String
类型。
StringBuilder
对象在做字符串连接操作时是在原来的字符串上进行修改,改善了性能。这一点我们平时使用中也许都知道,连接操作频繁的时候,使用StringBuilder
对象。
六十一:什么是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 完成一些底层操作
六十四: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)) //[1]
throw "Invalid argument(s)"; //[2]
char * strDestCopy=strDest; //[3]
while ((*strDest =*strSrc )!='