본문 바로가기
알고리즘

[알고리즘] A*알고리즘(Astar Algorithm)

by MY블로그 2023. 7. 12.

Astar(A*) Algorithm ?

A* 알고리즘은 그래프 탐색 알고리즘 종류의 하나 입니다.

최단 경로 문제를 해결하는데 주로 사용되며 A*알고리즘은 효율적으로 경로를 찾습니다.

다른 탐색(Search)알고리즘보다 빠르고 정확한 결과를 얻을 수 있습니다.

A*알고리즘은 *휴리스틱(heuristic)함수를 사용하여 탐색 중 경로의 우선순위(비용)를 결정합니다.

주어진 출발 노드(node)에서부터 목표의 노드(node)까지 가는 최단 경로(최저비용)를 찾아냅니다.

 

*휴리스틱 함수

현재 상태에서 목표 상태까지의 예상 비용 또는 예측 값을 제공하는 함수입니다.

보통 경로 탐색에서 비용 함수로 사용됩니다.

 

출처 : https://github.com/ysshah95/Astar-Algorithm-for-ACME-Robotics

 

A* 알고리즘의 간단한 예제

#include <iostream>
#include <vector>
#include <queue>
#include <cmath>

using namespace std;

// 노드 구조체 정의
struct Node {
    int x, y;   // 노드의 위치
    int g, h;   // g: 시작점부터 현재 위치까지의 비용, h: 현재 위치부터 목적지까지의 예상 비용
    Node* parent;   // 경로 추적을 위한 부모 노드 포인터

    Node(int x, int y) : x(x), y(y), g(0), h(0), parent(nullptr) {}

    // 총 비용을 반환하는 함수
    int getCost() const {
        return g + h;
    }
};

// 휴리스틱 함수: 현재 위치부터 목적지까지의 예상 비용 계산
int calculateHCost(const Node& current, const Node& goal) {
    return abs(current.x - goal.x) + abs(current.y - goal.y);
}

// A* 알고리즘 함수
vector<Node*> astar(Node* start, Node* goal) {
    // 우선순위 큐를 사용하여 노드를 관리
    priority_queue<Node*, vector<Node*>, function<bool(Node*, Node*)>> openList(
        [](Node* a, Node* b) {
            return a->getCost() > b->getCost();
        }
    );

    // 시작 노드를 openList에 추가
    openList.push(start);

    while (!openList.empty()) {
        // openList에서 가장 비용이 적은 노드 선택
        Node* current = openList.top();
        openList.pop();

        // 목적지에 도달하면 경로 반환
        if (current == goal) {
            vector<Node*> path;
            while (current != nullptr) {
                path.push_back(current);
                current = current->parent;
            }
            return path;
        }

        // 현재 노드의 이웃 노드들을 탐색
        vector<Node*> neighbors;
        // 이웃 노드들을 찾고 neighbors에 추가하는 코드를 작성

        for (Node* neighbor : neighbors) {
            // 이웃 노드의 비용 계산
            int gCost = current->g + 1; // 이웃 노드로 이동할 때마다 비용 1 증가

            // 이웃 노드가 openList나 closedList에 없으면 추가
            if (gCost < neighbor->g || neighbor->parent == nullptr) {
                neighbor->g = gCost;
                neighbor->h = calculateHCost(*neighbor, *goal);
                neighbor->parent = current;
                openList.push(neighbor);
            }
        }
    }

    // 경로를 찾을 수 없는 경우 빈 벡터 반환
    return vector<Node*>();
}

int main() {
    // 그래프 노드 생성
    Node* start = new Node(0, 0);
    Node* goal = new Node(5, 5);
    // 그래프에 이웃 노드들을 추가하는 코드를 작성

    // A* 알고리즘 실행
    vector<Node*> path = astar(start, goal);

    // 경로 출력
    if (!path.empty()) {
        for (int i = path.size() - 1; i >= 0; --i) {
            cout << "(" << path[i]->x << ", " << path[i]->y << ") ";
        }
        cout << endl;
    } else {
        cout << "경로를 찾을 수 없습니다." << endl;
    }

    // 동적으로 할당한 노드들을 해제
    delete start;
    delete goal;
    // 이웃 노드들을 해제하는 코드를 작성

    return 0;
}

