201521123082 《Java程序设计》第7周学习总结 201521123082 《Java程序设计》第7周学习总结

标签(空格分隔): Java


1. 本周学习总结

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

201521123082 《Java程序设计》第7周学习总结
201521123082 《Java程序设计》第7周学习总结


2. 书面作业

1.ArrayList代码分析###

1.1 解释ArrayList的contains源代码####

Answer:
先贴上源码:
201521123082 《Java程序设计》第7周学习总结
201521123082 《Java程序设计》第7周学习总结
个人解析:
ArrayList中允许存放相同的元素,我们可以使用contains()方法来存放不重复有序的元素集合。
在源码中我们可以看出contains方法中调用indexOf方法来判断是否包含对象。在indexOf方法中,如果对象为空,elementData[i]也为空,返回i;如果对象不为空,调用equals方法,进行比较,如果与element[i]相同,返回i。


1.2 解释E remove(int index)源代码####

Answer:
先贴上源码:
201521123082 《Java程序设计》第7周学习总结
201521123082 《Java程序设计》第7周学习总结

remove(int index)删除ArrayList数组中指定位置上的元素并返回该元素。

关键解释如下:

/*numMoved就是index后面的所有的元素个数,即删除的元素后需要移动的元素个数 */
int numMoved = size - index - 1; 

/*将index后面的所有元素全部往前移动 */
if (numMoved > 0) 
    System.arraycopy(elementData, index+1, elementData, index, 
    numMoved); 
    
/*将最后多余的引用置为null. */
    elementData[--size] = null; 

1.3 结合1.1与1.2,回答ArrayList存储数据时需要考虑元素的类型吗?####

Anser:
1.结合1.1和1.2以及上课内容可以知道ArrayList是一个数组列表,ArrayList只能包含对象类型(Object)。
2.ArrayList在存放的时候可以是不同种类型的元素,因为ArrayList可以存储Object。所以任何类型的元素都是可以在ArrayList储存。

ArrayList对象的大小可按照其中存储的数据进行动态增减,所以在声明ArrayList对象时并不需要指定它的长度。但是由于ArrayList会把插入其中的所有数据当作为object类型来处理,在存储或检索值类型时会发生装箱和拆箱操作,带来很大的性能耗损。另外,ArrayList没有泛型的实现,也就是ArrayList不是类型安全的,在我们使用ArrayList处理数据时,很可能会报类型不匹配的错误。


1.4 分析add源代码,回答当内部数组容量不够时,怎么办?####

Answer:
先贴上源码:

 /**
     * Appends the specified element to the end of this list.
     *
     * @param e element to be appended to this list
     * @return <tt>true</tt> (as specified by {@link Collection#add})
     */
    public boolean add(E e) {
        ensureCapacityInternal(size + 1);  // Increments modCount!!
        elementData[size++] = e;
        return true;
    }
    
private void ensureCapacityInternal(int minCapacity) {
        if (elementData == EMPTY_ELEMENTDATA) {
            minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);
        }

        ensureExplicitCapacity(minCapacity);
    }

    private void ensureExplicitCapacity(int minCapacity) {
        modCount++;

        // overflow-conscious code
        if (minCapacity - elementData.length > 0)
            grow(minCapacity);
    }

    /**
     * The maximum size of array to allocate.
     * Some VMs reserve some header words in an array.
     * Attempts to allocate larger arrays may result in
     * OutOfMemoryError: Requested array size exceeds VM limit
     */
    private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;

    /**
     * Increases the capacity to ensure that it can hold at least the
     * number of elements specified by the minimum capacity argument.
     *
     * @param minCapacity the desired minimum capacity
     */
    private void grow(int minCapacity) {
        // overflow-conscious code
        int oldCapacity = elementData.length;
        int newCapacity = oldCapacity + (oldCapacity >> 1);
        if (newCapacity - minCapacity < 0)
            newCapacity = minCapacity;
        if (newCapacity - MAX_ARRAY_SIZE > 0)
            newCapacity = hugeCapacity(minCapacity);
        // minCapacity is usually close to size, so this is a win:
        elementData = Arrays.copyOf(elementData, newCapacity);
    }

    private static int hugeCapacity(int minCapacity) {
        if (minCapacity < 0) // overflow
            throw new OutOfMemoryError();
        return (minCapacity > MAX_ARRAY_SIZE) ?
            Integer.MAX_VALUE :
            MAX_ARRAY_SIZE;
    }

