java中方法的使用

news2024/9/30 15:25:23

方法的使用

  • 方法的概念
    • 什么是方法
    • 方法定义
    • 方法的调用过程
    • 实参和形参的关系
  • 方法重载
    • 为什么需要方法重载
    • 方法重载的概念
    • 方法签名
  • 递归
    • 递归的概念
    • 递归过程分析
    • 递归练习

方法的概念

什么是方法

方法就是一个代码片段,类似于C语言的函数。
方法存在的意义:

  1. 是能够模块化的组织代码(当代码规模比较复杂的时候)
  2. 做到代码被重复使用,一份代码可以在多个位置使用
  3. 让代码更好理解更简单
  4. 直接调用现有方法使用,不用重复写一份代码

方法定义

语法格式:

//方法定义
修饰符 返回值类型 方法名称([参数类型 形参...]){
	方法体代码;
	[return 返回值];
}

例子1:实现一个函数,用于检测年份是否为闰年

public static boolean isLeapYear(int year){
        if ((year % 400 == 0) || ((year % 100 != 0) && (year % 4 == 0))){
            return true;
        }
        return false;
    }
public static void main(String[] args) {
        int year = 2023;
        if (isLeapYear(year)){
            System.out.println("是闰年");
        }else{
            System.out.println("不是闰年");
        }
    }

注意:

  • 返回值类型:如果方法有返回值,返回值类型必须要与返回的实体类型一致,如果没有返回值,必须写成void
  • 方法名字:采用小驼峰命名(首个单词的首个字母小写,后续单词首字母大写)
  • 参数列表:如果方法没有参数。()中什么都不写,如果有参数,需指定参数类型,多个参数之间用逗号隔开
  • 方法体:方法内部要执行的语句
  • 在java当中,方法必须写在类当中
  • 在java中,方法不能嵌套定义
  • 在java中,没有方法声明一说

方法的调用过程

调用方法–>传递参数–>找到方法地址–>执行被调方法的方法体–>被调方法结束返回–>回到主调方法继续往下执行
注意:

  • 定义方法的时候,不会执行方法的代码,只有调用的时候才会执行
  • 一个方法可以被多次调用
    例子:计算 1! + 2! + 3! + 4! + 5!
//计算 1! + 2! + 3! + 4! + 5!
    public static int func(int n){
        int result = 1;
        for (int i = 1; i <= n; i++) {
            result *= i;
        }
        return result;
    }

    public static void main(String[] args) {
        int sum = 0;
        for (int i = 1; i <= 5; i++) {
            sum += func(i);
        }
        System.out.println(sum);
    }

实参和形参的关系

java当中方法的形参相当于数学函数的自变量,用来接收函数调用时传递的值,形参的名字是可以随意取的,对方法没有任何影响,形参只是方法在定义时需要借助的一个变量,用来保存方法在调用时传递过来的值。

public static int add(int a, int b){    //a, b是形参
        return a + b;
    }
public static void main(String[] args){
        int ret = add(2,3); //2,3是实参,a,b分别用来保存2,3,在调用时传给形参a和b
        System.out.println(ret);
    }

注意:在java中,实参的值永远是拷贝到形参中,形参和实参本质是两个实体。
例子:交换两个整型变量

public static void swap(int a, int b){
            int tmp = a;
            a = b;
        b = tmp;
        System.out.println("swap:a =  " + a + " b = " + b);
    }

    public static void main(String[] args) {
        int a =10;
        int b = 20;
        swap(a, b);
        System.out.println("main:a =  " + a + " b = " + b);
        //运行结果:
        //swap:a =  20 b = 10
        //main:a =  10 b = 20
    }

