【数据结构与算法】1、学习动态数组数据结构(基本模拟实现 Java 的 ArrayList 实现增删改查)

简介: 【数据结构与算法】1、学习动态数组数据结构(基本模拟实现 Java 的 ArrayList 实现增删改查)


一、什么是数据结构

(1) 概念

🍃 数据结构是计算机存储组织数据的方式

(2) 分类

🎉 线性结构

  • 线性表(数组、链表、栈、队列、哈希表)

🎉 树形结构

  • 二叉树
  • AVL 树
  • 红黑树
  • B 树
  • Trie
  • 哈夫曼树
  • 并查集

🎉 图形结构

  • 邻接矩阵
  • 邻接表

二、线性表

🎁 线性表是具有 n 个相同类型元素的有限序列(n >= 0)

  • a1 是首节点(首元素), an 是尾结点(尾元素)
  • a1 是 a2 的前驱
  • a2 是 a1 的后继

常见的线性表有:

🎗️ 数组

🎗️ 链表

🎗️ 栈

🎗️ 队列

🎗️ 哈希表(散列表)

三、数组(Array)

(1) 数组的底层结构

🎁 数组是一种顺序存储的线性表,全部元素的内存地址是连续的

// 【new】向堆空间申请一段存储空间
int[] array = new int[]{11, 22, 33};

(2) 数组缺点

  • 👓数组有个致命的缺点:无法动态修改容量
  • 👓数组创建完毕后,能够存储的数据就固定了
  • 👓数组操纵元素的方式不够面向对象

🎐 自己实现动态数组,弥补数组的缺点

四、动态数组(Dynamic Array)接口设计

public interface List<E> {
    /**
     * 元素的数量
     */
    int size();
    /**
     * 是否为空
     */
    boolean isEmpty();
    /**
     * 是否包含某个元素
     */
    boolean contains(E element);
    /**
     * 添加元素到最后面
     */
    void add(E element);
    /**
     * 返回 index 位置对应的元素
     */
    E get(int index);
    /**
     * 设置 index 位置的元素
     */
    E set(int index, E element);
    /**
     * 往 index 位置添加元素
     */
    void add(int index, E element);
    /**
     * 删除 index 位置对应的元素
     */
    E remove(int index);
    /**
     * 返回元素的下标
     */
    int indexOf(E element);
    /**
     * 清空数组
     */
    void clear();
}

五、动态数组的设计和基本代码实现

(1) 成员变量

Java 中的成员变量会自动初始化,比如:

🎑 int 类型自动初始化为 0

🎑 对象类型自动初始化为 null

🎑 size 记录动态数组中元素的个数

🎑 elements 用于实际存储数据

🎑 动态数组的底层是数组

(2) 代码

/**
 * 只支持存储 int 类型数据的动态数组
 */
public class ArrayListInt {
    private int size;
    private int[] elements;
    public static final int DEFAULT_CAPACITY = 10;
    public static final int ELEMENT_NOT_FOUND = -1;
    public ArrayListInt() {
        this(DEFAULT_CAPACITY);
    }
    public ArrayListInt(int capacity) {
        capacity = (capacity < DEFAULT_CAPACITY) ? DEFAULT_CAPACITY : capacity;
        elements = new int[capacity];
    }
    public int size() {
        return size;
    }
    public boolean isEmpty() {
        return size == 0;
    }
    /**
     * 获取 index 索引处的元素
     */
    public int get(int index) {
        if (index < 0 || index >= size) {
            throw new IndexOutOfBoundsException("size: " + size + ", " + "index: " + index);
        }
        return elements[index];
    }
    /**
     * 设置 index 位置的元素
     *
     * @param index   下标
     * @param element 要设置的元素
     * @return 原来的元素ֵ
     */
    public int set(int index, int element) {
        // 获取 index 位置的元素
        int old = get(index);
        elements[index] = element;
        return old;
    }
    /**
     * 返回元素的索引
     */
    public int indexOf(int element) {
        // 如果数组为空, 直接找不到
        if (isEmpty()) return ELEMENT_NOT_FOUND;
        for (int i = 0; i < size; i++) {
            if (element == elements[i]) return i;
        }
        return ELEMENT_NOT_FOUND;
    }
    /**
     * 检查数组中是否包含 element 元素
     */
    public boolean contains(int element) {
        return indexOf(element) != ELEMENT_NOT_FOUND;
    }
    /**
     * 清除全部元素
     * 只要【size = 0】就无法获取到数组中的任何元素了
     */
    public void clear() {
        size = 0;
    }
}

