201621123033 《Java程序设计》第10周学习总结

1. 本周学习总结

1.1 以你喜欢的方式(思维导图或其他)归纳总结异常相关内容。

2. 书面作业

本次PTA作业题集异常

1. 常用异常

结合题集题目7-1回答

1.1 自己以前编写的代码中经常出现什么异常、需要捕获吗(为什么)?应如何避免?

经常出现空指针错误(NullPointerException),NullPointerException是RuntimeException的一种,不需要捕获。
如何避免NullPointerException:

  1. 从已知的String对象中调用equals()和equalsIgnoreCase()方法,而非未知对象。
    unknownObject.equals("knownObject")→错误
    "knownObject".equals(unknownObject)→正确
  2. 当valueOf()和toString()返回相同的结果时,使用前者
    因为调用null对象的toString()会抛出空指针异常,而使用valueOf(),传递一个null给valueOf()将会返回“null”。

1.2 什么样的异常要求用户一定要使用捕获处理?

直接继承Exception的异常。

2. 处理异常使你的程序更加健壮

题集题目7-2

2.1 实验总结。并回答:怎么样才能让你的程序更加健壮?

异常应该是出现在输入的时候输入非法字符,所以try应该放在int value = Integer.parseInt(line);之前,如果此时出现异常,则不会执行下一步的arr[i] = value;。另一方面,因为如果出现了异常,输入的值不会放入数组中,所以此时下标应该不增加,所以为了抵消for循环里的i++,应该在catch语句里加上一句i--。
使程序更加健壮的话,给程序加上异常处理,使程序遇到异常地时候不会“错误”地执行下去,同时也能知道出现了什么异常,从而去改进代码。

3. throw与throws

题集题目7-3
阅读Integer.parsetInt源代码

3.1 Integer.parsetInt一开始就有大量的抛出异常的代码,这种做法有什么好处?

可以告知程序员错误最早出现在什么地方,而且一遇到异常就抛出,程序就不会“错误”地执行下去。

3.2 结合自己编写的程序与3.1,分析自己编写的方法抛出异常时一般需要传递给调用者一些什么信息?

抛出异常时应该告知调用者该异常出现的原因。

if (s == null) {
            throw new NumberFormatException("null");
        }

        if (radix < Character.MIN_RADIX) {
            throw new NumberFormatException("radix " + radix +
                                            " less than Character.MIN_RADIX");
        }

        if (radix > Character.MAX_RADIX) {
            throw new NumberFormatException("radix " + radix +
                                            " greater than Character.MAX_RADIX");
        }

如Integer.parseInt就根据不同的错误情况抛出对应的信息。

4. 用异常改进ArrayIntegerStack

题集题目6-3

4.1 结合6-3代码,回答使用抛出异常的方式代表程序运行时出错有什么好处?比单纯的返回错误值,有何优点?

可以让调用者得到更多关于该错误的信息,如6-3代码↓,抛出的异常让调用者能直观地了解到是什么错误,而改进前单纯的返回错误值,调用者很难判断到底是哪里出错了。

    @Override
	public Integer push(Integer item) {
		if(item == null)
			return null;
		if(top==capacity){
			throw new FullStackException();
		}
		arrStack[top++]=item;
		return item;
	}
	@Override
	public Integer pop() {
		if(top==0)
			throw new EmptyStackException();
		top--;
		return arrStack[top];
	}

4.2 如果一个方法内部的内码抛出的是RuntimeException类型的异常,那么方法声明是否应该使用throws关键字,如果使用throws关键字声明该方法抛出的异常,能给我们带来什么好处吗?

对大多数Exception子类来说,Java编译器会强迫你声明在一个成员函数中抛出的异常的类型。如果异常的类型是Error或RuntimeException,或它们的子类,这个规则不起作用,因为这在程序的正常部分中是不期待出现的。如果你想明确地抛出一个RuntimeException,你必须用throws语句来声明它的类型。

该方法不需要使用throws关键字,若使用的话,能明确地抛出声明的异常,但其实感觉也没太大好处......

5. 函数题-多种异常的捕获

题集题目6-1

5.1 结合6-1代码,回答:一个try块中如果可能抛出多种异常,且异常之间可能有继承关系,捕获时需要注意些什么?

根据不同异常进行不同处理,但父类异常要放在子类异常之后,若放在子类异常之前则会报错(如下图)

5.2 一个try块中如果可能抛出多种异常,使用Java8的多重异常捕获语法需要注意些什么?

父类异常要放在子类异常之后,若使用单个catch块捕获异常,父类异常要放在子类异常的右边↓
catch(NumberFormatException|IllegalArgumentException|Exception e)

6. 为如下代码加上异常处理

