一.Unity部分

1. Mask和RectMask2D 区别

 区别点1:
  mask:可以处理不规则图形遮罩 依赖Image组件 剪裁范围是Image大大小
  rectMask2D:只能处理矩形遮罩效果,不依赖Image组件,剪裁范围是他RectTransform的rect大小

 区别点2:
  mask:会在首尾(首=mask节点 尾=maks节点的子节点遍历完毕) –产生2个drawcall 并且Mask会增加一层的overdraw
  rectMask2D:rectMask2D 不产生drawcall

 区别点3:
  mask:多个maks之间如何符合合批的条件 就可以合批 (mask1的首和mask2的首 合批 mask1的尾和mask2的尾合批 首尾不能合批–需要同渲染层级depth 同材质,同图集)
  rectMask2D:多个rectMask2D之间不能合批

 相同点4:
  rectMask2D:mask和rectMask2D的内部和外部不能合批

2. 手指/鼠标如何判断向左向右向上向下

在平面组成的2D坐标系上,分为四个象限,再分为8块,

例如:当一个方向为东北或者西北上面时,记为“向上滑”

unity matarial matallic设置_值类型

if (m_Dir.y < m_Dir.x && m_Dir.y > -m_Dir.x)//向右
	    dir = FingerDragDir.right;
	else if (m_Dir.y > m_Dir.x && m_Dir.y < -m_Dir.x)//向左
	    dir = FingerDragDir.left;
	else if (m_Dir.y > m_Dir.x && m_Dir.y > -m_Dir.x)//向上
	    dir = FingerDragDir.up;
	else if (m_Dir.y < m_Dir.x && m_Dir.y < -m_Dir.x)//向下
	    dir = FingerDragDir.down;
	在1部分 m_Dir.y < m_Dir.x-->向右--》
	x=斜边*cos(度数)  y=斜边*sin(度数)
	度数在[0,45]之间 cos(度) < sin(度)--》y<x

3. 向量的点乘和叉乘

点乘:
 点积的计算结果是个标量,只有大小 没有方向,也称为标量积
 公式:UV=UxVx+UyVy+UzVz==》|U|*|v|*cos(θ)==》U·V=V·U
 几何意义:
  P向量在Q向量上的投影与Q向量的模的乘积,反映两个向量在方向的相似性,结果越大越相似
  U * V=0–》U和V相互垂直
  U * V>0–》U和V夹角小于90度(0,90),V在U的前方,方向基本相同
  U * V<0–》U和V夹角大于90度(90,180), V在U的后方,方向基本相反
 用途:
  ①计算2个向量之间的夹角 θ∈【0-180】

//如果u、v都是单位向量,u.v=cos(a)==>a:
    private float GetAngle(Vector3 u, Vector3 v)
    {
        var dot = Vector3.Dot(u.normalized, v.normalized);//计算角度
        var cross = Vector3.Cross(u.normalized, v.normalized);//叉乘判断顺时针【+】逆时针【-】
        // 通过反正弦函数获取 向量 u、b 夹角(默认为 弧度)  
        float radians = Mathf.Acos( dot);
        // 将弧度转换为 角度  
        var angle = radians * Mathf.Rad2Deg;

        if (cross.z < 0)//逆时针
            angle = -1 * angle;

        return angle;
    }//==>Vector3.SingleAngle
    
	// 通过向量直接获取两个向量的夹角(默认为 角度), 此方法范围 [0 - 180]  
	float angle = Vector3.Angle(a, b);

  ②判断敌人在自己的前方还是后方

   Vector3.Dot(transform.forward, target.position-transform.position)

   返回值为正(0<夹角<90)时,目标在自己的前方,反之(90<180)在自己的后方

  ③计算投影向量

   P向量在Q向量上的投影与Q向量的模的乘积

unity matarial matallic设置_主线程_02

叉乘:
 叉乘结果是向量,其模是两个向量为边的平行四边形面积,叉积只能用于3D向量,2D向量没有叉积
 公式:UV=(UxVy-UyVx)+(UxVz-UzVx)+(UyVz-UzVy)》|U|*|v|*sin(θ)**》U·V=-V·U
 计算方式:
  对2个3D向量 u、v 计算叉积 结果w 向量 同时垂直于 u和v
  左手拇指法则:提起左手,将拇指之前的 4个手指指向第一个向量u,然后朝着v的方向 沿角度 0<a<π 弯曲手指,此时拇指所指方向 就是w的方向
 用途:
  ①计算2个向量之间的夹角 θ∈【0-90】

//如果u、v都是单位向量,u.v=sin(a)==>a:
	private float GetAngle(Vector3 u, Vector3 v)  
	{  
	    // 计算 u、v 单位向量的叉积,得到夹角正弦值,|u.normalized|*|v.normalized|=1;  
	    var result = Vector3.Cross(u.normalized, v.normalized);  
	    // 通过反正弦函数获取 向量 u、b 夹角(默认为 弧度)  
	    float radians = Mathf.Asin(result);  
	    // 将弧度转换为 角度  
	    var angle = radians * Mathf.Rad2Deg;  
		return angle;
	}

  ②计算平面法线 【法线:垂直一个平面的向量】
  ③求转向【顺时针还是逆时针】

