2018-2019-2 20175308实验二《面向对象程序设计》实验报告

20175308 实验二《Java面向对象程序设计》实验报告

一、前期准备:

单元测试和TDD:

(一)单元测试

我们首先要会写三种代码:

  • 伪代码
  • 产品代码
  • 测试代码
    Java编程中,我们首先写伪代码,它与具体的编程语言无关,从意图层面来解决问题,是产品代码最自然的、最好的注释。当我们写好产品代码后,我们还要写测试代码。Java编程中程序员对类实现的测试即为单元测试。在多人合作完成软件时,如何能让自己负责的模块功能定义尽量明确,且模块的质量得到稳定的、量化的保证?单元测试就是一个很有效的解决方案。

    (二)TDD(Test Driven Devlopment, 测试驱动开发)

    编程中,我们往往不会选择先写产品代码再写测试代码,而是应该先写测试代码,然后再写产品代码,从而保证写出来的代码就是正确的。这种开发方法就叫做测试驱动开发(TDD),TDD的一般步骤如下:

  • 明确当前要完成的功能,记录成一个测试列表
  • 快速完成编写针对此功能的测试用例
  • 测试代码编译不通过(没产品代码呢)
  • 编写产品代码
  • 测试通过
  • 对代码进行重构,并保证测试通过(重构下次实验练习)
  • 循环完成所有功能的开发

    Junit的安装与使用:

  1. 打开idea,Preferences中点击Plugins,在market中搜索junit,如图点选JUnitGenerator V2.0进行安装,安装后会显示installed
  2. 在安装后,按照老师博客中的操作建立与src同等级的test,用于存放各程序的test文件
  3. 新建立的文件可以点击小灯泡,建立test文件,或者选中类名,Go to>test,新建test文件并勾选要测试的方法即可

二、实验步骤:

任务一:在一个MyUtil类中解决一个百分制成绩转成“优、良、中、及格、不及格”五级制成绩的功能。

  • 伪代码:
百分制转五分制:
如果成绩小于60,转成“不及格”
如果成绩在60与70之间,转成“及格”
如果成绩在70与80之间,转成“中等”
如果成绩在80与90之间,转成“良好”
如果成绩在90与100之间,转成“优秀”
其他,转成“错误”
  • 产品代码:
public class MyUtil{
    public static String percentage2fivegrade(int grade){
        //如果成绩小于60,转成“不及格”
        if (grade<60&&grade>=0)
            return "不及格";
            //如果成绩在60与70之间,转成“及格”
        else if (grade < 70&&grade>=60)
            return "及格";
            //如果成绩在70与80之间,转成“中等”
        else if (grade < 80&&grade>=70)
            return "中等";
            //如果成绩在80与90之间,转成“良好”
        else if (grade < 90&&grade>=80)
            return "良好";
            //如果成绩在90与100之间,转成“优秀”
        else if (grade <= 100&&grade>=90)
            return "优秀";
            //其他,转成“错误”
        else
            return "错误";
    }
}
  • 测试代码:
import org.junit.Test;
import junit.framework.TestCase;
public class MyUtilTest extends TestCase {
    @Test
    public void testNormal() {
        assertEquals("不及格", MyUtil.percentage2fivegrade(55));
        assertEquals("及格", MyUtil.percentage2fivegrade(65));
        assertEquals("中等", MyUtil.percentage2fivegrade(75));
        assertEquals("良好", MyUtil.percentage2fivegrade(85));
        assertEquals("优秀", MyUtil.percentage2fivegrade(95));
    }
    @Test
    public void testException(){
        assertEquals("错误",MyUtil.percentage2fivegrade(-55));
        assertEquals("错误",MyUtil.percentage2fivegrade(105));
    }
    @Test
    public void testBoundary(){
        assertEquals("不及格", MyUtil.percentage2fivegrade(0));
        assertEquals("及格",MyUtil.percentage2fivegrade(60));
        assertEquals("中等",MyUtil.percentage2fivegrade(70));
        assertEquals("良好",MyUtil.percentage2fivegrade(80));
        assertEquals("优秀",MyUtil.percentage2fivegrade(90));
        assertEquals("优秀",MyUtil.percentage2fivegrade(100));
    }
}
  • 系统反馈结果,如果测试失败,IDEA会指出具体哪个测试用例出现错误,修改代码直至所有测试均通过
    测试成功截图

