看注释吧,写的很清楚了

using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace ZoneTest
{
public class Zone
{
public int mId;
public float Width;
public float Height;
public List<Zone> VisibleZoneList = new List<Zone>(); public Zone(int id, float width, float height)
{
mId = id;
Width = width;
Height = height;
} public void AddVisibleZone(Zone zone)
{
VisibleZoneList.Add(zone);
}
}
public class Scene
{
private List<Zone> mZoneList = new List<Zone>(); public Scene(float w,float h)
{
mSceneWidth = w;
mSceneHeight = h;
}
public const float ZONE_SIDE = 1.0f;
//场景宽高
private float mSceneWidth;
private float mSceneHeight; //格子列数和行数
private int mZoneColumnCount;
private int mZoneLineCount; public Zone GetZone(int id)
{
if (id < 0 || id > mZoneList.Count)
{
return null;
}
return mZoneList[id];
} //Zone示意图
/*
| mSceneWidth |
----------------- --
| | | | |ZONE_SIDE
-----------------
| | | | |
----------------- mSceneHeight
| | | | |
----------------- -- ----------------- line 0 ----------------- line 1 ----------------- line 2 ----------------- line 3 column0 column1 column2 column3
| | | | | | | | | | | | | | | | *
*/ public bool InitZone()
{
//计算场景内zone行和列
mZoneLineCount = (int)Math.Ceiling((double)mSceneHeight / (double)ZONE_SIDE);
mZoneColumnCount = (int)Math.Ceiling((double)mSceneWidth / (double)ZONE_SIDE); //创建zone
for (int i = 0; i < mZoneLineCount; i++)
{
for (int j = 0; j < mZoneColumnCount; j++)
{
int id = i * mZoneColumnCount + j;
Zone zone = new Zone(id, ZONE_SIDE, ZONE_SIDE);
mZoneList.Add(zone);
}
} //最大周围多少个格子被可见 MaxNearByZoneNumber*MaxNearByZoneNumber 个
const int MaxNearByZoneNumber = 3;
//可见格子的起始偏移
const int Offset = MaxNearByZoneNumber / 2; //把周围可见格子加入列表
/*
*    -------------
* |0 |1 |2 |
* -------------
* |3 |4 |5 |
* -------------
* |6 |7 |8 |
* -------------
*/ //0的可见格子为 0134
//4的可见格子为 0123456789
//7的可见格子为 345678 //遍历所有格子行
for (int i = 0; i < mZoneLineCount; i++)
{
//遍历所有格子列
for (int j = 0; j < mZoneColumnCount; j++)
{
//当前要判断可见性的格子id
int id = i * mZoneColumnCount + j;
Zone zone = mZoneList[id]; //遍历周围 MaxNearByZoneNumber*MaxNearByZoneNumber个格子
//判断[x,y]这个格子是否出界 //行
for (int n = 0; n < MaxNearByZoneNumber; n++)
{
int y = i - Offset + n;
if (y < 0 || y >= mZoneLineCount) continue;//y出界 //列
for (int m = 0; m < MaxNearByZoneNumber; m++)
{ int x = j - Offset + m;
if (x < 0 || x >= mZoneColumnCount) continue;//x出界 //算当前格子id
int visibleZoneId = y * mZoneColumnCount + x; //if (visibleZoneId == id) continue;//是自己格子也要加入,自己属于自己可见格子 zone.AddVisibleZone(mZoneList[visibleZoneId]);
}
} }
} return true;
} public void Print()
{
for (int i = 0; i < mZoneLineCount; i++)
{
string str = "";
//遍历所有格子列
for (int j = 0; j < mZoneColumnCount; j++)
{
str += (i * mZoneColumnCount + j).ToString()+"\t";
}
Debug.Print(str);
}
} public void PrintZone(int i)
{
Debug.Print("[{0}]---------------------",i);
var zone = mZoneList[i];
string str = "";
foreach(var z in zone.VisibleZoneList)
{ str+=z.mId.ToString()+"\t";
}
Debug.Print(str);
Debug.Print("[{0}]---------------------",i);
}
}
class Program
{
static void Main(string[] args)
{ Scene scene = new Scene(5,7);
scene.InitZone(); scene.Print(); int oldId = 5;
int newId = 10; Zone oldZone = scene.GetZone(oldId);
Zone newZone = scene.GetZone(newId); /*
//比较计算应该删除我的zone
List<Zone> deleteMeZone = new List<Zone>();
foreach (var zone in oldZone.VisibleZoneList)
{
if (!newZone.VisibleZoneList.Contains(zone))
{
deleteMeZone.Add(zone);
Debug.Print(zone.mId.ToString());
}
}
Debug.Print("-----------");
//比较计算应该创建我的
List<Zone> createMeZone = new List<Zone>();
foreach (var zone in newZone.VisibleZoneList)
{
if (!oldZone.VisibleZoneList.Contains(zone))
{
createMeZone.Add(zone);
Debug.Print(zone.mId.ToString());
}
}
*/ scene.PrintZone(4); scene.PrintZone(16); scene.PrintZone(28); scene.PrintZone(30); }
}
}

  

