C解法

输入解析:

每个灯光的编号 id 和它的两个角点坐标 (x1, y1) 和 (x2, y2),程序需要通过这些数据计算出灯光的中心坐标 (x, y) 和半径 r。
排序规则:

首先根据 y 坐标对所有灯光进行排序。
然后根据灯光的 y 坐标差值判断是否处于同一行,如果是,则按 x 坐标排序;如果不是,则将当前行的灯光按 x 坐标输出。
分组灯光:

程序将灯光分为几组:如果灯光 y 坐标差值小于等于当前灯光的半径,则认为它们在同一行。
对每行内的灯光进行 x 坐标排序,然后输出它们的编号

#include 
#include 

// 定义灯光的结构体,包含灯光编号、x、y 坐标及半径
typedef struct {
    int id;  // 灯光编号
    int x;   // 灯光中心的 x 坐标
    int y;   // 灯光中心的 y 坐标
    int r;   // 灯光的半径
} Light;

// 比较函数:按 y 坐标升序排序
int compareY(const void* a, const void* b) {
    Light* lightA = (Light*)a;
    Light* lightB = (Light*)b;
    return lightA->y - lightB->y;
}

// 比较函数:按 x 坐标升序排序
int compareX(const void* a, const void* b) {
    Light* lightA = (Light*)a;
    Light* lightB = (Light*)b;
    return lightA->x - lightB->x;
}

// 确定并输出灯光的排序顺序
void determineOrder(Light lights[], int n) {
    // 根据 y 坐标对灯光进行排序
    qsort(lights, n, sizeof(Light), compareY);

    // 定义一个数组来保存当前行的灯光
    Light sameRow[n];
    int sameRowCount = 0;

    // 将第一个灯光设为基准灯光
    Light reference = lights[0];
    sameRow[sameRowCount++] = reference;

    // 遍历剩余的灯光,判断是否和基准灯光在同一行
    for (int i = 1; i < n; i++) {
        Light current = lights[i];
        // 如果当前灯光和基准灯光在同一行,y 坐标差值 <= 基准灯光的半径
        if (current.y - reference.y <= reference.r) {
            sameRow[sameRowCount++] = current;  // 加入当前行
        }
        else {
            // 如果当前灯光不在同一行,按 x 坐标排序当前行的灯光
            qsort(sameRow, sameRowCount, sizeof(Light), compareX);

            // 输出当前行的所有灯光编号
            for (int j = 0; j < sameRowCount; j++) {
                printf("%d ", sameRow[j].id);
            }
            sameRowCount = 0;  // 清空当前行

            // 更新基准灯光为当前灯光,开始新的行
            reference = current;
            sameRow[sameRowCount++] = reference;
        }
    }

    // 如果最后一行灯光还没有输出,进行输出
    if (sameRowCount > 0) {
        // 按 x 坐标排序
        qsort(sameRow, sameRowCount, sizeof(Light), compareX);
        // 输出这一行的所有灯光编号
        for (int j = 0; j < sameRowCount; j++) {
            printf("%d ", sameRow[j].id);
        }
    }

    // 换行,结束输出
    printf("\n");
}

// 主函数
int main() {
    // 读取灯光的数量
    int n;
    scanf("%d", &n);

    // 定义一个数组来存储灯光信息
    Light lights[n];

    // 读取每个灯光的信息
    for (int i = 0; i < n; i++) {
        int id, x1, y1, x2, y2;
        // 读取灯光编号和两个角的坐标
        scanf("%d %d %d %d %d", &id, &x1, &y1, &x2, &y2);
        lights[i].id = id;
        // 计算灯光的中心坐标和半径
        lights[i].x = (x1 + x2) / 2;
        lights[i].y = (y1 + y2) / 2;
        lights[i].r = (x2 - x1) / 2;
    }

    // 调用函数进行排序并输出结果
    determineOrder(lights, n);
    return 0;
}

 class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">

JS解法

灯光根据 y 坐标排序,若 y 坐标差距小于等于某个基准灯光的半径 r,则认为这些灯光在同一行。
同一行内的灯光按照 x 坐标进行排序,然后输出它们的编号。
如果灯光属于不同的行,则按 y 坐标分组并按 x 坐标排序。

const readline = require("readline");

// 创建读取接口
const rl = readline.createInterface({
    input: process.stdin,
    output: process.stdout,
});

let data = []; // 存储输入数据
let numberOfLights; // 灯光数量

// 监听每行输入
rl.on("line", (line) => {
    data.push(line); // 将每行输入存储到 data 数组中

    if (data.length === 1) {
        numberOfLights = parseInt(data[0]); // 第一行是灯光数量,转换为数字
    }

    // 当读取到足够的输入数据后,开始处理
    if (numberOfLights && data.length === numberOfLights + 1) {
        data.shift(); // 删除第一行(灯光数量)

        // 处理灯光数据
        const lights = data.map((str) => {
            const [id, x1, y1, x2, y2] = str.split(" ").map(Number);
            return { id, midX: (x1 + x2) / 2, midY: (y1 + y2) / 2, rad: (x2 - x1) / 2 };
        });

        // 输出处理结果
        console.log(computeOrder(lights));

        // 清空数据,准备下一个测试用例
        data = [];
    }
});

// 计算灯光的顺序
function computeOrder(lightArray) {
    // 按照 midY (y 坐标) 排序
    lightArray.sort((lightA, lightB) => lightA.midY - lightB.midY);

    const finalOrder = []; // 存储最终的灯光编号顺序
    let rowLights = []; // 存储当前行的灯光
    let currentLight = lightArray[0]; // 基准灯光为第一个灯光
    rowLights.push(currentLight); // 将基准灯光加入当前行

    // 遍历所有灯光
    for (let i = 1; i < lightArray.length; i++) {
        const nextLight = lightArray[i];

        // 如果当前灯光和基准灯光的 y 坐标差值小于等于基准灯光的半径,则认为它们在同一行
        if (nextLight.midY - currentLight.midY <= currentLight.rad) {
            rowLights.push(nextLight); // 将灯光加入当前行
        } else {
            // 当前灯光与基准灯光不在同一行,输出当前行的灯光编号并重置
            finalizeRow(rowLights, finalOrder);
            rowLights = [nextLight]; // 新行开始,加入当前灯光
            currentLight = nextLight; // 更新基准灯光为当前灯光
        }
    }

    // 处理最后一行
    if (rowLights.length > 0) {
        finalizeRow(rowLights, finalOrder);
    }

    return finalOrder.join(" "); // 返回最终的灯光编号顺序
}

// 输出当前行的灯光编号,并按 x 坐标排序
function finalizeRow(row, result) {
    row.sort((a, b) => a.midX - b.midX) // 按 x 坐标升序排序
        .forEach((light) => result.push(light.id)); // 将排序后的灯光编号加入结果数组
}

 class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">

注意:

如果发现代码有用例覆盖不到的情况,欢迎反馈!会在第一时间修正,更新。
解题不易,如对您有帮助,欢迎点赞/收藏

注:本文转载自blog.csdn.net的CodeClimb的文章"https://blog.csdn.net/CodeClimb/article/details/145065170"。版权归原作者所有,此博客不拥有其著作权,亦不承担相应法律责任。如有侵权,请联系我们删除。
复制链接

评论记录:

未查询到任何数据!