数据结构与算法——Java实现 6.递归

news2024/9/21 2:34:23

要学会试着安静下来

                        —— 24.9.17

一、递归的定义

计算机科学中,递归是一种解决计算问题的方法,其中解决方案取决于同一类问题的更小子集

说明:

        ① 自己调用自己,如果说每个函数对应着一种解决方案,自己调用自己意味着解决方案是一样的(有规律的)

        ② 每次调用,函数处理的数据会较上次缩减(子集),而且最后会缩减至无需继续递归

        ③ 内层函数调用(子集处理)完成,外层函数才能算调用完成(内层完成,外层才能完成——先处理子问题

二、递归的解题思路

1.确定能否使用递归求解

2.推导出递推关系,即父问题与子问题的关系,以及递归的结束条件

① 深入到最里层叫

② 从最里层出来叫

③ 在的过程中,外层函数内的局部变量(以及方法参数)并未消失,的时候还可以用到

④ 大问题慢慢递到小问题,再从小问题计算结果归回大问题

递的时候是顺序的,归的时候是逆序的

三、递归——习题

1.阶乘

用递归的方法求阶乘

阶乘的定义:n! = 1*2*.3*(n-2)*(n-1)*n,其中n为自然数,当然0! = 1

递推关系:f(n) = 1,n = 1;f(n) = n * f(n-1), n > 1

public class demo1Factorial {
    // 递归函数
    public static int factorial(int n) {
        if (n == 1){
            return 1;
        }
        return n * factorial(n-1);
    }

    public static void main(String[] args) {
        demo1Factorial demo1Factorial = new demo1Factorial();
        System.out.println(demo1Factorial.factorial(5));
        System.out.println(factorial(6));
        System.out.println(factorial(7));
        System.out.println(factorial(8));
    }
}

2.反向打印字符串

用递归反向打印字符串,n为字符在整个字符串 str 中的索引位置。

递:n从0开始,每次n+1,一直递到n == str.length()-1。

归:从n==str.length()开始归,从归打印,自然是逆序的

递推关系:f(m)=1rn1)osns.h()-1n=str.length()

注:因为是先递再归,所以反向打印字符串输出语句的位置应该放在调用递归函数之后

public class demo2ReversePrintStr {
    // 静态方法反向打印字符串
    public static void reversePrintStr(String str,int n) {
        if (n == str.length()){
            return;
        }
        // System.out.print(str.charAt(n)+" ");
        reversePrintStr(str,n+1);
        System.out.print(str.charAt(n)+" ");
    }

    public static void main(String[] args) {
        reversePrintStr("YYSHlcl",0);
    }
}

3.二分查找

输入元素,返回元素在有序数组中的位置

前提:待查找数组是有序数组

public class demo3BinarySearch {
    public static int binarySearch(int[] arr, int key) {
        int target = Finding(arr,key,0,arr.length-1);
        return target;
    }

    private static int Finding(int[] arr, int key,int i,int j) {
        if (i > j){
            return -1;
        }
        int m = (i + j) >>> 1;
        if (key == arr[m]) {
            return m;
        }else if (key < arr[m]) {
            j = m-1;
            return Finding(arr,key,i,j);
        }else {
            i = m+1;
            return Finding(arr,key,i,j);
        }
    }

    public static void main(String[] args) {
        int[] arr = {7,9,11,12,24,34,65,81,98};
        int res = binarySearch(arr, 81);
        System.out.println("结果为:"+res);
        System.out.println(binarySearch(arr, 7));
        System.out.println(binarySearch(arr, 9));
        System.out.println(binarySearch(arr, 81));
    }
}

4.递归冒泡排序

将数组划分成两部分 [0…j][j+1…a.length-1]

左边 [0…j] 是未排序部分

右边 [j+1…a.length-1]是已排序部分

未排序区间内,相邻的两个元素比较,如果前一个大于后一个,则交换位置

package Day7Recursion;

import java.util.Arrays;

public class demo4BubbleSort {

    public static void sort(int[] arr) {
        bubble(arr,arr.length-1);
    }

    // j代表未排序区域的右边界
    private static void bubble(int[] arr, int j) {
        if (j == 0){
            return;
        }
        for (int i = 0; i < j; i++) {
            if (arr[i] > arr[i + 1]) {
                int temp = arr[i];
                arr[i] = arr[i + 1];
                arr[i + 1] = temp;
            }
        }
        bubble(arr, j - 1);
    }