//当角色位置发生改变

protected void UpdateZone()
{
//安全检查
if (null == Scene || null == Zone)
return; //获得我的zone id
int zoneId = Scene.Pos2ZoneId(mPosition.X, mPosition.Y); Zone newZone = Scene.GetZone(zoneId);
if (null==newZone)
{
Logger.Fatal("[{0}] move out of scene[{1},{2}]", GetName(), GetPosition().X, GetPosition().Y);
}
//如果我的zone没变化
if (zoneId == Zone.Id) return; //比较计算应该删除我的zone
List<Zone> deleteMeZone = new List<Zone>();
foreach (var zone in Zone.VisibleZoneList)
{
if (!newZone.VisibleZoneList.Contains(zone))
{
deleteMeZone.Add(zone);
}
} //比较计算应该创建我的
List<Zone> createMeZone = new List<Zone>();
foreach (var zone in newZone.VisibleZoneList)
{
if (!Zone.VisibleZoneList.Contains(zone))
{
createMeZone.Add(zone);
}
} //发包删除我
foreach(var zone in deleteMeZone)
{
zone.PushAction2AllPlayer((player)=>{
player.Proxy.DeleteObj(ObjId, (int)ReasonType.VisibilityChanged);
},ObjId);
} //把该删除的通知我
if (GetObjType() == ObjType.PLAYER)
{
ObjPlayer player = this as ObjPlayer;
foreach (var zone in deleteMeZone)
{
zone.PushAction((obj) =>
{
if(obj!=this)
{
player.Proxy.DeleteObj(obj.ObjId, (int)ReasonType.VisibilityChanged);
}
});
} } //发包创建我
CreateObjMsg msg2Other = new CreateObjMsg();
ObjData data = DumpObjData(ReasonType.VisibilityChanged);
msg2Other.Data.Add(data);
foreach (var zone in createMeZone)
{
zone.PushAction2AllPlayer((player) =>
{
if (IsVisibleTo(player))
{
player.Proxy.CreateObj(msg2Other);
}
}, ObjId);
} //把周围所有人通知给我
if(GetObjType()==ObjType.PLAYER)
{
ObjPlayer player = this as ObjPlayer; CreateObjMsg msg2Me = new CreateObjMsg();
foreach (var zone in createMeZone)
{
zone.PushAction((obj) =>
{
if(obj!=this)
{
if (obj.IsVisibleTo(this))
{
msg2Me.Data.Add(obj.DumpObjData(ReasonType.VisibilityChanged));
}
}
});
}
player.Proxy.CreateObj(msg2Me);
}
Zone.RemoveObj(this);
newZone.AddObj(this);
SetZone(newZone); }

  

