数据结构(1):使用面向对象模拟数组

简介: 数组是一种常用的数据结构,数组具有不可变性,创建后的数组的长度固定,通过索引访问数组中的元素,访问速度快,删除添加效率低。通过面向对象模拟数组,模拟的数组具有以下功能:添加新元素展示查找元素所在位置根据索引获取元素根据索引删除元素修改指定位置的元素同时使用两个算法对数组进行操作:有序添加元素二分查找法1.

数组是一种常用的数据结构,数组具有不可变性,创建后的数组的长度固定,通过索引访问数组中的元素,访问速度快,删除添加效率低。

通过面向对象模拟数组,模拟的数组具有以下功能:

  1. 添加新元素
  2. 展示
  3. 查找元素所在位置
  4. 根据索引获取元素
  5. 根据索引删除元素
  6. 修改指定位置的元素

同时使用两个算法对数组进行操作:

  1. 有序添加元素
  2. 二分查找法

1.创建数组类 MyArray.java

数据如何存储呢?在类中添加一个数组类型的私有属性用来保存数据,同时添加一个变量存储有效数据的长度(也就是元素的个数)

创建数组的时候需要指定数组的长度,所以要添加两个构造方法:

1.无参构造方法设置数组默认长度
2.有参构造方法指定数组长度

public class MyArray {
    //存储元素
    private long[] arr;
    //表示有效数据的长度
    private int elements;

    //无参构造默认50个长度
    public MyArray() {
        arr=new long[50];
    }

    public MyArray(int maxsize) {
        arr=new long[maxsize];
    }
}

2.编写添加数据的方法

elements 属性的默认值是 0,第一次向对象中添加元素也是添加到索引为0 的元素中,添加后将 elements 的长度加1,就能一直向数组中添加元素了,添加元素的个数取决于 arr 的长度。

public void insert(long value) {
    arr[elements]=value;
    elements++;
}

3.编写展示数据的方法

简单的展示数组中的元素即可,使用 for 循环遍历

public void display() {
    System.out.print("[");
    for (int i = 0; i < elements; i++) {
        System.out.print(arr[i]+" ");
    }
    System.out.println("]");
}

4.编写查找数据的方法

思路:使用循环遍历数组 arr ,将要查找的数据和 arr 中每个元素进行比较。如果相等,则跳出循环。循环结束后,如果循环次数等于元素个数说明没有找到数据,返回-1。否则返回循环次数(即找到的索引)。

public int search(long value) {
    int i ;
    for (i = 0; i < elements; i++) {
        if(value==arr[i]) {
            break;
        }
    }
    //遍历到末尾说明没有找到
    if (i==elements) {
        return -1;
    }else {
        return i;
    }
}

5.根据索引获取元素

思路:这相对比较简单了,直接给 arr 索引就能获取到元素。但是要注意传入的索引必须可用,不能用的索引可以抛出异常。

public long get(int index) {
    //如果索引大于可用,或索引小于0 都是无效的索引
    if (index>=elements||index<0) {
        //抛出数组越界异常
        throw new ArrayIndexOutOfBoundsException();
    }else {
        return arr[index];
    }
}

6.根据索引删除元素

思路:和获取元素时一样,先检查索引是否可用。如果可用,就从要删除元素的位置开始向后遍历,每次都将下一个元素的值赋值给当前元素。也就相当于要删除的元素被下一个元素覆盖,下一个元素被下下一个元素覆盖,以此类推。元素移动完成后,将可用元素长度 elements 减1。

public void delete(int index) {
    //如果索引大于可用,或索引小于0 都是无效的索引
    if (index>=elements||index<0) {
        throw new ArrayIndexOutOfBoundsException();
    }else {
        for(int i=index;i<elements;i++) {
            arr[index]=arr[i+1];
        }
        elements--;
    }
}

7.修改指定位置的元素

思路:和获取差不多,就是把获取改为修改

public void change(int index,int newValue) {
    //如果索引大于可用,或索引小于0 都是无效的索引
    if (index>=elements||index<0) {
        throw new ArrayIndexOutOfBoundsException();
    }else {
        arr[index]=newValue;
    }
}

8.完整代码

public class MyArray {
    private long[] arr;
    //表示有效数据的长度
    private int elements;

    public MyArray() {
        arr=new long[50];
    }

    public MyArray(int maxsize) {
        arr=new long[maxsize];
    }

    /**
     * 添加数据
     * @param value
     */
    public void insert(long value) {
        arr[elements]=value;
        elements++;
    }

    /**
     * 显示数据
     */
    public void display() {
        System.out.print("[");
        for (int i = 0; i < elements; i++) {
            System.out.print(arr[i]+" ");
        }
        System.out.println("]");
    }

    /**
     * 查找数据
     */
    public int search(long value) {
        int i ;
        for (i = 0; i < elements; i++) {
            if(value==arr[i]) {
                break;
            }
        }
        //遍历到末尾说明没有找到
        if (i==elements) {
            return -1;
        }else {
            return i;
        }
    }

    /**
     * 查找数据,根据索引来查
     */
    public long get(int index) {
        //如果索引大于可用,或索引小于0 都是无效的索引
        if (index>=elements||index<0) {
            throw new ArrayIndexOutOfBoundsException();
        }else {
            return arr[index];
        }
    }