위의 코드는 'Node' 구조체를 정의 하고 'astar' 함수를 사용하여 알고리즘을 구현한 예제 입니다.

'astar' 함수는 시작 노드와 목적지 노드를 입력으로 받고난뒤 경로를 반환하도록 되어있습니다.

이웃 노드를 참색하며 탐색한 노드들의 비용을 계산한뒤 우선순위 큐의 'openList' 에 추가합니다.

주의할 점!

위의 예제는 단순화된 형태일뿐입니다. 코드작성의 목적에따라 수정이 필요하며 이웃 노드를 찾는 방법과 이웃노드를 추가하는 방식은 구체적인 문제에 따라 다를 수 있습니다.

 

2D FrameWork 에서 사용했던 A*

아래의 코드는 이전 2D 에서 사용하였던 A*알고리즘 구현 코드입니다.

접은 글로 첨부 하도록 합니다.

더보기
Main.cpp

아래의 코드는 메인에서 마우스 우클릭을 조건으로 길찾기 알고리즘이 실행되는 코드 구간입니다.
우클릭시 갈수있는 길이있어서 길의 컨테이너가 비어있지않을경우 움직이는 구조 입니다.

//우클릭햇을때
if (INPUT->KeyDown(VK_RBUTTON))
{
    //   출발점, 도착점
    Int2 sour, dest;
    //찾았는가?
    bool isFind;
    isFind = map->WorldPosToTileIdx(player->col->GetWorldPos(), sour);
    isFind &= map->WorldPosToTileIdx(INPUT->GetMouseWorldPos(), dest);

    if (isFind)
    {
        if (map->PathFinding(sour, dest, PlWay))
        {
            //길 사이즈만큼 반복
            for (int i = 0; i < PlWay.size(); i++)
            {
                cout << "Way" << i << ":" << PlWay[i]->idx.x <<
                    "," << PlWay[i]->idx.y << endl;
            }

            g = 0.0f;
            plSour = player->col->GetWorldPos();
            PlWay.pop_back(); //출발지 빼기
            plDest = PlWay.back()->Pos;
        }
    }
}

//가야될 길이 존재할 때
if (!PlWay.empty())
{
    player->col->SetWorldPos(Utility::Lerp(plSour, plDest, g));
    g += DELTA * 3.0f;

    //도착지를 지났을때
    if (g > 1.0f)
    {
        g = 0.0f;
        plSour = plDest;
        PlWay.pop_back(); //맨뒷길 빼기

        if (!PlWay.empty())
            plDest = PlWay.back()->Pos;
    }
}
아래의 헤더와 cpp 파일은 A*알고리즘이 실제 구현된 코드 입니다.
// 헤더
#pragma once
#define IMAGESLOT 5
enum TileState
{
	TILE_NONE,
	TILE_WALL,
	TILE_DOOR,
	TILE_TRAP,
	TILE_WATER,
	TILE_SIZE
};

class Tile
{
public:
	int         F, G, H;//타일 비용
	Int2        idx;    //타일 인덱스
	Tile*		P; //나를 갱신시킨 타일
	bool        isFind; //검사한적이 있는가?
	int         state;
	Vector2     Pos;

	void ClearCost();         //비용 초기화
	void ClacH(Int2 DestIdx); //H계산해라
	void ClacF(); //F계산해라
};

//typedef
//               타일주소, 비교값
using PTile = pair<Tile*, int>;

//비교함수
struct compare
{
	//연산자 오버로딩
	bool operator()(PTile& a, PTile& b)
	{
		//오름차순
		return a.second > b.second;
	}
};

class ObTileMap : public GameObject
{
protected:
	VertexTile*				vertices;
	ID3D11Buffer*			vertexBuffer;
	Int2					tileSize; // 4x4 10x10
	vector<vector<Tile>>    Tiles;

public:
	ObImage*				tileImages[IMAGESLOT];
	string					file;

public:
	ObTileMap();
	virtual ~ObTileMap();
	virtual void        ResizeTile(Int2 TileSize);
	virtual void        Load();
	virtual void        CreateTileCost();
public:
	void                Render() override;
	void                SetTile(Int2 TileIdx, Int2 FrameIdx, int ImgIdx = 0,
		int TileState = TILE_NONE, Color color = Color(0.5f, 0.5f, 0.5f, 0.5f));
	void RenderGui(Int2& GuiPickingIdx, int& ImgIdx);
	bool WorldPosToTileIdx(Vector2 WPos, Int2& TileIdx);
	void                Save();
	Int2				GetTileSize() { return tileSize; }

