Java - 基本数据类型 / 对象的赋值、引用、参数传递篇

简介: Java - 基本数据类型 / 对象的赋值、引用、参数传递篇

案例一:

public class Solution {
    private int a;
    private int b;
    public void setA(int a) {
        this.a = a;
    }
    public void setB(int b) {
        this.b = b;
    }
}
     Solution s1 = new Solution();
     s1.setA(1);
     s1.setB(2);
     Solution s2 = s1;
     s2.setA(-1);

image.png


Ps1:对于Solution s1 = new Solution();这条语句,这条语句执行的动作是创建一个对象,我们都很明白,但是它确包含了四个步骤:


右边“new Solution”,表示以Solution类为模板,在堆空间中创建一个Solution类对象;

“()”,这对括号表示,在对象创建后,会立马调用Solution类的构造函数,由于没有给参数,所以会调用默认的无参构造。

左边的“Solution s1 ”创建了一个Solution类的引用变量,也就是说用来指向Solution对象的对象引用。这和C语言中的指针可以理解为一个意思。

“=”,这个等号操作符表示使对象引用s1指向刚创建的Solution对象。


Ps2:为了形象地说明对象、引用及它们之间的关系,可以做一个比喻:

A有一栋房子,他把地址告诉了B,B就知道了A家的地址,然后B又把A家的地址告诉了C。此时,B和C都知道了A家的地址,但是这栋房子确是属于A的。如果此时,B去A家拿了一个东西,然后C去了后会发现少了一样东西。所以B和C只是知道A家的地址,可以随时访问,但是确不是独有的。


Ps3:两个结论:

(1)一个对象引用可以指向0个或1个对象(一个人可以知道房子的地址,也可以不知道)。

(2)一个对象可以有N个引用指向它(可以有N个人知道房子的地址)。


再理解了对象和引用的关系后,再来看参数传递。程序设计语言中,将参数传递给方法(或函数)有两种方法。按值传递(call by value)表示方法接受的是调用者提供的值;按引用调用(call by reference)表示方法接受的是调用者提供的变量地址。Java程序设计语言都是采用按值传递。【但是Java只有一种参数传递方式:那就是按值传递】,即Java中传递任何东西都是传值。如果传入方法的是基本类型的东西,你就得到此基本类型的一份拷贝。如果是传递引用,就得到引用的拷贝。


案例二:


/

public class Solution {
    // 基本类型的参数传递
    public static void fun1(int m){
        m = 100;
    }
    // 参数为对象,不改变引用的值
    public static void fun2(StringBuffer s){
        s.append("fun2");
    }
    // 参数为对象,改变引用的值
    public static void fun3(StringBuffer s){
        s = new StringBuffer("fun3");
    }
    public static void main(String[] args) {
        int i = 1;
  fun1(i);
        System.out.println(i);// i = 1
        StringBuffer ss = new StringBuffer("main");
        System.out.println(ss.toString());// main
        fun2(ss);
        System.out.println(ss.toString());// mainfun2
        fun3(ss);
        System.out.println(ss.toString());// mainfun2
    }
}

Ps:以上结果中:


fun1方法的参数是基本类型,尽管在 fun1 中参数 m 的值发生了改变,但是并不影响 i。

fun2方法的参数是一个对象,当把ss传给参数s时,s得到的是ss的拷贝,所以s和ss指向的是同一个对象,因此,使用s操作对象,ss也会受影响。

fun3方法的参数虽然也是一个对象,当ss传给参数s时,s得到的是ss的拷贝,但是,在fun3中改变了s指向的对象,给s重新赋值后,s与ss已经没有关系,它和ss指向了不同的对象,所以不管对s做什么操作,ss都不会受影响。


案例三:

public class ParamTest {
    public static void main(String[] args) {
        /*
        *Test1: Methods can't modify numeric parameters
        */
        System.out.println("Testing tripleValue:");
        double percent = 10;
        System.out.println("Before: percent=" + percent);
        tripleValue(percent);
        System.out.println("After: percent=" + percent);
        /*
        *Test2: Methods can change the state of object parameters
        */
        System.out.println("\nTesting tripleSalary");
        Employee harry = new Employee("Harry", 50000);
        System.out.println("Before: salary=" + harry.getSalary());
        tripleSalary(harry);
        System.out.println("After: salary=" + harry.getSalary());
        /*
        *Test3: Methods can't attach new objects to object parameters
        */
        System.out.println("\nTesting swap");
        Employee a = new Employee("Alice", 30000);
        Employee b = new Employee("Bob", 60000);
        System.out.println("Before: a=" + a.getName());
        System.out.println("Before: b=" + b.getName());
        swap(a, b);
        System.out.println("After: a=" + a.getName());
        System.out.println("After: b=" + b.getName());
    }
    public static void tripleValue(double x) {  
        x *= 3;
        System.out.println("End of method: x=" + x);
    }
    public static void tripleSalary(Employee x) {
        x.raiseSalary(200);
        System.out.println("End of method: salary=" + x.getSalary());
    }
    public static void swap(Employee x, Employee y) {
        Employee temp = x;
        x = y;
        y = temp;
        System.out.println("End of method: x=" + x.getName());
        System.out.println("End of method: y=" + y.getName());
    }
}
class Employee {
    private String name;
    private double salary;
    public Employee(){}
    public Employee(String name, double salary){
        this.name = name;
        this.salary = salary;
    }
    public String getName() {
        return name;
    }
    public double getSalary() {
        return salary;
    }
    public void raiseSalary(double byPercent){
        double raise = salary * byPercent / 100;
        salary += raise;
    }
}