判断顺、逆时针,是在 2D 平面内的,所以需指定一个平面,下面以X、Z轴组成的平面为例,--》判断Y轴上的正负,即为“叉乘的正负值”,判断 b 在 a 的顺时针或者逆时针方向。
	Vector3 c = B.position - A.position;
    CrossTest(A.forward, c.normalized);
	private void CrossTest(Vector3 tmpA, Vector3 tmpB)
	{
	    Vector3 result = Vector3.Cross(tmpA, tmpB);
	    if(result.y> 0)
	        Debug.Log("tmpB在 tmpA的顺时针方向");
	    if (result.y == 0)
	        Debug.Log("tmpB在tmpA同向或反向");
	    if (result.y < 0)
	        Debug.Log("tmpB在tmpA的逆时针方向");
	}
	
	Vector3.Cross(transform.forward, target.position-transform.position).y
	返回值为正时,目标在自己的右方,反之在自己的左方

  ④计算出两向量组成的平行四边形面积

unity matarial matallic设置_unity面试题_03

根据平行四边形公式  S=a*h,h为高,a为底。a = |a|
	又因为h = |b|*sinθ,则 a*h = |a|*|b|*sinθ 即 |aXb| = S
  1. Unity的协程作用及用途
     ①协程是:主线程运行的同时开启另一段逻辑处理,来协助当前程序的执行,协程不是多线程。Unity的协程是在每帧结束之后去检测yield的条件是否满足
     ②主要用于:延时、WWW、等待异步加载、等待某个条件是否满足
     ③协程和多线程的区别:
      多线程是独立于主线程的,同一时间可以有多个多线程运行。协程只能存在于主线程,同一时间最多执行一个协程。
     ④协程和子程序(方法)区别:
      如加载文件,子程序会卡主主线程,直到加载完毕,才执行下面程序。协程类似异步,主线程执行加载文件,只加载0.1s然后挂起执行其他程序,主线程下次再来,恢复协程,以此往复,不卡主线。
  2. 防止UI穿透
/// <summary>
/// 检测是否点击3d对象【自己实现的 UI防穿透、移动端有效】
/// </summary>
public static bool CheckIsClick3DObj_Self()
{//EventSystem.current.IsPointerOverGameObject Unity自带的 移动端无效
    EventSystem eventSystem = EventSystem.current;
    PointerEventData eventData = new PointerEventData(eventSystem);
    if (Application.isMobilePlatform)//移动端
    {
        if (Input.touchCount > 0)
        {
            eventData.pressPosition = Input.GetTouch(0).position;
            eventData.position = Input.GetTouch(0).position;
        }
    }
    else//Unity_Editor 下或者电脑上能运行
    {
        eventData.pressPosition = Input.mousePosition;
        eventData.position = Input.mousePosition;
    }
    List<RaycastResult> list = new List<RaycastResult>();
    eventSystem.RaycastAll(eventData, list);
    //graphicRayCaster.Raycast(eventData, list);
    return list.Count > 0;
}
  1. transform.forward和Vector3.forword的区别:
  2. unity matarial matallic设置_主线程_04

  3. 7.2D和3D混合界面 的渲染顺序
  4. unity matarial matallic设置_主线程_05

二.C#部分

    常见的冒泡、选择、插入、希尔、归并、快速排序
    详细讲解

    值类型:
     数字类型:整形(u)byte (u)short (u)int (u)long 浮点类型double float decimal
     布尔类型:bool
     复合类型:enum struct
    引用类型:
     数组:Array、字符串:string、类:class、接口:interface、委托:delegate、object

    区别:
     1.值类型的变量和值在栈中分配,引用类型变量在栈中分配,引用实例在堆中分配
      譬如:Object A=new A(); 变量A存放在栈区,栈区A的值存放的是引用对象的地址,引用对象的内容是存放在堆
     2.值类型数据读取速度快,引用类型数据读取速度满
     3.栈中的内存自动回收,堆中的内存由.net的GC自动释放
     4.值类型继承System.TypeValue,引用类型继承System.Object
     5.值类型变量在赋值或传递参数时,是重新开辟空间,将值拷贝给新的变量
     6.引用类型变量的数据一修改,所有指向同一地址的变量的数据都会修改

     抽象类和接口都不能实例化,继承都必须实现所有抽象方法
     抽象类不是完全抽象可以有抽象和未抽象的方法,接口是完全抽象
     抽象类可以定义属性、构造函数,接口不能定义
     接口是多继承的,类是单继承的
     接口是不能用访问修饰符修饰,抽象类必须用public修饰

     类是引用类型,存储在堆中,内存GC回收,结构体是值类型,存储在栈中,内存自动回收。
     定义成员类可以初始化,结构体只有static\cost的可以,数据成员不行
     类的变量可以设置null,结构体不行
     类定义无参构造和私有无参构造,结构体只能定义有参构造,且必须为所有成员赋初始值
     类有析构函数,结构体没有
     类可以被继承,结构体不能被继承。
     结构体适用于数据量小,层级简单。类适用于数据量大或层级复杂比如多继承。

三.Lua部分

四.优化部分

    1.将同一画面图片放到同一图集中

    2.图片和文字尽量不要交叉,会产生多余drawcall(相同材质和纹理的UI元素是可以合并的)

    3.UI层级尽量不要重叠太多

    4.取消勾选不必要的射线检测RaycastTarget

    5.将动态的UI元素和静态的UI元素放在不同的Canvas中,减少canvas网格重构频率