您的位置:首页 > 编程语言 > Java开发

20145221 《Java程序设计》第三周学习总结

2016-03-20 14:45 736 查看

20145221 《Java程序设计》第三周学习总结

教材学习内容总结

第四章部分已在假期完成,详见博客:

《Java程序设计》第四章-认识对象

第五章部分

何谓封装

封装实际上使用方法将类的数据隐藏起来,控制用户对类的修改和访问数据的程度,仅对外提供公共访问方式,隐藏对象细节,将对象当作黑箱进行操作。

书中介绍了一个银行钱款存取的例子,我觉得非常形象,将对象进行封装,其实是为了更好的利用对象。

你可以在你定义的类中,创建属性、方法,如果人家需要用到相应的功能,只需要拿着你的“类设计图”去设计一个“对象实例”,再根据你公开提供的方法,就可以使用相应的功能了。

封装对象,感觉有点像C语言中的模块化编程,不过C语言中一般将自定义函数和主函数放在一个程序中来调用,但Java中做到了一类一文件,而且在Java中更灵活方便。

我假设了一个情景,比如你和你的小伙伴要做一个大程序,不可能等着他编完他的部分了,你再去编写,这样效率极低。但是有了封装对象,你可以根据设计好的功能,先撰写相应的方法,最后直接调用就行,大大提高了编程的效率。

也可以像书中提到的,使用private对关键信息进行隐藏,设置访问数据的程度,防止用户的恶意使用。

类语法细节

public

用在类前,表示的这个类是公开类,这样就可以在其他包的类中使用。

用在方法前,这样就表示其它包中的方法可以直接调用这个方法。这里需要注意的是,如果类上没有声明public关键字,类中的方法就算是public,也等于是包权限了,因为类本身是包权限,其它包就根本无法使用类,更别说当中定义的方法。

用在函数前,这表示其它包中的类可以直接调用这个构造函数。

用在成员变量前,表示这个成员变量是公开的,这样别人可以直接调取你类中的变量。

private

用在成员变量前,这是一种保护机制,不让自己这个类以外的方法去随便使用存取这个类的数据,可以保护私有数据,只能通过调用自己类的方法去操纵这些数据,这样会很安全。

private如果用在类或方法前,程序就会报错(我试着将类的前面写成private)。不过,其实也可以用在方法或构造函数声明上,私有方法或构造函数通常是类内部某个共享的演算流程,外借不用知道私有方法的存在。一般用在内部类声明中。

public & private

一般来说,我自己感觉public用在类、方法、函数前,private用在成员变量前,这样用户就可以通过你类中提供的方法和函数来进行操作,但是不会改变类中私有数据。这样,既能满足用户的使用,又能保护自己的数据不被随意更改。

方法重载(Overload)

根据自己的理解,重载就好比是用相同的词表达多种不同的含义。

只要参数类型和个数不同,就可以定义多个构造函数。这样程序设计人员不用苦恼方法名称的设计,可用一致的名称来调用类似功能的方法。

与返回值无关,构造函数不一定关心它的返回值,可能只是想利用其中的某个功能。所以,如果像下面这样调用方法:
f();
此时Java如何才能判断该调用哪一个
f()
呢?因此,根据方法的返回值来区分重载方法是行不通的。

this与super

除了被声明为static的地方外,this关键字可以出现在类中的任何地方,在对象建立后为“这个对象”的参考名称。

super一般用在父类中,指代父类对象的参考名称。

如果定义一个新的类:A,这个A继承了类B,也就是说B是A的父类。那么如果A中 有个方法:aa();B中也有个方法: aa();那么在A 中用this.aa()调用的就是A中定义的方法,而super.aa()调用的就是A的父类B中定义的方法aa();。

final

用在类前,表明这个类不能被继承。也就是说,如果一个类你永远不会让他被继承,就可以用final进行修饰。final类中的成员变量可以根据需要设为final,但是要注意final类中的所有成员方法都会被隐式地指定为final方法。在使用final修饰类的时候,要注意谨慎选择,除非这个类真的在以后不会用来继承或者出于安全的考虑,尽量不要将类设计为final类。

用在方法前,使用final方法的原因有两个:第一个原因是把方法锁定,以防任何继承类修改它的含义;第二个原因是效率。在早期的Java实现版本中,会将final方法转为内嵌调用。但是如果方法过于庞大,可能看不到内嵌调用带来的任何性能提升。在最近的Java版本中,不需要使用final方法进行这些优化了。因此,如果只有在想明确禁止 该方法在子类中被覆盖的情况下才将方法设置为final的。注:类的private方法会隐式地被指定为final方法。

用在成员变量前,修饰变量是final用得最多的地方,也是本文接下来要重点阐述的内容。首先了解一下final变量的基本语法:对于一个final变量,如果是基本数据类型的变量,则其数值一旦在初始化之后便不能更改;如果是引用类型的变量,则在对其初始化之后便不能再让其指向另一个对象。

static:所有对象公有的

static用于修饰成员变量和成员函数,被声明为static的成员,不会让个别对象拥有,而是属于类。