    public static void main(String[] args) {
        int[] arr = { 5, 4, 3, 2, 1 };
        System.out.println(Arrays.toString(arr));
        bubble(arr,arr.length-1);
        System.out.println(Arrays.toString(arr));
        sort(arr);
        System.out.println(Arrays.toString(arr));

    }
}

将已排序的部分省略,提高效率

    private static void adbubble(int[] arr, int j) {
        if (j == 0){
            return;
        }
        int x = 0;
        for (int i = 0; i < j; i++) {
            if (arr[i] > arr[i + 1]) {
                int temp = arr[i];
                arr[i] = arr[i + 1];
                arr[i + 1] = temp;
                x = i;
            }
        }
        adbubble(arr,x);
    }

 5.插入排序

插入排序(Insertion Sort)是一种简单直观的排序算法。它的工作原理是通过构建有序序列,对于未排序数据,在已排序序列中从后向前扫描,找到相应位置并插入。插入排序在实现上,通常采用in-place排序(即只需用到O(1)的额外空间的排序),因而在从后向前扫描过程中,找到排序位置后,需要将已排序元素逐步向后挪位,为最新元素提供插入空间。

插入排序:将元素放在一个已排序的元素队列中,使其仍然有序,类比于扑克牌,将插入位置找到后,插入元素

import java.util.Arrays;

public class demo5InsertSort {
    public static void sort(int[] arr) {
        insertSort(arr,1);
    }

    // low代表未排序区域的下边界
    private static void insertSort(int[] arr,int low) {
        // 代表所有元素都已经被排序
        if (low == arr.length) {
            return;
        }
        int t = arr[low];
        int i = low - 1; // 已排序区域指针

        while (i >= 0 && arr[i] > t){ // 循环找到插入位置
            arr[i+1] = arr[i];
            i--;
        }
        // 找到插入位置
        arr[i+1] = t;

        insertSort(arr,low+1);
    }

    public static void main(String[] args) {
        int[] arr = {111,27,36,45,51,63,7,81,9};
        sort(arr);
        System.out.println(Arrays.toString(arr));
    }
}

四、多路递归

上面写到的递归都是只调用了一次递归函数,我们也称之为单路递归

如果每个递归函数包含多个自身的调用,我们称之为多路递归

1.斐波那契数列

思路:多路进行递归,将问题转化为子问题,逐步小化问题

public class demo1MoreFib {

    public static int fib(int n) {
        if (n==0){
            return 0;
        }
        if (n==1){
            return 1;
        }
        int x = fib(n-1);
        int y = fib(n-2);
        return x + y;
    }

    public static void main(String[] args) {
        int fib = fib(9);
        System.out.println("第九项斐波那契数列返回结果为:"+fib);
    }
}

斐波那契数列的时间复杂度推导:

2.兔子问题

第一个月,有一对未成熟的兔子

第二个月,它们成熟

第三个月,它们能产下一对新的小兔子

所有兔子遵循相同规律,求第n 个月的兔子数

注意:注意停止迭代条件与开始数列条件,与斐波那契数列的变化

import java.util.Scanner;

public class demo2FibRabbit {
    public static int fib(int n) {
        if (n==1||n==2){
            return 1;
        }
        return fib(n-1)+fib(n-2);
    }

    public static void main(String[] args) {
        System.out.println("请您输入你想要得知的月份:");
        Scanner sc = new Scanner(System.in);
        int month = sc.nextInt();
        int rab = fib(month);
        System.out.println("第"+month+"月的小兔子有"+rab+"只");
    }
}

3.青蛙爬楼梯

楼梯有 n. 阶

青蛙要爬到楼顶,可以一次跳一阶,也可以一次跳两阶

只能向上跳,问有多少种跳法

import java.util.Scanner;

public class demo3frag {
    public static int fibFrag(int n) {
        if (n == 1){
            return 1;
        }
        if (n == 2){
            return 2;
        }
        return fibFrag(n-1) + fibFrag(n-2);
    }

    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        System.out.println("请输入您想要让青蛙上几楼");
        int n = sc.nextInt();
        int method = fibFrag(n);
        System.out.println("共用"+method+"种跳法");
    }
}

