MMO可见格子算法
看注释吧,写的很清楚了
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可见格子算法的更多相关文章
- java算法 蓝桥杯 格子位置
问题描述 输入三个自然数N,i,j (1<=i<=N,1<=j<=N),输出在一个N*N格的棋盘中,与格子(i,j)同行.同列.同一对角线的所有格子的位置. 输入格式 输入共三 ...
- 算法笔记_135:格子取数问题(Java)
目录 1 问题描述 2 解决方案 1 问题描述 有n*n个格子,每个格子里有正数或者0,从最左上角往最右下角走,只能向下和向右走,一共走两次(即从左上角往右下角走两趟),把所有经过的格子里的数加起 ...
- Java实现 蓝桥杯VIP 算法提高 格子位置
算法提高 格子位置 时间限制:1.0s 内存限制:512.0MB 问题描述 输入三个自然数N,i,j (1<=i<=N,1<=j<=N),输出在一个N*N格的棋盘中,与格子(i ...
- 蓝桥杯-算法训练--ALGO-8 操作格子
问题描述 有n个格子,从左到右放成一排,编号为1-n. 共有m次操作,有3种操作类型: 1.修改一个格子的权值, 2.求连续一段格子权值和, 3.求连续一段格子的最大值. 对于每个2.3操作输出你所求 ...
- [蓝桥杯]ALGO-8.算法训练_操作格子
题目描述: 问题描述 有n个格子,从左到右放成一排,编号为1-n. 共有m次操作,有3种操作类型: .修改一个格子的权值, .求连续一段格子权值和, .求连续一段格子的最大值. 对于每个2.3操作输出 ...
- 算法笔记_196:历届试题 剪格子(Java)
目录 1 问题描述 2 解决方案 1 问题描述 问题描述 如下图所示,3 x 3 的格子中填写了一些整数. +--*--+--+|10* 1|52|+--****--+|20|30* 1|**** ...
- 算法笔记_185:历届试题 格子刷油漆(Java)
目录 1 问题描述 2 解决方案 1 问题描述 问题描述 X国的一段古城墙的顶端可以看成 2*N个格子组成的矩形(如下图所示),现需要把这些格子刷上保护漆. 你可以从任意一个格子刷起,刷完一格,可 ...
- 算法笔记_064:蓝桥杯练习 操作格子(Java)
目录 1 问题描述 2 解决方案 1 问题描述 问题描述 有n个格子,从左到右放成一排,编号为1-n. 共有m次操作,有3种操作类型: 1.修改一个格子的权值, 2.求连续一段格子权值和, 3.求 ...
- SLG, 菱形格子的算法.(递归版
class GeoPoint{ public: int x; int y; public: bool operator == (const GeoPoint& p){ return p.x = ...
随机推荐
- django,python,svn_web
- Nginx for Windows 使用笔记
一.常见启动错误 1. "No mapping for the Unicode character exists in the target multi-byte code page&quo ...
- iOS钥匙串
钥匙串 苹果的"生态圈",钥匙串访问,使用 AES 256 加密算法,能够保证用户密码的安全 钥匙串访问SDK,是苹果在 iOS 7.0.3 版本以后公布的 钥匙串访问的接口是纯 ...
- JQ实现右下角scrollTop()事件
废话不多说,先贴代码 <script> $(document).ready(function(){ $("#id").hide(); $(function(){ // ...
- asp.net core 笔记
dnvm use 1.0.0-rc1-final -r clr 切换版本
- NetBeans如何关联两个项目
在实际工作中,有的项目需要关联其他项目 找到项目-->右键-->属性-->包含路径-->添加文件夹 添加你要的项目即可.
- jquery+css3实现3d滚动旋转
最近有个漂亮妹子一直在向我推销潭州的视频,BUT我以前就看了一次觉得挺水的,就对那个妹子表示吾孤高.你们那玩意没意义的很弱.然后那个漂亮妹子锲而不舍的对我发链接,这个效果会吗,这个幻灯会写吗...我也 ...
- 【转载】Unity 优雅地管理资源,减少占用内存,优化游戏
转自:星辰的<Unity3D占用内存太大的解决方法> 最近网友通过网站搜索Unity3D在手机及其他平台下占用内存太大. 这里写下关于Unity3D对于内存的管理与优化. Unity3D ...
- iOS 滑动性能优化
iOS 滑动性能优化 目录 一. 减少图层的Blend操作 1. UIView的背景色避免使用clearColor 2. 控件贴图避免使用带alpha的图片 3. UIImageView 使用时避免半 ...
- PHP安装laravel(win+linux)
作为一名不优秀的程序猿,忙碌的四月终于结束了,五一大假的最后一天,终于有时间来整理整理这段时间的收获了. 一.laravel介绍 首先看看http://www.sitepoint.com/网站做的一个 ...