首页 > 其他 > 详细

4.2 方法详解

时间:2020-02-22 22:11:06      阅读:53      评论:0      收藏:0      [点我收藏+]

一、方法的所属性

方法由传统的函数发展而来,方法与传统的函数显著不同:在结构化编程中,函数是一等公民,这个程序由一个个函数组成;在面向对象编程语言里,类才是一等公民,整个系统由一个个类组成。因此在Java语言里,方法不能独立存在,方法必须属于类或对象。
方法的所属性:
(1)方法类似于函数。但与函数不同的是,方法不能存在,方法必须定义在类里面。
(2)方法一定要有执行者,必须通过类或对象来调用方法。从逻辑上来看,该方法属于类本身,应该用类来调用
    如果该方法有static修饰,该方法属于类本身,应该用类调用;
    如果该方法无static修饰,该方法属于对象本身,应该用对象调用
【规则】如果调用同一个类中方法,可以省略调用者,此时系统会默认添加调用者。如果方法是无static方法,添加this作为默认调用者;如果方法是static方法,添加类作为调用者。

class Method_attribute 
{
    //定义一个普通方法
    public void nonStaticMethod()
    {
        System.out.println("这是一个普通方法");
    }
    //定义一个static方法
    public static void StaticMethod()
    {
        System.out.println("这是一个类方法");
    }
    //在同一个类中一个方法调用另外一个方法
    public void test()
    {
        this.nonStaticMethod();
        this.StaticMethod();
        StaticMethod();//省略的是主调类
    }
    public static void main(String[] args) 
    {
        var p=new Method_attribute();
        //此时test()方法中的两个this代表对象p
        p.test();        
    }
}
---------- 运行Java捕获输出窗 ----------
这是一个普通方法
这是一个类方法
这是一个类方法
输出完成 (耗时 0 秒) - 正常终止

二、方法参数的传递机制

Java里方法是不能独立存在的,调用方法时必须使用类或对象作为主调者。如果声明方法时,包含了形参声明,则调用方法时必须给这些形参指定参数值,调用参数时传给形参参数值也称为实参。
Java里面参数传递方式只有一种:值传递。所谓值传递,就是将实际参数的副本(复制品)传入方法内,而参数本身不会受到影响。

class ParamTransferTest 
{
    public static void swap(int a,int b)
    {
        //实现变量a和b的值交换
        //定义一个临时变量来保存a的值
        var temp=a;
        a=b;
        b=temp;
        System.out.println("swap方法里,a的值为:"+a+",b的值为:"+b);
    }
    public static void main(String[] args) 
    {
        int a=6;
        int b=9;
        swap(a,b);
        System.out.println("交换结束后,a的值为:"+a+",b的值为:"+b);
    }
}
---------- 运行Java捕获输出窗 ----------
swap方法里,a的值为:9,b的值为:6
交换结束后,a的值为:6,b的值为:9

输出完成 (耗时 0 秒) - 正常终止

Java程序从main()方法开始执行,main()方法开始定义了a、b两个局部变量,如图所示:
技术分享图片
程序从main()函数开始执行,当程序进入swap()方法时,系统分配了两个栈区,将mian()方法中的变量a、b的副本传入swap()方法,而不是a、b本身。
程序在swap()方法中,进行变量a、b交换的值,交换结束后,内存中的存储情况:
技术分享图片
两个示意图可以发现,mian()方法栈区中a、b值并未发生改变,程序只改变swap()栈区中的变量a、b。这就是值传递的实质:当系统开始执行方法时,系统为形参初始化,就是把实参变量的值赋给方法的形参变量,方法里操作的并不是实参变量。
再看一个例子:

class DataWrap
{
    int a;
    int b;
}

public class ReferenceTransferTest 
{
    public static void swap(DataWrap dw)
    {
        //下面实现的dw的两个成员的变量值的交换
        var temp=dw.a;
        dw.a=dw.b;
        dw.b=temp;
        System.out.println("swap()方法里,a成员变量的值是:" + dw.a + ",b成员变量的值是:"+dw.b);
    }

    public static void main(String[] args) 
    {
        var dw=new DataWrap();
        dw.a=6;
        dw.b=9;
        swap(dw);
        System.out.println("swap()方法里,a成员变量的值是:"+dw.a+",b成员变量的值是:"+dw.b);
    }
}
---------- 运行Java捕获输出窗 ----------
swap()方法里,a成员变量的值是:9,b成员变量的值是:6
swap()方法里,a成员变量的值是:9,b成员变量的值是:6

输出完成 (耗时 0 秒) - 正常终止

