华为od(D卷) 堆内存申请

题目描述

有一个总空间为100字节的堆,现要从中新申请一块内存,内存分配原则为:优先分配紧接着前一块已使用的内存,分配空间足够时分配最接近申请大小的空闲内存。

输入描述

第1行是1个整数,表示期望申请的内存字节数。
第2到第N行是用空格分割的两个整数,表示当前已分配的内存的情况,每一行表示一块已分配的连续内存空间,每行的第1个和第2个整数分别表示偏移地址和内存块大小,如:
0 1
3 2
表示0偏移地址开始的1个字节和3偏移地址开始的2个字节已被分配,其余内存空闲。

输出描述

若申请成功,输出申请到内存的偏移
若申请失败,输出-1。

备注
1.若输入信息不合法或无效,则申请失败

2.若没有足够的空间供分配,则申请失败

3.堆内存信息有区域重叠或有非法值等都是无效输入

示例1

输入:
1
0 1
3 2

输出:
1

说明
堆中已使用的两块内存是偏移从0开始的1字节和偏移从3开始的2字节,空闲的两块内存是偏移从1开始2个字节和偏移从5开始95字节根据分配原则,新申请的内存应从1开始分配1个字节,所以输出偏移为1。

思路

代码

public class Demo19 {
    public static void main(String[] args) {
        // 检查空闲区域是否能满足申请
        int heapSize = 100;  // 总堆大小
        int lastOffsetEnd = 0;


        Scanner scanner = new Scanner(System.in);

        // 读取期望的内存字节数
        int requestSize = Integer.parseInt(scanner.nextLine());
        if (requestSize > heapSize) {
            System.out.println(-1);
        }
        List<int[]> allocatedBlocks = new ArrayList<>();


        // 读取已分配的内存块信息
        String in = scanner.nextLine();

        while (!in.equals("")) {
            int[] ints = Arrays.stream(in.split(" ")).mapToInt(Integer::parseInt).toArray();
            int offset = ints[0];
            int size = ints[1];
            allocatedBlocks.add(new int[] {offset, size});
            in = scanner.nextLine();
        }

        allocatedBlocks.sort(Comparator.comparingInt(a -> a[0]));

        // 检查非法输入,如堆内存块重叠或偏移、大小不合法
        if (!isValidInput(allocatedBlocks)) {
            System.out.println(-1);
            return;
        }


        // 遍历已分配块,检查相邻块之间的空闲区域
        for (int[] block : allocatedBlocks) {
            int currentOffset = block[0];
            int currentSize = block[1];

            // 计算当前块之前的空闲空间大小
            int freeSpace = currentOffset - lastOffsetEnd;

            // 如果空闲空间足够,返回起始偏移
            if (freeSpace >= requestSize) {
                System.out.println(lastOffsetEnd);
                return;
            }

            // 更新上一个内存块的结束位置
            lastOffsetEnd = currentOffset + currentSize;
        }

        // 检查最后一个块之后的空闲空间
        if (heapSize - lastOffsetEnd >= requestSize) {
            System.out.println(lastOffsetEnd);
        } else {
            System.out.println(-1);
        }
    }

    // 检查输入是否合法
    private static boolean isValidInput(List<int[]> allocatedBlocks) {
        for (int i = 1; i < allocatedBlocks.size(); i++) {
            int[] prevBlock = allocatedBlocks.get(i - 1);
            int[] currentBlock = allocatedBlocks.get(i);

            // 检查是否有重叠或者非法大小
            if (currentBlock[0] < prevBlock[0] + prevBlock[1] || currentBlock[1] <= 0) {
                return false;
            }
        }
        return true;
    }
}

  • 3
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
华为OD机试:堆内存申请华为OD机试中,堆内存申请是其中一个重要的考点。堆内存是程序运行时动态分配的内存空间,用于存储程序运行中产生的数据。 在C/C++语言中,我们使用`malloc()`函数来申请堆内存。`malloc()`函数接受一个参数,表示要申请的内存空间大小(单位是字节),并返回一个指向申请到的堆内存的指针。 例如,我们可以使用以下代码申请一个大小为100字节的堆内存空间: ``` int* ptr = (int*)malloc(100); ``` 在申请堆内存后,我们可以通过指针`ptr`来访问和操作这段堆内存空间。还需要注意的是,申请到的堆内存空间在使用完毕后,需要使用`free()`函数来释放它,以防止内存泄漏: ``` free(ptr); ``` 另外,为了避免内存泄漏和悬挂指针等问题,我们还需要注意以下几点: 1. 在申请堆内存时,应该检查`malloc()`函数返回的指针是否为`NULL`,以确保内存申请成功。 2. 在使用完申请到的内存后,一定要记得使用`free()`函数来释放内存,避免内存泄漏。 3. 避免使用已经释放的内存,也就是悬挂指针的问题,这样的操作可能导致程序崩溃或非预期的行为。 总之,在华为OD机试中,对于堆内存申请的考察主要包括如何正确申请和释放堆内存、避免悬挂指针和内存泄漏的问题。在编写代码时,我们应该牢记这些原则,以确保程序的正确性和稳定性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值