可以看到,在swap交换之后,形参a,b值发生了改变,但是main方法中a和b还是交换之前的值,即没有交换成功。
原因:因为实参a,b是main方法中的变量,其空间在main方法的栈中,而形参a和b是swap方法中的两个变量,a和b的空间在swap方法运行时的栈中,因此实参a和b与形参a和b是两个没有关联性的变量,在swap方法调用时,只是将实参a和b值拷贝一份传递给了形参a和b,因此对形参a和b操作不会对实参a和b产生任何影响
注意:对于基础类型,形参相当于实参的拷贝,即传值调用
解决方法:传引用类型参数(例如通过数组来解决)(java中,拿不到栈上变量的地址)

public static void swap(int[] arr){
        int tmp = arr[0];
        arr[0] = arr[1];
        arr[1] = tmp;
    }

    public static void main(String[] args) {
        int[] arr = {10,20};
        swap(arr);
        System.out.println("main:"+arr[0] + " " + arr[1]);

    }

方法重载

为什么需要方法重载

public static int add(int a, int b){
        return a + b;
    }

    public static void main(String[] args) {
        int x = 10;
        int y = 20;
        int ret1 = add(x,y);
        double a = 2.0;
        double b = 1.0;
        double ret2 = add(a,b);//编译报错:不兼容的类型: 从double转换到int可能会有损失
    }

任何解决呢
我们可能会采取再写另外一个方法

public static int add(int a, int b){
        return a + b;
    }

    public static double addDouble(double a, double b){
        return a + b;
    }

    public static void main(String[] args) {
        int x = 10;
        int y = 20;
        int ret1 = add(x,y);
        double a = 2.0;
        double b = 1.0;
        double ret2 = addDouble(a,b);//编译报错:不兼容的类型: 从double转换到int可能会有损失
    }

但是随着需求和要求增加,每增加新类型参数就需要新写一个方法,重新起一个方法名,想许多不同的方法名是一件比较烦人的事,能否都是同一个方法名呢?

方法重载的概念

在java中,方法是可以重载的,在java中,如果多个方法的名字相同,参数列表不同,则说该几种方法被重载了。
例子:

public static int add(int a, int b){
        return a + b;
    }

    public static double add(double a, double b){
        return a + b;
    }

    public static double add(double a, double b, double c){
        return a + b + c;
    }

    public static void main(String[] args) {
        int x = 10;
        int y = 20;
        int ret1 = add(x,y);//调用add(int,int)
        double a = 2.0;
        double b = 1.0;
        double c = 3.0;
        double ret2 = add(a,b);//调用add(double,double)
        double ret3 = add(a,b,c);//调用add(double,double,double)
    }

注意:

  1. 方法名必须相同
  2. 参数列表必须不同(参数的个数、参数的类型、类型的次序)
  3. 与返回值类型是否相同无关
  4. 如果两个数仅仅只是因为返回值类型不同,是不能构成重载的,会编译出错
  5. 编译器在编译代码时,会对实参类型进行推演,根据推演的结果来确定调用哪个方法

方法签名

方法签名:经过编译器编译修改过之后方法的最终名字。具体方式:方法全路径名+参数列表+返回值类型,构成方法完整的名字。

public static int add(int a, int b){
        return a + b;
    }

    public static double add(double a, double b){
        return a + b;
    }

    public static double add(double a, double b, double c){
        return a + b + c;
    }

    public static void main(String[] args) {
        int x = 10;
        int y = 20;
        int ret1 = add(x,y);//调用add(int,int)
        double a = 2.0;
        double b = 1.0;
        double c = 3.0;
        double ret2 = add(a,b);//调用add(double,double)
        double ret3 = add(a,b,c);//调用add(double,double,double)
    }

上述代码经过编译之后,然后使用JDK自带的javap反汇编工具查看,具体操作:

  1. 先对工程进行编译生成.class字节码文件
  2. 在控制台中进入到要查看的.class所在的目录
  3. 输入:javap -v 字节码文件名字即可
    在这里插入图片描述
