Java第六次作业


一、学习总结

1.用思维导图对本周的学习内容进行总结。

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" );
       }
    }
  • 运行结果:
  • e.getMessage方法会获得具体的异常名称。在此程序中getMessage 方法的输出结果:
  • e.printStackTrace方法会打出详细异常,异常名称,出错位置,以便于调试用。在此程序中printStackTrace方法的输出结果:
  • 传播过程:就此题来说,method3();发生异常,抛给指定异常对象Exception并传到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);
        }
    }
  • 删除第一个对象运行结果:
  • 删除最后一个对象的运行结果:
  • 原因:
    - 显然在删除第一个和最后一个对象时都没有进行删除操作。
    - 经过查阅相关资料,迭代时Iterator会产生一个指向元素对象的单链索引表,若此时对象集合执行删除操作,对象数量改变,但是索引表的内容不会同步改变。
    - 所以当索引指针往后移动时就找不到要迭代的对象。
    - 所以在迭代时,不允许被迭代的对象被改变。
  • (用Iterator 的remove()方法将迭代器所返回的元素删除)在遍历时删除最后一个对象的实现:
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("Three book"))
            {
                it.remove();
            }
        }
        System.out.println("移除元素之后:"+books);
    }
}
  • 运行结果:
    原始元素之后:[One book, Two book, Three book]
    One book
    Two book
    Three book
    移除元素之后:[One book, Two book]

4.HashSet存储的元素是不可重复的。运行下面的程序,分析为什么存入了相同的学生信息?如果要去掉重复元素,应该如何修改程序。

    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 Test
    {
        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);                
        }
    }
  • 运行结果:
    [Student id=2, name=Rose, Student id=1, name=Jack, Student id=2, name=Rose]
  • 原因:没有在Student类里面重写hashCode()和 equals()方法。HashSet依靠Object类的hashCode()方法和equals()方法完成重复元素的判断,所以在HashSet中的元素所属的类必须重写这两个方法。
  • 修改后:
package test;
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 ;
    }
	@Override
	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;
	}
	@Override
	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 Test
{
    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);                
    }
}
  • 修改后运行结果:
    [Student id=2, name=Rose, Student id=1, name=Jack]

二、实验总结:

1.模拟KTV点歌系统

分别用LinkedList和ArrayList集合,实现一个模拟KTV点歌系统的程序。实现以下功能:
(1)显示歌曲列表
(2)添加歌曲到列表
(3)删除歌曲
(4)将歌曲置顶
(5)将歌曲前移一位
(6)退出
题目扩展:歌曲包括曲名、演唱者。增加排序显示歌曲列表功能

  • 程序设计思路:
    - 创建一个歌曲类,分别有歌曲名和演唱者两个成员,并创建两个成员为参数的构造方法,以及相应getter、setter方法、toString方法。
    - 创建一个测试类,有增加歌曲方法、删除歌曲方法、歌曲置顶方法、歌曲前移一位方法、显示歌曲列表方法和主方法。在主方法中循环让用户选择几个功能(y/n控制循环结束)来实现点歌系统。
  • 问题1:执行置顶方法后,再输出错误不能置顶。
  • 原因:因为add(E o)方法是将对象加入到了双链表尾,而不是加入链表的头。
  • 解决方案:将link.add(link.get(i));改为link.add(0,link.get(i)); 部分代码如下:
public static void top(LinkedList<Song> link){
	Scanner in=new Scanner(System.in);
	String name;
	int i;
	System.out.println("请输入你要置顶的歌曲名:");	
	name=in.next();		
	for(i=0;i<link.size();i++){
		if(name.equals(link.get(i).getSong())){
			link.add(0,link.get(i));
			link.remove(i+1);
			System.out.println("置顶成功!");
			break;
		}
	}
	if(i==link.size()){
		System.out.println("您输入的歌曲名错误!");
	}
}
  • 问题2:
  • 原因:没有考虑当集合中本身只有一个元素时不能前移问题。
  • 解决方案:将集合中只有一个元素的情况分为类处理。部分代码如下:
public static void forward(LinkedList<Song> link){//歌曲前移一位方法
	Scanner in=new Scanner(System.in);
	int i;
	String song,per;
	if(link.size()!=0&&link.size()!=1){
		showSong(link);
		System.out.println("请输入你要前移的歌曲此时对应的序号:");
		song=in.next();
		per=in.next();
		for(i=0;i<link.size();i++){
			if(song.equals(link.get(i).getSong())&&per.equals(link.get(i).getPer())){
				link.add(i-1,link.get(i));
				link.remove(i+1);
				System.out.println("前移成功!");
				break;
			}
		}
		if(i==link.size())
			System.out.println("对不起,歌曲列表中没有这首歌!");	
	}
	else if(link.size()==1)
		System.out.println("对不起,歌曲列表中只有一首歌不能前移!");
	else
		System.out.println("对不起,您还未点歌曲!");
}

2.模拟微博用户注册

用HashSet实现一个模拟微博用户注册的程序。用户输入用户名、密码、确认密码、生日(格式yyyy-mm-dd)、手机号码(11位,13、15、17、18开头)、邮箱信息进行微博的注册。要求对用户输入的信息进行验证,输入信息正确后,验证是否重复注册,如果不是则注册成功,否则注册失败。
提示:
(1)设计一个用户类存储用户注册信息
(2)设计一个校验信息类,定义校验方法完成对输入信息的校验。学习使用正则表达式完成对生日、手机号码和邮箱的验证。
(3)设计一个用户注册类模拟注册过程。用HashSet存储用户数据列表,定义一个initData()方法添加初始用户信息。在main方法中完成用户注册功能。

  • 程序设计思路:
    - 创建一个用户信息类,有相关私有成员、以这些成员为参数的构造方法和相应的getter、setter、toString方法,还有重写的hashCode()和equals(Object obj)方法。
    - 创建一个校验信息类,分别有检验用户是否重复方法、检验确认密码与密码是否相符方法、检验生日格式是否正确方法、检验手机号码格式是否正确方法和检验邮箱格式是否正确方法。
    - 创建一个用户注册类,有initData方法和主方法。在initData方法中让用户进行输入信息并效验,效验正确返回1,效验错误打印是哪些错误并返回0。在主方法中让用户循环注册,注册成功判断是否还继续注册,失败后重新注册(用一个临时变量来控制循环结束)来实现用户注册。
  • 问题1:当微博注册出现两个错误时只能提示第一个错误。
  • 原因:将几个错误用了if elseif···来判断的,思路错乱。
  • 解决方案:先用if判断错误是否发生,再在这个判断里面用if if···语句逐个看是哪几种错误部分代码如下:
if(Information.checkUserName(userName,users)==0||Information.checkConfirmPassword(confirmPassword,password)==0||Information.checkBrithday(brithday)==0||Information.checkPhoneNumber(phoneNumber)==0||Information.checkEmail(email)==0){
	if(Information.checkUserName(userName,users)==0){
		System.out.println("您输入的用户名已被占用!");
	}
	if(Information.checkConfirmPassword(confirmPassword,password)==0){
		System.out.println("您输入的密码与确认密码不符!");
	}
	if(Information.checkBrithday(brithday)==0){
		System.out.println("您输入的生日格式不符!");
	}
	if(Information.checkPhoneNumber(phoneNumber)==0){
		System.out.println("您输入的手机号码格式不符!");
	}
	if(Information.checkEmail(email)==0){
		System.out.println("您输入的邮箱格式不符!");
	}
}
  • 问题2:输入重名时,不提示错误仍注册成功。
  • 原因:检查是否重名的方法的内容写错。
  • 解决方案:
public int checkUserName(String userName,Set<User> users){//检验用户是否重复方法
	Iterator<User> iterator=users.iterator();
	int temp=1;
	while(iterator.hasNext()){
		if(userName.equals(iterator.next())){
			temp=0;
		}
	}
	return temp;
}
  • 问题3:想不出邮箱的检验正则表达式的格式。
  • 原因:未仔细阅读并理解课件
  • 解决方案:在课件上找到正则规则的内容并进行理解(\w需要转义字符前面添加一个斜杠‘\’),email格式如下:
"\\w+@\\w+.(com|cn|net|gov|edu|org)"

三、代码托管(务必链接到你的项目)

  • 码云commit历史截图
    上传实验项目代码到码云,在码云项目中选择“统计-commits”,设置搜索时间段,搜索本周提交历史,并截图。

posted @ 2017-05-04 14:40  暖心物语  阅读(596)  评论(0编辑  收藏  举报