MMORPG大型游戏设计与开发(服务器 游戏场景 地图和区域)

地图的数据以及区域的信息是场景的重要组成部分,这些数据同时存在客户端和服务器,而且都是由编辑器生成的。那么保存的文件数据结构是怎样的?一张3D的场景地图又是怎样处理这些数据的?同时告诉大家这里同样只是讲的理论与设计,理论和设计往往都很空洞,但是却很灵活,需要靠每个人怎么运用。

一些图片

区域和格子

从上面的截图可以看出游戏场景其实是由格子来区分的,不管是矩形的格子还是其他形状的格子也好,一张地图不可能只有一个点(即多点组成一张地图)。在3D场景中似乎格子的位置总伴随着高度信息,所以让人感觉有些迷茫,其实我们可以将这些格子平铺成一张图。

如上面的最后一张图,是地形的剖面图,不过是横切面,如果我们看一下纵切面的话,就可以将3D的地图进行2D的转换。其实地图还是由一张平面图组成,只是多了Y轴的数据,也就是我们常说的地表高度。

地图数据

由指定格式的数据组成,在服务器和客户端的主要作用是用来寻路(点击地图走路,以及自动寻路)。

地图区域

一个场景一般情况下会有区域的划分,因为这些区域会有自己特殊的事件,如一个玩家加入该区域会产生某个事件,就像我们玩游戏的时候忽然触发了剧情一样,这都是区域的事件。

数据结构

有了指定的数据结构后,文件才能被正常的读取,而地图的数据一般是由编辑器生成,所以也必须规定文件的数据结构。

1、地图

1. 文件数据

武侠世界/天龙八部的场景地图数据格式为:[文件头][单元数据][单元数据]…… 单元数据的数量为地图横长* 地图纵长。

code.
typedef struct map_header_struct {
  int16_t flag; //文件标记,用来区分是不是地图数据文件
  int32_t xsize; //X方向大小 横长
  int32_t zsize; //Y方向大小 纵长
} map_header_t; //文件头信息

typedef struct map_unit_struct {
  int16_t flag; //标识信息
                //00000000|00000000
                //               ||_ WalkDisable  -是否禁止地面上行走的OBJ穿越  [0 可穿越  1不可穿越]
                //               |__ FlyDisable   -是否禁止空中飞行的OBJ穿越    [0 可穿越  1不可穿越]
  int8_t height; //高度
  int8_t reserved; //预留字段
} map_unit_t; //单元数据信息

2. 对象数据

整个对象也就是整张地图的数据,在武侠世界/天龙八部中采用的是左手坐标系,而在服务器其实用不着空间的坐标也就是Y轴的数据,将地图切割后就是一张平面的图,这张平面的图一般都是矩形。

X坐标 左为0、右为最大(也可以说是地图横长)

Y坐标 上为0、下为最大(也可以说是地图纵长)

code.
/* 完整的地图示意图 */
//                (0, z)   (x, z)
//                ___________
//   y  z         |         |
//   | /          |         |
//   |/           |         |
//   +-------x    |_________|
//
//               (0, 0)    (x, 0)

2、区域

地图划分成一个个格子,而组成区域的也是格子,以下为一个区域的基本数据。

1. RECT(矩形格子)

left 左、top 高度、right 右、bottom 底部。

2. 区域ID

3. 脚本ID

算法(选择排序)

1、简单选择排序

简单选择排序基本思想是将最小的数据移动到第一个位置并交换数据,比如有N个元素,一次将最小的元素移动到第一个位置并交换数据,那么剩下的就只剩下N-1个数据,将剩下的N-1的序列的数据最小的数据移动到该剩下序列的第一个位置,这样移动N次之后,序列便成了有序的序列。

code.

#include <stdio.h>
#include <stdint.h>
#include <inttypes.h>

/**
 * 简单选择排序基本思想是将最小的数据移动到第一个位置并交换数据,比如有N个元素,
 * 第一次将最小的元素移动到第一个位置并交换数据,那么剩下的就只剩下N-1个数据
 * 再将剩下的N-1的序列的数据最小的数据移动到该剩下序列的第一个位置,这样移动N次
 * 之后,序列便成了有序的序列。
 */

//简单选择排序
void selectsort(int32_t array[], int32_t length);
//数组打印
void displayarray(int32_t array[], int32_t length);

int32_t main(int32_t argc, char *argv[]) {
  int32_t array[] = {32, 1, 6, 9, 37, 88, 47, 35, 99};
  int32_t length = sizeof(array) / sizeof(array[0]);
  selectsort(array, length);
  printf("sort result: ");
  displayarray(array, length);
  return 0;
}

void selectsort(int32_t array[], int32_t length) {
  int32_t i, j, k;
  int32_t temp;
  //将第i个元素与第i+1...length个元素比较,将最小的元素放在第i个位置
  for (i = 0; i < length - 1; ++i) {
    j = i;
    for (k = i + 1; k < length; ++k) { //最小元素的序号为j
      if (array[k] < array[j])
        j = k;
    }
    if (j != i) { //如果序号i不等于j,则需要加序号i和序号j的元素交换
      temp = array[i];
      array[i] = array[j];
      array[j] = temp;
    }
    printf("the %d times sort result: ", i + 1);
    displayarray(array, length);
  }
}