关键代码处的解释:

// 确定ArrarList的容量。  
    // 若ArrayList的容量不足以容纳当前的全部元素,设置 新的容量=“(原始容量x3)/2 + 1”  
    public void ensureCapacity(int minCapacity) {  
        // 将“修改统计数”+1,该变量主要是用来实现fail-fast机制的  
        modCount++;  
        int oldCapacity = elementData.length;  
        // 若当前容量不足以容纳当前的元素个数,设置 新的容量=“(原始容量x3)/2 + 1”  
        if (minCapacity > oldCapacity) {  
            Object oldData[] = elementData;  
            int newCapacity = (oldCapacity * 3)/2 + 1;  
            //如果还不够,则直接将minCapacity设置为当前容量
            if (newCapacity < minCapacity)  
                newCapacity = minCapacity;  
            elementData = Arrays.copyOf(elementData, newCapacity);  
        }  
    }  
 
    // 添加元素e  
    public boolean add(E e) {  
        // 确定ArrayList的容量大小  
        ensureCapacity(size + 1);  // Increments modCount!!  
        // 添加e到ArrayList中  
        elementData[size++] = e;  
        return true;  
    }  

引用及参考:ArrayList简介
java中数组列表ArrayList的使用


1.5 分析private void rangeCheck(intindex)源代码,为什么该方法应该声明为private而不声明为public?####

Answer:
源码:

/**
     * Checks if the given index is in range.  If not, throws an appropriate
     * runtime exception.  This method does *not* check if the index is
     * negative: It is always used immediately prior to an array access,
     * which throws an ArrayIndexOutOfBoundsException if index is negative.
     */
    private void rangeCheck(int index) {
        if (index >= size)
            throw new IndexOutOfBoundsException(outOfBoundsMsg(index));
    }

个人理解:
用private声明别的用户或者类不能调用,而public则任何用户和类都可以调用。rangeCheck用于边界检查,在类内调用就可以了。

关于rangeCheck(intindex)源码和ArrayList的学习可以参考:ArrayList 的实现原理


2.HashSet原理###

2.1 将元素加入HashSet(散列集)中,其存储位置如何确定?需要调用那些方法?####

Answer:
个人认识:散列集只能存储不重复的对象.散列表采用Hash算法决定集合元素的存储位置。
(1)链表是HashSet的存储形式,我们每添加一个元素时,调用hashCode()方法得到其hashCode值,然后确定存储位置。
(2)hashCode()方法和equals()方法。


2.2 选做:尝试分析HashSet源代码后,重新解释1.1####

Answer:
贴上源码和解释:

public class HashSet<E> 
	 extends AbstractSet<E> 
	 implements Set<E>, Cloneable, java.io.Serializable 
 { 
	 // 使用 HashMap 的 key 保存 HashSet 中所有元素
	 private transient HashMap<E,Object> map; 
	 // 定义一个虚拟的 Object 对象作为 HashMap 的 value 
	 private static final Object PRESENT = new Object(); 
	 ... 
	 // 初始化 HashSet,底层会初始化一个 HashMap 
	 public HashSet() 
	 { 
		 map = new HashMap<E,Object>(); 
	 } 
	 // 以指定的 initialCapacity、loadFactor 创建 HashSet 
	 // 其实就是以相应的参数创建 HashMap 
	 public HashSet(int initialCapacity, float loadFactor) 
	 { 
		 map = new HashMap<E,Object>(initialCapacity, loadFactor); 
	 } 
	 public HashSet(int initialCapacity) 
	 { 
		 map = new HashMap<E,Object>(initialCapacity); 
	 } 
	 HashSet(int initialCapacity, float loadFactor, boolean dummy) 
	 { 
		 map = new LinkedHashMap<E,Object>(initialCapacity 
			 , loadFactor); 
	 } 
	 // 调用 map 的 keySet 来返回所有的 key 
	 public Iterator<E> iterator() 
	 { 
		 return map.keySet().iterator(); 
	 } 
	 // 调用 HashMap 的 size() 方法返回 Entry 的数量,就得到该 Set 里元素的个数
	 public int size() 
	 { 
		 return map.size(); 
	 } 
	 // 调用 HashMap 的 isEmpty() 判断该 HashSet 是否为空,
	 // 当 HashMap 为空时,对应的 HashSet 也为空
	 public boolean isEmpty() 
	 { 
		 return map.isEmpty(); 
	 } 
	 // 调用 HashMap 的 containsKey 判断是否包含指定 key 
	 //HashSet 的所有元素就是通过 HashMap 的 key 来保存的
	 public boolean contains(Object o) 
	 { 
		 return map.containsKey(o); 
	 } 
	 // 将指定元素放入 HashSet 中,也就是将该元素作为 key 放入 HashMap 
	 public boolean add(E e) 
	 { 
		 return map.put(e, PRESENT) == null; 
	 } 
	 // 调用 HashMap 的 remove 方法删除指定 Entry,也就删除了 HashSet 中对应的元素
	 public boolean remove(Object o) 
	 { 
		 return map.remove(o)==PRESENT; 
	 } 
	 // 调用 Map 的 clear 方法清空所有 Entry,也就清空了 HashSet 中所有元素
	 public void clear() 
	 { 
		 map.clear(); 
	 } 
	 ... 
 }
  • 对于 HashSet 而言,它是基于 HashMap 实现的
  • HashSet封装了一个 HashMap 对象来存储所有的集合元素,所有放入 HashSet 中的集合元素实际上由 HashMap 的 key 来保存,而 HashMap 的 value 则存储了一个 PRESENT,它是一个静态的 Object 对象。
    HashSet 的绝大部分方法都是通过调用 HashMap 的方法来实现的,因此 HashSet 和 HashMap 两个集合在实现本质上是相同的。

3.ArrayListIntegerStack###

题集jmu-Java-05-集合之5-1 ArrayListIntegerStack

3.1 比较自己写的ArrayListIntegerStack与自己在题集jmu-Java-04-面向对象2-进阶-多态、接口与内部类中的题目5-3自定义接口ArrayIntegerStack,有什么不同?(不要出现大段代码)####

Answer:

  • ArrayIntegerStack是内部实现数组,实例化的时候需要规定大小
  • ArrayListIntegerStack是动态数组,可以用list可以自动扩容
  • ArrayList可以直接找到元素,进行删除,返回的操作

源码:

public ArrayIntegerStack(int n) {
		this.stack = new Integer[n];
	}
public ArrayListIntegerStack(){
		list =new ArrayList<Integer>();
	}

3.2 简单描述接口的好处.####

Answer:
接口是一种规范,同样的方法和操作可以有不同的实现,也可以供不同的类使用。上面一题就是很好的例子。


4.Stack and Queue###

4.1 编写函数判断一个给定字符串是否是回文,一定要使用栈,但不能使用java的Stack类(具体原因自己搜索)。请粘贴你的代码,类名为Main你的学号。####

Answer:

//ArrayListStringStack的代码
interface StringStack {
	public String push(String item);

	public String pop();

	public String peek();

	public boolean empty();

	public int size();
}

class ArrayListStringStack implements StringStack {
	private List<String> list;

	public ArrayListStringStack() {
		list = new ArrayList<String>();
	}

	@Override
	public String push(String item) {
		if (item == null)
			return null;
		list.add(item);
		return item;
	}

	@Override
	public String pop() {
		if (list.isEmpty())
			if (list.size() == 0)
				return null;
		return list.remove(list.size() - 1);
	}

