原地址:http://www.cnblogs.com/88999660/archive/2013/03/14/2959439.html

官方文档对CullingMask的注释只是说了通过位移运算符,可以添加选中层。 假设要摄像机只显示第10层,11层,12层。写成:

camera.cullingMask = 1<<10 + 1<<11 + 1<<12;

但是为什么要这样觉得很奇怪。于是研究了一下。
通过print发现,随便一个层,它前面所有层的和,不会大于它自身的值。 这样的话,可以只用一个int变量保存多个类似布尔型的数据,节省内存。
然后我写了一个读取CullingMask值的函数。智商拙技。如果用二进制去做更快 传入值是CullingMask的值,输出的int数组是同时选中了哪几个层。
[2012/12/10补充]      最近又发现2个LayerMask的”隐藏方法“,帮助文档里应该没有: public static extern string LayerToName (int layer); public static extern int NameToLayer (string layerName); 可以在层名和层序号之间互相转换,都是静态方法直接用类名调用。


例子下载: CalcLayer_D.unitypackage (5 K) 下载次数:17 
当cullingMask为Nothing时,值是0。 当cullingMask设置为everything时,值是-1。这时如果有层要关闭,就减去这个层的值。 比如第8层的值是 256。那关闭第8层后的值是 -257[-1-(1<<9)] 因为设置everything时的数值比较特别,和算法没关系。所以代码我就不做修改了。如果要使用负数可以自己转换一下。

public int[] calcMask(int val)
{
int[] result = null; int flag1 = 0;
int flag2 = 0;
List<int> tmpLayers = new List<int>();
List<int> resultLayers = new List<int>(); for(int i=1; i<=val; i*=2)
{
tmpLayers.Add(i);
}//把val和首个层之间的层添加入数组.数组元素从小到大顺序
tmpLayers.Add(tmpLayers[tmpLayers.Count-1]*2); //使用递归计算选中的层
recursiveCalcMask(val, tmpLayers, ref resultLayers); for(int i=0; i< resultLayers.Count; i++)
{
resultLayers[i] = (int)Mathf.Log(resultLayers[i], 2);
//用Mathf.Log以2为底数。去求图层编号。
} result = new int[resultLayers.Count];
resultLayers.CopyTo(result); return result;
} public void recursiveCalcMask(int val, List<int> arr, ref List<int> outArr)
{
for(int i= 0; i<arr.Count; i++)
{
if(arr[i] == val)
{
outArr.Add(val);
break;
}
else if(arr[i] < val && arr[i+1] > val)
{
recursiveCalcMask(val - arr[i], arr, ref outArr);
outArr.Add(arr[i]);
}
}
}