byte[] content = null;
FileInputStream fis = new FileInputStream("testfis.txt");
int bytesAvailabe = fis.available();//获得该文件可用的字节数
if(bytesAvailabe>0){
    content = new byte[bytesAvailabe];//创建可容纳文件大小的数组
    fis.read(content);//将文件内容读入数组
}
System.out.println(Arrays.toString(content));//打印数组内容

6.1 改正代码,并增加如下功能。当找不到文件时,需提示用户找不到文件xxx,请重新输入文件名,然后尝试重新打开。 如果是其他异常则提示打开或读取文件失败!

注1:里面有多个方法均可能抛出异常。
功能2:需要添加finally关闭文件。无论上面的代码是否产生异常,总要提示关闭文件ing。如果关闭文件失败,提示关闭文件失败!

//改正代码
byte[] content = null;
		FileInputStream fis=null;
		try{
			fis = new FileInputStream("testfis.txt");
			int bytesAvailabe = fis.available();//获得该文件可用的字节数
			if(bytesAvailabe>0){
			    content = new byte[bytesAvailabe];//创建可容纳文件大小的数组
			    fis.read(content);//将文件内容读入数组
			}
		}catch(FileNotFoundException e){
			System.out.println("找不到文件testfis.txt,请重新输入文件名");
		}catch(Exception e){
			System.out.println("打开或读取文件失败!");
		}finally{
			System.out.println("关闭文件ing");
			try{
				fis.close();
			}catch(Exception e){
				System.out.println("关闭文件失败!");
			}
		}
		System.out.println(Arrays.toString(content));//打印数组内容

运行结果:

6.2 结合题集6-2代码,要将什么样操作放在finally块?为什么?使用finally关闭资源需要注意一些什么?

将必须执行的操作放在finally块,如6-2中在发生异常前开启了相关资源,那么遇到异常时代码可能无法正确关闭资源,所以这时就要把关闭资源的代码块放到finally块中,因为finally块中的程序一定会执行。
使用finally关闭资源的时候注意也要进行异常捕获。

6.3 使用Java7中的try-with-resources来改写上述代码实现自动关闭资源。简述这种方法有何好处?

byte[] content = null;		
		try(FileInputStream fis = new FileInputStream("testfis.txt")){
			
			int bytesAvailabe = fis.available();//获得该文件可用的字节数
			if(bytesAvailabe>0){
			    content = new byte[bytesAvailabe];//创建可容纳文件大小的数组
			    fis.read(content);//将文件内容读入数组
			}
		}catch(FileNotFoundException e){
			System.out.println("找不到文件testfis.txt,请重新输入文件名");
		}catch(Exception e){
			System.out.println("打开或读取文件失败!");
		}
		
		System.out.println(Arrays.toString(content));//打印数组内容

好处:代码更简洁,资源会在try块执行完后、catch块及finally块执行前关闭,保证资源能被关闭。

7. 面向对象设计作业-图书馆管理系统(分组完成,每组不超过3个同学)

登录lib.jmu.edu.cn,对图书进行搜索。然后登录图书馆信息系统,查看我的图书馆。如果让你实现一个图书借阅系统,尝试使用面向对象建模。

7.1 该系统的使用者有谁?

学生、教职工、管理员

7.2 主要功能模块(不要太多)及每个模块的负责人。下周每个人要提交自己负责的模块代码及运行视频。

学生及教职工:

  • 登录系统
  • 个人信息
  • 当前借阅信息
  • 借还书历史
  • 搜索
  • 借书
  • 还书
  • 退出系统

管理员:

  • 登录系统
  • 增加图书
  • 下架图书
  • 修改图书信息
  • 退出系统

7.3 该系统的主要的类设计及类图(可用)

7.4 你准备如何存储图书信息、解决信息、读者信息等。

初步:使用ArrayList来存储图书信息等,同时可以存储在文件中。
但认为使用Map、Set等数据结构来存储或许会更好。

3.码云及PTA

题目集:异常

3.1. 码云代码提交记录

在码云的项目中,依次选择“统计-Commits历史-设置时间段”, 然后搜索并截图

3.2 截图PTA题集完成情况图

需要有两张图(1. 排名图。2.PTA提交列表图)


3.3 统计本周完成的代码量

需要将每周的代码统计情况融合到一张表中。

周次 总代码量 新增代码量 总文件数 新增文件数
3 547 547 12 12
5 971 424 20 8
6 1577 606 29 9
7 2187 1216 41 21
8 2793 606 49 8
9 3040 247 56 7
10 3617 577 63 7
11 3911 294 71 8

posted on 2017-11-25 15:11  HolmeC  阅读(168)  评论(1编辑  收藏  举报

导航