	@Override
	public String peek() {
		if (list.size() == 0)
			return null;
		return list.get(size() - 1);
	}

	@Override
	public boolean empty() {
		if (list.size() == 0)
			return true;
		return false;
	}

	@Override
	public int size() {
		return list.size();
	}

	@Override
	public String toString() {
		return list.toString();
	}

}

主代码及实现结果:
201521123082 《Java程序设计》第7周学习总结
201521123082 《Java程序设计》第7周学习总结


4.2 题集jmu-Java-05-集合之5-6 银行业务队列简单模拟。(不要出现大段代码)####

Aswer:

题目要求A窗口每处理完2个顾客时,B窗口处理完1个顾客。给定到达银行的顾客序列,请按业务完成的顺序输出顾客序列。假定不考虑顾客先后到达的时间间隔,并且当不同窗口同时处理完2个顾客时,A窗口顾客优先输出。
其实就是A、B两个队Queue1和Queue2分别按要求输出,关键是判断两个队列的元素是最后一个输出。
源码和实现结果如下:
201521123082 《Java程序设计》第7周学习总结
201521123082 《Java程序设计》第7周学习总结


5.统计文字中的单词数量并按单词的字母顺序排序后输出###

题集jmu-Java-05-集合之5-2 统计文字中的单词数量并按单词的字母顺序排序后输出 (不要出现大段代码)

5.1 实验总结####

Answer:
关键源码:

while (input.hasNext()) {
			String word = input.next();
			if (word.equals("!!!!!"))
				break;
			//不同的单词加入集合中
			else if (!str.contains(word))
				str.add(word);
		}

运行结果:
201521123082 《Java程序设计》第7周学习总结
201521123082 《Java程序设计》第7周学习总结


6.选做:加分考察-统计文字中的单词数量并按出现次数排序###

题集jmu-Java-05-集合之5-3 统计文字中的单词数量并按出现次数排序(不要出现大段代码)

6.1 伪代码####

Answer:

//步骤1
Map<String, Integer> map = new TreeMap<String,Integer>();
/*1.输入单词放入map直到出现“!!!”停止
.....
.....
*/

//步骤2
List<Map.Entry<String, Integer>> arrayList = new ArrayList<Map.Entry<String,Integer>>(map.entrySet());
/*用Collections.sort()方法对arrayList存放Map的键值进行排序
......
......
*/

//步骤3
for (Map.Entry<String, Integer> entry : arrayList) {
/*
按要求输出
......
......
*/
}

201521123082 《Java程序设计》第7周学习总结
201521123082 《Java程序设计》第7周学习总结


6.2 实验总结####

Answer:
了解Map和List的区别,题目中无法对Map的值排序,转为list后sort来实现功能。
对于Map,有以下比较好的链接:
map的详细用法


7.面向对象设计大作业-改进###

7.1 完善图形界面(说明与上次作业相比增加与修改了些什么)####

Answer:
在上次作业中页面设计已完成得差不多,查看下图:
201521123082 《Java程序设计》第7周学习总结
201521123082 《Java程序设计》第7周学习总结
本次完善了注册界面设计,如下:
201521123082 《Java程序设计》第7周学习总结
201521123082 《Java程序设计》第7周学习总结
完善购物车信息,用表格显示的设计:
201521123082 《Java程序设计》第7周学习总结
201521123082 《Java程序设计》第7周学习总结

可以用Set储存用户账号和商品,可以避免出现账号相同和物品重复的现象。


3. 码云上代码提交记录及PTA实验总结##

题目集:jmu-Java-05-集合

3.1. 码云代码提交记录
在码云的项目中,依次选择“统计-Commits历史-设置时间段”, 然后搜索并截图
201521123082 《Java程序设计》第7周学习总结
201521123082 《Java程序设计》第7周学习总结


3.2. PTA实验
编程(5-1, 5-2, 5-3(选做), 5-6)
实验总结已经在作业中体现,不用写。