#include<bits/stdc++.h>
using namespace std;
// Định nghĩa kích thước lưới
#define ROW 9
#define COL 10
// Cấu trúc tọa độ (Point)
typedef pair<int, int> Pair;
// Cấu trúc lưu trữ thông tin của một ô (Node)
struct cell {
// Tọa độ cha (parent) để truy vết
// pair<int, int> parent;
int parent_i, parent_j;
// f = g + h
double f, g, h;
};
// --- Các hàm hỗ trợ ---
// Kiểm tra xem ô (row, col) có nằm trong lưới không
bool isValid(int row, int col) {
return (row >= 0) && (row < ROW) && (col >= 0) && (col < COL);
}
// Kiểm tra xem ô có phải là tường (bị chặn) không
bool isUnBlocked(const vector<vector<int>>& grid, int row, int col) {
// 1 = bị chặn, 0 = không bị chặn
return grid[row][col] == 0;
}
// Kiểm tra xem có phải là ô đích không
bool isDestination(int row, int col, Pair dest) {
return (row == dest.first && col == dest.second);
}
// Tính Heuristic (h) - dùng khoảng cách Manhattan
double calculateHValue(int row, int col, Pair dest) {
// return sqrt(pow(row - dest.first, 2) + pow(col - dest.second, 2)); // Euclidean
return abs(row - dest.first) + abs(col - dest.second); // Manhattan
}
// Truy vết đường đi từ đích về
void tracePath(const vector<vector<cell>>& cellDetails, Pair dest) {
cout << "Đường đi là: " << endl;
stack<Pair> path;
int row = dest.first;
int col = dest.second;
while (!(cellDetails[row][col].parent_i == row && cellDetails[row][col].parent_j == col)) {
path.push({row, col});
int temp_row = cellDetails[row][col].parent_i;
int temp_col = cellDetails[row][col].parent_j;
row = temp_row;
col = temp_col;
}
path.push({row, col}); // Thêm nút Start
while (!path.empty()) {
Pair p = path.top();
path.pop();
cout << "-> (" << p.first << ", " << p.second << ") ";
}
cout << endl;
}
// --- Hàm A* chính ---
// Kiểu dữ liệu cho Hàng đợi ưu tiên (OPEN list)
// pair<f_score, pair<x, y>>
typedef pair<double, Pair> pPair;
void aStarSearch(const vector<vector<int>>& grid, Pair src, Pair dest) {
// 1. Kiểm tra tính hợp lệ của Start và Goal
if (!isValid(src.first, src.second) || !isValid(dest.first, dest.second)) {
cout << "Start hoặc Goal không hợp lệ" << endl;
return;
}
if (!isUnBlocked(grid, src.first, src.second) || !isUnBlocked(grid, dest.first, dest.second)) {
cout << "Start hoặc Goal bị chặn" << endl;
return;
}
if (isDestination(src.first, src.second, dest)) {
cout << "Chúng ta đã ở đích" << endl;
return;
}
// 2. Khởi tạo
// Mảng 2D lưu chi tiết các ô
vector<vector<cell>> cellDetails(ROW, vector<cell>(COL));
// Mảng 2D cho CLOSE list (đã xử lý hay chưa)
vector<vector<bool>> closedList(ROW, vector<bool>(COL, false));
// Khởi tạo tất cả các ô
for (int i = 0; i < ROW; ++i) {
for (int j = 0; j < COL; ++j) {
cellDetails[i][j].f = FLT_MAX;
cellDetails[i][j].g = FLT_MAX;
cellDetails[i][j].h = FLT_MAX;
cellDetails[i][j].parent_i = -1;
cellDetails[i][j].parent_j = -1;
}
}
// Khởi tạo ô bắt đầu
int i = src.first, j = src.second;
cellDetails[i][j].g = 0.0;
cellDetails[i][j].h = calculateHValue(i, j, dest);
cellDetails[i][j].f = cellDetails[i][j].g + cellDetails[i][j].h;
cellDetails[i][j].parent_i = i;
cellDetails[i][j].parent_j = j;
// Hàng đợi ưu tiên (OPEN list)
// Sắp xếp theo f tăng dần (min-heap)
priority_queue<pPair, vector<pPair>, greater<pPair>> openList;
// Thêm nút bắt đầu vào OPEN list
openList.push({cellDetails[i][j].f, {i, j}});
bool foundDest = false;
// 3. Vòng lặp chính
while (!openList.empty()) {
// Lấy nút có f thấp nhất
pPair p = openList.top();
openList.pop();
// Lấy tọa độ
i = p.second.first;
j = p.second.second;
// Đánh dấu là đã xử lý
closedList[i][j] = true;
// 4. Duyệt các hàng xóm (4 hướng)
// (Thay đổi để duyệt 8 hướng nếu cần)
int dRow[] = {-1, 1, 0, 0};
int dCol[] = {0, 0, 1, -1};
for (int k = 0; k < 4; ++k) {
int n_i = i + dRow[k];
int n_j = j + dCol[k];
// Nếu hàng xóm hợp lệ
if (isValid(n_i, n_j)) {
// 5. Kiểm tra Đích
if (isDestination(n_i, n_j, dest)) {
// Đặt cha cho ô đích và truy vết
cellDetails[n_i][n_j].parent_i = i;
cellDetails[n_i][n_j].parent_j = j;
cout << "Tìm thấy đường đi!" << endl;
tracePath(cellDetails, dest);
foundDest = true;
return; // Thoát hàm
}
// Nếu hàng xóm chưa xử lý VÀ không bị chặn
else if (!closedList[n_i][n_j] && isUnBlocked(grid, n_i, n_j)) {
// 6. Tính toán chi phí
double gNew = cellDetails[i][j].g + 1.0; // Chi phí mỗi bước là 1
double hNew = calculateHValue(n_i, n_j, dest);
double fNew = gNew + hNew;
// 7. Cập nhật đường đi tốt hơn
if (cellDetails[n_i][n_j].f == FLT_MAX || cellDetails[n_i][n_j].f > fNew) {
// Thêm vào OPEN list
openList.push({fNew, {n_i, n_j}});
// Cập nhật chi tiết
cellDetails[n_i][n_j].f = fNew;
cellDetails[n_i][n_j].g = gNew;
cellDetails[n_i][n_j].h = hNew;
cellDetails[n_i][n_j].parent_i = i;
cellDetails[n_i][n_j].parent_j = j;
}
}
}
}
}
// 8. Không tìm thấy đường
if (!foundDest) {
cout << "Không tìm thấy đường đi" << endl;
}
}
// --- Hàm main để chạy ---
int main() {
/* Bản đồ
0 = Đường đi
1 = Tường (Obstacle)
*/
vector<vector<int>> grid = {
{0, 0, 1, 0, 0, 0, 0, 1, 0, 0},
{0, 0, 0, 0, 0, 0, 0, 1, 0, 0},
{0, 0, 0, 0, 1, 0, 0, 0, 0, 0},
{0, 0, 0, 0, 1, 0, 0, 1, 0, 0},
{0, 0, 1, 1, 1, 0, 0, 1, 0, 0},
{0, 0, 0, 0, 1, 0, 0, 1, 0, 0},
{0, 0, 0, 0, 1, 0, 0, 1, 0, 0},
{0, 0, 0, 0, 1, 0, 0, 1, 0, 0},
{0, 0, 0, 0, 0, 0, 0, 0, 0, 0}
};
// Điểm bắt đầu (Start)
Pair src = {8, 0};
// Điểm kết thúc (Goal)
Pair dest = {0, 9};
aStarSearch(grid, src, dest);
return 0;
}
#include<bits/stdc++.h>
using namespace std;