	int                 GetTileState(Int2 TileIdx);
	bool                PathFinding(Int2 sour, Int2 dest, OUT vector<Tile*>& way);
};




//cpp
#include "framework.h"

ObTileMap::ObTileMap()
{
    SafeRelease(vertexBuffer);
    SafeDeleteArray(vertices);
    for (int i = 0; i < 4; i++)
        SafeDelete(tileImages[i]);

    tileSize.x = 20;
    tileSize.y = 20;
    scale = Vector2(50.0f, 50.0f); //한 타일크기
    file = "map1.txt";

    //IMAGESLOT 수정 시 초기화 필수!
    tileImages[0] = new ObImage(L"Tile.png");
    tileImages[0]->maxFrame = Int2(8, 6);
    tileImages[1] = new ObImage(L"Tile2.png");
    tileImages[1]->maxFrame = Int2(11, 7);
    tileImages[2] = new ObImage(L"Tile3.png");
    tileImages[2]->maxFrame = Int2(40, 40);
    tileImages[3] = nullptr;
    tileImages[4] = nullptr;

    ResizeTile(tileSize);
}

ObTileMap::~ObTileMap()
{
    SafeRelease(vertexBuffer);
    SafeDeleteArray(vertices);

    for (int i = 0; i < IMAGESLOT; i++)
        SafeDelete(tileImages[i]);
}
//정점 찍는 함수
void ObTileMap::ResizeTile(Int2 TileSize)
{
    VertexTile* Vertices = new VertexTile[TileSize.x * TileSize.y * 6];

    //Init 세로
    for (int i = 0; i < TileSize.y; i++)
    {
        //가로
        for (int j = 0; j < TileSize.x; j++)
        {
            //y * size + x
            int tileIdx = TileSize.x * i + j;
            //타일좌표 ( tileSize.x * y좌표 + x좌표)
            //정점 좌표 ( tileSize.x * y좌표 + x좌표) * 6

            //0
            Vertices[tileIdx * 6].position.x = 0.0f + j;
            Vertices[tileIdx * 6].position.y = 0.0f + i;
            Vertices[tileIdx * 6].uv = Vector2(0.0f, 1.0f);
            //1                             
            Vertices[tileIdx * 6 + 1].position.x = 0.0f + j;
            Vertices[tileIdx * 6 + 1].position.y = 1.0f + i;
            Vertices[tileIdx * 6 + 1].uv = Vector2(0.0f, 0.0f);
            //2                             
            Vertices[tileIdx * 6 + 2].position.x = 1.0f + j;
            Vertices[tileIdx * 6 + 2].position.y = 0.0f + i;
            Vertices[tileIdx * 6 + 2].uv = Vector2(1.0f, 1.0f);
            //3
            Vertices[tileIdx * 6 + 3].position.x = 1.0f + j;
            Vertices[tileIdx * 6 + 3].position.y = 1.0f + i;
            Vertices[tileIdx * 6 + 3].uv = Vector2(1.0f, 0.0f);
            //4
            Vertices[tileIdx * 6 + 4].position.x = 1.0f + j;
            Vertices[tileIdx * 6 + 4].position.y = 0.0f + i;
            Vertices[tileIdx * 6 + 4].uv = Vector2(1.0f, 1.0f);
            //5
            Vertices[tileIdx * 6 + 5].position.x = 0.0f + j;
            Vertices[tileIdx * 6 + 5].position.y = 1.0f + i;
            Vertices[tileIdx * 6 + 5].uv = Vector2(0.0f, 0.0f);
        }
    }

    //Copy
    if (vertices)
    {
        Int2 Min = Int2(min(TileSize.x, tileSize.x), min(TileSize.y, tileSize.y));
        for (int i = 0; i < Min.y; i++)
        {
            for (int j = 0; j < Min.x; j++)
            {
                int SrcIdx = tileSize.x * i + j;
                int DestIdx = TileSize.x * i + j;
                for (int k = 0; k < 6; k++)
                {
                    Vertices[DestIdx * 6 + k] = vertices[SrcIdx * 6 + k];
                }
            }
        }
    }

    SafeDeleteArray(vertices);
    vertices = Vertices;
    tileSize = TileSize;
    SafeRelease(vertexBuffer);
    //CreateVertexBuffer
    {
        D3D11_BUFFER_DESC desc;
        desc = { 0 };
        desc.Usage = D3D11_USAGE_DEFAULT;
        desc.ByteWidth = sizeof(VertexTile) * tileSize.x * tileSize.y * 6;
        desc.BindFlags = D3D11_BIND_VERTEX_BUFFER;
        D3D11_SUBRESOURCE_DATA data = { 0 };
        data.pSysMem = vertices;
        HRESULT hr = D3D->GetDevice()->CreateBuffer(&desc, &data, &vertexBuffer);
        Check(hr);
    }
}

