zoukankan      html  css  js  c++  java
  • 20175316盛茂淞 2018-2019-2 《Java程序设计》第6周学习总结

    20175316盛茂淞 2018-2019-2 《Java程序设计》第6周学习总结

    教材学习内容总结

    第7章 内部类与异常类

    1.使用 try、catch

    • Java中所有信息都会被打包为对象,如果愿意,可以尝试(try)捕捉(catch)代表错误的对象后做一些处理
    try{
        ...(需要尝试捕捉的程序代码)
    }
    catch(... ex){
        ...(发生错误时执行的代码)
    }
    
    • JVM 会尝试执行 try 区块中的程序代码。如果发生错误,执行流程会跳离错误发生点,然后比较 catch 括号中声明的类型,是否符合被抛出的错误对象类型,如果是的话,就执行catch 区块中的程序代码
    • try、catch 用法举例:
    import java.util.*;
    
    public class Average2
    {
        public static void main(String[] args)
        {
           try
           {
               Scanner console = new Scanner(System.in);
               double sum = 0;
               int count = 0;
               while (true)
               {
                   int number = console.nextInt();
                   if (number ==0)
                   {
                       break;
                   }
                   sum += number;
                   count++;
               }
               System.out.printf("平均 %.2f%n",sum / count);
           }
           catch (InputMismatchException ex)
           {
               System.out.println("必须输入整数");
           }
        }
    }
    
    • 有时错误可以在捕捉处理之后,尝试恢复程序正常执行流程,例如:
    import java.util.*;
    
    public class Average3
    {
        public static void main(String[] args)
        {
            Scanner console = new Scanner(System.in);
            double sum = 0;
            int count = 0;
            while (true)
            {
                try
                {
                    int number = console.nextInt();
                    if (number == 0)
                    {
                        break;
                    }
                    sum += number;
                    count++;
                }
                catch (InputMismatchException ex)
                {
                    System.out.printf("略过非整数输入:%s%n", console.next());
                }
            }
            System.out.printf("平均 %.2f%n", sum / count);
        }
    }
    

    2.异常继承架构

    • Throwable 定义了取得错误信息、堆栈追踪等方法,有两个子类:java.lang.Error 与 java.lang.Exception

    • 异常处理:程序设计本身的错误,建议使用 Exception 或其子类实例来表现,所以通常称错误处理为异常处理

    • 单就语法与继承架构上来说,如果某个方法声明会抛出 Throwable 或子类实例,只要不是属于 Error、ava.lang.RuntimeException 或其子类实例,你就必须明确使用 try、catch语法加以处理,或者用 throws 声明这个方法会抛出异常,否则会编译失败

    • 受检异常:Exception 或其子对象,但非属于 RuntimeException 或其子对象,称为受检异常

    • 执行期异常(非受检异常):因为编译程序不会强迫一定得在语法上加以处理,亦称为非受检异常

    • 规则表达式:String 的 matches() 方法中设定了 "d*",这是规则表示式,表示检查字符串中的字符是不是数字,若是则 matches() 返回 true

    • 如果父类异常对象在子类异常对象前被捕捉,则 catch 子类异常对象的区块将永远不不会被执行

    3.多重捕捉语法:

    try{
            做一些事...
        }catch(IOException | InterruptedException | ClassCastException e){
    //catch 区块会在发生 IOException、InterruptedException、ClassCastException 时执行
            e.printStackTrace();
        }
    
    • catch 括号中列出的异常不得有继承关系,否则会发生编译错误

    5.catch or throw?

    • 如果方法设计流程中发生异常,而设计时没有充足的信息知道该如何处理,那么可以抛出异常,让调用方法的客户端来处理。为了告诉编译程序这个事实,必须用 throws 声明此方法会抛出的异常类型或父类型,编译程序才会让你通过编译。例如:
    public class FileUtil {
        public static String readFile(String name)
            throws FileNotFoundException{
            StringBuilder text = new StringBuilder();
            Scanner console = new Scanner(new FileInputStream(name));
            while(console.hasNext()){
                text.append(console.nextLine())
                        .apend('
    ');
            }
            return text.toString();
        }
    }
    
    • catch区块进行完部分错误处理之后,可以使用throw(注意不是throws)将异常再抛出。如:
    import java.util.Scanner;
    
    public class FileUtil
    {
        public static String readFile(String name) throws FileNotFoundException
        {
            StringBuilder text = new StringBuilder();
            try
            {
                Scanner console = new Scanner(new FileInputStream(name));
                while (console.hasNext())
                {
                    text.append(console.nextLine())
                    .append('
    ');
                }
            }
           catch (FileNotFoundException ex)
            {
               ex.printStackTrace();
                throw ex;
            }
            return text.toString();
        }
    }```
    
    - 如果抛出的是受检异常,表示你认为客户端有能力且应处理异常,此时必须在方法上使用 throws 声明;
    
    
    - 如果抛出的异常是非受检异常,表示你认为客户端调用方法的时机出错了,抛出异常是要求客户修正这个漏洞再来调用方法,此时也就不使用 throws 声明
    
    
    - 如果使用继承时,父类某个方法声明throws 某些异常,子类重新定义该方法时可以:
    
    1. 不声明 throws任何异常

    2. throws父类该方法中声明的某些异常

    3. throws父类该方法中声明异常的子类

    但是不可以:
    4. throws父类方法中未声明的其他异常

    1. throws父类方法中声明异常的父类
    - 自定义异常
    - 自定义异常类别时,可以继承Throw、Error 或 Exception或其子类,如果不是继承自Error或 RuntimeException,那么就会是受检异常
    - 自定义受检异常:
    

    public class CustomizedException extends Exception{
    ...
    }

    - 错误发生时:
    
    • 无足够信息处理异常:就现有信息处理完异常后,重新抛出异常

    • 已针对错误做了某些处理:考虑自定义异常,用以更精确地表示出未处理的错误

    • 客户端有能力处理未处理的错误:自定义受检异常、填入适当错误信息并重新抛出,并在方法上使用 throws加以声明

    • 客户端没有准备好就调了方法造成未处理错误:自定义受检异常、填入适当错误信息并重新抛出

    #### 6.异常堆栈
    
    - 在多重方法调用下,异常发生点可能是在某个方法之中,若想得知异常发生的根源,以及多重方法调用下的堆栈传播,可以利用异常对象自动收集的堆栈追踪来取得相关信息
    
    - 查看堆栈追踪最简单的方法,就是直接调用异常对象的printStackTrace(),例如:
    

    public class StackTraceDemo1
    {
    public static void main(String[] args)
    {
    try
    {
    c();
    }
    catch (NullPointerException ex)
    {
    ex.printStackTrace();
    }
    }

    static void c()
    {
        b();
    }
    
    static void b()
    {
        a();
    }
    
    static String a()
    {
        String text = null;
        return text.toUpperCase();
    }
    

    }

    - 如果并不知道调用的顺序,当异常发生而被捕捉后,可以调用 printStackTrace()在控制台显示堆栈追踪
    
    - 如果想要取得个别的堆栈元素进行处理,则可以使用getStackTrace(),这会返回 StackTraceElement 数组,数组中索引0为异常根源的相关信息,之后为各方法调用中的信息,可以使用StrackTraceElement的 getClassName()、getFileName()、getLineNumber()、getMethodName() 等方法取得对应的信息
    
    - 要善用堆栈追踪,前提是程序代码中不可有私吞异常的行为
    
    ### 第10章 输入、输出流
    #### 输入流
    
    - Java语言定义了许多类专门负责各种方式的输入或者输出,这些类都被放在java.io包中。其中,
    
    所有输入流类都是抽象类InputStream(字节输入流),或者抽象类Reader(字符输入流)的子类;
    
    而所有输出流都是抽象类OutputStream(字节输出流)或者Writer(字符输出流)的子类。
    
    - InputStream类是字节输入流的抽象类,是所有字节输入流的父类,InputStream类具有层次结构如下图所示;
    ![](https://img2018.cnblogs.com/blog/1272669/201904/1272669-20190407211921456-2037482460.png)
    
    - java中的字符是Unicode编码的,是双字节的。InputStream是用来处理字节的,在处理字符文本时很不方便。Java为字符文本的输入提供了专门的一套类Reader。Reader类是字符输入流的抽象类,所有字符输入流的实现都是它的子类。
    ![](https://img2018.cnblogs.com/blog/1272669/201904/1272669-20190407212116296-431728053.png)
    
    - 输出流OutputStream类是字节输入流的抽象类,此抽象类表示输出字节流的所有类的超类。
    ![](https://img2018.cnblogs.com/blog/1272669/201904/1272669-20190407212134212-929690206.png)
    
    - Writer类是字符输出流的抽象类,所有字符输出类的实现都是它的子类。
    ![](https://img2018.cnblogs.com/blog/1272669/201904/1272669-20190407212203148-1803924063.png)
    
    - File类是IO包中唯一代表磁盘文件本身的对象。通过File来创建,删除,重命名文件。File类对象的主要作用就是用来获取文本本身的一些信息。如文本的所在的目录,文件的长度,读写权限等等。(有的需要记忆,比如isFile(),isDirectory(),exits();有的了解即可。使用的时候查看API)
    #### 详细如下
    - File类(File类的概述和构造方法)
    

    A:File类的概述
      File更应该叫做一个路径
      文件路径或者文件夹路径
      路径分为绝对路径和相对路径
        ** 绝对路径是一个固定的路径,从盘符开始
        相对路径相对于某个位置,在eclipse下是指当前项目下,在dos下 **
      查看API指的是当前路径
      文件和目录路径名的抽象表示形式
    B:构造方法
      File(String pathname):根据一个路径得到File对象
      File(String parent, String child):根据一个目录和一个子文件/目录得到File对象
      File(File parent, String child):根据一个父File对象和一个子文件/目录得到File对象

    File类(File类的创建功能)
    

    A:创建功能
      public boolean createNewFile():创建文件 如果存在这样的文件,就不创建了
      public boolean mkdir():创建文件夹 如果存在这样的文件夹,就不创建了    public boolean mkdirs():创建文件夹,如果父文件夹不存在,会帮你创建出来
    (使用createNewFile()文件创建的时候不加.txt或者其他后缀也是文件,不是文件夹;使用mkdir()创建文件夹的时候,如果起的名字是比如aaa.txt也是文件夹不是文件;)
    ** 注意事项:如果你创建文件或者文件夹忘了写盘符路径,那么,默认在项目路径下。 **

    
    - File类(File类的重命名和删除功能)
    

      A:重命名和删除功能
        public boolean renameTo(File dest):把文件重命名为指定的文件路径
        public boolean delete():删除文件或者文件夹
      B:重命名注意事项
        如果路径名相同,就是改名。
        如果路径名不同,就是改名并剪切。
      C:删除注意事项:
        Java中的删除不走回收站。
        要删除一个文件夹,请注意该文件夹内不能包含文件或者文件夹

    
    - File类(File类的判断功能)
    

      A:判断功能
        public boolean isDirectory():判断是否是目录
        public boolean isFile():判断是否是文件
        public boolean exists():判断是否存在
        public boolean canRead():判断是否可读
        public boolean canWrite():判断是否可写
        public boolean isHidden():判断是否隐藏

     
    - File类(File类的获取功能)
    

      A:获取功能
        public String getAbsolutePath():获取绝对路径
        public String getPath():获取路径
        public String getName():获取名称
        public long length():获取长度。字节数
        public long lastModified():获取最后一次的修改时间,毫秒值
        public String[] list():获取指定目录下的所有文件或者文件夹的名称数组
        public File[] listFiles():获取指定目录下的所有文件或者文件夹的File数组

    
    ## 教材学习中的问题和解决过程
    ### 问题一
    - 在学习使用Properties的时候我一开始不知道如何从文档中加载属性,代码如下:
    

    import java.io.FileInputStream;
    import java.io.IOException;
    import java.util.Properties;
    public class MapLoadProperties {
    public static void main(String[] args) throws IOException {
    Properties props = new Properties();
    props.load(new FileInputStream(args[0]));
    System.out.println(props.getProperty("CH5.username"));
    System.out.println(props.getProperty("CH5.password"));
    }
    }

    - 解决方案:
    - 从代码中我发现,后面输出的变量是```props.getProperty```,所以我觉得应该从这个变量入手,也就是从代码段```props.load(new FileInputStream(args[0]));```来研究输出的变量,我通过API中了解FileInputStream()调用的应该就是前面说的文档,那么该如何调用呢?
    - 一开始我尝试了直接用```properties```文件名替换掉```args[0]```,但是程序无法编译了,思考了一下我觉得应该是在调用文档的时候,文档名少了""的关系,果然加上了就可以编译了
    - 但是编译是可以编译了,程序抛出了问题:
    ![](https://img2018.cnblogs.com/blog/1272669/201904/1272669-20190407213119407-1419766927.png)
    - 虽然有了问题,但是解决方案却更清晰了,既然是提醒我找不到指定文件,那我告诉系统文件在哪里就好了!于是我将代码段修改了一下:
    

    props.load(new FileInputStream("C:/Users/Cai Ye/IdeaProjects/HelloWorld/out/production/HelloWorld/CH5/Mapperson.properties"));

    ### 问题二
    - finally块中的代码一定会被执行吗?
    - 想要验证finally块中的代码是不是一定会被执行,我的思路是在finally块前加一些终止类型的代码来看看能不能阻止它执行,例如return:
    - 原代码如下:
    

    public class TryCatchFinallyAutoClosableDemo {
    public static void main(String[] args) {
    try (Resource res = new Resource()) {
    res.doSome();
    } catch (Exception ex) {
    ex.printStackTrace();
    }
    finally {
    System.out.println("finally…");
    }
    }
    }
    class Resource implements AutoCloseable {
    void doSome() {
    System.out.println("做一些事情");
    }
    @Override
    public void close() throws Exception {
    System.out.println("资源被关闭");
    }
    }

    - 输出结果如下:
    ![](https://img2018.cnblogs.com/blog/1272669/201904/1272669-20190407213619789-31828735.png)
    - 在该代码段加上return:
    

    public static void main(String[] args) {
    try (Resource res = new Resource()) {
    res.doSome();
    } catch (Exception ex) {
    ex.printStackTrace();
    }
    finally {
    System.out.println("finally…");
    }
    }

    - 但是结果并没有改变,这证明finally块应该是都会被执行的。
    - 但是我在网上看到一种情况可以让finally块不执行,就是加上System.exit(),这段代码的意义是终止JVM……这太无赖了,脸JVM都被终止了,怎么可能执行别的呢,就像断电了一样……不做常规范围思考……
    
    ## 代码调试中的问题和解决过程
    - 自编小程序
    

    import java.util.Set;
    import java.util.TreeSet;
    import java.util.Iterator;

    public class Main {
    public static void main(String[] args){
    Set set = new TreeSet<>();
    set.add("B");
    set.add("A");
    set.add("D");
    set.add("C");
    set.add("E");
    set.add("F");
    Iterator iter = set.iterator();
    while (iter.hasNext()){
    String str = iter.next();
    if("A".equals(str)){
    iter.remove();
    }else {
    System.out.println(str);
    }
    }
    }
    }

    - 调试截图
    ![](https://img2018.cnblogs.com/blog/1272669/201904/1272669-20190407215406657-1356993414.png)
    
    ## [代码托管](码云学习项目链接)
    ![](https://img2018.cnblogs.com/blog/1272669/201904/1272669-20190407215530369-1092006795.png)
    
    ## 结对及互评
    - 20175329许钰玮上周很认真,结对任务我们分工合作,代码一起负责调试,我负责UML图,他负责其他,总体情况很满意。
    
    ## 感悟
    - 在本周的学习中,我反而在学习的过程中觉得自己有很多不会的地方,一方面可能是因为本周的学习内容相对比较难比较生疏,另一方面,我觉得也是因为我思考的多了,对代码想要了解的更加深入一点,对于出现的错误想要尽可能的解决。所以我觉得本周我对于代码的学习反而没有上周感觉的那么顺畅。
    - 本周的学习中,我认识到了API的作用真的很大,很多时候在代码出现问题的时候,我不了解代码的具体含义和一些引申的东西,所以在改错的时候觉得很困难,但是,如果使用API的话,改正错误代码的方向性就会比较明确,大大节约了我的时间,也让我在调试代码的时候更有条理性。
    
    ## 学习进度条
    
    |            | 代码行数(新增/累积)| 博客量(新增/累积)|学习时间(新增/累积)|重要成长|
    | --------   | :----------------:|:----------------:|:---------------:  |:-----:|
    | 目标        | 5000行            |   30篇           | 400小时            |       |
    | 第一周      | 200/200           |   2/2            | 20/20             |       |
    | 第二周      | 300/500           |   2/4            | 18/38             |       |
    | 第三周      | 500/1000          |   3/7            | 22/60             |       |
    | 第四周      | 300/1300          |   2/9            | 30/90             |       |
    | 第五周      | 300/1500          |   2/9            | 30/120            |       |
    | 第六周      | 6001800          |   2/9            | 60/160            |       |
  • 相关阅读:
    经典小程序源码及其下载地址
    基于cropper.js的图片上传和裁剪
    【组件】微信小程序input搜索框的实现
    如何打造个人技术影响力
    一位90后程序员的自述:如何从年薪3w到30w!
    状态模式(State)(开关灯,状态实例为类,不同状态,不同行为)
    责任链模式(Chain of Responsibility、Handler)(请求处理建立链)
    java中创建对象的五种方法
    PrintWrite
    观察者模式(Observer、Subject、ConcreteSubject、ConcreteObserver)(监护、订阅)
  • 原文地址:https://www.cnblogs.com/sms369/p/10667222.html
Copyright © 2011-2022 走看看