void displayarray(int32_t array[], int32_t length) {
  int32_t i;
  for (i = 0; i < length; ++i)
    printf("%4d", array[i]);
  printf("\n");
}

result.

2、堆排序

堆排序算法实现比较复杂,它主要适用于大规模的数据排序,比如在10万个数据元素中找出前10个最小或是最大的元素,使用该算法效率最高。

code.

#include <stdio.h>
#include <stdint.h>
#include <inttypes.h>

/**
 * 堆排序算法实现比较复杂,它主要适用于大规模的数据排序,比如在10万个数据元素
 * 中找出前10个最小或是最大的元素,使用该算法效率最高。
 */

//数组打印
void displayarray(int32_t array[], int32_t length);
//调整array[pos1...pos2],使其成为一个大顶堆
void adjustheap(int32_t array[], int32_t pos1, int32_t pos2);
//创建大顶堆
void createheap(int32_t array[], int32_t length);
//利用堆排序算法对数组进行排序
void heapsort(int32_t array[], int32_t length);

int32_t main(int32_t argc, char *argv[]) {
  int32_t array[] = {32, 1, 6, 9, 37, 88, 47, 35, 99};
  int32_t length = sizeof(array) / sizeof(array[0]);
  printf("before sort: ");
  displayarray(array, length);
  heapsort(array, length);
  printf("after sort: ");
  displayarray(array, length);
  return 0;
}

void displayarray(int32_t array[], int32_t length) {
  int32_t i;
  for (i = 0; i < length; ++i)
    printf("%4d", array[i]);
  printf("\n");
}

void adjustheap(int32_t array[], int32_t pos1, int32_t pos2) {
  int32_t i;
  int32_t temp;
  temp = array[pos1]; //临时存放根节点
  for (i = 2 * pos1 + 1; i <= pos2; i *= 2 + 1) {
    if (i < pos2 && array[i] < array[i + 1]) //从关键字较大的子节点向下筛选
      ++i; //i为关键字较大的节点的下标
    if (temp > array[i]) break; //如果子节点的值小于根节点的值,则不进行交换
    array[pos1] = array[i];
    pos1 = i;
  }
  array[pos1] = temp; //将根节点插入到正确的位置
}

void createheap(int32_t array[], int32_t length) {
  int32_t i;
  for (i = length / 2 - 1; i >= 0; --i) //从序号length / 2 - 1开始建立大顶堆
    adjustheap(array, i, length - 1);
}

void heapsort(int32_t array[], int32_t length) {
  int32_t i;
  int32_t temp;
  createheap(array, length); //创建堆
  for (i = length - 1; i > 0; --i) { //将堆顶元素与最后一个元素交换,重新调整堆
    temp = array[0];
    array[0] = array[i];
    array[i] = temp;
    printf("the %d sort times: ", length - i);
    displayarray(array, length);
    adjustheap(array, 0, i - 1); //将array[0...i-1]调整为大顶堆
  }
}

result.

时间: 2024-08-05 19:37:01

MMORPG大型游戏设计与开发(服务器 游戏场景 地图和区域)的相关文章

MMORGP大型游戏设计与开发(游戏服务器 游戏场景 概述)

我们在玩游戏的时候,我们进入游戏后第一眼往往都是看到游戏世界中的场景,当然除了个别例外,因为那些游戏将游戏场景隐藏了起来,如文字游戏中的地点一样.既然我们接触了游戏世界的核心,那么作为核心的场景又包括哪些内容呢? 一张截图 场景组成 这里以天龙八部/武侠世界作为参考,其组成主要为核心.事件.区域.聊天管道.搜索机.寻路器.副本.掉落. 核心 每个系统都有自己的核心部分,核心一般作为整体的控制的作用,在场景中数据逻辑的处理便放在核心部分,比如场景中的所有对象以及每个对象的数据,网络的同步等等. 事

MMORGP大型游戏设计与开发(客户端架构 part9 of vegine)