任务二:以TDD的方式研究学习StringBuffer

老师给出的StringBufferDemo代码如下

public class StringBufferDemo{
   public static void main(String [] args){
       StringBuffer buffer = new StringBuffer();
       buffer.append('S');
       buffer.append("tringBuffer");
       System.out.println(buffer.charAt(1));
       System.out.println(buffer.capacity();
       System.out.println(buffer.indexOf("tring"));
       System.out.println("buffer = " + buffer.toString());
  }
}    

对以上代码进行改写,得到每一步调用方法的返回来进行验证,改写代码如下:

public class StringBufferDemo{
    StringBuffer buffer = new StringBuffer();
    public StringBufferDemo(StringBuffer buffer){
        this.buffer = buffer;
    }
    public Character charAt(int i){
        return buffer.charAt(i);
    }
    public int capacity(){
        return buffer.capacity();
    }
    public int length(){
        return buffer.length();
    }
    public int indexOf(String buf) {
        return buffer.indexOf(buf);
    }
}

通过查阅相关资料和推测,各方法的作用如下:

  1. charAt(int i):得到字符串中第i个位置的字符,考虑到数组下标从0开始,字符串的位置也从0开始记。
  2. capacity():返回当前容量。容量指可用于最新插入的字符的存储量,超过这一容量就需要再次进行分配。
  3. length():得到字符串长度
  4. indexOf(String buf):得到buf字符串第一次出现的位置,该位置为buf中第一个字符的位置
    进行测试,测试代码如下:
import junit.framework.TestCase;
import org.junit.Test;
import static org.junit.Assert.*;
// 20175308
public class StringBufferDemoTest extends TestCase {
    StringBuffer a = new StringBuffer("StringBuffer");
    StringBuffer b = new StringBuffer("StringBufferStringBuffer");
    StringBuffer c = new StringBuffer("StringBuffer tested by 175308");
    @Test
    public void testCharAt() throws Exception{
        assertEquals('S',a.charAt(0));//验证返回是否是整个字符串中的第一个字符
        assertEquals('g',b.charAt(5));//验证返回是否是整个字符串的第六个字符
        assertEquals('t',c.charAt(16));
    }
    @Test
    public void testcapacity() throws Exception{//
        assertEquals(28,a.capacity());
        assertEquals(40,b.capacity());
        assertEquals(45,c.capacity());

    }
    @Test
    public void testlength() throws Exception{
        assertEquals(12,a.length());//验证字符串a的长度
        assertEquals(24,b.length());
        assertEquals(29,c.length());
    }
    @Test
    public void testindexOf(){
        assertEquals(6,a.indexOf("Buff"));
        assertEquals(3,b.indexOf("ing"));
        assertEquals(23,c.indexOf("1753"));
    }
}

测试成功截图

任务三:对设计模式示例进行扩充,体会OCP原则和DIP原则的应用,初步理解设计模式

我的学号后两位对6取余为2,应该使系统支持boolean类。在实验开始之前,首先要熟悉SOLID原则:

依据OCP原则和DIP原则,我们在扩充示例使其支持Boolean类时不应该直接添加或修改已有类中的方法(OCP原则要求软件实体对修改封闭),而是采用一种抽象工厂的方法来进行对于程序的扩充。具体代码如下:

abstract class Data {
    abstract public void DisplayValue();
}
class Integer extends  Data {
    int value;
    Integer() {
        value=100;
    }
    public void DisplayValue(){
        System.out.println (value);
    }
}
class Boolean extends Data{
    boolean value;
    Boolean(){
        value=true;
    }
    public void DisplayValue(){
        System.out.println(value);
    }
}
abstract class Factory {
    abstract public Data CreateDataObject();
}
class IntFactory extends Factory {
    public Data CreateDataObject(){
        return new Integer();
    }
}
class BooleanFactory extends Factory{
    public Data CreateDataObject(){
        return new Boolean();
    }
}
class Document {
    Data pd;
    Document(Factory pf){
        pd = pf.CreateDataObject();
    }
    public void DisplayData(){
        pd.DisplayValue();
    }
}
//Test class
public class MyDoc {
    static Document d;
    static Document e;
    public static void main(String[] args) {
        d = new Document(new IntFactory());
        d.DisplayData();
        e=new Document(new BooleanFactory());                                                   //20175308
        e.DisplayData();

    }
}

运行成功截图

任务四:以TDD的方式开发一个复数类Complex

由于在之前的结对项目中已经有过对定义分数类的经验,对于要求较为简单的复数类的定义也算信手拈来。