由于static成员是属于类,而非个别对象,所以在static成员中使用this,会是一种语意上的错误。

访问惯例:类名.static成员(一般不用参考名称)

import static 的用法

与import一样,这样做是为了偷懒,但要注意名称冲突的问题。

变长参数

在调用方法时,如果自变量的个数事先无法决定,则可以在声明参数列时在关键字后加上...,实际上不定长度自变量是编译程序蜜糖。

需要注意的是:函数如果有多个参数,变长参数必须是最后一个。

内部类

可以在内中再定义类,内部类亦可以使用public、protected、private声明。

一个被声明为static的内部类,通常是将外部类当作名称空间。例如:
Some.Other o = new Some.Other();


Java中方法都是传值(Pass By Value)的

传值通过画图理解起来更容易(课本P148-P150),代码如下:

代码调试中的问题和解决过程

第四章部分已在假期完成,详见博客:

《Java程序设计》第四章-认识对象

关于课本P121页斐波那契数列,考虑到了递归函数的调用,写了一个新代码如下:

import java.util.Scanner;

public class Fibonacci{
public static void main(String[] args){
System.out.printf("求几个费氏数?");
Scanner scanner = new Scanner(System.in);
int n = scanner.nextInt();
for(int i=0; i<n; i++)
System.out.print(fib(i+1) + " ");
System.out.println();
}
public static long fib(int x){
if(x==1)
return 0;
else if(x==2)
return 1;
else
return (fib(x-1) + fib(x-2));
}
}


分析:

区别寒假代码上述代码
数组元素的类型int型(可计算到n=47)long型(可计算到n=93)
自定义函数调用递归函数
运行速度一直都很快n<35时速度比较快,当n>50时速度明显减慢
结论:运用递归速度变慢的原因是,没一个斐波那契数的生成都需要从
fib(n)
递归到
fib(1)
,大大影响了代码的运行效率;寒假的代码,则通过数组存储前面每一个值,在进行后续的运算时直接调用即可。所以,一个程序的编写会有多种不同的算法,只有自己试验后,才能判断出代码的好坏,好的保留,坏的改进。

第五章书中部分代码调试

封装部分代码:

import java.util.Scanner;

public class CardApp {
public static void main(String[] args) {
CashCard[] cards = {
new CashCard("A001", 500, 0),
new CashCard("A002", 300, 0),
new CashCard("A003", 1000, 1)
};

Scanner console = new Scanner(System.in);
for(CashCard card : cards) {
System.out.printf("为 (%s, %d, %d) 储值:",
card.number, card.balance, card.bonus);
card.store(console.nextInt());
System.out.printf("明细 (%s, %d, %d)%n",
card.number, card.balance, card.bonus);
}
}
}


this的用法

class Other {
{
System.out.println("对象初始区块:");
}

Other() {
System.out.println("Other() 构造函数");
}

Other(int o) {
this();
System.out.println("Other(int o) 构造函数");
}
}

public class ObjectInitialBlock {
public static void main(String[] args) {
new Other(1);
}
}

效果截图:



import static的用法

import java.util.Scanner;
import static java.lang.System.in;
import static java.lang.System.out;

public class ImportStatic {
public static void main(String[] args) {
Scanner console = new Scanner(in);
out.print("请输入姓名:");
out.printf("%s 你好!%n", console.nextLine());
}
}

效果截图:



不定长度自变量

public class MathTool {
public static int sum(int... numbers) {
int sum = 0;
for(int number : numbers) {
sum += number;
}
return sum;
}
}


传值代码

public class CallByValue {
public static void main(String[] args) {
Customer c1 = new Customer("Justin");
some(c1);
System.out.println(c1.name);

Customer c2 = new Customer("Justin");
other(c2);
System.out.println(c2.name);
}

static void some(Customer c) {
c.name = "John";
}

static void other(Customer c) {
c = new Customer("Bill");
}
}

class Customer {
String name;
Customer(String name) {
this.name = name;
}
}

效果截图:



其他(感悟、思考等,可选)

这周最大的进步就是把书上的代码基本都敲了一遍,并且用git上传到了开源中国。





这两章的知识可能较前三章内容晦涩一些,毕竟有点深奥。说深奥其实也不算深奥,只是我以前可能没接触过对象这一知识,不能理解对象存在的价值。当看完这两章内容后,有了一个大致的了解。接下来就是对课本中的代码进行敲写一一体会,就能更加清楚书中所说含义。最后再看书上的代码时,头脑中就可以模拟当时编译运行的画面,对对象相关知识的理解更加透彻。学习Java肯定要靠大量的代码来加深印象巩固知识,只有坚持自主学习积极敲代码,才能更好地掌握这门奇妙的语言。

学习进度条

代码行数(新增/累积)博客量(新增/累积)学习时间(新增/累积)重要成长
目标5000行30篇400小时
第一周200/2001/620/20学会MarkdownPad2
第二周150/3501/715/35理解了补码机制
第三周500/8501/825/60初步了解了对象

参考资料

Java学习笔记(第8版)

《Java学习笔记(第8版)》学习指导

积极主动敲代码,使用JUnit学习Java

浅析Java中的final关键字
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: