用4种客户端架构(MVC、MVP、MVVM、FRP)分别实现局域网群聊的Android应用。
PPT可以在这里下载到。
完整报告可以请访问陈俊达的博客文章或者本仓库下的docs/报告.md
可以访问陈俊达的博客的PPT资源界面或者他专门放PPT的Repo以查找到他的更多PPT。
首先启动Server项目,然后使用多个模拟器/真机启动Client项目,输入用户名,即可开始聊天。
要修改登录后进入哪个窗口,修改app/assets/config.properties文件中chat_activity为对应的类名。
- 建立与多个客户端的连接
- 接受任何一个已连接的客户端发送的消息,并发送到其他所有客户端
- 发起到服务器的连接
- 接受用户输入文字信息
- 发送文字信息到服务器
- 接受服务器文字信息并显示
功能编号 | 实现的功能点简介 | 功能详细介绍 | 使用架构 | 使用本架构的代码实现 | 相对于上一个架构的优点 | 使用此架构实现本需求点的不足之处 | 更好的架构 | 更好的实现 |
---|---|---|---|---|---|---|---|---|
1 | 增加消息记录云备份功能 | 增加一个界面,界面中有一个按钮,点击按钮后将本地的消息记录发送到某个云服务器 | MVC | 增加一个Activity,Activity连接一个Controller。当用户点击备份按钮后,Activity将备份请求转发给C,C转发给M,M发起网络请求,且在结束后通知V修改界面。 | 当按下按钮后View无法及时响应(M还在等待响应,没有通知V数据变化) | MVP | P能够控制V(对比C不能控制V),所以当V(Activity)将备份请求给P时,P可以控制V进行及时的、恰当的显示(比如显示备份中字样) | |
2 | 撤回消息 | 某个客户端的用户选择一条消息撤回(只能是自己发的消息),其他用户的客户端上,如果该消息存在于浏览页面,这一消息会被置换为“该消息已被撤回” | MVP | 在P和V中都增加处理逻辑,P中当接受到撤回消息的请求时,调用V的撤回消息方法;V的响应方法中,将对应消息文本框的text设置成“已撤回” | 需要同时修改P和V。当业务逻辑复杂时(例如说一个消息可能具有很多状态(撤回也可以考虑成一个消息的状态)每个状态都需要在UI上进行特殊显示时),P和V可能需要增加很多处理函数。 | MVVM | 将消息状态和界面进行双向绑定,修改消息ViewModel的状态就会同时更改界面中对象消息的状态,不需要单独的逻辑处理。同时也可以把界面上的输入框和某个属性进行双向绑定,减少UI事件处理的代码。(这个其实可以算一个单独的功能) | |
3 | 过滤脏话 | 当客户端接受到信息时,根据预设的脏话列表(表现为正则表达式数组)匹配信息内容,如果匹配到信息里包含脏话,则信息修改为*** | MVVM | 修改在VM和UI组件的绑定处理函数中(就是,当VM改变时UI应该怎么改变),增加判断逻辑。若包含脏话,则将显示的信息设置为*** | 需要修改逻辑。若逻辑更加复杂,则可能使此绑定函数过于冗长,难以维护。 | FRP | 将接受到的信息看作一个流,增加判断逻辑只是在这个流上增加过滤函数而已;增加逻辑也只需要增加过滤函数而不是修改已有逻辑 | |
4 | 限制用户发送消息频率 | 限制用户在1s中只能发出一条信息,消息发送后1s内不允许发送。 | FRP | 将用户发送信息也考虑为一个流,使用throttle对发送消息函数进行节流 | 只需要增加流处理函数,不需要修改已有逻辑,也不需要手动写计时器 |
- 使用Java语言(Kotlin可能到时候他们看不懂)
- 所有客户端和服务器运行在同一台机器,使用Socket通信
在实际实现中,我们将采用以下形式:
- 首先使用MVC,MVP,MVVM和FRP四种架构分别实现基础需求
- 初始设计中考虑所有功能点的前置条件
- 功能1:客户端本地已经保存消息记录
- 功能2:需要项目配好junit,MVP中P对M和V应该是依赖接口,而不是依赖具体类
- 功能3:需要区分不同的请求类型;每个接受到的信息可以独立修改
- 功能4:倒是没有特殊的前置条件
- 功能5:倒是没有特殊的前置条件。
- 作业1:每个接受的的信息用单独的容器显示,而不是在一个TextBox里修改其文本(和功能3比较类似)
- 作业2:同上
- 作业3:这个让他们自己改吧
- 初始设计中考虑所有功能点的前置条件
- 复制1次MVC的原始代码(称为MVP-0),在复制的MVC代码中实现功能1
- 实现了功能1的MVC代码编号为代码MVC-1
- 复制2次MVP的原始代码(MVP-0),在第一份MVP代码中实现功能1,第二份实现功能2
- 实现了功能1的MVP代码编号为代码MVP-1,和MVC-1形成对照
- 实现了功能2的MVP代码编号为代码MVP-2
- 复制2次MVVM的原始代码(MVVM-0),在第一份代码中实现功能2,在第二份代码中实现功能3
- 实现功能2的MVVM代码编号为代码MVVM-2,和MVP-2形成对照
- 实现功能3的MVVM代码编号为代码MVVM-3
- 复制2次FRP的原始代码(FRP-0),在第一份代码中实现功能3,第二份代码中实现功能4
- 实现了功能4的FRP代码编号为FRP-3,和MVVM-3对照
- 实现了功能5的FRP代码编号为FRP-4
三选一
- MVP:在MVP中,除了之前提到的便于测试之外,由于将业务逻辑和View分离,增加功能也只需要增加一个Presenter和少量修改View即可实现。尝试在MVP的原始代码的基础上实现以下功能:
当所发送的信息为
![]({图片地址})
这样的格式时(即markdown的图片),将信息显示为对应的图片。
- MVVM:当消息过多时,有的消息可能没有在当前屏幕上显示出来。尝试在MVVM的原始代码的基础上实现如下功能:
显示不在当前屏幕中显示出来的消息数量,并随着用户滚动消息列表、发送和接受消息实时更新。(Tip:考虑绑定消息列表的当前位置属性)
- FRP:使用现有的XML界面文件,使用流从头实现基本需求,并实现功能3(撤回消息)