    /**
     * 删除数据 
     */
    public void delete(int index) {
        //如果索引大于可用,或索引小于0 都是无效的索引
        if (index>=elements||index<0) {
            throw new ArrayIndexOutOfBoundsException();
        }else {
            for(int i=index;i<elements;i++) {
                arr[index]=arr[i+1];
            }
            elements--;
        }
    }

    /**
     * 更新数据
     */
    public void change(int index,int newValue) {
        //如果索引大于可用,或索引小于0 都是无效的索引
        if (index>=elements||index<0) {
            throw new ArrayIndexOutOfBoundsException();
        }else {
            arr[index]=newValue;
        }
    }
}

9.有序添加元素

思路:修改 insert 方法,遍历 arr 数组,如果当前元素大于添加的数据,当前的位置就要存入的位置。从最后一个元素开始,逐个将元素向后位移,空出要存入的位置。存入要添加的元素后,将有效数据长度加1。

public void insert(long value) {
    int i;
    for(i=0;i<elements;i++) {
        if(arr[i]>value) {
            break;
        }
    }
    for (int j = elements; j > i; j--) {
        arr[j]=arr[j-1];
    }
    arr[i]=value;
    
    elements++;
}

10.二分查找法

思路:数据必须是有序的,才能使用二分查找法!可以结合有序添加元素一块使用,这里的序列是升序(从小到大)。二分查找是每次和一组数中间的数进行比较,如果大于就再和右边的数最中间的数比较,如果小于就和左边的数最中间的数比较。直到中间的数和要查找的数相等,否则就是没有这个数。

public int binarySearch(long value) {
    int mid=0;//中间值
    int low=0;
    int high = elements;
    
    while(true) {
        mid=(high+low)/2;
        if(arr[mid]==value) {
            return mid;
        }else if(low>high) {
            return -1;
        }else {
            if (value>arr[mid]) {
                high=mid+1;
            }else {
                high=mid-1;
            }
        }
    }
    
}

二分查找法图示

相关文章
|
1月前
|
存储 Java 程序员
数据结构之 - 深入了解数组数据结构
数据结构之 - 深入了解数组数据结构
38 6
|
1月前
|
存储 算法 搜索推荐
探索常见数据结构:数组、链表、栈、队列、树和图
探索常见数据结构:数组、链表、栈、队列、树和图
99 64
|
1月前
|
算法 程序员 索引
数据结构与算法学习七:栈、数组模拟栈、单链表模拟栈、栈应用实例 实现 综合计算器
栈的基本概念、应用场景以及如何使用数组和单链表模拟栈,并展示了如何利用栈和中缀表达式实现一个综合计算器。
30 1
数据结构与算法学习七:栈、数组模拟栈、单链表模拟栈、栈应用实例 实现 综合计算器
|
1月前
|
存储 算法 定位技术
数据结构与算法学习二、稀疏数组与队列,数组模拟队列,模拟环形队列
这篇文章主要介绍了稀疏数组和队列的概念、应用实例以及如何使用数组模拟队列和环形队列的实现方法。
21 0
数据结构与算法学习二、稀疏数组与队列,数组模拟队列,模拟环形队列
|
2月前
|
存储 Java
java数据结构,线性表顺序存储(数组)的实现
文章介绍了Java中线性表顺序存储(数组)的实现。线性表是数据结构的一种,它使用数组来实现。文章详细描述了线性表的基本操作,如增加、查找、删除、修改元素,以及其他操作如遍历、清空、求长度等。同时,提供了完整的Java代码实现,包括MyList接口和MyLinearList实现类。通过main函数的测试代码,展示了如何使用这些方法操作线性表。
|
4月前
|
存储
【数据结构OJ题】轮转数组
力扣题目——轮转数组
43 2
【数据结构OJ题】轮转数组
|
3月前
|
存储 Java 程序员
"揭秘HashMap底层实现:从数组到链表,再到红黑树,掌握高效数据结构的秘密武器!"
【8月更文挑战第21天】HashMap是Java中重要的数据结构,采用数组+链表/红黑树实现,确保高效查询与更新。构造方法初始化数组,默认容量16,负载因子0.75触发扩容。`put`操作通过计算`hashCode`定位元素,利用链表或红黑树处理冲突。`get`和`remove`操作类似地定位并返回或移除元素。JDK 1.8优化了链表转红黑树机制,提升性能。理解这些原理能帮助我们更高效地应用HashMap。
45 0
|
5月前
|
存储 算法 调度
【数据结构与算法】详解循环队列:基于数组实现高效存储与访问
【数据结构与算法】详解循环队列:基于数组实现高效存储与访问
|
5月前
|
存储 JavaScript 前端开发
JavaScript中的数组是核心数据结构,用于存储和操作序列数据
【6月更文挑战第22天】JavaScript中的数组是核心数据结构,用于存储和操作序列数据。创建数组可以使用字面量`[]`或`new Array()`。访问元素通过索引,如`myArray[0]`,修改同样如此。常见方法包括:`push()`添加元素至末尾,`pop()`移除末尾元素,`shift()`移除首元素,`unshift()`添加到开头,`join()`连接为字符串,`slice()`提取子数组,`splice()`进行删除、替换,`indexOf()`查找元素位置,`sort()`排序数组。还有其他如`reverse()`、`concat()`等方法。
130 2
|
5月前
|
算法 索引 Python
数据结构 数组部分小结
数据结构 数组部分小结

热门文章

最新文章