① get ()

🎈 该方法的作用:获取 index 索引处的元素

🎈 数组可通过索引获取到元素

🎈 要做参数(index)校验

② indexOf ()

🎈 返回某个元素的索引

🎈 遍历每个下标的元素,拿数组中的每个元素和参数元素进行比较

③ clear ()

🎈 清除全部元素(清空数组)

🎈 在当前的代码中,只要【size = 0】就无法获取到数组中的任何元素了(就可以理解为清空了数组)

六、add 方法和扩容

🎈 add(int element):往数组的尾部添加元素 element

🎈 add(int index, int element):往数组的 index 索引位置添加元素 element

(1) add (int element)

🕰️ 每次往尾部添加元素的时候,是往数组的索引为 size 位置添加元素

public class ArrayListInt {
    /**
     * 往数组尾部添加元素
     */
    public void add(int element) {
        // TODO 扩容检测
        elements[size++] = element;
    }
}

(2) 打印动态数组中的元素

☆写法1:

public class ArrayListInt {
    /**
     * 遍历打印数组中的元素
     */
    public String printElements() {
        StringBuilder sb = new StringBuilder();
        sb.append("{size=").append(size).append(", [");
        for (int i = 0; i < size; i++) {
            sb.append(elements[i]);
            if (i != size - 1) {
                sb.append(", ");
            }
        }
        sb.append("]}");
        return sb.toString();
    }
}

🎨 遍历获取数组中的各个元素,然后进行拼接

🎨 Java 中大量字符串拼接用 StringBuilder 最好

☆写法2:

public class ArrayListInt {
 
    /**
     * 遍历打印数组中的元素
     */
    public String printElements() {
        StringBuilder sb = new StringBuilder();
        sb.append("{size=").append(size).append(", [");
        for (int i = 0; i < size; i++) {
            // 不是第 0 个元素就先拼接【, 】
            if (i != 0) {
                sb.append(", ");
            }
            sb.append(elements[i]);
        }
        sb.append("]}");
        return sb.toString();
    }
}

🎨 不是第 0 个元素就先拼接【, 】

🎨 相比写法1每次循环少了一次减法运算

(3) add (int index, int element)

☆ 往 index 位置插入元素 element

🎁 把 index 位置到 size - 1 位置【[index, size-1]】的元素往后挪动【空出 index 位置的元素】

🎁 把 element 元素赋值到 index 索引位置

🎁 从 索引为 size - 1 处开始挪动

public class ArrayListInt {
    /**
     * 在 index 位置插入元素 element
     */
    public void add(int index, int element) {
        // 当 index 等于 size 的时候, 是往数组的尾部添加元素
        if (index < 0 || index > size) {
            throw new IndexOutOfBoundsException("size: " + size + ", " + "index: " + index);
        }
        // TODO 扩容检测
        // 挪动元素(从最后一个元素开始挪动)
        for (int i = size - 1; i >= index; i--) {
            elements[i + 1] = elements[i];
        }
        // 把元素 element 赋值到 index 索引位置
        elements[index] = element;
        size++; // 数组元素加1
    }
}

(4) 数组越界异常的封装

public class ArrayListInt {
    public void outOfBounds(int index) {
        throw new IndexOutOfBoundsException("index: " + index + " size: " + size);
    }
    public void rangeCheck(int index) {
        if (index < 0 || index >= size) {
            outOfBounds(index);
        }
    }
    public void rangeCheck4Add(int index) {
        if (index < 0 || index > size) {
            outOfBounds(index);
        }
    }
}

✏️ outOfBounds(int index):封装抛异常的方法

✏️ rangeCheck(int index):索引 index 不能小于 0 或 大于等于 size,否则都会数组越界

✏️ rangeCheck4Add(int index):添加元素的时候 index 是可以等于 size 的(此时是往数组的最后添加元素)

(5) MyJunit(接口测试)

🍃使用异常知识进行接口测试:当测试不通过的时候,会抛异常

🍃测试通过的时候,打印 Success!

