标签:
实 验 报 告
学号:20135324
姓名:杨舒雯
指导教师:娄嘉鹏
实验名称:Java敏捷开发与xp实现(有关代码所示图片都为本人亲自操作实验代码)
实验内容:
1. XP基础
2. XP核心实践
3. 相关工具
实验步骤
软件工程是把系统的、有序的、可量化的方法应用到软件的开发、运营和维护上的过程。软件工程包括下列领域:软件需求分析、软件设计、软件构建、软件测试和软件维护。 人们在开发、运营、维护软件的过程中有很多技术、做法、习惯和思想体系。软件工程把这些相关的技术和过程统一到一个体系中,叫“软件开发流程”。软件开发流程的目的是为了提高软件开发、运营、维护的效率,并提高软件的质量、用户满意度、可靠性和软件的可维护性。 光有各种流程的思想是不够的,我们还要有一系列的工具来保证这些思想能够在实践中有效率地运作。软件开发很重要的一点不是看你能对多少理论讲的头头是道,还要看你对相关工具应用的如何,比如Java中单元测试要和JUnit的应用结合起来,建模要和Umbrello或StarUML的应用结合起来。编程学习是一个习而学
的过程。 一个常见的公式是:软件工程=开发流程+工具
邹欣老师给出的两个公式:软件=程序+软件工程
和软件企业=软件+商业模式
开发流程大家可以参考学习邹欣老师的软件团队和开发流程。常见的开发流程有:
敏捷开发(Agile Development)是一种以人为核心、迭代、循序渐进的开发方法。“敏捷流程”是一系列价值观和方法论的集合。从2001年开始,一些软件界的专家开始倡导“敏捷”的价值观和流程,他们肯定了流行做法的价值,但是强调敏捷的做法更能带来价值。他们的敏捷开发宣言:
敏捷开发包括很多模式:
其中,极限编程(eXtreme Programming,XP)是是一种全新而快捷的软件开发方法。XP团队使用现场客户、特殊计划方法和持续测试来提供快速的反馈和全面的交流:
XP软件开发是什么样的
通过 XP准则来表达:
一项实践在XP环境中成功使用的依据通过XP的法则
呈现,包括:快速反馈、假设简单性、递增更改、提倡更改、优质工作。
XP软件开发的基石是XP的活动
,包括:编码、测试、倾听、设计。
项目成员用户成功执行XP活动的技术通过XP实践
来呈现,包括编程、团队、过程相关的12条实践:
我们关注其中的编码标准
,结对编程
,代码集体所有
,测试
,重构
等实践。上次实验已经讲过TDD,通过学习这些实践,可以形成以测试为核心的开发流程:
敏捷可以作为一种做事的方式,掌握好的在以后的工作中也会受益无穷。
编写代码一个重要的认识是“程序大多时候是给人看的”,编程标准使代码更容易阅读和理解,甚至可以保证其中的错误更少。编程标准包含:具有说明性的名字、清晰的表达式、直截了当的控制流、可读的代码和注释,以及在追求这些内容时一致地使用某些规则和惯用法的重要性。
编码标准中的版式就是一个很好的例子,版式虽然不会影响程序的功能,但会影响可读性。程序的版式追求清晰、美观,是程序风格的重要因素。 作为反例,体会一下国际C语言混乱代码大赛 (IOCCC, The International Obfuscated C Code Contest)的一个例子:
这真是一个可以编译的C语言程序,体会一下它的可读性。当然写出这样的代码也是牛人了。我们常见的是这样的代码:
程序没有最基本的缩进,让人读起来很费劲,这个问题在Eclipse中比较容易解决,我们单击Eclipse菜单中的source->Format 或用快捷键Ctrl+Shift+F就可以按Eclipse规定的规范缩进,
效果如下:
代码标准中很重要的一项是如何给包、类、变量、方法等标识符命名,能很好的命名可以让自己的代码立马上升一个档次。Java中的一般的命名规则有:
标识符名字应当直观且可以拼读,可望文知意,不必进行“解码”,一般采用英文单词或其组合,便于记忆和阅读,切忌使用汉语拼音来命名,用词要准确例如“当前值”应该起名currentValue
,写成nowValue
就不准确了,但还凑合,写成dqz
(dang qian zhi 首字母)就是笑话了。
标识符的长度“min-length && max-information”
的原则,比如:maxVal
比 maxValueUntilOverflow
要好些,可以通过去元音法把变量名变短,如returnValue
->rtnVal
,message
->msg
;一般全局变量用具有说明性的名字,局部变量用短名字:单字符的名字,常见的如i,j,k等用作局部变量。
其他的可以参考邹欣老师写的代码规范与代码复审.
关于代码标准,可以遵循以下原则:
有一些公司比如Google公开了自己的编码标准,可以作为学习不错的参考,大家参考一下范飞龙老师写的代码规范&代码风格,有兴趣的可以尝试如何在Eclipse中实践Google Java Style(中文版),也就是说如何做到“按一下快捷键Ctrl+Shift+F
就可以让自己的代码符合Google Java Style(中文版)的要求”,完成后单独写一篇Blog,有加分的。
(三)结对编程
结对编程是XP中的重要实践。在结对编程模式下,一对程序员肩并肩、平等地、互补地进行开发工作。他们并排坐在一台电脑前,面对同一个显示器,使用同一个键盘、同一个鼠标一起工作。他们一起分析,一起设计,一起写测试用例,一起编码,一起做单元测试,一起做集成测试,一起写文档等。 结对编程中有两个角色:
如何结对编程,为何要结对编程,大家参考一下结对编程和两人合作,重点是:
团队精神是好多地方都强调的一个精神,最小的团队就是一对一的二人团队了,培养团队精神从结对编程开始吧。社会生活中人与人相处最重要的是诚信,有同理心,互利。结对编程中大家会出现分歧,如何更有效地合作要做到对事不对人,掌握这些是可以终生受益的,如何影响小伙伴,大家参考一下两人合作:要会做汉堡包。
(四)版本控制
XP的集体所有制意味着每个人都对所有的代码负责;这一点,反过来又意味着每个人都可以更改代码的任意部分。结对编程对这一实践贡献良多:借由在不同的结对中工作,所有的程序员都能看到完全的代码。集体所有制的一个主要优势是提升了开发程序的速度,因为一旦代码中出现错误,任何程序员都能修正它。 这意味着代码要放到一个大家都能方便获取的地方,我们叫代码仓库。这引出另外一个话题叫版本控制(Version Control)。
不论是对于团队还是个体,版本控制都提供了很多好处。
流行的版本控制工具有CVS,SVN,Git等,更多的可以参考这里。Git是Linus除了Linux操作系统外的另外一个重要发明。
实验楼上线我的代码库功能,为大家提供实验环境内置的公开的git服务。学习的课程会自动创建一个公开的代码仓库,命名为shiyanlou_cs[课程ID],比如本课程的代码库命名shiyanlou_cs212,为启动实验时会在环境中自动执行git pull,获取课程仓库最新代码,存放在/home/shiyanlou/Code目录。git push操作需要手动完成,请务必在停止实验前push全部修改,否则代码就丢了。注意:
•代码库链接:http://git.shiyanlou.com/[您的专属用户名] •课程代码仓库的链接为:http://git.shiyanlou.com/[您的专属用户名]/shiyanlou_cs[课程ID] •实验环境中代码路径:/home/shiyanlou/Code/shiyanlou_cs[课程ID]
比如我的专属用户名是rocedu,则我的代码库链接为:http://git.shiyanlou.com/rocedu,我的《Java 程序设计》课程的代码仓库链接为:http://git.shiyanlou.com/rocedu/shiyanlou_cs212,该课程实验环境中代码路径为:/home/shiyanlou/Code/shiyanlou_cs212.
点击立即开通就可以开通我的代码库功能,进入页面完善信息即可,如果已经开通服务,请注意专属用户名必须是字母及数字的组合,建议包含学号信息。
使用方法如下:
$ cd /home/shiyanlou/Code/shiyanlou_cs212
# 修改代码文件
# 添加修改文件
$ git add 所有修改的文件
# 提交到环境中本地代码仓库
$ git commit -m ‘本次修改的描述‘
# push到git.shiyanlou.com,无需输入密码
$ git push
克隆其他用户代码仓库只需要知道对方的仓库链接,我们鼓励在别人代码基础上修改:
$ git clone http://git.shiyanlou.com/[对方的专属用户名]/[课程代码仓库名]
我们给一个HelloWorld的例子: 首先进入Code目录,你会发现有了shiyanlou_cs212目录,进入shiyanlou_cs212,如下图所示:
创建HelloWorld目录,如下图所示:
创建并编辑HelloWorld.java文件,如下图所示:
注意一点,往代码库提交的代码一定编译、运行、测试都没有问题的代码,我们上面测试代码没有问题了,就可以提交了:如图:我们可以先用git status查看一下代码状态,显示有未跟踪的代码,并建议用git add <file>...添加,我们使用git add HelloWorld.* 把要提交的文件的信息添加到索引库中。当我们使用git commit时,git将依据索引库中的内容来进行文件的提交。这只是在本地操作,关闭实验环境,会删除代码的,如果想把代码保存到远程托管服务器中,需要使用git push,实验完成前,一定不要忘了使用git push,否则就是相当于你在Word中编辑了半天文件最后却没有保存。 我们可以修改HelloWorld.java,如下图所示:
编译、运行、测试没有问题后进行提交,这儿使用的是git commit -a:,
我们可以通过git log查看代码提交记录:
谁在什么时候写了什么代码非常清楚。例子到此结束,有个问题是HelloWorld.class是不应该保存在代码库中的,我们只要有HelloWorld.java就行了,这怎么办?通过搜索引擎解决一下。
当然,Git不是只有程序员才用到,所有需要"Undo"的场合,版本控制都能帮助你。参考一下Git for Non-Programmers(中文版)。
更进一步的学习,参考Git使用、工具、原理、进阶的几个链接
(五)重构
我们先看看重构的概念:
重构(Refactor),就是在不改变软件外部行为的基础上,改变软件内部的结构,使其更加易于阅读、易于维护和易于变更 。
重构中一个非常关键的前提就是“不改变软件外部行为”,它保证了我们在重构原有系统的同时,不会为原系统带来新的BUG,以确保重构的安全。如何保证不改变软件外部行为?重构后的代码要能通过单元测试。如何使其更加易于阅读、易于维护和易于变更 ?设计模式给出了重构的目标。
重构重要吗?你看看Eclipse菜单中有个refactor菜单就知道了,重构几乎是现代IDE的标配了:
我们在编码标准中说“给标识符命名”是程序员一项重要技能,以前没有这个意识,现在知道了怎么办?没问题,上图中重构的第一项功能就是Rename,可以给类、包、方法、变量改名字。 例如这有个ABC类:
这个类,类名,方法名和方法的参数名都有问题,没有注释的话是无法理解代码的。我们可以使用Eclipse中的重构功能来改名。修改方法是,用鼠标单击要改的名字,选择Eclipse中菜单中的Refactor->Rename...:
重构完的效果如下:
功能不变,代码水平立马上了一个档次,体会到命名的威力了吧?
学过C语言的学生学Java时常犯的毛病是不会封装,该用类的地方都用了结构体。比如要定义一个类Student,会出现这样的代码:
Eclipse中菜单中的Refactor->Encapsulate Field...,如下图:
上面第34,35行还是有问题的,每次打印学生信息都这么写代码违反了DRY原则,造成代码重复,正常的重构可以使用Eclipse中的Extract Method...,如下图:
由于Java中所有的类都有个专门的toString方法,我们使用Eclipse中Source->Generate toString()... 给Student类产生一个toString方法,修改main的代码,结果如下:
大家想一想,这样重构后有什么好处?重构有什么问题吗?
我们要修改软件,万变不离其宗,无非就是四种动机:
第一种和第二种动机,都是源于客户的功能需求,而第四种是源于客户的非功能需求。软件的外部质量,其衡量的标准就是客户对软件功能需求与非功能需求的满意度。它涉及到一个企业、一个软件的信誉度与生命力,因此为所有软件企业所高度重视。要提高软件内部质量,毫无疑问就是软件修改的第三个动机:改善原有程序的结构。它的价值是隐性的,并不体现在某一次或两次开发中,而是逐渐体现在日后长期维护的软件过程中。 高质量的软件,可以保证开发人员(即使是新手)能够轻易看懂软件代码,能够保证日后的每一次软件维护都可以轻易地完成(不论软件经历了多少次变更,维护了多少年),能够保证日后的每一次需求变更都能够轻易地进行(而不是伤筋动骨地大动)。要做到这几点其实并不容易,它需要我们持续不断地对系统内部质量进行优化与改进。这,就是系统重构的价值。 下面一个重要问题是哪些地方需要重构?有臭味道(Bad Smell)的代码。 什么是臭味道?想象一下你打开冰箱门,出来一股臭味道你就知道冰箱里有东西腐坏了,要清除了。代码一样有臭味道:
臭味行列中首当其冲的就是Duplicated Code(重复的代码)。如果你在一个以上的地点看到相同的程序结构,那么当可肯定:设法将它们合而为一,程序会变得更好。
Duplicated Code
就是[同一个class内的两个方法含有相同表达式(expression)]。这时候你需要做的就是采用Extract Method
提炼出重复的代码,然后让这两个地点都调用被提炼出来的那一段代码。Extract Method
,然后再对被提炼出的代码使用Pull Up Method
,将它推入superclass内。Extract Method
将相似部分和差异部分割开,构成单独一个方法。然后你可能发现或许可以运用Form Template Method
获得一个Template Method
设计模式。Substitute Algorithm
将其它方法的算法替换掉。Duplicaded Code
,你应该考虑对其中一个使用Extract Class
,将重复代码提炼到一个独立class中,然后在另一个class内使用这个新class。但是,重复代码所在的方法也可能的确只应该属于某个class,另一个class只能调用它,抑或这个方法可能属于第三个class,而另两个classes应该引用这第三个class。你必须决定这个方法放在哪儿最合适,并确保它被安置后就不会再在其它任何地方出现。
其他Bad Smell
与相应的重构手法如下表所示:
Eclipse中Refactor
菜单中的重构手示:
更完整的手法可以参考《重构》作者Martin Fowler的博客。Eclipse中基本手法的使用大家可以参考任何人都可以重构来进行学习实践。
一个完整的重构流程包括:
我们结合Git给出一个比较完整的例子。
2.所选游戏:人机猜拳。用户通过屏幕下方的按钮来做出“石头、剪刀、布”的选择,电脑随机做出选择后,提示用户输赢或者平局。
import java.awt.Color;
import java.awt.EventQueue;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.util.Random;
import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.SwingConstants;
import javax.swing.border.EmptyBorder;
import javax.swing.border.LineBorder;
@SuppressWarnings("serial")
public class Game extends JFrame {
private JPanel contentPane;
private JPanel userPanel;
private JPanel computerPanel;
private JLabel userValue;
private JLabel computerValue;
private JLabel resultLable;
/**
* Launch the application.
*/
public static void main(String[] args) {
EventQueue.invokeLater(new Runnable() {
public void run() {
try {
Game frame = new Game();
frame.setVisible(true);
} catch (Exception e) {
e.printStackTrace();
}
}
});
}
/**
* Create the frame.
*/
public Game() {
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
setBounds(100, 100, 450, 300);
contentPane = new JPanel();
contentPane.setBorder(new EmptyBorder(5, 5, 5, 5));
setContentPane(contentPane);
contentPane.setLayout(null);
JButton shitouBtn = new JButton("石头");
shitouBtn.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
userValue.setText("石头");
computerValue.setText(getComputerValue());
compareValues();
}
});
shitouBtn.setBounds(144, 216, 146, 46);
contentPane.add(shitouBtn);
JButton buBtn = new JButton("布");
buBtn.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
userValue.setText("布");
computerValue.setText(getComputerValue());
compareValues();
}
});
buBtn.setBounds(288, 216, 146, 46);
contentPane.add(buBtn);
JButton jiandanBtn = new JButton("剪刀");
jiandanBtn.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
userValue.setText("剪刀");
computerValue.setText(getComputerValue());
compareValues();
}
});
jiandanBtn.setBounds(0, 216, 146, 46);
contentPane.add(jiandanBtn);
computerPanel = new JPanel();
computerPanel.setBackground(Color.WHITE);
computerPanel.setForeground(Color.BLACK);
computerPanel.setBorder(new LineBorder(new Color(0, 0, 0)));
computerPanel.setBounds(318, 60, 95, 87);
contentPane.add(computerPanel);
computerValue = new JLabel("");
computerPanel.add(computerValue);
userPanel = new JPanel();
userPanel.setBackground(Color.WHITE);
userPanel.setBorder(new LineBorder(new Color(0, 0, 0)));
userPanel.setBounds(64, 60, 95, 87);
contentPane.add(userPanel);
userValue = new JLabel("");
userPanel.add(userValue);
JLabel lblNewLabel = new JLabel("欢迎使用人机猜拳程序");
lblNewLabel.setHorizontalAlignment(SwingConstants.CENTER);
lblNewLabel.setBounds(10, 10, 414, 15);
contentPane.add(lblNewLabel);
JLabel label = new JLabel("结果");
label.setHorizontalAlignment(SwingConstants.CENTER);
label.setBounds(0, 154, 424, 15);
contentPane.add(label);
resultLable = new JLabel("");
resultLable.setHorizontalAlignment(SwingConstants.CENTER);
resultLable.setBounds(0, 179, 424, 15);
contentPane.add(resultLable);
JLabel label_2 = new JLabel("你出拳:");
label_2.setBounds(0, 67, 54, 46);
contentPane.add(label_2);
JLabel label_3 = new JLabel("电脑出拳:");
label_3.setBounds(242, 64, 77, 53);
contentPane.add(label_3);
}
//电脑出拳,生成一个随机数(0,1,2)代表:石头、剪刀、布
//电脑出拳,生成一个随机数(0,1,2)代表:石头、剪刀、布
public String getComputerValue() {
String[] quans = {"石头","剪刀","布"};
Random random = new Random();
int value = random.nextInt(100);
value = value%3;
return quans[value];
}
private void compareValues() {
String userText = userValue.getText();
String computerText = computerValue.getText();
String result = "PK";
if(userText.equals(computerText)) {
result = "棋逢对手!再来";
} else {
if("石头".equals(userText)) {
if("剪刀".equals(computerText)) {
result = "你赢了!!好棒!!";
}else if("布".equals(computerText)){
result = "你输了!!加油啊!";
}
}
if("剪刀".equals(userText)) {
if("布".equals(computerText)) {
result = "你赢了!!好棒!!";
}else if("石头".equals(computerText)){
result = "你输了!!加油啊!";
}
}
if("布".equals(userText)) {
if("石头".equals(computerText)) {
result = "你赢了!!好棒!!";
}else if("剪刀".equals(computerText)){
result = "你输了!!加油啊!";
}
}
}
resultLable.setText(result);
}
}
初始运行界面如下:
输:
赢:
平局:
步骤 |
耗时 |
百分比 |
需求分析 |
20min | 14% |
设计 |
20min | 14% |
代码实现 |
60min | 43% |
测试 |
30min | 21% |
分析总结 |
10min | 8% |
(七)实验总结
这次实验对我来说相对比较难的一次实验,尤其是最后的自己做的小游戏,查了许多的资料才艰难的完成。在实验中版本控制的内容中指导书的内容并不详细,并不能将HelloWorld存储进去。后就自己在实验楼里新建了一个空文档,将代码打进去,才可以接着做实验
标签:
原文地址:http://www.cnblogs.com/yswysw/p/4552890.html