<Java>第六次作业
Java第六次作业
(一)学习总结
1.用思维导图对本周的学习内容进行总结
参考资料: XMind
2.当程序中出现异常时,JVM会依据方法调用顺序依次查找有关的错误处理程序。可使用printStackTrace 和getMessage方法了解异常发生的情况。阅读下面的程序,说明printStackTrace方法和getMessage 方法的输出结果分别是什么?并分析异常的传播过程。
public class PrintExceptionStack {
public static void main( String args[] )
{
try {
method1();
} catch ( Exception e ) {
System.err.println( e.getMessage() + "\n" );
e.printStackTrace();
}
}
public static void method1() throws Exception
{
method2();
}
public static void method2() throws Exception
{
method3();
}
public static void method3() throws Exception
{
throw new Exception( "Exception thrown in method3" );
}
}
程序的运行结果为:
Exception thrown in method3
java.lang.Exception: Exception thrown in method3
at 练习.PrintExceptionStack.method3(PrintExceptionStack.java:23)
at 练习.PrintExceptionStack.method2(PrintExceptionStack.java:19)
at 练习.PrintExceptionStack.method1(PrintExceptionStack.java:15)
at 练习.PrintExceptionStack.main(PrintExceptionStack.java:7)
printStackTrace():将此throwable对象的堆栈跟踪输出至错误输出流,printStackTrace方法的输出结果
java.lang.Exception: Exception thrown in method3
at PrintExceptionStack.method3(PrintExceptionStack.java:20)
at PrintExceptionStack.method2(PrintExceptionStack.java:16)
at PrintExceptionStack.method1(PrintExceptionStack.java:12)
at PrintExceptionStack.main(PrintExceptionStack.java:4)
getMessage():返回此throwable详细信息字符串,getMessage 方法的输出结果
Exception thrown in method3
异常传播过程:method3()产生异常并抛出,method2() 调用method3()产生异常并抛出,method1() 调用method2()产生异常并抛出,主函数调用method1()产生异常,用try...catch...捕获。
3.阅读下面程序,分析程序的运行结果,解释产生错误的原因,如果删除的是books集合的最后一个对象,运行的结果又是什么?你能对此作出解释吗?如果在遍历时非要删除集合中的元素,应如何实现?
import java.util.*;
public class Test
{
public static void main(String[] args)
{
Collection<String> books = new ArrayList<String>();
books.add("One book");
books.add("Two book");
books.add("Three book");
System.out.println("原始元素之后:"+books);
Iterator<String> it = books.iterator();
while(it.hasNext())
{
String book = (String)it.next();
System.out.println(book);
if (book.equals("One book"))
{
books.remove(book);
}
}
System.out.println("移除元素之后:"+books);
}
}
运行结果为:
原始元素之后:[One book, Two book, Three book]
One book
删除的是books集合的最后一个对象,运行的结果是:
原始元素之后:[One book, Two book, Three book]
One book
Two book
Three book
移除元素之后:[One book, Two book, Three book]
错误原因:Iterator输出一个集合的内容时,集合操作中也存在一个remove()方法,再使用Iterator输出时由集合对象调用了自身的删除方法,再进行删除时,会使迭代器的大小发生改变,造成异常。
解决方法:使用迭代器本身的删除方法。
修改之后的程序
package 练习;
import java.util.*;
public class Test
{
public static void main(String[] args)
{
Collection<String> books = new ArrayList<String>();
books.add("One book");
books.add("Two book");
books.add("Three book");
System.out.println("原始元素之后:"+books);
Iterator<String> it = books.iterator();
while(it.hasNext())
{
String book = (String)it.next();
System.out.println(book);
if (book.equals("There book"))
{
it.remove();
}else{
System.out.println("book");
}
}
System.out.println("移除元素之后:"+books);
}
}
在遍历时非要删除集合中的元素: 使用Iterator删除指定内容,直接使用remove()方法,使用迭代输出一定要注意使用Iterator删除输出时是否调用了自身的删除方法。
4.HashSet存储的元素是不可重复的。运行下面的程序,分析为什么存入了相同的学生信息?如果要去掉重复元素,应该如何修改程序`
package 练习;
import java.util.*;
class Student {
String id;
String name;
public Student(String id, String name) {
this.id = id;
this.name = name;
}
public String toString() {
return "Student id=" + id + ", name=" + name ;
}
public class Test1
{
public static void main(String[] args)
{
HashSet<Student> set = new HashSet<Student>();
set.add(new Student("1","Jack"));
set.add(new Student("2","Rose"));
set.add(new Student("2","Rose"));
System.out.println(set);
}
}
原因: 添加的匿名对象引用的地址不同:因为内存空间不一样, 两个相同信息的对象添加时,分别进行实例化时两个对象的HashCode值不同,所以HashSet判断为不同元素,引用时会被当做不同的对象处理。
解决方案:在Student中重写hashCode()方法和equals()方法。可用源代码直接添加
package 练习;
import java.util.*;
class Student {
String id;
String name;
public Student(){}
public Student(String id, String name) {
this.id = id;
this.name = name;
}
public String toString() {
return "Student id=" + id + ", name=" + name;
}
public int hashCode() {
final int prime = 31;
int result = 1;
result = prime * result + ((id == null) ? 0 : id.hashCode());
result = prime * result + ((name == null) ? 0 : name.hashCode());
return result;
}
public boolean equals(Object obj) {
if (this == obj)
return true;
if (obj == null)
return false;
if (getClass() != obj.getClass())
return false;
Student other = (Student) obj;
if (id == null) {
if (other.id != null)
return false;
} else if (!id.equals(other.id))
return false;
if (name == null) {
if (other.name != null)
return false;
} else if (!name.equals(other.name))
return false;
return true;
}
}
public class Test1 {
public static void main(String[] args) {
HashSet<Student> set = new HashSet<Student>();
set.add(new Student("1", "Jack"));
set.add(new Student("2", "Rose"));
set.add(new Student("2", "Rose"));
System.out.println(set);
}
}
5.其他需要总结的内容。
(1)线程是指程序的运行流程,多线程机制是指可以同时运行多个程序块,提高程序运行的效率。
(2)进程消失,线程一定会消失,而线程消失消失进程不一定消失.
(3)在Java中实现多线程有两种操作:一种是继承THread类;另一种是实现Runnable接口。
(4)注意:启动线程不能直接调用run()方法,而是应调用从Thread类中继承的start()方法;线程启动时虽然调用的是start()方法,但实际调用的确是run()方法定义的主体;一个类通过继承Thread类实现,则只能调用一次start()方法;
(5)在Runnable中启动多线程还需依靠Thread类中的public Thread(Runnable target)和public Thread(Runnable target,String name)两个构造方法。
(二)实验总结
实验内容:
1.模拟KTV点歌系统
分别用LinkedList和ArrayList集合,实现一个模拟KTV点歌系统的程序。实现以下功能:
(1)显示歌曲列表
(2)添加歌曲到列表
(3)删除歌曲
(4)将歌曲置顶
(5)将歌曲前移一位
(6)退出
题目扩展:歌曲包括曲名、演唱者。增加排序显示歌曲列表功能。
程序思路:直接在测试类中添加歌曲,定义List和Collection对象,实例化后直接添加歌曲里的歌名,然后用Switch...catch...来判断KTV功能的实现,在集合中指定位置增加歌曲,使用public void add(int ides,E element),删除指定位置的歌曲,使用public E remove(int index)
2.模拟微博用户注册
用HashSet实现一个模拟微博用户注册的程序。用户输入用户名、密码、确认密码、生日(格式yyyy-mm-dd)、手机号码(11位,13、15、17、18开头)、邮箱信息进行微博的注册。要求对用户输入的信息进行验证,输入信息正确后,验证是否重复注册,如果不是则注册成功,否则注册失败。
提示:
(1)设计一个用户类存储用户注册信息
(2)设计一个校验信息类,定义校验方法完成对输入信息的校验。学习使用正则表达式完成对生日、手机号码和邮箱的验证。
(3)设计一个用户注册类模拟注册过程。用HashSet存储用户数据列表,定义一个initData()方法添加初始用户信息。在main方法中完成用户注册功能。
程序思路:新建一个用户类存储用户注册信息,设定用户的基本输入格式,创建一个校验信息类,使用正则表达式完成对生日、手机号码和邮箱的验证,最后建一个用户注册类模拟注册过程,定义一个initData()方法添加初始用户信息。在main方法中完成用户注册功能,在判断时要注意格式是否符合要求。
(三)代码托管
- 码云commit历史截图