这些笔录是我关于已完成的数据库功能测试的一些心得。其中的例子是用java语言编写的,但我认为这些想法对于大多数编程环境都普遍适用。当然,我仍致力于寻找更佳的解决方案。

现实的问题是这样的:你有一个SQL数据库,一些存储过程和一个介于应用程序和数据库之间的中间层。你怎样在其中插入测试代码从而保证在数据库中数据存取功能的实现?

一、 为什么会有这样的问题?

我猜想有些,可能不完全是大多数的数据库开发过程都是这样的:建立数据库,编写存取数据到数据库的代码,编译并运行,用一个查询语句查验所列的数据是否正确显示。如果能正确显示那大功告成了。

然而,这种靠眼睛来检测的弊端在于:你不经常进行这样的检验,而且这种检验是不完全的。存在这样的可能性,当你对系统进行了修改,过了几个月后,你无意中破坏了系统,从而导致数据的丢失。作为一个编程人员,你可能不会花很多时间来检查数据本身,这使错误的数据要经过较长的时间才能暴露出来。我曾经参与一个建立网站的项目,该项目中在注册时有一个必填数据在大半年中没有被发现未实际输入进数据库。尽管公司市场部曾经提出他们需要这一信息,但因为这项数据从来没有人去看它,直接导致了这一问题在很长时间内没有被发现。

自动化测试,由于它能经常测试而且测试范围较广,降低了数据丢失的风险。我发现它能使我更心安理得地休息。当然,自动化测试还有其他一些好处,他们本身是代码编写的范例,也可以作为文档,便于你修改别人编写的原始程序,从而减少检测所需的时间。

二、 什么是我们所谈论的测试?

设想有一个非常简单的用户数据库,包括用户电子信箱和一个标志,用来指示邮件地址是否被弹回。你的数据库程序应该包括插入、修改、删除和查询等方法

插入方法会调用一个存储过程将数据写入数据库。为了叙述方便,这里省去了一些细节,大致的程序如下所示:

public class UserDatabase 

... 

public void insert(User user) 

PreparedStatement ps = connection.prepareCall("{ call User_insert(?,?) }"); 

ps.setString(1, user.getEmail()); 

ps.setString(2, user.isBad()); // In real life, this would be a boolean. 

ps.executeUpdate(); 

ps.close(); 

... 

而我认为的测试代码应为:

public class TestUserDatabase extends TestCase 

... 

public void testInsert() 

// Insert a test user: 

User user = new User("some@email.address"); 

UserDatabase database = new UserDatabase(); 

database.insert(user); 

// Make sure the data really got there: 

User db_user = database.find("some@email.address"); 

assertTrue("Expected non-null result", db_user != null); 

assertEquals("Wrong email", "some@email.address", db_user.getEmail()); 

assertEquals("Wrong bad flag", false, db_user.isBad()); 

...