// Định nghĩa kích thước lưới
#define ROW 9
#define COL 10

// Cấu trúc tọa độ (Point)
typedef pair<int, int> Pair;

// Cấu trúc lưu trữ thông tin của một ô (Node)
struct cell {
    // Tọa độ cha (parent) để truy vết
    // pair<int, int> parent;
    int parent_i, parent_j;
    // f = g + h
    double f, g, h;
};

// --- Các hàm hỗ trợ ---

// Kiểm tra xem ô (row, col) có nằm trong lưới không
bool isValid(int row, int col) {
    return (row >= 0) && (row < ROW) && (col >= 0) && (col < COL);
}

// Kiểm tra xem ô có phải là tường (bị chặn) không
bool isUnBlocked(const vector<vector<int>>& grid, int row, int col) {
    // 1 = bị chặn, 0 = không bị chặn
    return grid[row][col] == 0;
}

// Kiểm tra xem có phải là ô đích không
bool isDestination(int row, int col, Pair dest) {
    return (row == dest.first && col == dest.second);
}

// Tính Heuristic (h) - dùng khoảng cách Manhattan
double calculateHValue(int row, int col, Pair dest) {
    // return sqrt(pow(row - dest.first, 2) + pow(col - dest.second, 2)); // Euclidean
    return abs(row - dest.first) + abs(col - dest.second); // Manhattan
}

