WordCount编码和测试

项目地址:https://github.com/handsomesnail/WordCount

PSP表格

PSP2.1 PSP阶段 预估耗时(分钟) 实际耗时(分钟)
Planning 计划 20 10
Estimate 估计任务需要多少时间 20 10
Development 开发 150 140
Analysis 需求分析 10 10
Design Spec 生成设计文档 10 0
Design Review 设计复审 10 0
Coding Standard 代码规范 10 0
Design 具体设计 20 20
Coding 具体编码 60 80
Code Review 代码复审 10 10
Test 测试 20 20
Reporting 报告 60 90
Test Report 测试报告 30 60
Size Measurement 计算工作量 20 20
Postmortem 总结 10 10
合计 230 240

解题思路

将问题分解为如下模块:

  • 解析命令行参数 (将输入转化为路径以及各种操作选项如-c -w -l等)
  • I/O操作 (将路径转化为相应路径下文件的字符串内容)
  • 字符串统计 (将各种操作选项抽象为各类处理字符串的方法)

程序设计实现

Program类
  • void Main(string[] args)

    Main函数作为程序入口
  • int Entrance(string[] args)

    测试入口, 返回值为错误码,正确执行返回0
  • string Read(string fileName)

    读取相对路径下名为fileName的文件并返回字符串
  • string Read(out string fileName)

    调用win32 api选取文件并读取返回字符串,同时返回选取的文件名
  • void Write(string fileName, string content)

    在相对路径下向名为fileName的文件追加内容为content的字符串
  • void LogArgumentError(int code)

    打印错误信息
  • int CountChar(string s)

    统计并返回字符串s的字符数
  • int CountWord(string s)

    统计并返回字符串s的单词数
  • int CountWord(string s,HashSet stopList)

    根据停用词表stopList统计并返回字符串s的单词数
  • int CountLine(string s)

    统计并返回字符串s的行数
  • void CountMoreAboutLine(string s, out int codeLineCount, out int emptyLineCount, out int commentLineCount)

    统计并返回行的详细信息(代码行/空行/注释行)
OpenFileName类

初始化打开或另存为对话框的信息,系统返回关于用户的选择信息到这个结构中

LocalDialog类
  • extern bool GetOpenFileName([In, Out] OpenFileName ofn)

    链接Comdlg32.dll中打开文件的系统函数

代码说明

/// <summary>统计字符数并打印</summary>
private static int CountChar(string s) {
return s.Length; //直接返回字符串长度
}
/// <summary>统计单词数并打印</summary>
private static int CountWord(string s) {
return CountWord(s, new HashSet<string>());
}
/// <summary>根据停用词表统计单词数并打印</summary>
private static int CountWord(string s,HashSet<string> stopList) {
if (s.Length == 0)
return 0;
int count = 1;
int i = -1;
StringBuilder sb = new StringBuilder();
foreach (char c in s) {
i++;
if (splitCharDic.Contains(c)) {
if (i < s.Length - 1 && !splitCharDic.Contains(s[i + 1])) {
if (!stopList.Contains(sb.ToString())) {
count++;
//Console.WriteLine(count + ":" + sb.ToString());
sb.Clear();
}
else {
sb.Clear();
}
}
}
else {
sb.Append(c);
}
}
return count;
}
/// <summary>统计行数并打印</summary>
private static int CountLine(string s) {
int count = 0;
//遍历直接统计'\n'数量
foreach (char c in s) {
if (count == 0) {
count++;
continue;
}
if (c == '\n') {
count++;
}
}
return count;
}
/// <summary>统计行的详细信息</summary>
private static void CountMoreAboutLine(string s, out int codeLineCount, out int emptyLineCount, out int commentLineCount) {
codeLineCount = 0;//代码行
emptyLineCount = 0;//空行
commentLineCount = 0;//注释行
int charInALine = 0; //当前行可见字符数
int i = -1;
CommentLineType CommentLine = CommentLineType.NonCommentLine;
bool cancelCommentLine = false;
foreach (char c in s) {
i++;
if (c == '\n'||i==s.Length-1) {
if (CommentLine==CommentLineType.SingleCommentLine) {
commentLineCount++;
CommentLine = CommentLineType.NonCommentLine;
}
else if ((CommentLine == CommentLineType.SeveralCommentLine|| cancelCommentLine )&& charInALine<=1) {
commentLineCount++;
}
else if (charInALine > 1) {
codeLineCount++;
}
else emptyLineCount++;
if (cancelCommentLine) {
cancelCommentLine = false;
}
charInALine = 0;
}
if (charInALine <= 1 && c == '/' && i > 0 && s[i - 1] == '/') {
CommentLine = CommentLineType.SingleCommentLine;
}
else if (c == '*' && i > 0 && s[i - 1] == '/') {
CommentLine = CommentLineType.SeveralCommentLine;
}
else if (c == '/' && i > 0 && s[i - 1] == '*') {
cancelCommentLine = true;
CommentLine = CommentLineType.NonCommentLine;
}
else if (CommentLine == CommentLineType.NonCommentLine && !displayedCharDic.Contains(c)) {
charInALine++;
}
}
}
/// <summary>通过图形界面读取文件 </summary>
public static string Read(out string fileName) {
OpenFileName ofn = new OpenFileName();
ofn.structSize = Marshal.SizeOf(ofn);
ofn.file = new string(new char[256]);
ofn.maxFile = ofn.file.Length;
ofn.fileTitle = new string(new char[64]);
ofn.maxFileTitle = ofn.fileTitle.Length;
ofn.title = "选择文件";
ofn.flags = 0x00080000 | 0x00001000 | 0x00000800 | 0x00000200 | 0x00000008;
if (LocalDialog.GetOpenFileName(ofn)) {
fileName = ofn.fileTitle;
using (FileStream fs = new FileStream(ofn.file, FileMode.Open, FileAccess.Read)) {
using (StreamReader sw = new StreamReader(fs, Encoding.UTF8)) {
return sw.ReadToEnd();
}
}
}
throw new System.Exception("用户没有选择文件");
}