void ObTileMap::Render()
{
    if (!isVisible)return;
    GameObject::Render();

    for (int i = 0; i < IMAGESLOT; i++)
    {
        if (tileImages[i]) //주소가 잇다면
        {
            //SRV 접근 안됨 obimage 에서 친구 먹여주기
            D3D->GetDC()->PSSetShaderResources(i, 1, &tileImages[i]->SRV);
            D3D->GetDC()->PSSetSamplers(i, 1, &tileImages[i]->sampler);
        }
    }
    tileMapShader->Set();

    UINT stride = sizeof(VertexTile);
    UINT offset = 0;

    //버텍스버퍼 바인딩
    D3D->GetDC()->IASetVertexBuffers(0, 1, &vertexBuffer, &stride, &offset);
    D3D->GetDC()->IASetPrimitiveTopology(D3D11_PRIMITIVE_TOPOLOGY_TRIANGLELIST);
    D3D->GetDC()->Draw(tileSize.x * tileSize.y * 6, 0);
}

//uv 자르는 함수
void ObTileMap::SetTile(Int2 TileIdx, Int2 FrameIdx, int ImgIdx, int TileState, Color color)
{
    int tileIdx = tileSize.x * TileIdx.y + TileIdx.x;

    //0 1 5 UV.x 가 최소값
    vertices[tileIdx * 6 + 0].uv.x = FrameIdx.x / (float)tileImages[ImgIdx]->maxFrame.x;
    vertices[tileIdx * 6 + 1].uv.x = FrameIdx.x / (float)tileImages[ImgIdx]->maxFrame.x;
    vertices[tileIdx * 6 + 5].uv.x = FrameIdx.x / (float)tileImages[ImgIdx]->maxFrame.x;

    //2 3 4 UV.x 최대
    vertices[tileIdx * 6 + 2].uv.x = (FrameIdx.x + 1.0f) / (float)tileImages[ImgIdx]->maxFrame.x;
    vertices[tileIdx * 6 + 3].uv.x = (FrameIdx.x + 1.0f) / (float)tileImages[ImgIdx]->maxFrame.x;
    vertices[tileIdx * 6 + 4].uv.x = (FrameIdx.x + 1.0f) / (float)tileImages[ImgIdx]->maxFrame.x;

    //UV.y 최소
    vertices[tileIdx * 6 + 3].uv.y = FrameIdx.y / (float)tileImages[ImgIdx]->maxFrame.y;
    vertices[tileIdx * 6 + 1].uv.y = FrameIdx.y / (float)tileImages[ImgIdx]->maxFrame.y;
    vertices[tileIdx * 6 + 5].uv.y = FrameIdx.y / (float)tileImages[ImgIdx]->maxFrame.y;

    //UV.y 최대
    vertices[tileIdx * 6 + 2].uv.y = (FrameIdx.y + 1.0f) / (float)tileImages[ImgIdx]->maxFrame.y;
    vertices[tileIdx * 6 + 0].uv.y = (FrameIdx.y + 1.0f) / (float)tileImages[ImgIdx]->maxFrame.y;
    vertices[tileIdx * 6 + 4].uv.y = (FrameIdx.y + 1.0f) / (float)tileImages[ImgIdx]->maxFrame.y;

    //이미지 슬롯이 바뀔 경우
    for (int i = 0; i < 6; i++)
    {
        vertices[tileIdx * 6 + i].tileMapIdx = ImgIdx;
        vertices[tileIdx * 6 + i].color = color;
        vertices[tileIdx * 6 + i].tileState = TileState;
    }

    D3D->GetDC()->UpdateSubresource(vertexBuffer, 0, NULL, vertices, 0, 0);
}