  • 伪代码:
// 定义属性并生成getter,setter
double RealPart;
double ImagePart;
// 定义构造函数
public Complex()
public Complex(double R,double I)

//Override Object
public boolean equals(Object obj)
public String toString()

// 定义公有方法:加减乘除
Complex ComplexAdd(Complex a)
Complex ComplexSub(Complex a)
Complex ComplexMulti(Complex a)
Complex ComplexDiv(Complex a)
  • 测试代码:
import org.junit.Test;
import junit.framework.TestCase;
public class ComplexTest extends TestCase {
    Complex a =new Complex(3.0,4.0);
    Complex b =new Complex( 2.0,-4.0);
    Complex c =new Complex(0.0,0.0);
    Complex d =new Complex(-3.0,0.0);
    Complex e =new Complex(-6.0,-0.8);
    @Test
    public void testgetRealPart()throws Exception{
        assertEquals(3.0,a.getRealPart());
        assertEquals(2.0,b.getRealPart());
        assertEquals(0.0,c.getRealPart());
        assertEquals(-3.0,d.getRealPart());
        assertEquals(-6.0,e.getRealPart());
    }
    @Test
    public void testgetImagePart()throws Exception{
        assertEquals(4.0,a.getImagePart());
        assertEquals(-4.0,b.getImagePart());
        assertEquals(0.0,c.getImagePart());
        assertEquals(0.0,d.getImagePart());
        assertEquals(0.8,e.getImagePart());

    }
    @Test
    public void testtoString()throws Exception{
        assertEquals("3.0+4.0i",a.toString());
        assertEquals("2.0-4.0i",b.toString());
        assertEquals("0",c.toString());
        assertEquals("-3.0",d.toString());
        assertEquals("-6.0-8.0i",e.toString());
    }
    @Test
    public void testComplexAdd()throws Exception{
        assertEquals("5.0",a.ComplexAdd(b).toString());
        assertEquals("2.0-4.0i",b.ComplexAdd(c).toString());
        assertEquals("-1.0-4.0i",b.ComplexAdd(d).toString());
    }
    @Test
    public void testComplexSub()throws Exception{
        assertEquals("1.0+8.0i",a.ComplexSub(b).toString());
        assertEquals("-2.0+4.0i",c.ComplexSub(b).toString());
        assertEquals("3.0",c.ComplexSub(d).toString());
    }
    @Test
    public void testComplexMulti()throws Exception{
        assertEquals("22.0-4.0i",a.ComplexMulti(b).toString());
        assertEquals("0",b.ComplexMulti(c).toString());
        assertEquals("18.0+2.4i",d.ComplexMulti(e).toString());
    }
    @Test
    public void testComplexDiv()throws Exception{
        assertEquals("-0.2-1.25i",a.ComplexDiv(b).toString());
        assertEquals("0",c.ComplexDiv(b).toString());
    }
    @Test
    public void testequals()throws Exception{
        assertEquals(true,a.equals(a));
        assertEquals(false,a.equals(b));

    }

}
  • 产品代码:
import com.sun.jdi.DoubleValue;
import java.util.zip.CheckedOutputStream;
import java.text.DecimalFormat;

public class Complex {
    double RealPart=0;
    double ImagePart=0;
    public Complex(){}
    public Complex(double RealPart,double ImagePart){
        this.RealPart=RealPart;
        this.ImagePart=ImagePart;

    }
    public double getRealPart(){
        return RealPart;
    }
    public double getImagePart(){
        return ImagePart;
    }
    public String toString(){
        String s = "";
        double r=RealPart;
        double i=ImagePart;
        if(r==0&&i==0){
            s="0";
        }
        else if(r==0&&i!=0){
            s=i+"i";
        }
        else if(r!=0&&i<0){
            s=r+""+i+"i";
        }
        else if(r!=0&&i==0){
            s=r+"";
        }
        else
        {
            s=r+"+"+i+"i";
        }
        return s;
    }
    public boolean equals(Object obj){
        if(this==obj){
            return true;
        }
        else return false;
    }
    DecimalFormat df = new DecimalFormat( "0.0");
    public Complex ComplexAdd(Complex a){
        return new Complex(RealPart+a.getRealPart(),ImagePart+a.getImagePart());
    }
    public Complex ComplexSub(Complex a){
        return new Complex(RealPart-a.getRealPart(),ImagePart-a.getImagePart());
    }
    public Complex ComplexMulti(Complex a){
        double r=RealPart*a.getRealPart()-ImagePart*a.getImagePart();
        double i =ImagePart*a.getRealPart()+RealPart*a.getImagePart();
        return new Complex(Double.valueOf(df.format(r)),Double.valueOf(df.format(i)));
    }
    public Complex ComplexDiv(Complex a){
        double r=(RealPart * a.ImagePart + ImagePart * a.RealPart) / (a.ImagePart * a.ImagePart + a.RealPart * a.RealPart);
        double i=(ImagePart * a.ImagePart + RealPart * a.RealPart) / (a.RealPart * a.RealPart + a.RealPart * a.RealPart);
        return new Complex(Double.valueOf(df.format(r)),Double.valueOf(df.format(i)));
    }

}

测试成功截图

PSP图

步骤 耗时 百分比
需求分析 2min 4%
设计 5min 10%
代码实现 12min 24%
测试 25min 50%
分析总结 6min 12%

任务五:对实验二中的代码进行建模

对于UML图的使用一直缺少锻炼,直到进行结对项目对项目进行建模才开始学着使用。有关UML图的使用可以在网络上找到很详尽的讲解:参考链接
任务要求类图中至少要有两个类,最为符合要求的便是MyDoc.java的代码了,UML图如下:

实验中遇到的问题及解决:

1.下载Junit后仍然出现标红无法使用。

有关这个问题的解决网络上给出了五花八门许许多多的答案,但我尝试过来一直都没有找到正确的解决方法,直到我又仔细的阅览了一下老师的博客,其实已经给了详尽的操作,里面提醒了一步倒包的操作,由于系统差别一时没有对应上。mac版IDEA解决方法如下:

  • File菜单栏找到Project Structure

  • 点击Dependencies,直接勾选即可

2.任务一中传入-55系统输出不及格而非错误。

此时是未经修改过的代码,此时代码如下:

public class MyUtil{
    public static String percentage2fivegrade(int grade){
        //如果成绩小于60,转成“不及格”
        if (grade<60)
            return "不及格";
            //如果成绩在60与70之间,转成“及格”
        else if (grade < 70)
            return "及格";
            //如果成绩在70与80之间,转成“中等”
        else if (grade < 80)
            return "中等";
            //如果成绩在80与90之间,转成“良好”
        else if (grade < 90)
            return "良好";
            //如果成绩在90与100之间,转成“优秀”
        else if (grade <= 100)
            return "优秀";
            //其他,转成“错误”
        else
            return "错误";
    }
}

当传入-55时,虽然-55的分数理论上一定是错误的,但是在代码中只会检测其是否小于60,满足条件后直接得出结果。该问题的解决其实很简单,只需在判断是否小于60之前先加一步判断正负的步骤即可(超出100的数字判断由最后的else负责)。出于C语言学习时的习惯,我用&&对每一步的判断条件进行连接,形成我最后的代码。

3.复数类中涉及double运算可能会给出一个极长位数的答案

在double类型的运算中,经常由于一些不可描述的原因产生如下的结果:

理论上心里知道咋回事就可以了,但为了看到百分百的Test passed,还是要含泪修改。
原有的代码为(主要出错在乘除法):


ComplexMulti(Complex a){
        return new Complex(RealPart*a.getRealPart()-ImagePart*a.getImagePart(),ImagePart*a.getRealPart()+RealPart*a.getImagePart());
    }
    public Complex ComplexDiv(Complex a){
        return new Complex((RealPart * a.ImagePart + ImagePart * a.RealPart) / (a.ImagePart * a.ImagePart + a.RealPart * a.RealPart), (ImagePart * a.ImagePart + RealPart * a.RealPart) / (a.RealPart * a.RealPart + a.RealPart * a.RealPart));
    }

经过查阅相关资料,java指定小数点后的位数可以通过导入import java.text.DecimalFormat;来实现,DecimalFormat df = new DecimalFormat( "0.0");就将小数点后的位数控制在了一位。
注意??:此时的df.format(r)为string类型,还要再进行一次类型转换
将产品代码修改为任务二中的版本后,再运行一次刚才出错的test,测试通过!

三、参考:

  • http://www.cnblogs.com/rocedu/p/6371315.html#SECUNITTEST
  • http://www.cnblogs.com/rocedu/p/6736847.html
  • https://blog.csdn.net/tastill/article/details/80346443
  • https://blog.csdn.net/xinqing5130/article/details/84099628

原文地址:https://www.cnblogs.com/darklord0604/p/10686137.html

时间: 2024-07-31 07:02:38

2018-2019-2 20175308实验二《面向对象程序设计》实验报告的相关文章

20165201 实验二面向对象程序设计

##20165201 实验二面向对象程序设计 Java面向对象程序设计-1 实验目的与要求: 参考http://www.cnblogs.com/rocedu/p/6371315.html#SECUNITTEST 完成单元测试的学习 提交最后三个JUnit测试用例(正常情况,错误情况,边界情况)都通过的截图,截图上要有画图加水印,输入自己的学号 测试用例至少要包含正常情况,错误情况,边界情况的测试 实验内容及结果截图: 伪代码(可以用中文,最好用英文~) 百分制转五分制: 如果成绩小于60,转成"

20165223 实验二 面向对象程序设计

目录 一.实验报告封面 二.具体实验内容 (一)初步掌握JUnit的使用 (二)IDEA中的JAVA程序开发 (三)自主练习 三.实验总结 四.PSP时间 一.实验报告封面 北京电子科技学院(BESTI) 实 验 报 告 ◆ ◇ ◆ ◇ 课程 Java程序设计 班级 1652班 姓名 蔡霓 学号 20165223 成绩 指导教师 娄嘉鹏 实验日期 2018年4月16日 实验密级 非密级 预习程度 已预习 实验时间 13:45 - 15:25 必修/选修 选修 实验序号 二 实验名称:面向对象程序

2017-2018-1 20155222 201552228 实验二 固件程序设计

2017-2018-1 20155222 201552228 实验二 固件程序设计 实验内容 实验三-并发程序-1 学习使用Linux命令wc(1) 基于Linux Socket程序设计实现wc(1)服务器(端口号是你学号的后6位)和客户端 客户端传一个文本文件给服务器 服务器返加文本文件中的单词数 上方提交代码 附件提交测试截图,至少要测试附件中的两个文件 实验三-并发程序-2 使用多线程实现wc服务器并使用同步互斥机制保证计数正确 上方提交代码 下方提交测试 对比单线程版本的性能,并分析原因

20145326《Java程序设计》实验二Java面向对象程序设计实验报告

20145326<Java程序设计>实验二Java面向对象程序设计实验报告 实验内容 初步掌握单元测试和TDD 理解并掌握面向对象三要素:封装.继承.多态 初步掌握UML建模 熟悉S.O.L.I.D原则 了解设计模式 实验步骤 一.单元测试 1. 三种代码·伪代码·产品代码·测试代码Example:需求:我们要在一个MyUtil类中解决一个百分制成绩转成“优.良.中.及格.不及格”五级制成绩的功能. 伪代码: 百分制转五分制: 如果成绩小于60,转成“不及格” 如果成绩在60与70之间,转成“

java的第二个实验——JAVA面向对象程序设计

java的第二个实验——JAVA面向对象程序设计 北京电子科技学院 实     验    报     告 课程:Java程序设计 班级:1352 姓名:林涵锦 学号:20135213 成绩:        指导教师:娄嘉鹏  实验日期:2015.4.14 实验密级: 预习程度:  实验时间:19:30~21:50 仪器组次:13 必修/选修:选修 实验序号:2 实验名称:JAVA面向对象程序设计 实验目的与要求: 1. 初步掌握单元测试和TDD 2. 理解并掌握面向对象三要素:封装.继承.多态

2018-2019-1 20165201 20165203 20165206 实验二 固件程序设计

- 2018-2019-1 20165201 20165203 20165206 实验二 固件程序设计 - 任务一 固件程序设计-1-MDK - 任务要求 0. 注意不经老师允许不准烧写自己修改的代码 1. 三人一组 2. 参考云班课资源中"信息安全系统实验箱指导书.pdf "第一章,1.1-1.5安装MDK,JLink驱动,注意,要用系统管理员身分运行uVision4,破解MDK(破解程序中target一定选ARM) 3. 提交破解程序中产生LIC的截图 4. 提交破解成功的截图 -

2018-2019-1 20165308 20165317 20165331 实验二 固件程序设计

2018-2019-1 20165308 20165317 20165331 实验二 固件程序设计 成员 20165308张士洋 20165317宁心宇 20165331胡麟 实验二 固件程序设计-1-MDK 0. 注意不经老师允许不准烧写自己修改的代码 1. 三人一组 2. 参考云班课资源中"信息安全系统实验箱指导书.pdf "第一章,1.1-1.5安装MDK,JLink驱动,注意,要用系统管理员身分运行uVision4,破解MDK(破解程序中target一定选ARM) 3. 提交破

2018-2019-1 20165227 20165228 20165237 实验二 固件程序设计

2018-2019-1 20165227 20165228 20165237 实验二 固件程序设计 实验步骤 任务一:固件程序设计-1-MDK 要求: 0.注意不经老师允许不准烧写自己修改的代码 1.三人一组 2.参考云班课资源中"信息安全系统实验箱指导书.pdf "第一章,1.1-1.5安装MDK,ULink驱动 注意:要用系统管理员身分运行uVision4,破解MDK(破解程序中target一定选ARM) 3.提交破解程序中产生LIC的截图 4.提交破解成功的截图 实验步骤截图:

2019-2020-1 20175312 20175321 20175333 实验二 固件程序设计

2019-2020-1 20175312 20175321 20175333 实验二 固件程序设计 本小组成员 20175312 20175321 20175333 实验二 固件程序设计-1-MDK 0. 注意不经老师允许不准烧写自己修改的代码 1. 三人一组 2. 参考云班课资源中"信息安全系统实验箱指导书.pdf "第一章,1.1-1.5安装MDK,JLink驱动,注意,要用系统管理员身分运行uVision4,破解MDK(破解程序中target一定选ARM) 3. 提交破解程序中产

2019-2020-1 20175334 20175322 20175315 实验二 固件程序设计

我们组陈煜扬同学的博客: 实验二 固件程序设计 补充截图: 原文地址:https://www.cnblogs.com/lxr1006/p/11877739.html