菜鸡的Java笔记 第十六 - java 引用传递
referenceDelivery
引用传递是整个java 的精髓,也是所有初学者最难学的地方
引用的本质:同一块堆内存可以被不同的栈内存所指向
下面通过三道程序来进行引用传递分析
范例:第一道引用范例
class Demo{
private int num = 10:
public Demo(int num){
this.num = num: }
public void setNum(int num){
this.num = num:
}
public int getNum(){
return this.num:
}
}
public class referenceDelivery{
public static void main(String args[]){
Demo demo = new Demo(100):
fun(demo): // Demo temp = demo
System.out.println(demo.getNum()):
}
public static void fun(Demo temp){
temp.setNum(30):
}
}
// 结果: 30
范例:第二道引用传递
public class referenceDelivery{
public static void main(String args[]){
String str = "hello":
fun(str):
System.out.println(str):
}
public static void fun(String temp){
temp = "world":
}
}
// 结果: hello
字符串一旦声明则不可改变,字符串对象内容的改变依靠的是引用的改变实现的
如果觉得以上的代码看起来很麻烦,那么不如换一个思路:不要把String当成引用类型,而只当成基本数据类型
public class referenceDelivery{
public static void main(String args[]){
int str = 10:
fun(str):
System.out.println(str):
}
public static void fun(int temp){
temp = 30:
}
}
//结果:10
范例:第三道引用传递
class Demo{
private String msg = "Hello":
public Demo(String msg){
this.msg = msg:
}
public void setMsg(String msg){
this.msg = msg:
}
public String getMsg(){
rrturn this.msg:
}
}
public class referenceDelivery{
public static void main(String args[]){
Demo demo = new Demo ("world"):
fun(demo):
System.out.println(demo.getMsg()):
}
public static void fun(String temp){
temp.setMsg("MLDN"):
}
}
//结果: MLDN
引用传递一定要耐心使用内存分析,String这种类型额数据需要特殊处理
*/
/* Object comparison 对象比较
对象比较的实现以及引用的操作
所谓的对象比较指的就是可以判断两个对象是否相等
实际上现在对象是否相同只能够依靠地址是否相同但是有些时候可能会出现地址不相同
但是内容相同的的情况,就好比String类的“==”与“equals()”的关系
那么我们希望可以实现一个属性自己的对象比较操作
然而要进行对象比较操作那么首先要做的事情就是必须将两个对象中的每个属性内容分别比较
如果属性内容全部相同那么就表示同一个对象,否则就认为不是同一个对象
范例:对象比较的基础实现
class Person{
private String name:
private int age:
public Person (String name,int age){
this.name:
this.age|:
}
public String getName(){
rrturn this.name:
}
public int getAge(){
rrturn .this,age:
}
}
public class referenceDelivery{
public static void main(String args[]){
Person perA = new Person("张三",20):
Person perB = new Person("李四",20):
if(perA.getName(),equals(perB,getName())&&perA,getAge()==perB.getAge()){
System.out.println("是同一个人"):
}else{
System.out.println("不是同一个人"):
}
}
//结果: 不是同一个人
现在的功能已经实现了,但是不能够仅仅只是控制在实现的范畴,需要更好的实现,那么以上的代码有什么问题
直接的反映就是代码写在主方法里面那么如果不写在主方法里面该写在哪
class Person{
private String name:
private int age:
public Person (String name,int age){
this.name:
this.age|:
}
public String getName(){
rrturn this.name:
}
public int getAge(){
rrturn .this,age:
}
// 此时需要接收一个要比较的对象(对象的本质就是数据的集合)
// 同时在本类方法里面还有一个隐藏的对象:this
// 此时的compare()方法接收了自己本类的引用
// 由于此时的person参数对象在Person类中了,所以可以直接访问私有属性
public boolean compare(Person person){
if(person == null){// 就不要比较了
rrturn false:
}
if(this == person){ // 防止自己与自己比较,地址比较
rrturn true:
}
if(this.name.equals(person,name)&&this.age == person.age){
rrturn true:
}
rrturn false:
}
}
public class referenceDelivery{
public static void main(String args[]){
Person perA = new Person("张三",20):
Person perB = new Person("李四",20):
if(perA.compare(perB)){
System.out.println("是同一个人"):
}else{
System.out.println("不是同一个人"):
}
}
对象比较更多的是在分析层次上使用,而在实际之中,只有String用的比较多
对象比较一定是一个类自己具备的功能,所以一定要在类中写一个比较方法
进行对象比较时一定是将对象中的每一个属性进行逐步判断,所有属性都相同了,就表示对象是相同的,相等的
*/
/* 引用传递实例分析
将程序与生活联系在一起
一直以来一直强调:程序是生活的浓缩。生活的抽象,也就是说日常中的一切概念都可以通过程序来进行描述
下面描述这样的一个程序:一个人有一辆车
在描述以上概念之前希望可以有一个对比,如果说现在你进行数据库的设计,要描述以上的操作形式,你的数据表该如何设计呢?
CREATE TABLE person(
pid NUMBER,
name VARCHAR2(50),
CONSTRAINT pk_pid PRIMARY KEY(pid)
);
CREATE TABLE car(
pid NUMBER,
cname VARCHAR2(50),
CONSTRAINT pk_pid PRIMARY KEY(pid),
COMSTRAINT fk_pid2 FOREIGN KEY(pid)REFERENCES person(pid)
)
如果说现在要是进行类的设计,也需要两个类:Person Car,那么就可以发现两者的关联:
表名称 = 类名称:
表的字段 = 类属性
表的一行记录 = 一个实例化对象
表的多行记录 = 对象数组
标的外键 = 引用设置
正是因为有这样的匹配关系,所以在实际的开发过程之中,简单JAVA类不是凭空设计的,往往都与数据表的结构一一对应
范例:定义类
class Person{
private int pid:
private String name:
// car 为null了表示这个人没有车
private Car car: // 一个人有一辆车
public Person(int pid,String name){
this.pid = pid:
this.name = name:
}
public String getPersonlnfo(){
return"身份号:"+this.pid+",姓名:"+this.name:
}
public void setCar(Car car){
this.car = car:
}
public Car getCar(){
return this.car;
}
}
class Car{
private String cname;
// 如果人有车了,那么必须同时修改车与人对的关系
private Person person; // 一辆车属于一个人
public Car(String cname){
this.cname = cname:
}
public void setPerson(Person person){
this.person = person:
}
public Person getPerson(){
return this.person:
}
public String getCarlofo(){
return "汽车名称:"+this.cname:
}
}
public class biji{
public static void main(String args[]){
// 第一步:根据已有的结构设置内容
Person per = new Person(450802,"少爷"):
Catr car = new Car("保时捷356A"):
per,setCar(car): // 现在人有车了
car.setPerson(per): // 现在车属于一个人了
// 第二步:根据关系取出数据
System.out.println(per,getPersonlnfo()):
System.out.println(per,getCar().per,getCarlnfo()): // 代码链
System.out.println(car.getPerson().getPersonlnfo()):
}
}
/* 结果: 身份号:450802,姓名:少爷
汽车名称:保时捷356A
身份号:450802,姓名:少爷
*/
此时程序中使用了自定义的数据类型,Person Car都是类。那么随后的操作主要是进行代码的测试
但是测试分为两步:第一步设置内容,第二步取出内容
代码链别看单词,就看每一个方法的返回值是什么东西。如果是一个类的对象则可以继续调用该类的方法
那么也可以针对于此代码进行进一步额扩展。每个人还有孩子,每个孩子也可能有车
实际上这里面需要注意的是每一个人的孩子一定还是人,并且具备有跟人同样的属性信息,那么就可以在Person里面设计一个孩子的属性
class Person{
private int pid:
private String name:
private Person child: //孩子
// car 为null了表示这个人没有车
private Car car: // 一个人有一辆车
public Person(int pid,String name){
this.pid = pid:
this.name = name:
}
public void setChild(Person child){
this.child = child:
}
public Person getChild(){
return this.child
}
public String getPersonlnfo(){
return"身份号:"+this.pid+",姓名:"+this.name:
}
public void setCar(Car car){
this.car = car:
}
public Car getCar(){
return this.car;
}
}
class Car{
private String cname;
// 如果人有车了,那么必须同时修改车与人对的关系
private Person person; // 一辆车属于一个人
public Car(String cname){
this.cname = cname:
}
public void setPerson(Person person){
this.person = person:
}
public Person getPerson(){
return this.person:
}
public String getCarlnfo(){
return "汽车名称:"+this.cname:
}
}
public class biji{
public static void main(String args[]){
// 第一步:根据已有的结构设置内容
Person per = new Person(450802,"少爷"):
Person chd = new Person(450803,"小少爷"): // 有了孩子
Catr car = new Car("保时捷356A"):
Car c = new Car("布加迪威龙银河版"):
per,setCar(car): // 现在人有车了
car.setPerson(per): // 现在车属于一个人了
per.setChild(chd): // 一个人有一个孩子
chd.setCar(c): // 孩子有车
// 第二步:根据关系取出数据
System.out.println(per.getChild().getPersonlnfo()):
// 通过父亲的车找到父亲的信息,再找到父亲的孩子的车的信息
System.out.println(car.getPerson().getChild().getCar().getCarlnfo):
}
}
结果: 身份号:450803,姓名:小少爷
汽车名称:布加迪威龙银河版
引用的关系可以描述出不同类之间的引用关系
在现实的生活中这样的设计实际上并不麻烦了,理论上任何的事物都可以进行这样的整合
范例:抽象出衣服
class 袖子{}
class 衣领{}
class 衣服{
private 袖子 左:
private 袖子 右:
private 衣领 对象:
}
范例:抽象电脑
class cpu{}
class 内存{}
class 显卡{}
class 硬盘{}
class 主板{
private CPU 对象 :
private 内存 对象[]:
private 硬盘 对象[]:
private 显卡 对象:
}
class 键盘{}
class 鼠标{}
class 主机{
private 主板 对象:
private 鼠标 对象;
private 键盘 对象:
}
此处也同样属于引用。这样的形式的代码在设计模式上讲称为合成设计模式
1.不要把程序只当程序
2.引用传递除了进行数据的分析之外,还需要掌握类与类的联系使用
3.代码链的使用必须熟练
菜鸡的Java笔记 第十六 - java 引用传递的更多相关文章
- 菜鸡的Java笔记 第二十六 - java 内部类
/* innerClass 从实际的开发来看,真正写到内部类的时候是在很久以后了,短期内如果是自己编写代码,几乎是见不到内部类出现的 讲解它的目的第一个是为了解释概念 ...
- 菜鸡的Java笔记 第二十八 - java 包的定义
包的主要作用以及定义 包的导入操作 系统常见的开发包 jar 程序命令 包的定义 在任何的操作系统之中都有一个统一的共识:同一个目录下不能够存在有相同的文 ...
- 菜鸡的Java笔记 第二十四 - java 接口的基本定义
1.接口的基本定义以及使用形式 2.与接口有关的设计模式的初步认识 3.接口与抽象类的区别 接口与抽象类相比,接口的使用几率是最高的,所有的 ...
- 菜鸡的Java笔记 第二十二 - java 对象多态性
本次只是围绕着多态性的概念来进行讲解,但是所讲解的代码与实际的开发几乎没有关系,而且多态一定是在继承性的基础上才可以操作的, 而本次将使用类继承的关系来描述多态的性质,实际的开发中不会出 ...
- 菜鸡的Java笔记 第十九 - java 继承
继承性的主要目的,继承的实现,继承的限制 继承是面向对象中的第二大主要特点,其核心的本质在于:可以将父类的功能一直沿用下去 为什么需要继承? ...
- 菜鸡的Java笔记 第十八 - java 代码块
代码块 code block content (内容) 在程序结构之中使用"{}"定义的内容就称为代码块,但是会根据其声明的位置以及关 ...
- 菜鸡的Java笔记 第十二 - java 构造方法与匿名对象
1.点 构造方法的作用以及定义要求 匿名对象的使用 构造方法: 只要出现()的都表示方法 构造方法就是类构造对象时调用的方法,主要用来实例化对象.> ...
- 菜鸡的Java笔记 第二十九 - java 单例设计模式
SingleCase 单例设计模式 1.单例设计模式的特点 2.多例设计模式的特点 内容 单例设计模式 现在如果说有这么一个程序类 class S ...
- Java笔记(二十六)……IO流上 字节流与字符流
概述 IO流用来处理设备之间的数据传输 Java对数据的操作时通过流的方式 Java用于操作流的对象都在IO包中 流按操作的数据分为:字节流和字符流 流按流向不同分为:输入流和输出流 IO流常用基类 ...
随机推荐
- 92仿<高频彩>源码带采集
需要的联系QQ 肆伍以柒柒九八一
- 解决连接云服务器的redis失败
在本地连接服务器redis的时候,发现连接失败,这是因为服务器上的redis开启保护模式运行,该模式下是无法进行远程连接的.只需要修改redis目录下的redis.conf文件,找到 protecte ...
- C++ostringstream用法
ostringstream用法 1.类型转换 要求包含头文件; 字符串和int之间的互相转换: int num = 100; string str = ""; std::ostri ...
- openlayer 4326与3857坐标互转之Java版
public class Transform { private static final double PI = Math.PI; private static final double merca ...
- 第29篇-调用Java主类的main()方法
在第1篇中大概介绍过Java中主类方法main()的调用过程,这一篇介绍的详细一点,大概的调用过程如下图所示. 其中浅红色的函数由主线程执行,而另外的浅绿色部分由另外一个线程执行,这个线程最终也会负责 ...
- 2020.12.14--Codeforces Round #104 (Div.2)补题
C - Lucky Conversion CodeForces - 146C Petya loves lucky numbers very much. Everybody knows that luc ...
- rocketmq高可用集群部署(RocketMQ-on-DLedger Group)
rocketmq高可用集群部署(RocketMQ-on-DLedger Group) rocketmq部署架构 rocketmq部署架构非常多,都是为了解决一些问题,越来越高可用,越来越复杂. 单ma ...
- Python在Linux下编译安装报错:Makefile:1141:install
正常情况下执行:./configuremake && make install可以直接安装python,但是在在更新了乌版图后居然报错了!!!检查了一圈,发现乌版图安装了python3 ...
- js--Symbol 符号基本数据类型
前言 ECMAScript 6 中新增了 Symbol 符号这一基本数据类型,那么Symbol 是用来干什么的,对开发又有什么帮助呢?本文来总结记录一下 Symbol 的相关知识点. 正文 Symbo ...
- Qt学习-模仿Qt实现一个colorbutton
1.mycolorbutton.h #include<QToolButton> #include<QtGlobal> #include<QColor> class ...