MMO可见格子算法的更多相关文章

  1. java算法 蓝桥杯 格子位置

    问题描述 输入三个自然数N,i,j (1<=i<=N,1<=j<=N),输出在一个N*N格的棋盘中,与格子(i,j)同行.同列.同一对角线的所有格子的位置. 输入格式 输入共三 ...

  2. 算法笔记_135:格子取数问题(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 有n*n个格子,每个格子里有正数或者0,从最左上角往最右下角走,只能向下和向右走,一共走两次(即从左上角往右下角走两趟),把所有经过的格子里的数加起 ...

  3. Java实现 蓝桥杯VIP 算法提高 格子位置

    算法提高 格子位置 时间限制:1.0s 内存限制:512.0MB 问题描述 输入三个自然数N,i,j (1<=i<=N,1<=j<=N),输出在一个N*N格的棋盘中,与格子(i ...

  4. 蓝桥杯-算法训练--ALGO-8 操作格子

    问题描述 有n个格子,从左到右放成一排,编号为1-n. 共有m次操作,有3种操作类型: 1.修改一个格子的权值, 2.求连续一段格子权值和, 3.求连续一段格子的最大值. 对于每个2.3操作输出你所求 ...

  5. [蓝桥杯]ALGO-8.算法训练_操作格子

    题目描述: 问题描述 有n个格子,从左到右放成一排,编号为1-n. 共有m次操作,有3种操作类型: .修改一个格子的权值, .求连续一段格子权值和, .求连续一段格子的最大值. 对于每个2.3操作输出 ...

  6. 算法笔记_196:历届试题 剪格子(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 如下图所示,3 x 3 的格子中填写了一些整数. +--*--+--+|10* 1|52|+--****--+|20|30* 1|**** ...

  7. 算法笔记_185:历届试题 格子刷油漆(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 X国的一段古城墙的顶端可以看成 2*N个格子组成的矩形(如下图所示),现需要把这些格子刷上保护漆. 你可以从任意一个格子刷起,刷完一格,可 ...

  8. 算法笔记_064:蓝桥杯练习 操作格子(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 有n个格子,从左到右放成一排,编号为1-n. 共有m次操作,有3种操作类型: 1.修改一个格子的权值, 2.求连续一段格子权值和, 3.求 ...

  9. SLG, 菱形格子的算法.(递归版

    class GeoPoint{ public: int x; int y; public: bool operator == (const GeoPoint& p){ return p.x = ...

随机推荐

  1. C#通用类型转换 Convert.ChangeType

    ];         object innerValue = ChangeType(value, innerType);         return Activator.CreateInstance ...

  2. XMLBEANS的使用总结

    在本文中,我们将详细了解最好的数据对象XMLBean.从传统角度来说,在Java应用程序中使用XML,就是在从 XML文档向Java导入数据的技术或从数据模型层向XML导出数据技术之间架起了一座桥梁. ...

  3. Memcached存储命令 - add

    Memcached add 命令用于将 value(数据值) 存储在指定的 key(键) 中. 如果 add 的 key 已经存在,则不会更新数据,之前的值将仍然保持相同,并且您将获得响应 NOT_S ...

  4. caller和callee

    我们先来看下caller. caller:返回一个对函数(该函数调用了当前函数)的引用. functionName.caller:functionName对象是所执行函数的名称. 说明 对于函数来说, ...

  5. Dirty Markup - 在线代码美化工具

    如果你需要一个帮助你规整书写混乱的代码的工具的话,我强烈推荐给你这个在线代码美化工具 - Dirty Markup.这个在线工具能够帮助你有效的处理HTML/HTML5,CSS和javascript代 ...

  6. iOS中assign,copy,retain之间的区别以及weak和strong的区别(面试)

    • copy: 用于希望保持一份传入值的拷贝,而不是值自身的情况,即把原来的对象完整的赋值到另外一地方,重新加载一内存区,一个地方变了不影响另一个地方的对象. • assign:  简单的直接赋值,相 ...

  7. mysql索引优化

    mysql 索引优化 >mysql一次查询只能使用一个索引.如果要对多个字段使用索引,建立复合索引. >越小的数据类型通常更好:越小的数据类型通常在磁盘.内存和CPU缓存中都需要更少的空间 ...

  8. Aptana 中去掉“Missing semicolon”提醒

    打开“窗口”下的“首选项” 然后找到“Aptana Studio”,在其下找到并点击Validation,在右侧窗口找到Javascript Syntax Validator,在下方的Options中 ...

  9. js读取修改创建txt文本类型文件(.ini)

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  10. apache 访问出现403 Forbidden

    在linux虚拟机的apache上新增一个虚拟目录/var/wordpress,想把理论网挂上去. 在配置文件httpd.conf中,把”Include conf/extra/httpd-vhosts ...