时间在人们的生活中是多么重要的东西,如果打乱了时间,不知道这个时间会成什么样子.在客户端中,自然也有时间模块,因为不同的时间可能会处理不同的事情,特别是在追求高度自由化的同时,时间也成为了一个很重要的核心.没有时间,游戏世界将失去平衡,没有时间的游戏,会让人觉得乏味而单调.好了,我不用再强调时间多重要,想必大家已经明白了.比如游戏中的时辰变化,天气变化,这个是有时间的吗,答案是肯定的.在此次设计中,时间的接口中又提供了哪些方法呢? CODE 模块time 文件system.h /** * PAP

MMORGP大型游戏设计与开发(客户端架构 part1 of vegine)

重写有些核心接口的时候,其实遇到了许多的问题,甚至一度的想过要放弃,但是最终还是坚持了下来.在客户端提供的这些接口中,可以清晰的看到客户端所依赖的各种模块的支持,以及各自之间的一些关联.下面只是介绍了vengine(微引擎)接口的基础模块框架,所谓的接口即对象设计中常见的Interface,为一个框架提供了清晰的规范支持. VEGINE FRAMEWORK 功能实现 该接口已全部实现,具体的实例只需要继承接口封装即可.上图只为简单的模块介绍,其实每个接口都有每个接口其特别的用处,这一点会在接下的

MMORGP大型游戏设计与开发(客户端架构)

首先为所有等待的朋友说一声歉意,实在让大家等的太久.客户端的设计本来就是一个大的工程,而且工作的关系,也没有太多时间在这方面做研究.不过在私下有空的时间,我还是继续着这方面的研究,很遗憾没有用期望的ogre+cegui最新的版本作为开发,这方面原因是新的版本资料实在不多,对于没有什么经验的人来说实在是一大难事,所以最终选择了同天龙八部/武侠世界版本接近的源码作为开发.好了,废话不多说,今天好介绍的是客户端的基本构架,天龙八部/武侠世界的设计模式. CLIENT 功能实现 本次功能实现了vengi

MMORGP大型游戏设计与开发(客户端架构 part14 of vegine)

渲染在客户端中具有着至关重要的地位,试想我们玩游戏的第一感觉是什么就会明白了,良好的画面效果对客户端来说是多么的迫切.没有学习过opengl或是direct3d这些渲染API的朋友们也不必担心,而学习过这些接口的朋友们现在可以安心了,没有必要为了一个小小的渲染弄的头疼.因为渲染引擎可以帮你解决大部分你所能想到的渲染效果,这也正是渲染引擎所诞生的直接原因,那就是为了便利. CODE /** * PAP Engine ( -- ) * $Id system.h * @link-- for the c

MMORGP大型游戏设计与开发(客户端架构 part6 of vegine)

客户端的变量模块部分主要是将一些常用可变的值集中管理,如窗口的大小,是否开启音乐,音量的大小等等.这些变量通常会应该到客户端的操作,一般来说变量改变的时候会调用一个回调进行处理.下面我们就看看该模块的常用方法吧. CODE 文件system.h /** * PAP Engine ( -- ) * $Id system.h * @link -- for the canonical source repository * @copyright Copyright (c) 2013-2014 viti

MMORGP大型游戏设计与开发(客户端架构 part4 of vegine)

昨天是七夕,祝大家都过的快乐,希望这句迟到的问候不会造成大家心中的困扰.这一节讲到了前端比较重要的模块,性能以及调试异常模块.一个应用的性能往往是最核心的部分,就像人身体的各个器官一样,一小部分也不能马虎,否则你得到的只是你想象不到的苦果.在这里,我们封装了性能采集,调试输出.变量打印,以及异常收集.希望大家会对这方面有所了解与进步. 结构 CODE ax模块,文件profile.h /** * PAP Engine ( -- ) * $Id profile.h * @link -- for t

MMORGP大型游戏设计与开发(客户端架构 part11 of vegine)

从早年的无声电影到现在的逼真3D大片,人类在科技上可谓是一再突破.不知道有没有人经历过那无声的日子,没有声音的世界,咱们的耳朵也就失去了它本有的用途了.在游戏世界中,声音元素成了必不可少的一部分,一个没有声音的游戏现在可谓是太少见了,而且存活下来的希望自然不高.当前在游戏中,特别是3D游戏中,声音分为3D音效和平面音效.3D音效中主要是指环境音效,比如说某个地方的流水发出的声音,某片树林里充满的鸟叫声.平面音效即咱们感官的普通音效,背景音乐,自身的技能.UI声音. CODE 模块sound 文件

MMORGP大型游戏设计与开发(客户端架构 part7 of vegine)

我在讲述某个东西的时候总喜欢从简单的入手,然后从互相关联的地方联合讲解,因为时间关系所以没能讲的十分详细,这点引以为憾,希望得到大家的谅解.这一节讲述的是微引擎(vengine)比较简单的一个模块,那便是鼠标指针的模块,方法也提供的不多,相信大多数熟悉VC的朋友们已经掌握了这方面的知识,我也就不在此班门弄虎了. CODE 文件system.h /** * PAP Engine ( -- ) * $Id system.h * @link -- for the canonical source re

MMORGP大型游戏设计与开发(客户端架构 part3 of vegine)

无论在何处在什么地方,我们都或多或少的接触到数学知识.特别是在客户端中,从打开界面的那一刻起就有太多与数学扯上的关联,如打开窗口的大小,窗口的位置,窗口里面的元件对象,以及UI的坐标等等.而在进入游戏之后,不仅有这些坐标,还有了世界的坐标,以及场景坐标,还有粒子对象的各种属性值.但为什么要扩展ogre的数学库呢?就让我们看看有哪些类型的吧. CODE 文件math/base.h /** * PAP Engine ( -- ) * $Id math.h * @link -- for the can