测试设计过程

  • 如何设计测试用例

    保证设计的测试用例应至少覆盖函数中所有的可执行语句
  • 哪些地方会导致程序高风险

    边界条件,未捕获的异常,未正确初始化的内存空间,空引用等等
  • 测试代码如何设计

    通过验证测试入口函数和相应分支的返回码覆盖代码的所有分支
namespace UnitTest {
[TestClass]
public class UnitTest1 {
[TestMethod]
public void TestMethod1() {
string[] input = new string[] { };
Assert.AreEqual(Program.Entrance(input), -1);
}
[TestMethod]
public void TestMethod2() {
string[] input = new string[] {"-q", "input.txt" };
Assert.AreEqual(Program.Entrance(input), 2);
}
[TestMethod]
public void TestMethod3() {
string[] input = new string[] {"-c", "-c" ,"input.txt"};
Assert.AreEqual(Program.Entrance(input), 1);
}
[TestMethod]
public void TestMethod4() {
string[] input = new string[] { "-c", "input.txt", "-a" };
Assert.AreEqual(Program.Entrance(input), 7);
}
[TestMethod]
public void TestMethod5() {
string[] input = new string[] { "-c", "input.txt", "-o", "output.txt", "-c" };
Assert.AreEqual(Program.Entrance(input), 4);
}
[TestMethod]
public void TestMethod6() {
string[] input = new string[] { "-c", "input.txt","-o" };
Assert.AreEqual(Program.Entrance(input), 5);
}
[TestMethod]
public void TestMethod7() {
string[] input = new string[] { "-c", "-w", "-l" ,"example.txt" };
Assert.AreEqual(Program.Entrance(input), 6);
}
[TestMethod]
public void TestMethod8() {
string[] input = new string[] { "-c", "-w", "-l", "input.txt" };
Assert.AreEqual(Program.Entrance(input), 0);
}
[TestMethod]
public void TestMethod9() {
string[] input = new string[] { "-l", "-w", "-c", "input.txt", "-o" ,"output.txt"};
Assert.AreEqual(Program.Entrance(input), 0);
}
[TestMethod]
public void TestMethod10() {
string[] input = new string[] { "-l", "-w", "-c", "-a", "input.txt", "-e", "stopList.txt", "-o", "output.txt" };
Assert.AreEqual(Program.Entrance(input), 0);
}
[TestMethod]
public void TestMethod11() {
string[] input = new string[] { "-l", "-w", "-c", "-a", "-s", "*.txt", "-e", "stopList.txt" };
Assert.AreEqual(Program.Entrance(input), 0);
}
[TestMethod]
public void TestMethod12() {
string[] input = new string[] { "-x" };
Assert.AreEqual(Program.Entrance(input), 0);
} }
}
  • 测试结果


参考文献链接:

[1]http://www.cnblogs.com/xinz/archive/2011/10/22/2220872.html

[2]http://www.cnblogs.com/xinz/p/5044037.html

[3]http://http