特殊字符数据类型
Vvoid
Zboolean
Bbyte
Cchar
Sshort
Iint
Jlong
Ffloat
Ddouble
[数组(以[开头,配合其他的特殊字符,表述对应数据类型的数组,几个[表述几维数组)
L引用类型,以L开头,以;结尾,中间是引用类型的全类名

递归

递归的概念

一个方法在执行过程中调用自身,即”递归“
解决递归问题:

  1. 有一个递归公式
  2. 找到当前递归问题的解决条件
    例子:用递归求N的阶乘
public static int func(int n){
        if (n == 1){
            return 1;
        }
        return n * func(n - 1);
    }

    public static void main(String[] args) {
        int n = 5;
        int ret = func(5);
        System.out.println(ret);
        //结果是120
    }

递归过程分析

理解清楚递归,首先必须理解清楚“方法的执行过程”,尤其是“方法执行结束之后,回到调用位置继续往下执行”
在这里插入图片描述

关于调用栈
方法调用的时候,会有一个“栈”这样的内存空间描述当前的调用关系,称为调用栈。
每一次的方法调用就称为一个“栈帧”,每个栈帧中包含了这次调用的参数是哪些,返回到哪里继续执行等信息。

递归练习

例子1:按顺序打印数字的每一位(例如1234,打印1 2 3 4)

//按顺序打印数字的每一位(例如1234,打印1 2 3 4)
    public static void print(int n){
        if (n > 9){
            print(n / 10);
        }
        System.out.print(n % 10 + " ");
    }
    public static void main(String[] args) {
        int num = 1234;
        print(num);
    }

例子2:递归求1 + 2 + 3 + … + 9 + 10

//递归求1 + 2 + 3 + ... + 9 + 10
    public static int sum(int n){
        if (n == 1){
            return 1;
        }
        return n + sum(n -1);
    }

    public static void main(String[] args) {
        int num = 10;
        int ret = sum(num);
        System.out.println(ret);
        //计算结果是55
    }

例子3:写一个递归方法,输入一个非负整数,返回组成它的数字之和. 例如,输入 1729, 则应该返回1+7+2+9,它的和是19

//写一个递归方法,输入一个非负整数,返回组成它的数字之和. 例如,输入 1729, 则应该返回1+7+2+9,它的和是19
    public static int sum(int n){
        if (n < 10) {
            return n;
        }
        return n % 10 + sum(n / 10);
    }

    public static void main(String[] args) {
        int num = 1729;
        int ret = sum(num);
        System.out.println(ret);
    }

例子4:求斐波那契数列的第N项

//求斐波那契数列的第N项
    public  static int fib(int n){
        if (n == 1 || n == 2){
            return 1;
        }
        return fib(n - 2) + fib(n-1);
    }

    public static void main(String[] args) {
        System.out.println(fib(5));
    }

当我们求fib(40)的时候发现程序运行速度很慢,原因是进行了大量的重复运算

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

    public static void main(String[] args) {
        System.out.println(fib(40));
        System.out.println(count );
    }
    //计算结果:102334155
	//			39088169(fib(3)计算的次数)

我们可以使用循环求斐波那契数列,避免冗余的计算。

public static int fib(int n){
        int num1 = 1;
        int num2 = 1;
        for (int i = 2; i < n; i++) {
            int tmp = num1 + num2;
            num1 = num2;
            num2 = tmp;
        }
        return num2;
    }

    public static void main(String[] args) {
        System.out.println(fib(40));
    }

———————————————————————————————————————————
对于方法的学习我们先了解到这里,后续我们继续学习和了解java的内容。

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

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

相关文章

MySQL 面试相关问题

1. MySQL 基础问题1.1 为什么用MySQL&#xff1f;1.2 表属性类型 varchar 和 char 的区别&#xff1f;1.2 什么时候用 varchar 和 char&#xff1f;1.3 Datetime 和 Timestamp 的区别&#xff1f;1.4 一个SQL语句的执行过程&#xff0c;表述下&#xff1f; 2. MySQL 存储引擎相…

Docker部署gitlab私有仓库后查看root默认密码以及修改external_url路径和端口的方法

文章目录 1、docker部署最新版gitlab2、进入gitlab容器3、修改路径地址ip和端口4、检验效果 1、docker部署最新版gitlab #docker安装命令 docker run --detach \--name gitlab \--restart always \-p 1080:80 \-p 10443:443 \-p 1022:22 \-v /gitlab/config:/etc/gitlab \-v …

预算有限?如何挑选经济适用的安全管理系统?

如今&#xff0c;无论是信息安全、生产安全还是人员安全&#xff0c;都直接关系到企业的稳定运营和长远发展。然而&#xff0c;对于许多中小企业而言&#xff0c;高昂的安全管理系统投入往往成为一大难题。那么&#xff0c;在预算有限的情况下&#xff0c;如何挑选一款既经济适…

04 效用评测层 显性指标+标准方法 对推荐效果定量评测

你好&#xff0c;我是大师兄。前面三节课我们介绍了推荐系统的数据采集层和数据加工层的相关内容&#xff0c;本节课我们重点介绍下推荐系统效用评价层的评测方法和评测指标。 这里的“效用评测”&#xff0c;通俗点说是指推荐系统输出数据的好坏是否符合用户和商业的预期&…

window 安装 openssl

文章目录 前言window 安装 openssl1. 下载2. 安装3. 配置环境变量4. 测试 前言 如果您觉得有用的话&#xff0c;记得给博主点个赞&#xff0c;评论&#xff0c;收藏一键三连啊&#xff0c;写作不易啊^ _ ^。   而且听说点赞的人每天的运气都不会太差&#xff0c;实在白嫖的话…

C++ 十进制与十六进制之间相互转换

十进制与十六进制之间相互转换 10_to_16 与二进制类似&#xff0c;十进制转十六进制对16整除&#xff0c;得到的余数的倒序即为转换而成的十六进制&#xff0c;特别地&#xff0c;如果超过10以后&#xff0c;分别用ABCDEF或abcdef来代替10、11、12、13、14、15。 代码1: #in…

Codeforces Round 954 (Div. 3) F. Non-academic Problem

思路&#xff1a;考虑缩点&#xff0c;因为是无向图&#xff0c;所以双连通分量缩完点后是一棵树&#xff0c;我们去枚举删除每一条树边的答案&#xff0c;然后取最小值即可。 #include <bits/stdc.h>using namespace std; const int N 3e5 5; typedef long long ll; …

【学术会议征稿】第四届机械自动化与电子信息工程国际学术会议(MAEIE 2024)

第四届机械自动化与电子信息工程国际学术会议&#xff08;MAEIE 2024&#xff09; 2024 4th International Conference on Mechanical Automation and Electronic Information Engineering 由安徽大学主办&#xff0c;安徽大学电气工程与自动化学院、安徽省人机共融系统与智能…

利用远程桌面进行开发,

现在的软硬件开发都涉及庞杂的软硬件环境和多种外设总线部署&#xff0c;这时我们利用远程工具和windows自带的wsl虚拟机环境再配合vscode的remote ssh远程开发模式&#xff0c;可自由的在linux windows android等平台上切换&#xff0c;让开发更顺畅&#xff0c;也可以更好的利…

数据结构--单向链表篇(python实现)

写在开头 链表&#xff08;Linked list&#xff09;是一种常见的基础数据结构&#xff0c;是一种线性表&#xff0c;但是并不会按线性的顺序存储数据&#xff0c;而是在每一个节点里存到下一个节点的指针(Pointer) 链表的优缺点 优点 不需要预先知道数据大小&#xff0c;实现灵…

简易限流实现

需求描述 写一个1秒两个的限流工具类&#xff0c;2r/s 使用semaphore 代码实现-类似令牌桶算法 public class LimitHelper {private int maxLimit;private Semaphore semaphore;private int timeoutSeconds;public LimitHelper(int maxLimit, int timeoutSeconds) {this.max…

向github远程仓库中push,要求使用token登录

Support for password authentication was removed on August 13, 2021. Please use a personal access token instead. 如上&#xff0c;当向github远程仓库push时&#xff0c;输入github的用户名和密码出现如上错误&#xff0c;要求使用token登录&#xff0c;此时只需要用户…

【uni-app+Vue3】 API请求封装:让接口调用更便捷

前言&#xff1a;uni-app是一款基于Vue.js框架的跨平台开发工具&#xff0c;可以将代码编译成H5、小程序、App等不同平台的应用。在进行uni-app开发时&#xff0c;网络请求是必不可少的环节。为了方便开发&#xff0c;我们可以封装一些网络请求方法&#xff0c;以便在多个页面中…

数据结构(初阶1.复杂度)

文章目录 一、复杂度概念 二、时间复杂度 2.1 大O的渐进表示法 2.2 时间复杂度计算示例 2.2.1. // 计算Func2的时间复杂度&#xff1f; 2.2.2.// 计算Func3的时间复杂度&#xff1f; 2.2.3.// 计算Func4的时间复杂度&#xff1f; 2.2.4.// 计算strchr的时间复杂度&#xff1f; …

RSRS研报复现——年化21.5%,含RSRS标准分,右偏标准分的Backtrader指标计算(代码+数据)

原创文章第583篇&#xff0c;专注“AI量化投资、世界运行的规律、个人成长与财富自由"。 继续Backtrader&#xff0c;今天讲讲指标扩展。 作为规则型的量化框架&#xff0c;指标是非常重要的元素&#xff0c;它是策略的基础。 我们来扩展一个经典的指标&#xff0c;RSR…

docker 安装教程

机缘 最近在开发用用到了docker,主要是将开发的项目发版到平台上&#xff0c;运用到docker发版&#xff0c;所以才会写这篇文章。 教程 1、安装Hype-v。 在安装之前&#xff0c;首先要检查下电脑系统是否安装Hyper-v功能。 注意&#xff1a;一定要检查电脑系统&#xff0c;…

在繁华与奇迹交织的深圳

在繁华与奇迹交织的深圳&#xff0c;有一本奇书悄然走红&#xff0c;它便是《华强北经济学》&#xff0c;由那位深谙市场脉搏、笔触带风的宋仕强先生所著。这本书&#xff0c;仿佛是华强北这片创业热土的缩影&#xff0c;用一个个生动鲜活的故事&#xff0c;诠释了什么是“从螺…

Go语言---正则表达式

正则表达式是一种进行模式匹配和文本操纵的复杂而又强大的工具。虽然正则表达式比纯粹的文本匹配效率低&#xff0c;但是它却更灵活。按照它的语法规则&#xff0c;只需构造出的匹配模式就能够从原始文本中筛选出几乎任何你想要得到的字符组合。go语言的通过regexp标准包来实现…

【C++报错已解决】Invalid Conversion from ‘const char*’ to ‘char*’

&#x1f3ac; 鸽芷咕&#xff1a;个人主页 &#x1f525; 个人专栏: 《C干货基地》《粉丝福利》 ⛺️生活的理想&#xff0c;就是为了理想的生活! 文章目录 引言 ❓ 一、问题描述 &#x1f469;‍&#x1f52c;1.1 报错示例 &#x1f3c6;1.2 报错分析 &#x1f4da;1.3 解决…

【python】PyQt5事件传递,鼠标动作捕获,键盘按键捕获原理与应用实战

✨✨ 欢迎大家来到景天科技苑✨✨ &#x1f388;&#x1f388; 养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; &#x1f3c6; 作者简介&#xff1a;景天科技苑 &#x1f3c6;《头衔》&#xff1a;大厂架构师&#xff0c;华为云开发者社区专家博主&#xff0c;…