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

20145221 《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),代码如下:

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

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

    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);
        }
    }

效果截图:
877163-20160320151127053-299497545.png

  • 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());
        }
    }   

效果截图:
877163-20160320145914303-1344313792.png

  • 不定长度自变量
    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;
        }
    }

效果截图:
877163-20160320145600740-1053289541.png

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

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

    877163-20160324104454698-1825050139.png

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

学习进度条

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

参考资料

转载于:https://www.cnblogs.com/20145221GQ/p/5297771.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值