// Truy vết đường đi từ đích về
void tracePath(const vector<vector<cell>>& cellDetails, Pair dest) {
    cout << "Đường đi là: " << endl;
    stack<Pair> path;

    int row = dest.first;
    int col = dest.second;

    while (!(cellDetails[row][col].parent_i == row && cellDetails[row][col].parent_j == col)) {
        path.push({row, col});
        int temp_row = cellDetails[row][col].parent_i;
        int temp_col = cellDetails[row][col].parent_j;
        row = temp_row;
        col = temp_col;
    }

    path.push({row, col}); // Thêm nút Start

    while (!path.empty()) {
        Pair p = path.top();
        path.pop();
        cout << "-> (" << p.first << ", " << p.second << ") ";
    }
    cout << endl;
}

// --- Hàm A* chính ---

// Kiểu dữ liệu cho Hàng đợi ưu tiên (OPEN list)
// pair<f_score, pair<x, y>>
typedef pair<double, Pair> pPair;

void aStarSearch(const vector<vector<int>>& grid, Pair src, Pair dest) {
    
    // 1. Kiểm tra tính hợp lệ của Start và Goal
    if (!isValid(src.first, src.second) || !isValid(dest.first, dest.second)) {
        cout << "Start hoặc Goal không hợp lệ" << endl;
        return;
    }

    if (!isUnBlocked(grid, src.first, src.second) || !isUnBlocked(grid, dest.first, dest.second)) {
        cout << "Start hoặc Goal bị chặn" << endl;
        return;
    }

    if (isDestination(src.first, src.second, dest)) {
        cout << "Chúng ta đã ở đích" << endl;
        return;
    }

    // 2. Khởi tạo
    // Mảng 2D lưu chi tiết các ô
    vector<vector<cell>> cellDetails(ROW, vector<cell>(COL));
    
    // Mảng 2D cho CLOSE list (đã xử lý hay chưa)
    vector<vector<bool>> closedList(ROW, vector<bool>(COL, false));

    // Khởi tạo tất cả các ô
    for (int i = 0; i < ROW; ++i) {
        for (int j = 0; j < COL; ++j) {
            cellDetails[i][j].f = FLT_MAX;
            cellDetails[i][j].g = FLT_MAX;
            cellDetails[i][j].h = FLT_MAX;
            cellDetails[i][j].parent_i = -1;
            cellDetails[i][j].parent_j = -1;
        }
    }

    // Khởi tạo ô bắt đầu
    int i = src.first, j = src.second;
    cellDetails[i][j].g = 0.0;
    cellDetails[i][j].h = calculateHValue(i, j, dest);
    cellDetails[i][j].f = cellDetails[i][j].g + cellDetails[i][j].h;
    cellDetails[i][j].parent_i = i;
    cellDetails[i][j].parent_j = j;

    // Hàng đợi ưu tiên (OPEN list)
    // Sắp xếp theo f tăng dần (min-heap)
    priority_queue<pPair, vector<pPair>, greater<pPair>> openList;

    // Thêm nút bắt đầu vào OPEN list
    openList.push({cellDetails[i][j].f, {i, j}});

    bool foundDest = false;

    // 3. Vòng lặp chính
    while (!openList.empty()) {
        // Lấy nút có f thấp nhất
        pPair p = openList.top();
        openList.pop();

        // Lấy tọa độ
        i = p.second.first;
        j = p.second.second;

        // Đánh dấu là đã xử lý
        closedList[i][j] = true;

        // 4. Duyệt các hàng xóm (4 hướng)
        // (Thay đổi để duyệt 8 hướng nếu cần)
        int dRow[] = {-1, 1, 0, 0};
        int dCol[] = {0, 0, 1, -1};

        for (int k = 0; k < 4; ++k) {
            int n_i = i + dRow[k];
            int n_j = j + dCol[k];

            // Nếu hàng xóm hợp lệ
            if (isValid(n_i, n_j)) {
                // 5. Kiểm tra Đích
                if (isDestination(n_i, n_j, dest)) {
                    // Đặt cha cho ô đích và truy vết
                    cellDetails[n_i][n_j].parent_i = i;
                    cellDetails[n_i][n_j].parent_j = j;
                    cout << "Tìm thấy đường đi!" << endl;
                    tracePath(cellDetails, dest);
                    foundDest = true;
                    return; // Thoát hàm
                }
                
                // Nếu hàng xóm chưa xử lý VÀ không bị chặn
                else if (!closedList[n_i][n_j] && isUnBlocked(grid, n_i, n_j)) {
                    
                    // 6. Tính toán chi phí
                    double gNew = cellDetails[i][j].g + 1.0; // Chi phí mỗi bước là 1
                    double hNew = calculateHValue(n_i, n_j, dest);
                    double fNew = gNew + hNew;

                    // 7. Cập nhật đường đi tốt hơn
                    if (cellDetails[n_i][n_j].f == FLT_MAX || cellDetails[n_i][n_j].f > fNew) {
                        
                        // Thêm vào OPEN list
                        openList.push({fNew, {n_i, n_j}});

                        // Cập nhật chi tiết
                        cellDetails[n_i][n_j].f = fNew;
                        cellDetails[n_i][n_j].g = gNew;
                        cellDetails[n_i][n_j].h = hNew;
                        cellDetails[n_i][n_j].parent_i = i;
                        cellDetails[n_i][n_j].parent_j = j;
                    }
                }
            }
        }
    }

    // 8. Không tìm thấy đường
    if (!foundDest) {
        cout << "Không tìm thấy đường đi" << endl;
    }
}

// --- Hàm main để chạy ---
int main() {
    /* Bản đồ
    0 = Đường đi
    1 = Tường (Obstacle)
    */
    vector<vector<int>> grid = {
        {0, 0, 1, 0, 0, 0, 0, 1, 0, 0},
        {0, 0, 0, 0, 0, 0, 0, 1, 0, 0},
        {0, 0, 0, 0, 1, 0, 0, 0, 0, 0},
        {0, 0, 0, 0, 1, 0, 0, 1, 0, 0},
        {0, 0, 1, 1, 1, 0, 0, 1, 0, 0},
        {0, 0, 0, 0, 1, 0, 0, 1, 0, 0},
        {0, 0, 0, 0, 1, 0, 0, 1, 0, 0},
        {0, 0, 0, 0, 1, 0, 0, 1, 0, 0},
        {0, 0, 0, 0, 0, 0, 0, 0, 0, 0}
    };

    // Điểm bắt đầu (Start)
    Pair src = {8, 0};

    // Điểm kết thúc (Goal)
    Pair dest = {0, 9};

    aStarSearch(grid, src, dest);

    return 0;
}