void ObTileMap::RenderGui(Int2& GuiPickingIdx, int& ImgIdx)
{
    if (ImGui::InputInt("ImgIdx", &ImgIdx))
    {
        ImgIdx = Utility::Saturate(ImgIdx, 0, IMAGESLOT - 1);

        if (not tileImages[ImgIdx])
        {
            ImgIdx = 0;
        }
    }

    Int2 MF = tileImages[ImgIdx]->maxFrame;
    ImVec2 size;

    size.x = 300.0f / (float)MF.x;
    size.y = 300.0f / (float)MF.y;

    //텍스쳐 좌표
    ImVec2 LT, RB;
    int index = 0;
    for (UINT i = 0; i < MF.y; i++)
    {
        for (UINT j = 0; j < MF.x; j++)
        {
            if (j != 0)
            {
                //같은줄에 배치
                ImGui::SameLine();
            }

            //텍스쳐 좌표
            LT.x = 1.0f / MF.x * j;
            LT.y = 1.0f / MF.y * i;
            RB.x = 1.0f / MF.x * (j + 1);
            RB.y = 1.0f / MF.y * (i + 1);

            ImGui::PushID(index);
            if (ImGui::ImageButton((void*)tileImages[ImgIdx]->SRV, size, LT, RB))
            {
                GuiPickingIdx.x = j;
                GuiPickingIdx.y = i;
            }
            index++;
            ImGui::PopID();
        }
    }
}

bool ObTileMap::WorldPosToTileIdx(Vector2 WPos, Int2& TileIdx)
{
    WPos -= GetWorldPos();

    Vector2 tileCoord;
    tileCoord.x = WPos.x / scale.x;
    tileCoord.y = WPos.y / scale.y;

    if ((tileCoord.x < 0) or (tileCoord.y < 0) or
        (tileCoord.x >= tileSize.x) or (tileCoord.y >= tileSize.y))
    {
        return false;
    }

    TileIdx.x = tileCoord.x;
    TileIdx.y = tileCoord.y;

    return true;
}

void ObTileMap::Save()
{
    ofstream fout;
    string path = "../Contents/TileMap/" + file;
    //ios::out 쓰기 용으로 열겟다
    fout.open(path.c_str(), ios::out);

    //열였냐
    if (fout.is_open())
    {
        for (int i = 0; i < IMAGESLOT; i++)
        {
            fout << i << " ";
            if (tileImages[i])
            {
                string imgFile = "";
                imgFile.assign(tileImages[i]->file.begin(), tileImages[i]->file.end());

                fout << imgFile << " ";
                fout << tileImages[i]->maxFrame.x << " " << tileImages[i]->maxFrame.y << endl;
            }
            else
            {
                fout << "N" << endl;
            }
        }

        fout << "TileSize " << tileSize.x << " " << tileSize.y << endl;
        fout << "TileScale " << scale.x << " " << scale.y << endl;
        fout << "TilePosition " << GetWorldPos().x << " " << GetWorldPos().y << endl;

        for (int i = 0; i < tileSize.y; i++)
        {
            //가로
            for (int j = 0; j < tileSize.x; j++)
            {
                int tileIdx = tileSize.x * i + j;
                fout << j << " " << i << " "
                    << vertices[tileIdx * 6 + 1].uv.x << " " << vertices[tileIdx * 6 + 1].uv.y << " "
                    << vertices[tileIdx * 6 + 2].uv.x << " " << vertices[tileIdx * 6 + 2].uv.y << " "
                    << vertices[tileIdx * 6].color.x << " " << vertices[tileIdx * 6].color.y << " "
                    << vertices[tileIdx * 6].color.z << " " << vertices[tileIdx * 6].color.w << " "
                    << vertices[tileIdx * 6].tileMapIdx << " " << vertices[tileIdx * 6].tileState << endl;
            }
        }

        fout.close();
    }
}