4.汉诺塔问题

 Tower of Hanoi,是一个源于印度古老传说:大梵天创建世界时做了三根金刚石柱,在一根柱子从下往上按大小顺序摞着 64 片黄金圆盘,大梵天命令婆罗门把圆盘重新摆放在另一根柱子上,并且规定:
        ①  一次只能移动一个圆盘
        ② 小圆盘上不能放大圆盘

思路

① 一个圆盘:圆盘1:a——>c

② 两个圆盘:圆盘1:a——>b、圆盘2:a——>c、圆盘1:b——>c

③ 三个圆盘:圆盘12:a——>b、圆盘3:a——>c、圆盘12:b——>c

④ 四个圆盘:圆盘123:a——>b、圆盘4:a——>c、圆盘123:b——>c

时间复杂度计算:T(n) = 2T(n-1) + c,T(1) = c
                             T(n) = c(2^n-1)

import java.util.LinkedList;

public class demo4HanoiTower {
    // 三个LinkedList集合代表三根柱子
    static LinkedList<Integer> a = new LinkedList<>();
    static LinkedList<Integer> b = new LinkedList<>();
    static LinkedList<Integer> c = new LinkedList<>();

    // 类似于添加元素放到柱子上
    // 初始化柱子
    static void init(int n){
        for (int i = n; i >= 1 ; i--) {
            a.addLast(i);
        }
    }

    // 设计递归函数   n:圆盘个数 a:原始柱子 b:借用柱子 c:目标柱子
    static void move(int n,LinkedList<Integer> a,
                     LinkedList<Integer> b,
                     LinkedList<Integer> c){
        if (n==0){
            return;
        }
        // 将n-1个盘子由a借助c移动到b上
        move(n - 1, a, c, b);
        c.addLast(a.removeLast()); // 中间步骤
        Print();
        // 将n-1个盘子由b借用a移动到c上
        move(n - 1, b, a, c);
    }

    public static void Print(){
        System.out.println("————————————————————————");
        System.out.println(a);
        System.out.println(b);
        System.out.println(c);
    }

    public static void main(String[] args) {
        init(3);
        Print();
        move(3,a,b,c);
    }

    // 时间复杂度计算:T(n) = 2T(n-1) + c,T(1) = c
    //              T(n) = c(2^n-1)
}

5.杨辉三角

分析

行i,列j,那么[i][j]的取值应该为[ i-1 ][ j-1 ] + [ i-1 ][ j ]

当 j = 0 或 i = j 时,[ i ][ j ]的取值为-1

public class demo5YangHui {

    public static int element(int i,int j){
        if (j == 0||j == i){
            return 1;
        }
        return element(i - 1,j - 1)+element(i - 1,j);
    }

    // 打印空格
    private static void printSpace(int n,int i){
        int num = (n - 1 - i) * 2;
        for (int j = 0; j < num; j++) {
            System.out.print(" ");
        }
    }

    public static void Print(int n){
        for (int i = 0; i < n; i++) {
            printSpace(n,i);
            for (int j = 0; j <= i; j++) {
                System.out.printf("%-4d",element(i,j));
            }
            System.out.println();
        }
    }

    public static void main(String[] args) {
        Print(9);
    }

}

 

6.杨辉三角——改进版1

用一个二维数组接收上次遍历的杨辉三角的数值,然后根据上一行的数据进行相加

用二维数组的空间复杂度换取递归的时间复杂度

public class demo6YangHuiEx1 {
    // 记忆法利用二维数组进行缓存,优化
    public static int element(int[][] triangle,int i,int j){
        if (triangle[i][j] > 0){
            return triangle[i][j];
        }
        if (j == 0 || i == j){
            triangle[i][j] = 1;
            return 1;
        }
        triangle[i][j] = element( triangle,i - 1,j - 1) + element(triangle,i - 1, j);
        return triangle[i][j];
    }

    // 打印空格 空格与高度和行号有关
    private static void printSpace(int n,int i){
        int num = (n - 1 - i) * 2;
        for (int j = 0; j < num; j++) {
            System.out.print(" ");
        }
    }