上面结果swap()方法里和mian()方法里的两个变量a、b都发生了改变。这很容易产生错觉:调用swap()方法时,传入的时dw本身,而不是它的复制品。
下面分析一下程序执行的过程:
(1)程序从mian()方法开始执行:
var dw=new DataWrap();
dw.a=6;
dw.b=9;
这里创建了一个DataWrap对象,并赋给引用变量dw。堆内存保存该对象本身,栈内存保存的该对象的引用变量。
技术分享图片
(2)执行swap(dw);
接下来main()方法开始调用swap()方法,mian(0方法并未结束,系统会为main()和swap()开辟两个栈区,用于存放mian()和swap()方法里的局部变量。调用swap(0方法时,dw作为实参传入swap()方法,同样采用值传递:把main()方法里的dw变量赋值给swap()方法里的dw形参,从而完成swap()方法的dw形参初始化。下图显示main()方法中实参dw传入swap()方法后的存储示意图:
技术分享图片
系统只复制了dw变量,但未复制DataWrap对象。
  不管程序在swap()方法中还是在mian()方法的操作dw变量时,实际操作的都是堆内存中的DataWrap对象,他们引用的是同一个变量。因此在swap()方法中交换了dw所引用的a、b两个成员变量后,可看到在main()方法中dw引用变量引用的a、b变量的值也发生了改变。
技术分享图片
实际上,我们在创建swap()方法时,将

public static void swap(DataWrap dw);中的dw换个符号表示理解起来就更容易,不易混淆

三、形参个数可变的方法

Java允许定义形参个数可变的参数,从而允许为方法指定数量不确定的形参。如果在定义方法时,在最后一个形参的类型后增加三点(...),则表示该形参可以接受多个参数值,多个参数被当作数组传入。

class  VarArgs
{
    public void test(int a,String... names)
    {
        System.out.println("a参数:"+a);
        System.out.println("names数组的长度:"+names.length);
        for(int i=0;i<names.length;i++)
        {
            System.out.println(names[i]);
        }
    }
    public static void main(String... args) 
    {
        VarArgs va=new VarArgs();
        va.test(2,"efdf","fdfs","fjgd");
        va.test(34,new String[]{"孙悟空","猪八戒"});
    }
}
class  VarArgs
{
    public void test(int a,String... names)
    {
        System.out.println("a参数:"+a);
        System.out.println("names数组的长度:"+names.length);
        for(int i=0;i<names.length;i++)
        {
            System.out.println(names[i]);
        }
    }
    public static void main(String... args) 
    {
        VarArgs va=new VarArgs();
        va.test(2,"efdf","fdfs","fjgd");
        va.test(34,new String[]{"孙悟空","猪八戒"});
    }
}
---------- 运行Java捕获输出窗 ----------
a参数:2
names数组的长度:3
efdf
fdfs
fjgd
a参数:34
names数组的长度:2
孙悟空
猪八戒

输出完成 (耗时 0 秒) - 正常终止

va.test(2,"efdf","fdfs","fjgd");   va.test(34,new String[]{"孙悟空","猪八戒"});
这两种多形参传入方法都可以,但是第一种方法更加简洁。

类型[] 形参名

类型...写法的好处是:
调用方法时更加方便,即可直接传入多个元素,系统会自动将它们封装为数组
也可直接传入数组
确定:类型... 这种写法只能作为形参列表的最后一个形参
【暗示】一个方法只能由一个形参个数可变的形参

再看一个例子:

class VarIntTest 
{
    public int add(int... num)
    {
        int result=0;
        for (int i=0;i<num.length;i++)
        {
            result+=num[i];
        }
        return result;
    }
    public static void main(String[] args) 
    {
        var p=new VarIntTest();
        System.out.println(p.add(1,2,3,4));
    }
}
---------- 运行Java捕获输出窗 ----------
10

输出完成 (耗时 0 秒) - 正常终止

四、递归方法

一个方法体可以调用它自身,被称为方法递归。方法递归包含一种隐式循环,它会重复执行某段代码,但这种重复执行无需循环控制。
例子:已知f(0)=,f(1)=4,f(n+2)=2f(n+1)+f(n),其中n是大于0的整数,求f(10)。
递归必须向一直方向进行,如果使用f(n)=f(n+2)-2
f(n+1),f(10)=f(12)-2f(11),朝着大段方向进行,大段是未知的。故递归有一条重要的规则:必须向着已知方向递归。
对上面的迭代式子进行变形,f(n)=2
f(n-1)+f(n-2)

class Recursive 
{
    public static int fn(int n)
    {
        if(n==1)
            return 1;
        else if(n==2)
            return 4;
        else
        {
            return fn(n-2)+2*fn(n-1);
        }
    }
    public static void main(String[] args) 
    {
        
        System.out.println(new Recursive().fn(5));
    }
}
---------- 运行Java捕获输出窗 ----------
53

输出完成 (耗时 0 秒) - 正常终止

五、方法重载

Java允许在同一个类里定义多个同名的方法,只要形参列表不同就行。如果同一个类中包含两个或以上的方法名相同,但形参列表不同,则称为方法重载。
因此在Java中确定一个方法需要三个因素:
1、调用者,也就是方法的所属者,既可以是类,也可以是对象。
2、方法名,方法的标识。
3、形参列表,当调用方法时,系统会根据传入实参列表匹配。

方法重载要求两同一不同:同一个类的方法名相同,形参列表不同
修饰符不同(如static修饰的方法和无static方法修饰的方法)方法,带返回值或不带返回值或返回值类型不同的方法,形参列表不同的方法都不能算重载。

class OverLoad 
{
    public void test()
    {
        System.out.println("这是一个无参数的test方法");
    }
    public void test(String... strs)
    {
        System.out.println("这是一个形参个数可变的test方法");
    }

    protected void test(int a)
    {
        System.out.println("这是一个带int参数的test方法");
    }
    /*
    public static void test()
    {
        System.out.println("这是一个static修饰的test方法");
    }
    //OverLoad.java:17: 错误: 已在类 OverLoad中定义了方法 test()
    */
    /*
    public int test()
    {
        System.out.println("带返回值的test方法");
    }
    //OverLoad.java:23: 错误: 已在类 OverLoad中定义了方法 test()
    */

    public static void main(String[] args) 
    {
        OverLoad ov=new OverLoad();
        ov.test();
        ov.test("hello","world");
        ov.test(5);
    }
}
---------- 运行Java捕获输出窗 ----------
这是一个无参数的test方法
这是一个形参个数可变的test方法
这是一个带int参数的test方法

输出完成 (耗时 0 秒) - 正常终止

4.2 方法详解

原文:https://www.cnblogs.com/weststar/p/12345967.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!