void ObTileMap::Load()
{
    ifstream fin;
    string path = "../Contents/TileMap/" + file;
    fin.open(path.c_str(), ios::in);
    string temp;

    if (fin.is_open())
    {
        for (int i = 0; i < IMAGESLOT; i++)
        {
            SafeDelete(tileImages[i]);

            int idx; string Imgfile;
            fin >> idx;
            fin >> Imgfile;

            if (Imgfile != "N")
            {
                wstring wImgFile = L"";
                wImgFile.assign(Imgfile.begin(), Imgfile.end());
                tileImages[i] = new ObImage(wImgFile);
                fin >> tileImages[i]->maxFrame.x >> tileImages[i]->maxFrame.y;
            }
        }

        Int2 TileSize;
        fin >> temp >> TileSize.x >> TileSize.y;

        ResizeTile(TileSize);

        fin >> temp >> scale.x >> scale.y;

        Vector2 pos;
        fin >> temp >> pos.x >> pos.y;
        SetWorldPos(pos);

        for (int i = 0; i < tileSize.y; i++)
        {
            //가로
            for (int j = 0; j < tileSize.x; j++)
            {
                int temp; Vector2 MinUV, MaxUV; Color color; float tileMapIdx, tileMapState;

                fin >> temp >> temp >> MinUV.x >> MinUV.y >> MaxUV.x >> MaxUV.y
                    >> color.x >> color.y >> color.z >> color.w >> tileMapIdx >> tileMapState;

                int tileIdx = tileSize.x * i + j;

                vertices[tileIdx * 6].uv = Vector2(MinUV.x, MaxUV.y);
                vertices[tileIdx * 6 + 1].uv = Vector2(MinUV.x, MinUV.y);
                vertices[tileIdx * 6 + 2].uv = Vector2(MaxUV.x, MaxUV.y);
                vertices[tileIdx * 6 + 3].uv = Vector2(MaxUV.x, MinUV.y);
                vertices[tileIdx * 6 + 4].uv = Vector2(MaxUV.x, MaxUV.y);
                vertices[tileIdx * 6 + 5].uv = Vector2(MinUV.x, MinUV.y);

                for (int k = 0; k < 6; k++)
                {
                    vertices[tileIdx * 6 + k].color = color;
                    vertices[tileIdx * 6 + k].tileMapIdx = tileMapIdx;
                    vertices[tileIdx * 6 + k].tileState = tileMapState;
                }
            }
        }
        D3D->GetDC()->UpdateSubresource(vertexBuffer, 0, NULL, vertices, 0, 0);

        fin.close();
    }
}

int ObTileMap::GetTileState(Int2 TileIdx)
{
    int tileIdx = tileSize.x * TileIdx.y + TileIdx.x;
    return vertices[tileIdx * 6].tileState;
}