public class MyJunit {
    public static void test(boolean boolean_) {
        try {
            if (!boolean_) throw new Exception("测试不通过");
            System.out.println("\nSuccess!");
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

(6) 扩容【 ensureCapacity() 】

📖① 申请全新的数组空间(容量适宜)

📖② 把就数据的数据复制到全新的数组中

📔 添加的时候才会考虑扩容操作

/**
     * 扩容检测
     *
     * @param capacity 数组容量至少是 capacity
     */
    private void ensureCapacity(int capacity) {
        int oldCapacity = elements.length;
        // 如果所需容量足够, 则不扩容
        if (oldCapacity >= capacity) return;
        // 申请全新的数组空间(新容量是旧容量的 1.5 倍)
        capacity = oldCapacity + (oldCapacity >> 1);
        int[] newElements = new int[capacity];
        // 把旧数组中的数据复制到新数组中
        for (int i = 0; i < size; i++) {
            newElements[i] = elements[i];
        }
        // elements 指针指向新数组
        elements = newElements;
        System.out.println(oldCapacity + "扩容为" + capacity);
    }

七、remove (int index)

📔 作用:删除 index 索引处的元素,返回之前 index 索引处的元素

📙 思路:① 用后面的元素把要 删除的索引 index 位置的元素覆盖掉

📙 ② 数组 size 减 1

📙 ③ 如何覆盖❓ 遍历

public class ArrayListInt {
    /**
     * 删除 index 位置的元素
     *
     * @param index 下标
     * @return 原来的元素
     */
    public int remove(int index) {
        // 取出 index 索引处原来的元素
        int old = get(index);
        // 覆盖掉 index 索引处的元素
        for (int i = index; i < size; i++) {
            elements[i] = elements[i + 1];
        }
        // 最后一个元素不被访问到的关键代码
        size--;
        return old;
    }
}

八、仅能存储 int 类型的动态数组 ArrayListInt 完整代码

/**
 * 只支持存储 int 类型数据的动态数组
 */
@SuppressWarnings("all")
public class ArrayListInt {
    private int size;
    private int[] elements;
    public static final int DEFAULT_CAPACITY = 10;
    public static final int ELEMENT_NOT_FOUND = -1;
    public ArrayListInt() {
        this(DEFAULT_CAPACITY);
    }
    public ArrayListInt(int capacity) {
        capacity = (capacity < DEFAULT_CAPACITY) ? DEFAULT_CAPACITY : capacity;
        elements = new int[capacity];
    }
    public int size() {
        return size;
    }
    public boolean isEmpty() {
        return size == 0;
    }
    /**
     * 获取 index 索引处的元素
     */
    public int get(int index) {
        rangeCheck(index);
        return elements[index];
    }
    /**
     * 设置 index 位置的元素
     *
     * @param index   下标
     * @param element 要设置的元素
     * @return 原来的元素ֵ
     */
    public int set(int index, int element) {
        // 获取 index 位置的元素
        int old = get(index);
        elements[index] = element;
        return old;
    }
    /**
     * 返回元素的索引
     */
    public int indexOf(int element) {  
        for (int i = 0; i < size; i++) {
            if (element == elements[i]) return i;
        }
        return ELEMENT_NOT_FOUND;
    }
    /**
     * 检查数组中是否包含 element 元素
     */
    public boolean contains(int element) {
        return indexOf(element) != ELEMENT_NOT_FOUND;
    }
    /**
     * 清除全部元素
     * 只要【size = 0】就无法获取到数组中的任何元素了
     */
    public void clear() {
        size = 0;
    }
    /**
     * 往数组尾部添加元素
     */
    public void add(int element) {
        add(size, element);
    }
    /**
     * 在 index 位置插入元素 element
     */
    public void add(int index, int element) {
        rangeCheck4Add(index);
        // 扩容检测, 保证容量至少是【size+1】
        ensureCapacity(size + 1);
        // 挪动元素(从最后一个元素开始挪动)
        for (int i = size - 1; i >= index; i--) {
            elements[i + 1] = elements[i];
        }
        // 把元素 element 赋值到 index 索引位置
        elements[index] = element;
        size++; // 数组元素加1
    }
    /**
     * 扩容检测
     *
     * @param capacity 数组容量至少是 capacity
     */
    private void ensureCapacity(int capacity) {
        int oldCapacity = elements.length;
        // 如果所需容量足够, 则不扩容
        if (oldCapacity >= capacity) return;
        // 申请全新的数组空间(新容量是旧容量的 1.5 倍)
        capacity = oldCapacity + (oldCapacity >> 1);
        int[] newElements = new int[capacity];
        // 把旧数组中的数据复制到新数组中
        for (int i = 0; i < size; i++) {
            newElements[i] = elements[i];
        }
        // elements 指针指向新数组
        elements = newElements;
        System.out.println(oldCapacity + "扩容为" + capacity);
    }
    /**
     * 删除 index 位置的元素
     *
     * @param index 下标
     * @return 原来的元素
     */
    public int remove(int index) {
        // 取出 index 索引处原来的元素
        int old = get(index);
        // 覆盖掉 index 索引处的元素
        for (int i = index; i < size - 1; i++) {
            elements[i] = elements[i + 1];
        }
        // 最后一个元素不被访问到的关键代码
        size--;
        return old;
    }
    public void outOfBounds(int index) {
        throw new IndexOutOfBoundsException("index: " + index + " size: " + size);
    }
    public void rangeCheck(int index) {
        if (index < 0 || index >= size) {
            outOfBounds(index);
        }
    }
    public void rangeCheck4Add(int index) {
        if (index < 0 || index > size) {
            outOfBounds(index);
        }
    }
    /**
     * 遍历打印数组中的元素
     */
    public String printElements() {
        StringBuilder sb = new StringBuilder();
        sb.append("{size=").append(size).append(", [");
        for (int i = 0; i < size; i++) {
            // 不是第 0 个元素就拼接【, 】
            if (i != 0) {
                sb.append(", ");
            }
            sb.append(elements[i]);
        }
        sb.append("]}");
        return sb.toString();
    }
}

九、泛型(让动态数组中能存放各种类型的数据)

🎁 可使用 Java 中的泛型让动态数据更加通用(在动态数组中能存放任何数据类型的数据)

(1) clear () 【对象数组】

🧣 当动态数组中可以存储任何类型的时候,对于 clear() 方法,仅仅把 size 设置为 0 是不够的

🧣 把 size 设置为 0 后,虽然使用动态数组的人无法获取到任何数据,但这些数据仍然存活在内存中【这些数据依然存在,只是你无法使用它而已】

🧣 最佳的方式是:把 size 设置为 0,并把这些内存都回收掉(置为 null)

/**
     * 清除全部元素
     */
    public void clear() {
        // 销毁堆空间的对象数据
        for (int i = 0; i < elements.length; i++) {
            elements[i] = null;
        }
        
        size = 0;
    }

(2) 对象的比较不用 【==】

🎁 两个 对象用 == 运算符进行比较的时候,比较的是两个对象的内存地址

🎁 若不想比较两个对象的内存地址,需要用 equals() 方法

public int indexOf(E element) {
        if (element == null) return ELEMENT_NOT_FOUND;
        // 如果数组为空, 直接找不到
        if (isEmpty()) return ELEMENT_NOT_FOUND;
        for (int i = 0; i < size; i++) {
            if (element.equals(elements[i])) return i;
        }
        return ELEMENT_NOT_FOUND;
    }

(3) remove (int index) 清空最后一个元素

/**
     * 删除 index 位置的元素
     *
     * @param index 下标
     * @return 原来的元素
     */
    public E remove(int index) {
        // 取出 index 索引处原来的元素
        E old = get(index);
        // 覆盖掉 index 索引处的元素
        for (int i = index + 1; i < size; i++) {
            elements[i - 1] = elements[i];
        }
        // 把最后一个元素置空
        elements[--size] = null;
        return old;
    }

(4) null 值处理

动态数组中应该要能够存放 null 值

/**
     * 返回元素的索引
     */
    public int indexOf(E element) {
        if (null == element) {
            for (int i = 0; i < size; i++) {
                if (elements[i] == null) return i;
            }
        } else {
            for (int i = 0; i < size; i++) {
                if (element.equals(elements[i])) return i;
            }
        }
        return ELEMENT_NOT_FOUND;
    }

(5) ArrayList 完整代码

/**
 * 泛型让动态数组中能存放任何数据类型的数据
 */
@SuppressWarnings("all")
public class ArrayList<E> {
    private int size;
    private E[] elements;
    public static final int DEFAULT_CAPACITY = 10;
    public static final int ELEMENT_NOT_FOUND = -1;
    public ArrayList() {
        this(DEFAULT_CAPACITY);
    }
    public ArrayList(int capacity) {
        capacity = (capacity < DEFAULT_CAPACITY) ? DEFAULT_CAPACITY : capacity;
        elements = (E[]) new Object[capacity];
    }
    public int size() {
        return size;
    }
    public boolean isEmpty() {
        return size == 0;
    }
    /**
     * 获取 index 索引处的元素
     */
    public E get(int index) {
        rangeCheck(index);
        return elements[index];
    }
    /**
     * 设置 index 位置的元素
     *
     * @param index   下标
     * @param element 要设置的元素
     * @return 原来的元素ֵ
     */
    public E set(int index, E element) {
        // 获取 index 位置的元素
        E old = get(index);
        elements[index] = element;
        return old;
    }
    /**
     * 返回元素的索引
     */
    public int indexOf(E element) {
        if (null == element) {
            for (int i = 0; i < size; i++) {
                if (elements[i] == null) return i;
            }
        } else {
            for (int i = 0; i < size; i++) {
                if (element.equals(elements[i])) return i;
            }
        }
        return ELEMENT_NOT_FOUND;
    }
    /**
     * 检查数组中是否包含 element 元素
     */
    public boolean contains(E element) {
        return indexOf(element) != ELEMENT_NOT_FOUND;
    }
    /**
     * 清除全部元素
     */
    public void clear() {
        // 销毁堆空间的对象数据
        for (int i = 0; i < elements.length; i++) {
            elements[i] = null;
        }
        size = 0;
    }
    /**
     * 往数组尾部添加元素
     */
    public void add(E element) {
        add(size, element);
    }
    /**
     * 在 index 位置插入元素 element
     */
    public void add(int index, E element) {
        rangeCheck4Add(index);
        // 扩容检测, 保证容量至少是【size + 1】
        ensureCapacity(size + 1);
        // 挪动元素(从最后一个元素开始挪动)
        for (int i = size - 1; i >= index; i--) {
            elements[i + 1] = elements[i];
        }
        // 把元素 element 赋值到 index 索引位置
        elements[index] = element;
        size++; // 数组元素加1
    }
    /**
     * 扩容检测
     *
     * @param capacity 数组容量至少是 capacity
     */
    private void ensureCapacity(int capacity) {
        int oldCapacity = elements.length;
        // 如果所需容量足够, 则不扩容
        if (oldCapacity >= capacity) return;
        // 申请全新的数组空间(新容量是旧容量的 1.5 倍)
        capacity = oldCapacity + (oldCapacity >> 1);
        E[] newElements = (E[]) new Object[capacity];
        // 把旧数组中的数据复制到新数组中
        for (int i = 0; i < size; i++) {
            newElements[i] = elements[i];
        }
        // elements 指针指向新数组
        elements = newElements;
        System.out.println(oldCapacity + "扩容为" + capacity);
    }
    /**
     * 删除 index 位置的元素
     *
     * @param index 下标
     * @return 原来的元素
     */
    public E remove(int index) {
        // 取出 index 索引处原来的元素
        E old = get(index);
        // 覆盖掉 index 索引处的元素
        for (int i = index + 1; i < size; i++) {
            elements[i - 1] = elements[i];
        }
        // 把最后一个元素置空
        elements[--size] = null;
        return old;
    }
    public void outOfBounds(int index) {
        throw new IndexOutOfBoundsException("index: " + index + " size: " + size);
    }
    public void rangeCheck(int index) {
        if (index < 0 || index >= size) {
            outOfBounds(index);
        }
    }
    public void rangeCheck4Add(int index) {
        if (index < 0 || index > size) {
            outOfBounds(index);
        }
    }
    /**
     * 遍历打印数组中的元素
     */
    public String printElements() {
        StringBuilder sb = new StringBuilder();
        sb.append("{size=").append(size).append(", [");
        for (int i = 0; i < size; i++) {
            // 不是第 0 个元素就拼接【, 】
            if (i != 0) {
                sb.append(", ");
            }
            sb.append(elements[i]);
        }
        sb.append("]}");
        return sb.toString();
    }
}

JDK 中内置了动态数组类:java.util.ArrayList

如有错误和疑问,欢迎和我交流

相关文章
|
2月前
|
Java
【Java集合类面试二十六】、介绍一下ArrayList的数据结构?
ArrayList是基于可动态扩展的数组实现的,支持快速随机访问,但在插入和删除操作时可能需要数组复制而性能较差。
|
9天前
|
Java
java数据结构,双向链表的实现
文章介绍了双向链表的实现,包括数据结构定义、插入和删除操作的代码实现,以及双向链表的其他操作方法,并提供了完整的Java代码实现。
java数据结构,双向链表的实现
|
9天前
|
存储 Java
java数据结构,线性表链式存储(单链表)的实现
文章讲解了单链表的基本概念和Java实现,包括头指针、尾节点和节点结构。提供了实现代码,包括数据结构、接口定义和具体实现类。通过测试代码演示了单链表的基本操作,如添加、删除、更新和查找元素,并总结了操作的时间复杂度。
java数据结构,线性表链式存储(单链表)的实现
|
3天前
|
存储 安全 Java
Java 数据结构类型总结
在 Java 中,常用的数据结构包括基础数据结构(如数组和字符串)、集合框架(如 Set、List 和 Map 接口的多种实现)、特殊数据结构(如栈、队列和双端队列)、链表(单链表、双链表和循环链表)以及图和树等。这些数据结构各有特点和适用场景,选择时需考虑性能、内存和操作需求。集合框架提供了丰富的接口和类,便于处理对象集合。
|
19天前
|
存储 Java 程序员
【数据结构】初识集合&深入剖析顺序表(Arraylist)
Java集合框架主要由接口、实现类及迭代器组成,包括Collection和Map两大类。Collection涵盖List(有序、可重复)、Set(无序、不可重复),Map则由键值对构成。集合通过接口定义基本操作,具体实现由各类如ArrayList、HashSet等提供。迭代器允许遍历集合而不暴露其实现细节。List系列集合元素有序且可重复,Set系列元素无序且不可重复。集合遍历可通过迭代器、增强for循环、普通for循环及Lambda表达式实现,各有适用场景。其中ArrayList实现了动态数组功能,可根据需求自动调整大小。
29 11
|
26天前
|
存储 C语言 C++
数据结构基础详解(C语言) 顺序表:顺序表静态分配和动态分配增删改查基本操作的基本介绍及c语言代码实现
本文介绍了顺序表的定义及其在C/C++中的实现方法。顺序表通过连续存储空间实现线性表,使逻辑上相邻的元素在物理位置上也相邻。文章详细描述了静态分配与动态分配两种方式下的顺序表定义、初始化、插入、删除、查找等基本操作,并提供了具体代码示例。静态分配方式下顺序表的长度固定,而动态分配则可根据需求调整大小。此外,还总结了顺序表的优点,如随机访问效率高、存储密度大,以及缺点,如扩展不便和插入删除操作成本高等特点。
|
9天前
|
存储 Java
java数据结构,线性表顺序存储(数组)的实现
文章介绍了Java中线性表顺序存储(数组)的实现。线性表是数据结构的一种,它使用数组来实现。文章详细描述了线性表的基本操作,如增加、查找、删除、修改元素,以及其他操作如遍历、清空、求长度等。同时,提供了完整的Java代码实现,包括MyList接口和MyLinearList实现类。通过main函数的测试代码,展示了如何使用这些方法操作线性表。
|
2月前
|
存储 算法 Java
"解锁Java对象数据结构的奥秘:从基础到实战,与热点技术共舞,让你的编程之路更激情四溢!"
【8月更文挑战第21天】Java以对象为核心,它是程序的基本单元与数据处理的基础。对象源自类,拥有属性(字段)和方法。对象在内存中分为对象头(含哈希码、GC信息等)和实例数据区(存储属性值)。例如,`Student`类定义了姓名、年龄等属性及相应的方法。通过`new`关键字实例化对象并调用其方法进行数据操作,是Java编程的关键技能。
29 0
|
5月前
|
存储 Java
Java数据结构:链表
Java数据结构:链表
41 2
|
4月前
|
算法 Java
Java数据结构与算法:双向链表
Java数据结构与算法:双向链表
下一篇
无影云桌面