    public static void Print(int n){
        int[][] triangle = new int[n][];
        for (int i = 0; i < n; i++) {
            triangle[i] = new int[i+1];
            // 空格与高度和行号有关
            printSpace(n,i);
            for (int j = 0; j <= i; j++) {
                // 格式化输出 宽度为4 从左对齐
                System.out.printf("%-4d",element(triangle,i,j));
            }
            System.out.println();
        }
    }

    public static void main(String[] args) {
        Print(5);
    }
}

 

7.杨辉三角——改进版2

利用一维数组记忆存储,用少量的空间复杂度换取递归的时间复杂度

这种算法也称为动态规划算法

public class demo7YangHuiEx2 {
    // 记忆法利用一维数组进行缓存,优化

    // 打印空格 空格与高度和行号有关
    private static void printSpace(int n,int i){
        int num = (n - 1 - i) * 2;
        for (int j = 0; j < num; j++) {
            System.out.print(" ");
        }
    }

    private static void createRow(int[] row,int i){
        if (i == 0){
            row[0] = 1;
            return;
        }
        for (int j = i; j > 0 ; j--) {
            row[j] = row[j]+row[j-1];
        }
    }

    public static void Print(int n){
        int[] row = new int[n];
        for (int i = 0; i < n; i++) {
            createRow(row, i);
            printSpace(n,i);
            for (int j = 0; j <= i; j++) {
                // 格式化输出 宽度为4 从左对齐
                System.out.printf("%-4d",row[j]);
            }
            System.out.println();
        }
    }

    public static void main(String[] args) {
        Print(7);
    }
}

 

递归避免爆栈问题 —— 用循环代替递归

  • 循环的空间复杂度通常较低,主要取决于循环体内声明的局部变量和使用的数据结构。
  • 递归的空间复杂度较高,主要由递归调用的最大深度和每次递归所需的辅助空间决定。在设计和分析递归算法时,需要特别注意空间复杂度的控制,以避免栈溢出等问题。

五、递归时间复杂度的计算

其中,

        ① T(n)是问题的运行时间,n是数据规模

        ② a是子问题个数

        ③ T(n/b)是子问题运行时间,每个子问题被拆成原问题数据规模的n/b

        ④ f(n)是除递归外执行的计算

令x = logba,即 x = log子问题缩小倍数 子问题个数

则 

1.公式求解

例一

例二

例三

例四

例五

二分查找递归

    private static int Finding(int[] arr, int key,int i,int j) {
        if (i > j){
            return -1;
        }
        int m = (i + j) >>> 1;
        if (key == arr[m]) {
            return m;
        }else if (key < arr[m]) {
            j = m-1;
            return Finding(arr,key,i,j);
        }else {
            i = m+1;
            return Finding(arr,key,i,j);
        }
    }

子问题个数:a = 1

子问题数据规模缩小倍数:b = 2

除递归外执行的计算是常数级 c = 0

T(n) = T(n/2) + n^b

此时 x = c = 0,时间复杂度是T(n^0*logn) = T(logn) 

归并排序

    // 归并排序
    private void split(int arr[],int i,int j,int arr2[]) {
        if(j-i <= 1){
            return;
        }
        int m = (i+j)/2;

        // 递归
        split(arr,i,m,arr2);
        split(arr,m,j,arr2);

        // 合并
        merge(arr,i,m,j,arr2);
    }

子问题个数 a = 2

子问题数据规模缩小倍数 b = 2

除递归外,主要时间花在合并上,它可以用f(n) = n表示

T(n) = 2T(n/2) + n

此时 x = c = 1,时间复杂度是Θ(nlogn)

快速排序递归

    // 快速排序递归
    private static void algorithm quicksort(int[] arr,int low,int high){
        if (low >= high || low < 0 || high >= arr.length){
            return;
        }
        
        // 分区
        int p = partition(arr,low,high);
        
        // 递归
        quicksort(arr,low,p-1);
        quicksort(arr,p+1,high);
    }

子问题个数a = 2
子问题数据规模缩小倍数
        如果分区分的好,b = 2
        如果分区没分好,例如分区1的数据是0,分区2的数据是n-1

除递归外,主要时间花在分区上,它可以用f(n) = n表示

情况1

        分区分的好        T(n)=2T(n/2)+n