// 길찾기 알고리즘
bool ObTileMap::PathFinding(Int2 sour, Int2 dest, OUT vector<Tile*>& way)
{
    //둘중에 하나가 벽이면 갈 수 있는길이 없다.
    if (Tiles[dest.x][dest.y].state == TILE_WALL ||
        Tiles[sour.x][sour.y].state == TILE_WALL)
    {
        return false;
    }

    //기존에 있던 길은 전부 비운다.
    way.clear();

    //출발지 목적지 같으면
    if (sour == dest)
    {
        return false;
    }

    //타일 비용 초기화
    for (int i = 0; i < tileSize.x; i++)
    {
        for (int j = 0; j < tileSize.y; j++)
        {
            Tiles[i][j].ClearCost();
        }
    }

    //             원소타입, 컨테이너타입, 비교함수
    priority_queue<PTile, vector<PTile>, compare> List;

    //리스트에 출발지를 추가
    Tile* pTemp = &Tiles[sour.x][sour.y];
    pTemp->G = 0; //출발지 현재비용은 0
    pTemp->ClacH(dest);//목적지까지 예상비용 만들기
    pTemp->ClacF(); //총예상비용 만들기

    //          주소 , 총 예상비용
    List.push({ pTemp ,pTemp->F });

    //도착점까지 비용이 생길때 까지 반복
    while (1)
    {
        //꺼내야될 F값이 더이상 없을때
        if (List.empty())
        {
            return false;
        }

        //탐색노드 받아오기
        PTile Temp = List.top();

        //탐색목록에서 제외
        Temp.first->isFind = true;

        //맨윗값 빼버리기
        List.pop();

        //맨윗값이 도착점이면 종료(길찾기 종료조건)
        if (Temp.first->idx == dest)
        {
            break;
        }

        //인접 타일 비용검사
        vector<Int2> LoopIdx;

        //왼쪽타일이 존재할때
        if (Temp.first->idx.x > 0)
        {
            LoopIdx.push_back(Int2(Temp.first->idx.x - 1,
                Temp.first->idx.y));
        }
        //아래쪽타일이 존재할때
        if (Temp.first->idx.y > 0)
        {
            LoopIdx.push_back(Int2(Temp.first->idx.x,
                Temp.first->idx.y - 1));
        }

        //오른쪽타일이 존재할때
        if (Temp.first->idx.x < tileSize.x - 1)
        {
            LoopIdx.push_back(Int2(Temp.first->idx.x + 1,
                Temp.first->idx.y));
        }

        //위쪽타일이 존재할때
        if (Temp.first->idx.y < tileSize.y - 1)
        {
            LoopIdx.push_back(Int2(Temp.first->idx.x,
                Temp.first->idx.y + 1));
        }

        //상하좌우타일 비용검사
        for (int i = 0; i < LoopIdx.size(); i++) //LoopIdx.size() 4개
        {
            Tile* loop = &Tiles[LoopIdx[i].x][LoopIdx[i].y];

            //벽이 아닐때
            if (loop->state != TILE_WALL)
            {
                //예상비용 만들기
                loop->ClacH(dest);

                //현재 가지고있는 비용이 클때
                if (loop->G > Temp.first->G + 10)
                {
                    //비용갱신
                    loop->G = Temp.first->G + 10;
                    loop->ClacF();

                    //누구로부터 갱신인지 표시
                    loop->P = Temp.first;

                    //이 타일이 찾은적 없던 타일이면 리스트에 추가
                    if (!loop->isFind)
                    {
                        List.push({ loop,loop->F });
                    }
                }
            }
        }
        LoopIdx.clear();
    }

    //도착지가 설정이 되었을때
    Tile* p = &Tiles[dest.x][dest.y]; //이게 도착점
    //dest 3,0 2,0 1,0 0,0
    while (1)
    {
        way.emplace_back(p);

        p = p->P;

        if (p == nullptr)
        {
            break;
        }
    }
    return true;
}

void ObTileMap::CreateTileCost()
{
    for (int i = 0; i < Tiles.size(); i++)
    {
        Tiles[i].clear();
    }
    Tiles.clear();

    Tiles.resize(tileSize.x);

    for (int i = 0; i < Tiles.size(); i++)
    {
        Tiles[i].resize(tileSize.y);
    }

    Vector2 half = scale * 0.5f;
    Update();
    for (int i = 0; i < tileSize.x; i++)
    {
        for (int j = 0; j < tileSize.y; j++)
        {
            Tiles[i][j].idx = Int2(i, j);
            Tiles[i][j].state = GetTileState(Tiles[i][j].idx);
            Tiles[i][j].Pos.x = i * scale.x + GetWorldPos().x + half.x;
            Tiles[i][j].Pos.y = j * scale.y + GetWorldPos().y + half.y;
        }
    }
}

void Tile::ClearCost()
{
    F = G = H = INT_MAX;

    P = nullptr;
    isFind = false;
}

void Tile::ClacH(Int2 DestIdx)
{
    int tempX = abs(idx.x - DestIdx.x) * 10;
    int tempY = abs(idx.y - DestIdx.y) * 10;

    H = tempX + tempY;
}

void Tile::ClacF()
{
    F = G + H;
}

A*알고리즘을 보기쉽게 정리해놓은 블로그를 참고하도록 합니다.

https://recall.tistory.com/40

 

A* 알고리즘(A star algorithm) grid map 개념 및 구현

A* algorithm이란? A* 알고리즘(A* star algorithm)은 주어진 출발 노드(node)에서부터 목표 노드(node)까지 가는 최단 경로를 찾아내는 그래프 탐색 알고리즘 중 하나이다. 주어진 지도(map)에서 출발 지점부

recall.tistory.com

 

댓글