本文描述了MySQL一种利用第三方数据库开发电子贸易和其它复杂动态网站的有效工具MySQL 是一种快速多线程和全功能的 SQL服务器除了描述MySQL系统的基本体系结构以外本文还提供了以Tcl和C++编写的简单示例帮助您开发支持数据库的Web应用
一个必须存储或访问大量信息的应用程序可以从使用第三方数据库产品中受益匪浅在对信息的访问必须在程序的多个实例上进行时更是如此基于Web的应用(包括电子贸易)就是它的良好例证
为什么使用独立数据库?
Web服务器必须使其处理脚本有办法来存储有关供其以后访问的状态信息尽管有可能使用比较原始一些的方法例如转储到文本文件或开发自制的迷你数据库但只有成熟的数据库应用才能提供更为复杂的Web应用所需的所有服务因为有一些免费获得的软件包可用于该目的所以编写定制的特定于应用的数据库引擎并无太大好处 另外使用第三方数据库还使Web开发者不必投入到开发和维护数据库的任务中
MySQL数据库
通过使用脚本语言和编译型系统语言(例如 C)将数据库集成到Linux应用就可能相当容易可免费获得的MySQL(在GNU Public License下发行)数据库提供了一系列复杂的SQL功能并易于集成到应用中MySQL是快速多线程的并支持ANSI和ODBC SQL标准加上第三方软件MySQL就支持用于事务处理应用的事务安全的表
注什么是事务处理?
事务是需要以原子方式执行的对数据库所做的一系列更改它们要么必须全部执行要么一个都不执行 例如在Web上销售产品时所有必需的数据库更改组成一个事务
数据库需要同时减去客户帐户余额和产品库存否则失败并且一个操作都不执行
无论服务器出于何种原因发生崩溃都不应该引起事务被部分执行例如帐单多算产品没有交付或者库存不实等都有可能是部分完成的事务的结果
支持事务处理的数据库可以将一组数据库代码封装在一个事务中在事务执行期间的任何失败会让数据库回滚到事务开始之前的状态
这是通过维护所有数据库操作的日志以及其原始状态表的副本来实现的在失败后下一次重新启动服务器时允许回滚操作 这种时间和空间上的开销是事务安全数据库系统所必需的一种折衷
单一MySQL服务器控制着一系列数据库它们都可以通过服务器以类似方式来访问 每个数据库实际上都是一组任意数量的表概念与其它SQL数据库的用户类似每个表都由带类型的数据列组成 数据可以是整数实数值字符串或其它类型包括原始二进制流 表中的每一行都是存储在数据库中的一个记录
MySQL被设计和构造成客户机/服务器 服务器mysqld可以在能从因特网访问到的任何机器上运行(最好与Web服务器在同一台或最接近的一台机器上以确保合理的响应时间)MySQL客户机使用请求来与MySQL服务器联系修改或查询服务器所拥有的数据库 在支持数据库的Web应用程序中数据库客户机是Web服务器或由Web服务器产生的CGI脚本 这些客户机可以用高级脚本语言或低级系统语言编写只要存在这种语言的数据库API即可在Linux中大多数脚本语言是以C 实现的因为存在MySQL C API所以要将MySQL支持添加到任何现有的脚本语言或工具应该很容易绝大部分脚本语言已经完成了这一步
MySQL API
MySQL API可用于各种语言包括几乎所有编写网站后端所实际使用的语言 使用这些API我们可以构建由Web服务器控制的 MySQL客户机
API(用于数据库访问)以基于连接的模式工作客户机必须做的第一件事是打开与MySQL服务器的连接 这包括适当地使用服务器认识的用户名和口令来对连接进行身份认证建立了连接后服务器选择要使用的特定数据库 确定了初始化后客户机应用程序(就我们来说是服务器方CGI脚本)就能自由地与数据库以两种方式中的一种进行交互可以运行常规SQL命令包括添加和删除表以及向它们添加记录也可以对返回结果的数据库运行查询 查询生成一组与查询匹配的记录然后客户机可以逐一访问记录直到查看完所有记录或者客户机取消暂挂的记录检索一旦脚本完成了对数据库的操作后与服务器的连接就被关闭
要构建集成数据库访问的网站需要编写CGI脚本来根据数据库状态生成动态结果Web服务器启动CGI脚本然后将适当格式化的HTML输出到它们的标准输出流中Web服务器捕捉到HTML后将它发送回客户机如同请求是对静态HTML页面进行的那样 在生成 HTML 的过程中脚本可以修改数据库也可以查询并将结果合并到它们的输出中
作为简单解释上述过程的一个示例下面的代码(以C和Tcl编写)查询一个包含某公司供销售的产品清单的数据库 这绝没有使用两种语言MySQL API的所有特性但提供了快速简易扩展的示例可以对数据库内容执行任何SQL命令 在该例中脚本显示了低于特定价格的所有产品 在实践中用户可能在Web浏览器中输入该价格然后将它发给服务器 我们省去了从环境变量中进行读取来确定 HTML 表单值的细节因为它与不支持数据库的 CGI 脚本中执行的情况没有什么差别 为清晰起见我们假设事先设置了特定一些参数(例如要查询的价格)
以下代码是使用免费获得的Tcl Generic Database Interface以Tcl实现的这样一种接口的好处在于Tcl是解释型的可以对代码进行迅速开发和快速修改
Tcl示例
#This code prints out all products in the database
# that are below a specified price (assumed to have been determined
# beforehand and stored in the variable targetPrice)
# The output is in HTML table format appropriate for CGI output
#load the SQL shared object library the Tcl interpreter could also
#have been compiled with the library making this line unnecessary
load /home/aroetter/tclsql/sqlso
#these are well defined beforehand or they could
#be passed into the script
set DBNAME clientWebSite;
set TBLNAME products;
set DBHOST
set DBUSER mysqluser
set DBPASSWD abigsecret
set targetPrice ;
#connect to the database
set handle [sql connect $DBHOST $DBUSER $DBPASSWD]
sql selectdb $handle $DBNAME ;# get test database
#run a query using the specified sql code
sql query $handle select * from $TBLNAME where price <= $targetPrice
#print out html table header
puts <table border=>
puts <th>Product Id <th width=>Description <th>Price (\$)
#output table rows each fetchrow retrieves one result
#from the sql query
while {[set row [sql fetchrow $handle]] != } {
set prodid [lindex $row ]
set descrip [lindex $row ]
set price [lindex $row ]
puts <tr><td>$prodid <td align=center>$descrip <td>$price
}
puts </table>
#empty the query result buffer should already be empty in this case
sql endquery $handle
#close the db connection in practice this same connection
#is used for multiple queries
sql disconnect $handle
下面的代码是使用正式MySQL C++ API MySQL++以C++编写的等价脚本该版本的优势在于它是编译型的因此比解释语言更快经常用在特定站点的数据库代码应该以C或C++编写然后由脚本或直接由Web服务器访问以改进整体运行时间
C++示例
#include
#include
#include
const char *DBNAME = clientWebSite;
const char *DBTABLE = products;
const char *DBHOST = ;
const char *DBUSER = mysqluser;
const char *DBPASSWD = abigsecret:
int main() {
try {
//open the database connection and query
Connection con(DBNAME DBHOST DBUSER DBPASSWD);
Query query = conquery();
//write valid sql code to the query object
query << select * from << DBTABLE;
//run the query and store the results
Result res = querystore();
//write out the html table header
cout << <table border=>\n;
cout << <th>Product Id <th width=>Description
<< <th>Price ($) << endl;
Result::iterator curResult;
Row row;
//iterate over each result and put it into an html table
for (curResult = resbegin(); curResult != resend(); curResult++) {
row = *curResult;
cout << <tr><td align=center> << row[]
<< <td> << row[]