WordCount编码和测试的更多相关文章

  1. WordCount 编码与测试

    word count github 项目地址:https://github.com/liuqiang666/wordCount PSP表格 PSP2.1  PSP阶段  预估耗时(小时)  实际耗时( ...

  2. WordCount编码与测试

    1. github项目地址:https://github.com/wwwwu/WordCount 2.PSP表格: PSP2.1 PSP阶段 预估耗时 (分钟) 实际耗时 (分钟) Planning ...

  3. 软件质量与测试——WordCount编码实现及测试

    1.GitHub地址       https://github.com/noblegongzi/WordCount 2.PSP表格 PSP2.1 PSP 阶段 预估耗时 (分钟) 实际耗时 (分钟) ...

  4. WordCount编码测试

    Github项目地址:https://github.com/LantyrLYL/WordCount PSP表格: PSP2.1 PSP阶段 预估耗时 (分钟) 实际耗时 (分钟) Planning 计 ...

  5. 软件测试第2周个人作业:WordCount编码测试

    一.Github地址 https://github.com/zhouyubei/WordCount 二.PSP表格 PSP2.1 PSP阶段 预估耗时 (分钟) 实际耗时 (分钟) Planning ...

  6. WordCount程序与测试

    Github地址: https://github.com/hcy6668/wordCount PSP表格: PSP PSP阶段 预估耗时(分钟) 实际耗时(分钟) Planning 计划 60 40 ...

  7. WordCountPro 编码与测试

    WordCountPro github项目地址:https://github.com/handsomesnail/WordCountPro PSP表格 PSP2.1  PSP阶段  预估耗时(小时) ...

  8. WordCount程序及测试

    Github地址:https://github.com/CG0317/WordCount PSP表: PSP2.1 PSP阶段 预估耗时 (分钟) 实际耗时 (分钟) Planning 计划  30 ...

  9. mysql字符集编码乱码测试如下

    创建三个表tb_latin1,tb_utf8,tb_gbk,编码分别为latin1/utf8/gbk “你好a”字符串编码如下GBK : %C4%E3 %BA%C3 %61UTF-8 : %E4%BD ...

随机推荐

  1. Unity3D教程:制作与载入AssetBundle

    通常我们在游戏程式执行过程,并不希望一次将全部的资源都载入,而比较希望实际上有使用到的才载入,以免占用多余的记忆体,所以我们可能会尽量规划好不同功能的场景,在需要时才载入场景并释放掉前个场景中不需要的 ...

  2. 爬虫利器 Puppeteer

    http://wintersmilesb101.online/2017/03/24/use-phantomjs-dynamic/    一起学爬虫 Node.js 爬虫篇(三)使用 PhantomJS ...

  3. ZIP 算法详解 (转!)

    zip 的压缩原理与实现(lz77 算法压缩) 无损数据压缩是一件奇妙的事情,想一想,一串任意的数据能够根据一定的规则转换成只有原来 1/2 - 1/5 长度的数据,并且能够按照相应的规则还原到原来的 ...

  4. 异常:java.lang.IllegalStateException: No instances found of configserver(里面是一个微服务名)

    今天本地测试代码时出现了个异常,该异常出现的原因是:微服务启动的顺序出现了问题: 应该先启动本地eureka,然后在启动本地配置中心,然后在启动具体的微服务.

  5. Windows SID理解

    Windows安全性要依赖于几个基本元素.:访问令牌.SID.安全描述符.访问控制列表.密码. 访问令牌:访问令牌在本质上定义了两 上“P”:Permissions(权限)和Privilege(特权) ...

  6. extern关键字祥解

    1 基本解释:extern可以置于变量或者函数前,以标示变量或者函数的定义在别的文件中,提示编译器遇到此变量和函数时在其他模块中寻找其定义.此外extern也可用来进行链接指定. 也就是说extern ...

  7. kubernetes 学习 创建cronjob

    POM.xml <?xml version="1.0" encoding="UTF-8"?> <project xmlns="htt ...

  8. 基于opencv+ffmpeg的镜头分割

    镜头分割常常被用于视频智能剪辑.视频关键帧提取等场景. 本文给出一种解决镜头分割问题的思路,可分为两个步骤: 1.根据镜头分割算法对视频进行分割标记 核心在于镜头分割算法,这里简单描述一种算法思路:r ...

  9. 【转】ruby中nil?, empty? and blank?的选择

    In Ruby, you check with nil? if an object is nil:article = nil article.nil? # => true empty? chec ...

  10. highcharts图表显示鼠标选择的Y轴提示线

    tooltip: { shared: true, crosshairs: [true, false] },