·        此时x=c=1,时间复杂度Θ(nlogn)
情况2

        分区没分好        T(n)=T(n-1)+T(1)+n

        不成比例,此时不能用主定理求解

2.展开求解

递归式不能用公式求解,每次递归时间复杂度不同,所以将式子展开,求出每次递归的时间复杂度,进行累计求出最后的时间复杂度

例一 递归求和

例二  递归冒泡排序

例三 递归快排

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2150881.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

数据建模无法满足复杂业务需求?别慌,数据开发平台可以完美互补!

前言 数据可视化建模无论是对于企业的数字化转型&#xff0c;还是对数据资源的价值开发来说&#xff0c;都是至关重要的工具&#xff0c;小兵在前文《数据可视化建模平台介绍》。中有详细介绍过他的能力&#xff0c;包括面向多源异构的企业数据&#xff0c;为企业提供数据集成…

web基础—dvwa靶场(十一)CSP Bypass

CSP Bypass(CSP 绕过) 内容安全策略&#xff08;CSP&#xff09;用于定义脚本和其他资源可以从何处加载或执行&#xff0c;本模块将指导您根据开发人员犯下的常见错误来绕过该策略。 这些漏洞都不是 CSP 中的实际漏洞&#xff0c;它们都是实现 CSP 的方式中的漏洞。 绕过内容安…

xtop:如何debug fail reason

我正在「拾陆楼」和朋友们讨论有趣的话题,你⼀起来吧? 拾陆楼知识星球入口 fix_xx_violations之后,工具会报告fail reason summary,通过man reason_name可以获知fail原因&#x

STM32(十六):MPU6050简介

MPU6050 MPU6050是一个6轴姿态传感器&#xff0c;可以测量芯片自身X、Y、Z轴的加速度、角速度参数&#xff0c;通过数据融合&#xff0c;可进一步得到姿态角&#xff0c;常应用于平衡车、飞行器等需要检测自身姿态的场景。 3轴加速度计&#xff08;Accelerometer&#…

攻防世界--->gametime

做题笔记。 前言&#xff1a; 因为有意思&#xff0c;所以&#xff0c;&#xff0c;&#xff0c;打通关了。。哈哈哈。 题外话&#xff1a;哦&#xff0c;程序结果还在&#xff0c;是因为我是在WSL—Debian上运行的。你还别说&#xff0c;真挺好用的&#xff0c;vm虚拟机能不…

Node-RED和物联网分析:实时数据处理和可视化平台

这篇论文的标题是《Node-RED and IoT Analytics: A Real-Time Data Processing and Visualization Platform》&#xff0c;发表在《Tech-Sphere Journal of Pure and Applied Sciences (TSJPAS)》2024年第一期上。论文主要探讨了Node-RED和物联网分析在物联网(IoT)实时数据处理…

列表、数组排序总结:Collections.sort()、list.sort()、list.stream().sorted()、Arrays.sort()

列表类型 一.Collections.sort() Collections.sort()用于List类型的排序&#xff0c;其提供了两个重载方法&#xff1a; 1.sort(List<T> list) &#xff08;1&#xff09;List指定泛型时只能指定引用数据类型&#xff0c;也就是说无法用于基本数据类型的排序。 &am…

9.20作业

手动封装一个顺序表&#xff08;SeqList&#xff09;,分文件编译实现 有私有成员&#xff1a; 顺序表数组的起始地址 ptr、 顺序表的总长度&#xff1a;size、顺序表的实际长度&#xff1a;len 成员函数&#xff1a; 初始化 init(int n) 判空&#xff1a;empty 判满&#xff1…

Rust语言入门第七篇-控制流

文章目录 Rust语言入门第七篇-控制流If 表达式基本结构特点和规则示例 let 语句中使用 ifloop 循环基本结构特点示例综合示例 while 循环基本结构特点示例综合示例 与 loop 循环的区别 for 循环基本结构详细说明特点示例综合示例 match 表达式match表达式的语法结构示例代码 Ru…

Mysql存储过程详细解读

目录 存储过程介绍 创建与调用 查看与删除 变量 系统变量 用户自定义变量 ​编辑局部变量 ​编辑​编辑IF判断 存储过程参数​编辑​编辑​编辑 CASE ​编辑 WHILE​编辑 ​编辑REPEAT​编辑​编辑 LOOP 游标 条件处理程序 存储函数 存储过程介绍 创建与调用 查…

