1. Trang chủ
  2. » Công Nghệ Thông Tin

Hướng dẫn sử dụng MySQL part 10 potx

27 395 0

Đang tải... (xem toàn văn)

Tài liệu hạn chế xem trước, để xem đầy đủ mời bạn chọn Tải xuống

THÔNG TIN TÀI LIỆU

Thông tin cơ bản

Định dạng
Số trang 27
Dung lượng 188,66 KB

Nội dung

DRAFT, 10/4/01 1 Perl The Perl programming language has gone from a tool primary used by Unix systems administrators to the most widely used development platform for the World Wide Web. Perl was not designed for the web, but its ease of use and powerful text handling abilities have made it a natural for Web application development. Similarly MySQL, with its small footprint, speed and large feature set, has been very attractive to web developments that need to serve thousands of transactions a day. Therefore, it was only a natural that a Perl interface to MySQL was developed that allowed for the best of both worlds. Note: At the time of this writing Perl has standardized on the DBI suite of modules for all database interaction, including MySQL. However, many legacy systems still use an older interface to MySQL called MySQL.pm. This module is not compatible with the DBI standard and is no longer actively developed. All new development should certainly use the standard DBI modules, and any sites using MySQL.pm should consider upgrading to DBI for any future development. DBI The recommended method for accessing MySQL databases from Perl is the DBD/DBI interface. DBD/DBI stands for DataBase Driver/DataBase Interface. The name arises from the two-layer implementation of the interface. At the bottom is the database driver layer. Here, modules exist for each type of database accessible from Perl. On top of these database dependent driver modules lies a database independent interface layer. This is the interface that you use to access the database. The advantage of this scheme is that the programmer only has to learn one API, the database interface layer. Every time a new database comes along, someone needs only to write a DBD module for it and it will be accessible to all DBD/DBI programmers. As with all Perl modules, you must use the DBI to get access: #!/usr/bin/perl use strict; use warnings; use DBI; When running and MySQL Perl programs, you should always include the 'use warnings' statement early in the script. With this present, DBI DRAFT, 10/4/01 2 will redirect all MySQL specific error messages to STDERR so that you can see any database errors without checking for them explicity in your program. All interactions between Perl and MySQL are conducted through what is known as a database handle. The database handle is an object—represented as a scalar reference in Perl—that implements all of the methods used to communicate with the database. You may have as many database handles open at once as you wish. You are limited only by your system resources. The connect() method used a connection format of DBI:servertype:database:hostname:post (hostname and port are optional), with additional arguments of username and password to create a handle: # We will use the variable name 'dbh' to indicate a database handle. # This is a very common idiom among DBI users. my $dbh = DBI->connect('DBI:mysql:mydata', undef, undef); my $dbh = DBI->connect('DBI:mysql:mydata:myserver', undef, undef); my $dbh = DBI->connect('DBI:mysql:mydata', 'me', 'mypass'); The servertype attribute is the name of the DBD database-specific module, which in our case will be 'mysql' (note capitalization). The database is the name of a database within the server, and the hostname and port determine the location of the server. If connection via a Unix socket on the local machine, the path of the socket can be used instead of a numerical port. The first version used above creates a connection to the MySQL server on the local machine via a Unix-style socket. This is the most efficient way to communicate with the database and should be used if you are connecting to a local server. If the hostname is supplied it will connect to the server on that host using the standard port unless the port is supplied as well. If you do not provide a username and password when connecting to a MySQL server, the user executing the program must have sufficient privileges within the MySQL database. Note: Perl 5 has two difference calling conventions for modules. With the object-oriented syntax, the arrow symbol “->” is used to reference a method in a particular class or object (as in DBI->connect). Another method is the indirect syntax, in which the method name is followed by the class name, then the arguments. The las connect method above would be written as connect DBI 'DBI:mysql:mydata', 'me', 'mypass'. Because of conventions used in early versions of the MySQL Perl modules, a lot of older Perl code that interfaces with MySQL will have lines in it like SelectDB $dbh 'test' wher a simple $dbh->selectdb('test') would do. If you haven't guess, we are partial to the object-oriented syntax, if only because the arrow makes the relationship between class/object and method clear. Once you have connected to the MySQL server, the database handle $dbh in all of the examples in this section – is the gateway to the database server. For instance to prepare a SQL query: $dbh->prepare($query); MySQL allows clients to use any number of different databases during a session, and different databases can even be accessed simultaneously during a query. Each connection also has a default database, which can DRAFT, 10/4/01 3 be changed at any time. This is the database that is used if no specific database name is given. However, sometimes it is necessary to access two databases that reside on entire separate servers at the same time. To enable this, DBI allows a program to open any number of simultaneous database handles and use them side-by-side. Chapter XX, Perl Reference, describes the full range of method and variables supplied by DBI. As an example of the use of DBI consider the following simple programs. In example XX-1, datashow.cgi is a CGI program which accepts a hostname as a parameter ”localhost” is assumed if no parameter is present. The program then displays all of the databases available on that host. Example 10-1. The CGI program database.cgi shows all of the databases on a MySQL server. #!/usr/bin/perl use warnings; use strict; use CGI qw(:standard); use CGI::Carp; # Use the DBI module use DBI; my ($server, $sock, $host); $server = param('server') || ''; # Prepare the MySQL DBD driver my $driver = DBI->install_driver('mysql'); my @databases = $driver->func($server, '_ListDBs'); # If @databases is undefined we assume that means that # the host does not have a running MySQL server. However, there # could be other reasons for the failer. You can find a complete # error message by checking $DBI::errmsg. if (not @databases) { print header, start_html('title'=>”Information on $server”, 'bgcolor' => 'white' ); print <<END_OF_HTML; <h1>$server</h1> $server does not appear to have a running MySQL server. </body></html> END_OF_HTML exit(0); } print header, start_html('title'=>”Information on $host”, 'bgcolor'=>'white'); print <<END_OF_HTML; <h1>$host</h1> <p> $host\'s connection is on socket $sock. <p> DRAFT, 10/4/01 4 Databases:<br> <ul> END_OF_HTML foreach (@databases) { print “<li>$_\n”; } print <<END_OF_HTML; </ul> </body></html> END_OF_HTML exit(0); You probably noticed in this example that we never created a database handle. This is because we never required a connection to a specific database on the server. Instead, we only needed a list of the available databases. To do this, we first loaded the DBD driver for MySQL. This step is normally performed automatically when you make an explicit connection to the server. However, since we are not making a connection, we need to explicitly load the driver in order to use any of its functions. Once we have loaded the DBD driver, we can make use of any methods it provides. Most methods require a connection to the database server, but a few do not. In our case we want to get a list of databases on a particular server, which is a function that does not require a pre-existing database connection. We call the '_ListDBs' function as parameter to the 'func' method of the driver. This is different than standard DBI methods that are called directly as methods against a database handle. However, as complete as DBI is, there are some features it does not provide, especially if they are specific to a certain database server. On such feature is the ability to list the available databases on a server. The database servers supported by DBI do not have a common concept of a 'database'. For many of them, being able to list the available databases would not be as useful as it is for MySQL. For this reason, the DBI does not provide a standard method for listing the available databases on a database server. However, the author of DBI anticipated that DBI would not be able to provide every piece of functionality present in every supported database server. Therefore DBI was given the ability to run database server-specific functions. This should generally be avoided, as code that uses database server specific functionality can not be directly ported to a new database server if need be. But sometime it is necessary to resort to database server specific functionality to get the job done. Database server specific functions are accessed via the 'func' method that is present in most DBI objects. In our case, we have a driver object created from the DBD MySQL module. Through the 'func' method on this object, we are able to call the MySQL-specific 'ListDBs' function and retrieve a list of the databases on a specific database and server. Once we have that information, we can create an HTML response page that lists the databases available on a MySQL server. Now that we know what databases are available to use, the next step is to see what tables we can use. In Example XX-2 tableshow.cgi accepts the name of a database server DRAFT, 10/4/01 5 (default is “localhost”) and the name of a database on that server. The program then shows all of the available tables on that server. Example XX-2. The CGI program tableshow.cgi shows all of the tables within a database. #!/usr/bin/perl -w use strict; use warnings; use CGI qw(:standard); use CGI::Carp; # Use the DBI module use DBI; my $db = param('db') or die “Database not supplied!”; my $host = param(‘host’) || ‘localhost’; # Connect to the requested server. my $dbh = DBI->connect(“DBI:mysql:$db:$host”, undef, undef); # If $dbh does not exist, the attempt to connect to the database # server failed. The server may not be running, or the given # database may not exist. if (not $dbh) { print header, start_html('title'=>'Information on $host => $db’ “bgcolor” => 'white'); print <<END_OF_HTML; <h1>$host</h1> <H2>$db</h2> The connection attemp failed for the followign reason:<br> $DBI::errstr </body></html> END_OF_HTML exit(0); } print header, start_html('title'=>'Information on $host => $db', 'bgcolor' => 'write' ); print <<END_OF_HTML; <h1>$host</h1> <h2>$db</h2> <p> Tables:<br> <ul> END_OF_HTML # $dbh->listtable returns an array of the tables that are available # in the current database. my @tables = $dbh->tables; foreach (@tables) { print “<li>$_\n”; } print <<END_OF_HTML; </ul> </body></html> END_OF_HTML exit(0); DRAFT, 10/4/01 6 In this example, we created an actual connection to a MySQL server for the first time. This connection was made to the server and port number given as parameters from the client browser. If no specific hostname and port number are given, the Unix socket /tmp/mysql.sock on the localhost is used by default. Once we have created an active connection to the desired database, we can interact with that database using the standard DBI methods. In our case, we want to obtain a list of tables that are available within the database. DBI provides the 'tables' method that returns a list of tables within a database. Notice that at the end of the script we do not explicitly close the database handle or do any other cleanup. The DBI module will automatically close and cleanup any connections at the end of script. Now that we know the names of all of the databases and tables available to us, we can take the last step and look at the structure and data within each table. Example XX-3 shows all of the information about a specific table, including its data. Example XX-3. The CGI program tabledump.cgi Shows Information About a Specific Table use struct; use warnings; use CGI qw(:standard); use CGI::Carp; # Use the DBI module use DBI; my ($db, $table, $host); $host = param('host') || ''; $db = param('db') or die “Database not supplied!”; $table = param(‘table’) or die ‘Table not supplied!’; # Connect to the requested server. my $dbh = DBI->connect(“DBI:mysql:$db:$host”, undef, undef); # WE now prepare a query for the server asking for all of the # data in the table. my $table_data = $dbh->prepare(“select * from $table”); # Now send the query to the server. $table_data->execute; # If the return value is undefined, the table must not exist. # (Or it could be empty; we don't check for that.) if (not $table_data) { print header, start_html('title'=> “Information on $host => $db => $table”, 'bgcolor'=>'white'); print <<END_OF_HTML; <h1>$host</h1> <h2>$db</h2> The table '$table' does not exist in $db on $host. </body></html> END_OF_HTML exit(0); } DRAFT, 10/4/01 7 # At this point, we know we have datat to display. First we show # the layout of the table. print header, start_html('title'=> ”Information on $host => $db => $table”, 'bgcolor'=>'white'); print <<END_OF_HTML; <h1>$host</h1> <h2>$db</h2> <h3>$table</h3> <p> <table border> <caption>Fields</caption> <tr> <th>Field</th><th>Type</th><th>Size</th><th>NOT NULL</th> </tr> <ul> END_OF_HTML # $table_data->NAME returns a reference to an array of the fields # of the database. my @fields = @{$table_data->NAME}; # $table_data->TYPE returns an array reference of the types of # fields. The types returned here are in SQL standard notation, # not MySQL specific. my @types = @{$table_data->TYPE}; # $table_data->is_not_null returns a boolean array reference # indicating which fields have the 'NOT NULL' flag. Notice the # term 'NULLABLE' has the opposite context as 'NOT NULL' my @nullable = @{$table_data->NULLABLE}; # $table_data->PRECISION returns an array reference of the lengths # of the fields. This is defined when the table is created. # For CHAR-type fields, this is the maximum number of characters. # For numeric fields this is the maximum number of significant digits. my @length = @{$table_data->PRECISION}; # All of the above arrays were returned in the same order, so that # fields[0], $types[0], $not_null[0] and $length[0] all refer to # the same field. foreach my $field (0 $#fields) { print <<END_OF_HTML; <tr> <td>$fields[$field]</td><td>$types[$field]</td><td> END_OF_HTML print $length[$field] if $types[$field] eq 'SQL_CHAR'; print '<td>'; print 'N' if not $nullable[$field]; print “</tr>\n”; } print <<END_OF_HTML; </table> <p> <b>Data</b><br> <ol> END_OF_HTML # Now we step through the data, row by row, using # DBI::fetchrow_array(). We save the data in an array that has # the same order as the informational arrays @fields, @types, etc.) # we created earlier. DRAFT, 10/4/01 8 While ( my @data = $table_data->fetchrow_array ) { print “<li>\n<ul>”; for (0 $#data) { print “<li>$fields[$_] => $data[$_]</li>\n”; } print “</ul></li>”; } print <<END_OF_HTML; </ul> </body></html> END_OF_HTML This is the most complex of the scripts by far. As in the tables script, we start of by connecting to the database using the parameters passed to us from the client browser. We then use that database connection to execute a SQL query that retrieves all of the data from a table. The first step in executing a SQL query is to prepare it. DBI provides the 'prepare' method within database handle object. The prepare method takes a SQL query and stores it (either locally or on the database server) until execution. On database servers that store the query on the database server itself, it is possible to perform operations on the query before executing it. However, MySQL does not support that ability yet, and prepared queries are simply stored within the database driver until execution. The result of the prepare method is an object known as a statement handle. A statement handle is a Perl program's interface to a SQL query, much like a database handle is the interface to the database server itself. While the statement handle is created when the SQL query is prepared, it is not possible to do anything useful with it until the query has been executed. A query is executed by using the 'execute' method on a statement handle. That is, one a statement handle has been created using 'prepare', calling 'execute' on that handle will cause the query to be sent to the database server and executed. The result of executing a query depends on the type of query. If the query is a non-SELECT query that returns no data (such as INSERT, UPDATE and DELETE) the execute method will return the number of rows that were affected by the query. That is, for an insert query (that inserts one row of data), the execute query will return '1' if the query was successful. For SELECT queries, the execute method simply returns a true value if the query was successful and a false value if there was an error. The data returned from the query is then available using various methods within the statemet handle itself. In addition to the data returned from a SELECT query, the statement handle also contains various information about the data (called meta-data). The meta-data associated with the query can be accessed via various properties in the statement handle. In our example we use several of those properties to build a table containing information about the table in question: $statement_handle->NAME DRAFT, 10/4/01 9 A reference to an array of the names of the columns in the result set. Since our query is selecting all of the columns from a table, this contains the names of all of the columns in the table. $statement_handle->TYPE A reference to an array of the SQL types of the columns in the result set. These types are returned as ANSI SQL standard types. While these are often the same as the MySQL data types, many of the more unusual MySQL data types (such as NUMERIC, and TEXT) are represented as simpler ANSI standard types. $statement_handle->NULLABLE A reference to an array of boolean values indicating whether the columns in the result set can contain NULL values. Note that this has the opposite meaning as the 'NOT NULL' which is used when defining MySQL columns. Thus, a NOT NULL column will have a value of false in the NULLABLE array, and vice versa. $statement_handle->PRECISION A reference to an array of the maximum lengths of the columns in the result set. These maximum values are defined when the table is created. For character-based columns, this is the maximum number of characters. For numeric columns, this is the number of significant digits. After printing a table of this meta-data, the program then displays all of the data in the table, row by row. This is done by using the fetchrow_array method on the statement handle containing the data. The fetchrow_array method reads a single row of data from the result set and then advances an internal pointer so that the next call to fetchrow_array will return the next row of data. This continues until there are no rows left, at which time the method will return a false value. Each row of data is returned as an array, in the order defined in the query. In our case, the query simply specifies 'SELECT *', so we don't know the order in which the fields were defined. However, it is guaranteed that the order of this array is the same as the order of the arrays of meta-data generated earlier. Therefore, we can loop through the data array and use the same indices on the meta-data arrays to describe the columns. An Example DBI Application DBI allows for the full range of SQL queries supported by MySQL. As an example, consider a database used by a school to keep track of student records, class schedules, test scores and so on. The database would contain several tables, one for class information, one for student information, one containing a list of tests, and a table for each test. MySQL's ability to access data across tables—such as the table-joining feature—enables all of these tables to be used together as a coherent whole to form a teacher's aide application. To begin with, we are interested in created tests for the various subjects. To do this we need a table that contains names and ID numbers for the tests. We also need a seperate table for each test. This table will contain the scores for all of the students as wll as a perfect score for comparison. The test table has the following structure: CREATE TABLE test ( DRAFT, 10/4/01 10 id INT NOT NULL AUTO_INCREMENT, name CHAR(100), subject INT, num INT ) The individual tests have table structures like this: CREATE TABLE t7 ( id INT NOT NULL, q1 INT, q2 INT, q3 INT, q4 INT, total INT ) The table name is t followed by the test ID number from the test table. The user determines the number of questions when he or she creates the table. The total field is the sum of all of the questions. The program that access and manipulates the test information is the CGI program test.cgi. This program, which follows, allows only for adding new tests. Viewing tests and changing tests is not implemented but is left as an exercise. Using the other scripts in this chapter as a reference, completing this script should be only a moderate challenge. As it stands, this script effective demonstrates the capabilities of DBI. #!/usr/bin/perl use warnings; use strict; use CGI qw(:standard); # Use the DBI module. use DBI; # DBI::connect uses the format 'DBI:driver:database', in our case # we are using the MySQL driver and accessing the 'teach' database. my $dbh = DBI->connect('DBI:mysql:teach'); # The add action itself is broken up into three seperate functions. # The first function, add, prints out the template form for the # user to create a new test. sub add { $subject = param('subject') || ''; $subject = '' if $subject eq 'all'; print header, start_html('title'=>'Create a New Test', 'bgcolor'=>'white'); print <<END_OF_HTML; <h1>Create a New Test</h1> <form action=”test.cgi” method=”post”> <input type=”hidden” name=”action” value=”add2”> Subject: END_OF_HTML my @ids = (); my %subjects = (); my $out2 = $dbh->prepare(“SELECT id, name FROM subject ORDER BY name”); $out2->execute; [...]... shift; my $i = shift; return 1 if $$sth-> {mysql_ is_auto_increment}->[$i]; return 0; } sub get_pk_value { my $self = shift; my $dbh = shift or die "mysql: :get_pk needs a Database Handle "; my $mysqlPk = "Select last_insert_id() as pk"; my $mysqlSth = $dbh->prepare($mysqlPk); $mysqlSth->execute(); my $mysqlHR = $mysqlSth->fetchrow_hashref; my $pk = $mysqlHR->{"pk"}; $mysqlSth->finish; return $pk; } 1; This... The database-specific functionality for MySQL is placed in a utility class called simply 'mysql' package BM: :mysql; use strict; use warnings; sub new { my $proto = shift; my $class = ref($proto) || $proto; my $self = { }; 24 DRAFT, 10/ 4/01 bless($self, $class); return $self; } sub is_pk ($$$) { my $self = shift; my $sth = shift; my $i = shift; return 1 if $$sth-> {mysql_ is_pri_key}->[$i]; return 0; } sub... performing logic is the responsibility of the Controller Since our purpose here is to examine using Perl to interface with MySQL, it is clear that the Model is the only layer that direct affects us This frees us from having to deal with code that is extraneous to our central purpose 14 DRAFT, 10/ 4/01 Designing the Model Designing the Model layer is one of the most important tasks of creating an MVC application... seperate method is that it can be used externally by other Model classes that need to create new 'publisher' objects • 22 create DRAFT, 10/ 4/01 This method inserts a new row into the table with the data from this object In this case, the primary key for this table is a MySQL AUTO_INCREMENT field with automatically creates a new value Therefore, this method only inserts the value of the 'name' field After... database and other utility matters # this is a static database helper class # that allows us to make connections to the database package CBDB::DB; use strict; use warnings; use BM: :mysql; our our our our $VERSION = '0.1'; $URL = "DBI :mysql: database=CBDB;host=localhost"; $USER = "myuser"; $PASSWORD = "mypass"; my %types = ( 'creator' => { 'name' => 1, 'id' => 4 }, ‘book’ => { 'title' => 1, 'publisher_id' =>... => 4 }, 'publisher' => { 'name' => 1, 'id' => 4 }, 'role' => { 'name' => 1, 'id' => 4 }, ); sub getDB { 23 DRAFT, 10/ 4/01 my $dbh = DBI->connect($URL,$USER,$PASSWORD); return $dbh; } sub get_pk_value { my $dbh = shift or die "DB::get_pk_value needs a Database Handle "; my $dbd = new BM: :mysql( ); return $dbd->get_pk_value( $dbh ); } sub getType { my $table = shift; my $col = shift; return $types{$table}{$col};... class uses yet another class for the database specific operation (see BM: :mysql, below) • get_type This method returns the SQL type of a column within a table used by the application All of the fields in all of the tables are stored within this superclass so that any subclass can access the type of any of the fields from any time mysql As mentioned above, an attempt has been made to abstract out any database... Enter the point value for each of the questions The points need not add up to 100 END_OF_HTML for (1 $num) { print qq%$_: %; if (not $_ % 5) { print “\n”; } } print . running and MySQL Perl programs, you should always include the 'use warnings' statement early in the script. With this present, DBI DRAFT, 10/ 4/01 2 will redirect all MySQL specific. DBI->connect('DBI :mysql: mydata', undef, undef); my $dbh = DBI->connect('DBI :mysql: mydata:myserver', undef, undef); my $dbh = DBI->connect('DBI :mysql: mydata',. 'DBI :mysql: mydata', 'me', 'mypass'. Because of conventions used in early versions of the MySQL Perl modules, a lot of older Perl code that interfaces with MySQL will

Ngày đăng: 02/07/2014, 12:20

TỪ KHÓA LIÊN QUAN