其他分享
首页 > 其他分享> > BUAA OO Unit3总结

BUAA OO Unit3总结

作者:互联网

BUAA OO Unit3` 总结

1 架构设计

1.1 Homework9

第九次作业训练目标是实现简单社交关系的模拟和查询。本次作业我一共实现了MyPersonMyGroupMyNetwork以及六个异常类。

具体关系如下:

image

1.2 Homework10

Homework9相比,本次作业添加了一个MyMessage类。

image

1.3 Homework11

新增了三种信息类型:MyRedEnvelopeMessageMyNoticeMessageMyEmojiMssage

image

2 性能问题

2.1 容器选择

基本上都选择了速度最快的HashMap容器

例如MyNetwork

private ArrayList<Person> people = new ArrayList<>();
private HashMap<Integer,Group> groups = new HashMap<>();
private HashMap<Integer,Message> messages = new HashMap<>();
private HashSet<Integer> emojiIdList = new HashSet<>();
private HashMap<Integer,Integer> emojiHeatList = new HashMap<>();
private HashMap<Person,Person> parent = new HashMap<>();
private HashMap<Person,HashSet<Person>> graph = new HashMap<>();

2.2 从dfs到并查集的实现

难点函数是isCircle():本质是查询两个结点之间有没有通路。

我一开始使用的是dfs深度优先搜索的算法,最糟糕的时间复杂度是O(n^2)。后来我优化成了并查集算法。

并查集的本质是通过一个一维数组来维护一个森林。开始时森林中的每一个节点都是孤立的,各自形成一个树。之后,进行若干次的合并操作,每次合并将两个树合并为一个更大的树。

实现方法:

parent保存着结点和它的父节点。

private HashMap<Person,Person> parent;

addRelation()函数中:

添加而父结点的关系。

	if (parent.get(getPerson(id1)) == null && parent.get(getPerson(id2)) == null) {
 		parent.put(getRoot(getPerson(id1)),getRoot(getPerson(id2)));
	} else if (parent.get(getPerson(id1)) == null) {
 		if (!getRoot(getPerson(id2)).equals(getPerson(id1))) {
			parent.put(getPerson(id1),getRoot(getPerson(id2)));
    	}
	} else if (parent.get(getPerson(id2)) == null) {
    	if (!getRoot(getPerson(id1)).equals(getPerson(id2))) {
        	parent.put(getPerson(id2),getRoot(getPerson(id1)));
    	}
	} else {
    	if (!getRoot(getPerson(id1)).equals(getRoot(getPerson(id2)))) {
        	parent.put(getRoot(getPerson(id1)),getRoot(getPerson(id2)));
   		}
	}

采用getRoot()函数进行查询操作,判断其父结点是不是一致的,如果一致两个结点之间就存在路径,即isCircle()返回值是true

在查询过程中会有路径压缩操作。

   public Person getRoot(Person person) {
        Person root = person;
        while (parent.get(root) != null) {
            root = parent.get(root);
        }
        return root;
    }

2.3 krustal算法

难点函数是queryLeastConnection(int id):本质是找到和id联通的最大点集,构造该点集的最小生成树。我使用了krustal算法。

Krustal算法查找最小生成树的方法是:将连通网中所有的边按照权值大小做升序排序,从权值最小的边开始选择,只要此边不和已选择的边一起构成环路,就可以选择它组成最小生成树。对于 N 个顶点的连通网,挑选出 N-1 条符合条件的边,这些边组成的生成树就是最小生成树。

2.4 Dijkstra堆优化算法

难点函数是sendIndirectMessage():本质是查询两个结点之间的最短路径。

Dijkstra算法详解:先建立一个dist数组表示和起点相连的各个点之间的最小值,先加入起点及其相连的点之间的距离,其他的赋值为inf(无穷大),开始在dist数组中扫描到距离最小的点,将其标记如最小生成树中,再看是否可以对其他点进行松弛操作;一直重复到所有点全部被标记结束。操作完之后dist[i]中存的就是和起点到 i 点的最小值。

部分实现代码:

public int main() {
	Person nextPerson;
	nextPerson = this.person1;
	while (!(nextPerson.equals(person2))) {
		int weight = minHeapWeight.get(1);
        Person person = this.get();
        while (flag.get(person.getId())) {
            weight = minHeapWeight.get(1);
            person = this.get();
        }
        this.flag.put(person.getId(),true);
        this.path.put(person,weight);
        //更新距离
        for (Person peo : graph.get(person)) {
			if (!this.path.containsKey(peo)) {
				this.add(peo,peo.queryValue(person) + this.path.get(person));
				this.path.put(peo,peo.queryValue(person) + this.path.get(person));
			} else if (this.path.get(peo) > peo.queryValue(person) + this.path.get(person)) {
				this.add(peo,peo.queryValue(person) + this.path.get(person));
			}
		}
		nextPerson = person;
    }
	return path.get(person2);
}

2.5 动态维护

实现函数为queryGroupValueSum():

MyGroup类中:

private int valueSum;
public void addPerson(Person person) {
	if (!this.hasPerson(person)) {
		for (Integer id:people.keySet()) {
			valueSum = valueSum + (people.get(id).queryValue(person) * 2);
        }
        this.ageSum += person.getAge();
        people.put(person.getId(),person);
	}
}

在查询的时候直接返回valueSum即可

@Override
public int getValueSum() {
	return this.valueSum;
}

3 bug修复情况

3.1 Homework9

这次作业完成的比较仓促,所以由有两个bug:

  1. 并查集在添加parent结点的时候没有考虑到本身就是最顶层节点的情况,所以会导致在并查集查询的时候出现死循环tle的情况。
  2. 纯属手误,输出信息打错了。(汗流光了

3.2 Homework10

这次作业主要问题是性能问题:

将一些变量查询方法改为动态维护之后节省了很多时间,就不tle了。

3.3 Homework11

Dijkstra堆优化算法有一个小bug,导致有一个点tle

4 测试数据准备

集中测试:

边缘数据:

构造异常数据:

5 Network扩展任务

假设出现了几种不同的Person

如此Network可以支持市场营销,并能查询某种商品的销售额和销售路径等 请讨论如何对Network扩展,给出相关接口方法,并选择3个核心业务功能的接口方法撰写JML规格(借鉴所总结的JML规格模式)

addProduct(Product product):添加产品

/*@ public normal_behavior
      @ requires !(\exists int i; 0 <= i && i < products.length; peoducts[i].equals(product));
      @ assignable products;
      @ ensures products.length == \old(products.length) + 1;
      @ ensures (\forall int i; 0 <= i && i < \old(products.length);
      @          (\exists int j; 0 <= j && j < products.length; products[j] == (\old(products[i]))));
      @ ensures (\exists int i; 0 <= i && i < products.length; products[i] == product);
      @ also
      @ public exceptional_behavior
      @ signals (EqualProductIdException e) (\exists int i; 0 <= i && i < products.length;
      @                                     products[i].equals(product));
      @*/
    public void addProduct(/*@ non_null @*/Product product) throws EqualProductIdException;

queryProductValue(Product product):查询商品的value

/*@ public normal_behavior
      @ requires (\exists int i; 0 <= i && i < products.length; 
      @          products[i].getId() == product.getId());
      @ assignable \nothing;
      @ ensures (\exists int i; 0 <= i && i < products.length; 
      @         products[i].getId() == product.getId() && \result == value[i]);
      @ also
      @ public normal_behavior
      @ requires (\forall int i; 0 <= i && i < products.length; 
      @          products[i].getId() != product.getId());
      @ ensures \result == 0;
      @*/
    public /*@ pure @*/ int queryProductValue(Product product);

sendPurchaseMessage():Customer直接通过Advertiser给相应Producer发一个购买消息

/*@ public normal_behavior
  @ requires containsPurchaseMessage(id);
  @ assignable purchaseMessages;
  @ assignable people[*].purchaseMessages;
  @ ensures !containsPurchaseMessage(id) && purchaseMessages.length == \old(purchaseMessages.length) - 1 &&
  @         (\forall int i; 0 <= i && i < \old(purchaseMessages.length) && \old(purchaseMessages[i].getId()) != id;
  @         (\exists int j; 0 <= j && j < purchaseMessages.length; purchaseMessages[j].equals(\old(purchaseMessages[i]))));
  @ ensures (\forall int i; 0 <= i && i < people.length; person[i].likeType(\old(((PurchaseMessage)getMessage(id))).getType) ==> 
  @         ((\forall int j; 0 <= j && j < \old(person[i].getMessages().size());
  @          person[i].getMessages().get(j+1) == \old(person[i].getMessages().get(j))) &&
  @         (person[i].getMessages().get(0).equals(\old(getMessage(id)))) &&
  @         (person[i].getMessages().size() == \old(person[i].getMessages().size()) + 1)));
  @ also
  @ public exceptional_behavior
  @ signals (MessageIdNotFoundException e) !containsPurchaseMessage(id);
  @*/
public /*@ pure @*/ void sendPurchaseMessage(int id) throws PurchseMessageIdNotFoundException;

6 本单元学习体会

标签:OO,getRoot,HashMap,parent,get,BUAA,person,Unit3,getPerson
来源: https://www.cnblogs.com/fxtpn/p/16340281.html