GNN-RAG:用于大模型推理的图神经检索

GNN-RAG&#xff1a;用于大模型推理的图神经检索 秒懂大纲提出背景解法拆解全流程优化创意总结 论文&#xff1a;GNN-RAG: Graph Neural Retrieval for Large Language Model Reasoning 代码&#xff1a;https://github.com/cmavro/GNN-RAG 秒懂大纲 ├── GNN-RAG【主题】…

【刷题日记】43. 字符串相乘

43. 字符串相乘 其实就是大数乘法题&#xff0c;这道题用草稿纸演练一下&#xff0c;其实很好找到方法&#xff0c;模拟大数乘法即可。需要注意的是进位和迭代值&#xff0c;还有注意向下取整和去除前导0&#xff08;容易遗漏&#xff09;。去除前导0的时候还要注意如果全是0&…

命令行运行python时找不到模块怎么解决

问题&#xff1a; 新建了一个项目&#xff0c;目录结构如下&#xff1a; 然后在pycharm中运行glovar是没有问题的&#xff0c;但是在命令行中运行就会提示找不到init模块。 这是因为在pycharm中运行的时候&#xff0c;pycharm会自动将项目所在目录添加到了sys.path中&#xf…

天宝Trimble RealWorks2024.0.2注册机 点云后处理软件 点云三维重建软件

一、功能特色 1、强大的点云数据处理平台 Trimble Realworks2024是市面上先进的点云数据处理软件&#xff0c;能够配准、可视化、浏览和直接处理市面上几乎所有主流品牌扫描仪点云数据&#xff0c;包括Leica、Riegl、ZF、Faro、Topcon等。 2、业界领先的无目标全自动配准 T…

Apache Flink 流批融合技术介绍

摘要&#xff1a;本文整理自阿里云高级研发工程师、Apache Flink Contributor 周云峰老师在 Apache Asia CommunityOverCode 2024中的分享。内容主要分为以下三个部分&#xff1a; 从流批一体到流批融合流批融合的技术解决方案社区进展及未来展望 一、从流批一体到流批融合 1&…

记忆化搜索专题——算法简介力扣实战应用

目录 1、记忆化搜索算法简介 1.1 什么是记忆化搜索 1.2 如何实现记忆化搜索 1.3 记忆化搜索与动态规划的区别 2、算法应用【leetcode】 2.1 题一&#xff1a;斐波那契数 2.1.1 递归暴搜解法代码 2.1.2 记忆化搜索解法代码 2.1.3 动态规划解法代码 2.2 题二&#xff1…

网络模型的保存与读取

文章目录 一、模型的保存二、文件的加载三、模型加载时容易犯的陷阱 一、模型的保存 方式1:torch.save(vgg16, “vgg16_method1.pth”) import torch import torchvision.modelsvgg16 torchvision.models.vgg16(pretrainedFalse) torch.save(vgg16, "vgg16_method1.pth…

oracle数据库启动

文章目录 背景一、步骤1.登录oracle用户2.启动监听服务3.启动数据库 背景 oracle数据库启动 一、步骤 1.登录oracle用户 代码如下&#xff08;示例&#xff09;&#xff1a; su - oracle2.启动监听服务 代码如下&#xff08;示例&#xff09;&#xff1a; lsnrctl start成…

【C++】STL----vector常见用法

&#x1f525;个人主页&#x1f525;&#xff1a;孤寂大仙V &#x1f308;收录专栏&#x1f308;&#xff1a;C从小白到高手 &#x1f339;往期回顾&#x1f339;&#xff1a;[C]string类 &#x1f516; 流水不争&#xff0c;争的是滔滔不息。 文章目录 一、vector的介绍vector…

MATLAB绘图基础8:双变量图形绘制

参考书&#xff1a;《 M A T L A B {\rm MATLAB} MATLAB与学术图表绘制》(关东升)。 8.双变量图形绘制 8.1 散点图 散点图用于显示两个变量间的关系&#xff0c;每个数据点在图上表示为一个点&#xff0c;一个变量在 X {\rm X} X轴&#xff0c;一个变量在 Y {\rm Y} Y轴&#…