image.png


Ps:从以上例题可以总结Java中方法参数的使用情况:

  1. 一个方法不能修改一个基本数据类型的参数(即数值型或布尔型)。
  2. 一个方法可以改变一个对象(数组)参数的状态。
  3. 一个方法不能让对象参数(数组)引用一个新的对象。
目录
相关文章
|
15天前
|
安全 Java 编译器
Java对象一定分配在堆上吗?
本文探讨了Java对象的内存分配问题,重点介绍了JVM的逃逸分析技术及其优化策略。逃逸分析能判断对象是否会在作用域外被访问,从而决定对象是否需要分配到堆上。文章详细讲解了栈上分配、标量替换和同步消除三种优化策略,并通过示例代码说明了这些技术的应用场景。
Java对象一定分配在堆上吗?
|
8天前
|
存储 缓存 Java
大厂面试必看!Java基本数据类型和包装类的那些坑
本文介绍了Java中的基本数据类型和包装类,包括整数类型、浮点数类型、字符类型和布尔类型。详细讲解了每种类型的特性和应用场景,并探讨了包装类的引入原因、装箱与拆箱机制以及缓存机制。最后总结了面试中常见的相关考点,帮助读者更好地理解和应对面试中的问题。
27 4
|
6天前
|
存储 消息中间件 NoSQL
使用Java操作Redis数据类型的详解指南
通过使用Jedis库,可以在Java中方便地操作Redis的各种数据类型。本文详细介绍了字符串、哈希、列表、集合和有序集合的基本操作及其对应的Java实现。这些示例展示了如何使用Java与Redis进行交互,为开发高效的Redis客户端应用程序提供了基础。希望本文的指南能帮助您更好地理解和使用Redis,提升应用程序的性能和可靠性。
15 1
|
19天前
|
Java API
Java 对象释放与 finalize 方法
关于 Java 对象释放的疑惑解答,以及 finalize 方法的相关知识。
39 17
|
19天前
|
存储 安全 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第22天】在Java的世界里,对象序列化和反序列化是数据持久化和网络传输的关键技术。本文将带你了解如何在Java中实现对象的序列化与反序列化,并探讨其背后的原理。通过实际代码示例,我们将一步步展示如何将复杂数据结构转换为字节流,以及如何将这些字节流还原为Java对象。文章还将讨论在使用序列化时应注意的安全性问题,以确保你的应用程序既高效又安全。
|
21天前
|
存储 Java 关系型数据库
[Java]“不同族”基本数据类型间只能“强转”吗?
本文探讨了不同位二进制表示范围的计算方法,重点分析了Java中int和char类型之间的转换规则,以及float与int类型之间的转换特性。通过具体示例说明了显式和隐式转换的条件和限制。
33 0
[Java]“不同族”基本数据类型间只能“强转”吗?
|
18天前
|
存储 缓存 NoSQL
一篇搞懂!Java对象序列化与反序列化的底层逻辑
本文介绍了Java中的序列化与反序列化,包括基本概念、应用场景、实现方式及注意事项。序列化是将对象转换为字节流,便于存储和传输;反序列化则是将字节流还原为对象。文中详细讲解了实现序列化的步骤,以及常见的反序列化失败原因和最佳实践。通过实例和代码示例,帮助读者更好地理解和应用这一重要技术。
16 0
|
20天前
|
存储 Java 编译器
[Java]基本数据类型与引用类型赋值的底层分析
本文详细分析了Java中不同类型引用的存储方式,包括int、Integer、int[]、Integer[]等,并探讨了byte与其他类型间的转换及String的相关特性。文章通过多个示例解释了引用和对象的存储位置,以及字符串常量池的使用。此外,还对比了String和StringBuilder的性能差异,帮助读者深入理解Java内存管理机制。
18 0
|
6天前
|
安全 Java 测试技术
Java并行流陷阱:为什么指定线程池可能是个坏主意
本文探讨了Java并行流的使用陷阱,尤其是指定线程池的问题。文章分析了并行流的设计思想,指出了指定线程池的弊端,并提供了使用CompletableFuture等替代方案。同时,介绍了Parallel Collector库在处理阻塞任务时的优势和特点。