Sleash

"Найти ближайший дом"

В теме 4 сообщения

Всем снова доброго времени суток. на этот раз хотел бы опубликовать свою версию поиска ближайшего свободного дома в PAWN.

Для начала приступим к enum, НО предупрежу, что у вас может быть не такие переменные и прошу их заменить, если будите копи-пастить мой код.

enum hInfo
{
Float:hEntrancex,   // Координаты входа по оси X
Float:hEntrancey,   // Координаты входа по оси Y
Float:hEntrancez,   // Координаты входа по оси Z
    hOwned,         // Имеет ли дом кто-то (ДА - 1 || НЕТ - 0)
};
new HouseInfo[2000][hInfo];

Дальше уже сам код поиска, сделаю в стиле команды, хотя изначально у меня это был пункт в /gps

CMD:nearesthouse(playerid)
{
    for(new Float:radi = 0.0; radi < 8485.0; radi++)
    {
        new n; // переменная "Найдено"
        for(new bp = 0; bp < sizeof(HouseInfo); bp++)
        {
            new Float:px, Float:py, Float:pz;
            GetPlayerPos(playerid, px,py,pz);
            if(PosToPoint(radi, px, py, pz, HouseInfo[bp][hEntranceX], HouseInfo[bp][hEntranceY], HouseInfo[bp][hEntranceZ]) && HouseInfo[bp][hOwned] == 0)
            {
                SetPlayerCheckpoint(playerid, HouseInfo[bp][hEntranceX], HouseInfo[bp][hEntranceY], HouseInfo[bp][hEntranceZ], 3.0);
                SendClientMessage(playerid, COLOR_WHITE, "{9ACD32}[GPS] {FFFFFF}Место на карте помечено красной меткой");
                n = 1// устанавливаем "найдено"
                break;
            }
        }
        if(n == 1// Если найден дом
        {
            break// закрываем цикл
        }
    }
}

Давайте разберём одну неясную тут вещь: переменная n;

Дело в том, что тут 2 цикла:

1. Берёт определённый радиус в округе игрока.

    1.2. Ищет незанятый дом в данном радиусе.

И если при нахождении такого дома мы можем с помощью break; выпрыгнуть из этого цикла. то уже первый цикл снова его запустит, и поэтому нам нужна переменная, которая будет сообщать вне цикла 1.2 найден ли был дом.

Теперь перейдём к стоку, который используется в команде (просто киньте его в конец мода):

stock PosToPoint(Float:radi, Float:oldposx, Float:oldposy, Float:oldposz, Float:x, Float:y, Float:z)
{
    new Float:tempposx, Float:tempposy, Float:tempposz;
    tempposx = (oldposx -x);
    tempposy = (oldposy -y);
    tempposz = (oldposz -z);
    if (((tempposx < radi) && (tempposx > -radi)) && ((tempposy < radi) && (tempposy > -radi)) && ((tempposz < radi) && (tempposz > -radi))) return true;
    return false;
}

 

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

@Sleash`en, задача поиска ближайшего объекта достаточно часто может пригождаться. То, что Вы привязались к домам, лишь вносит лишние зависимости. Наверное, стоило двигаться в более глобальном направлении: решение задачи поиска ближайшего объекта. И делать это непосредственно на примере системы домов.

 

Давайте посчитаем, сколько раз выполниться следующий участок кода:

new Float:px, Float:py, Float:pz;
GetPlayerPos(playerid, px,py,pz);
if(PosToPoint(radi, px, py, pz, HouseInfo[bp][hEntranceX], HouseInfo[bp][hEntranceY], HouseInfo[bp][hEntranceZ]) && HouseInfo[bp][hOwned] == 0)
{
    SetPlayerCheckpoint(playerid, HouseInfo[bp][hEntranceX], HouseInfo[bp][hEntranceY], HouseInfo[bp][hEntranceZ], 3.0);
    SendClientMessage(playerid, COLOR_WHITE, "{9ACD32}[GPS] {FFFFFF}Место на карте помечено красной меткой");
    n = 1// устанавливаем "найдено"
    break;
}

Если на сервере есть хотя бы 100 домов (то есть sizeof HouseInfo = 100), тогда указанный код будет исполняться (по крайней мере его часть) 8485*100=848500 раз. А зачем? Можно было бы обойтись 100 итерациями. Для этого было достаточно лишь получать расстояние между объектами, сохраняя минимальное, и потом установить точку на дом, расстояние до которого минимально. Что-то вроде такого:

Спойлер

stock getDistanceFromPointToPoint(Float:xA, Float:yA, Float:zA, Float:xB, Float:yB, Float:zB) {
    return (floatsqroot(floatpower(xB - xA, 2) + floatpower(yB - yA, 2) + floatpower(zB - zA, 2)));
}

stock setPosToNereastHouse(playerid) {
    new houseid = -1;
    new Float:minDistance = cellmax;

    new Float:x, Float:y, Float:z;
    GetPlayerPos(playerid, x, y, z);

    for (new i = 0, Float:distance; i < sizeof HouseInfo; i++) {
        distance = getDistanceFromPointToPoint(x, y, z, HouseInfo[i][hX], HouseInfo[i][hY], HouseInfo[i][hZ]);
        if (distance < minDistance) {
            houseid = i;
            minDistance = distance;
        }
    }

    if (houseid == -1) {
        return SendClientMessage(playerid, -1"К сожалению, ближайший дом не найден") & 0;
    }

    SetPlayerCheckpoint(playerid, HouseInfo[houseid][hX], HouseInfo[houseid][hY], HouseInfo[houseid][hZ]);
    SendClientMessage(playerid, -1"Точка до ближайшего дома установлена");
    return 1;
}

 

 

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

@Cawfee Вы правы, но я был "ограничен", точнее брал то, что было под рукой, к тому же и не думал о стоке о поиске расстояния.

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

При добавлении данной команды в свой мод результата никого не будет. Невозможно угадать как реализовано та или иная система у человека. Какой-то непонятный массив, непонятные координаты, производится цикл, когда можно обойтись и без него. Грубо говоря, очередная вырезка. Хочу посоветовать на будущее, никогда не создавайте переменные внутри цикла, к хорошему это уж точно не приведёт. Да и количество максимальных домов стоило вывести под константу, чтобы потом не мучатся и не искать каждую строку дабы изменить максимальное значение. ^_^

Отредактировано пользователем LoX_1337

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

Для публикации сообщений создайте учётную запись или авторизуйтесь

Вы должны быть пользователем, чтобы оставить комментарий

Создать учетную запись

Зарегистрируйте новую учётную запись в нашем сообществе. Это очень просто!


Регистрация нового пользователя

Войти

Уже есть аккаунт? Войти в систему.


Войти

  • Последние посетители   0 пользователей онлайн

    Ни одного зарегистрированного пользователя не просматривает данную страницу

  • Похожий контент

    • gooding
      От gooding


      Скачать файл PAR RP | Доработанный мод Cruela RP | Без багов
      Всем здравствуйте, уважаемые читатели. Я много времени потратил на доработку данного игрового мода, чтобы его выставить на форум.
      Лог доработок можно посмотреть внизу, а скриншоты игрового мода внизу и верху.
      Основа данного мода: Cruela RP
       
       

      Добавил gooding Добавлено 20.11.2024 Категория Моды Автор GOODING (Я)  
    • gooding
      От gooding
      Всем здравствуйте, уважаемые читатели. Я много времени потратил на доработку данного игрового мода, чтобы его выставить на форум.
      Лог доработок можно посмотреть внизу, а скриншоты игрового мода внизу и верху.
      Основа данного мода: Cruela RP
       
       

    • Max Luilchack
      От Max Luilchack
      Нужен человек который сможет сделать примерную карту Украины для сервера crmp mobile. Если есть желающие пишите в дс: lamarlml
    • Yashich
      От Yashich
      Возникла проблема, не работает команда /giverank Что здесь может работать не так? Заранее спасибо за ответ)
      Вот строки кода связанные с этой командой (мод Аризона)
       
      CMD:giverank(playerid, params[]) {     if(PI[playerid][pMember]<=0)return 0;     if(PI[playerid][pRank] < 9 && PI[playerid][pLeader] == 0) return SCM(playerid, COLOR_RED, !"Отказано в доступе!");     if(sscanf(params,"ud",params[0],params[1])) return SCM(playerid, COLOR_RED, !"Используйте: /giverank [id] [rank]");     if(!ProxDetectorS(8.0, playerid, params[0])) return SCM(playerid, COLOR_GREY, !"Вы должны находиться рядом с игроком!");     if(PI[params[0]][pLeader] > 0 ) return SCM(playerid, COLOR_RED, !"Вы не можете изменять ранг лидеру");     if(params[1] < 1) return SCM(playerid, COLOR_GREY, !"Нельзя повысить до этого ранга");     if(Clother[params[0]]!=-1)return 0;     if(GetPlayerVirtualWorld(playerid)!=GetPlayerVirtualWorld(params[0]))return 0;     if(!IsPlayerConnected(params[0])) return SCM(playerid, COLOR_GREY, !"Игрок не найден");     if(PI[playerid][pRank] <= PI[params[0]][pRank]) return SCM(playerid, COLOR_GREY, !"Вы не можете повысить/понизить этого игрока");     if(PI[playerid][pRank] == 9 && params[1] > 8) return SCM(playerid, COLOR_RED, !"Нельзя выдать ранг больше 8");     if(params[1] > 9) return SCM(playerid, COLOR_GREY, !"Нельзя повысить до этого ранга");     if(PI[params[0]][pMember] != PI[playerid][pMember]) return SCM(playerid, COLOR_GREY, !"Игрок находиться в другой организации!");     str_1[0] = EOS;     f(str_1,100,"Лидер %s %s до %i ранга",PN(playerid),PI[params[0]][pRank] > params[1] ? ("понизил"):("повысил"),params[1]);     SCM(params[0],COLOR_LIGHTBLUE,str_1);     f(str_1,100,"Вы %sи игрока %s до %i ранга",PI[params[0]][pRank] > params[1] ? ("понизил"):("повысил"),PN(params[0]),params[1]);     SCM(playerid,COLOR_LIGHTBLUE,str_1);     PI[params[0]][pRank] = params[1];     UpdatePlayerDataInt(params[0], "Rank", params[1]);     if(PI[params[0]][pSex] == 2) if(TeamDuty{params[0]}) SetPlayerSkinEx(params[0],GOrgSkins[GetTeamID(params[0])][params[1]]), TogglePlayerControllable(params[0],true);     else if(TeamDuty{params[0]}) SetPlayerSkinEx(params[0],MOrgSkins[GetTeamID(params[0])][params[1]]), TogglePlayerControllable(params[0],true);     return 1; }