• Feed RSS

Simple PHP Class-Based Querying

Simple PHP Class-Based Querying: "

Though it is usually advisable to use some sort of framework or CMS, sometimes a project is small enough such that those options would weigh down the development. However, even in smaller projects, separating presentational elements from backend querying should not be ignored. This tutorial will walk you through creating a basic class-based querying engine for PHP and MySQL.


Step 1. Setup the Project

The first thing we are going to want to do is make some specific files and directories. Here’s how I like to setup my projects. You can, of course, feel free to change the names and structure to your liking. Just make sure you change the require’s later on as well.

Make directories

We’ll need a new directory to hold everything. In this case, I called it tut. Inside of that, I put my configuration files in a directory called conf. Then, I will make an inc directory (short for include) and put a “class” directory inside of that.

Add Files

Then, inside /conf, we will make config.php. Inside /inc/class we will make DAL.php. Finally, in the root directory, we will make index.php.

DAL stands for “Data Access Layer” or “Data Access Link”.

In multi-tiered architecture, it is essentially used to translate database query results into objects and vice-versa.


Step 2. Setup the Database

We need to make a database and populate it with some data. For the purposes of this tutorial, it will just be a two-table database with a single one-to-many relationship. This is just so we can show our querying engine spanning at least one relationship.

Create tables

So, in a database named “tut”, let’s make a table called makes and a table called models. The makes table will have fields “id” and “name” and the models table will have fields “id”,”make”, and “name”.

Add some data

Now we can just add some makes (like Ford, Chevy, etc.) as data in the makes table and some models that those manufacturers are responsible for.

This tutorial assumes you have some working knowledge of databases and SQL, so I won’t go into details about the relation/foreign key setup.


Step 3. The Database Connection

Usually, I don’t like working with raw constants in PHP. I will typically define a bunch of things then make some functions to hook into those constants. For this example, let’s just keep things simple and use the constants.

Define connection variables

In our /conf/config.php file, let’s setup our database connection variables. While we are at it, let’s throw an include to our DAL.php script.

<?php

// Include DAL
require_once(dirname(dirname(__FILE__)) . '/inc/class/DAL.php');

// Database
define ( 'DB_HOST', 'localhost' );
define ( 'DB_USER', 'root' );
define ( 'DB_PASSWORD', 'password1' );
define ( 'DB_DB', 'tut' );

?>

This setup assumes you are running MySQL on it’s default port.

Create connection function

Now, inside /inc/class/DAL.php, we will make a function that we will use to connect to our database.

The connection, as well as all forthcoming queries, will live inside a class named DAL. Wrapping all database involvement inside a single class allows us to manipulate our queries later without needing to touch business or presentation layer scripts. Also, it provides some degree of mock namespacing.

In the case of this class, we will add a constructor even though it doesn’t need to do anything.

<?php

class DAL {

 public function __construct(){}

 private function dbconnect() {
   $conn = mysql_connect(DB_HOST, DB_USER, DB_PASSWORD)
     or die ("
Could not connect to MySQL server');

   mysql_select_db(DB_DB,$conn)
     or die ('
Could not select the indicated database");

   return $conn;
 }

}

?>

Notice that the scope of the dbconnect method is private. This is because we should not need to connect to the database from outside our DAL. Instead, we will have public query methods which will call the dbconnect from inside the DAL. A little confusing? No worries, read on.


Step 4. Create Generic Query Tools

To abstract our queries so that we can reuse short pieces of code, we will need two things. First, we will need some sort of “generic query result” class. Second, we will need a generic querying method inside our DAL.

Create generic query result class

The purpose of all of this is to be able to convert SQL queries into objects and minimize use of the ugly while($row = mysql_fetch_array($result)) loop. Objects are far easier to work with and allow us to use properties instead of array keys.

In short, we want to make a class that will create property names on the fly and store data associated with those properties.

We will put this class inside our /inc/class/DAL.php script. Since it is a new class, it will be outside the DAL class.

class DALQueryResult {

 private $_results = array();

 public function __construct(){}

 public function __set($var,$val){
   $this->_results[$var] = $val;
 }

 public function __get($var){
   if (isset($this->_results[$var])){
     return $this->_results[$var];
   }
   else{
     return null;
   }
 }
}

Create generic query method

Now, inside our DAL class, we need to make a generic querying method that will turn SELECT queries into DALQueryResult objects.

Basically, we want to turn each returned field name into a property of the DALQueryResult object.

private function query($sql){

 $this->dbconnect();

 $res = mysql_query($sql);

 if ($res){
   if (strpos($sql,'SELECT') === false){
     return true;
   }
 }
 else{
   if (strpos($sql,'SELECT') === false){
     return false;
   }
 else{
   return null;
 }
 }

 $results = array();

 while ($row = mysql_fetch_array($res)){

   $result = new DALQueryResult();

   foreach ($row as $k=>$v){
     $result->$k = $v;
   }

   $results[] = $result;
 }
 return $results;
}

Here is a private function that accepts a SQL query. It connects to the database and runs the query. Then, it checks to see if there are any results. If there are not any results, it returns null on a SELECT query, false on other queries. If the query was successful and the query was not a SELECT query, it will return true. If it was a SELECT, then it converts the results into an array of DALQueryResult objects. This mimics the results that one would normally get from a mysql_query.


Step 5. Write a Specific Query

Now we are ready to actually write a SQL query. DAL queries should be very specific both in name and purpose. Let’s make one that finds all models of a given make.

This will be our first public method.

public function get_models_by_make_name($name){
 $sql = "SELECT models.id as id, models.name as name, makes.name as make FROM models INNER JOIN makes ON models.make=makes.id WHERE makes.name='$name'";
 return $this->query($sql);
}

Here we are just writing the query and returning the result in the form of DALQueryResult objects. Our generic query method takes care of the itterations and decision making.

Finished DAL

At this point, our DAL.php script is finished. It should look like the following.

<?php

class DALQueryResult {

 private $_results = array();

 public function __construct(){}

 public function __set($var,$val){
   $this->_results[$var] = $val;
 }

 public function __get($var){
   if (isset($this->_results[$var])){
   return $this->_results[$var];
 }
   else{
   return null;
 }
 }
}

class DAL {

 public function __construct(){}

 public function get_models_by_make_name($name){
   $sql = "SELECT models.id as id, models.name as name, makes.name as make FROM models INNER JOIN makes ON models.make=makes.id WHERE makes.name='$name'";
   return $this->query($sql);
 }

 private function dbconnect() {
   $conn = mysql_connect(DB_HOST, DB_USER, DB_PASSWORD)
    or die ("<br/>Could not connect to MySQL server");

   mysql_select_db(DB_DB,$conn)
    or die ("<br/>Could not select the indicated database");

 return $conn;
 }

 private function query($sql){

   $this->dbconnect();

   $res = mysql_query($sql);

   if ($res){
     if (strpos($sql,'SELECT') === false){
       return true;
     }
   }
   else{
     if (strpos($sql,'SELECT') === false){
       return false;
     }
     else{
       return null;
     }
   }

   $results = array();

   while ($row = mysql_fetch_array($res)){

     $result = new DALQueryResult();

     foreach ($row as $k=>$v){
       $result->$k = $v;
     }

     $results[] = $result;
   }
   return $results;
 }
}

?>

Step 6. Use the DAL

Now, let’s finally head over to our /index.php script and display our results using the DAL. All we need to do is include our /conf/config.php file, instantiate the DAL, and do something with the data. Here’s an example.

<?php
// include configuration
require_once(dirname(__FILE__) . '/conf/config.php');

// instanciate a new DAL
$dal = new DAL();

// array of makes to check
$makes = array('Ford','Chevy','Honda');

// cycle through the makes
foreach ($makes as $make){
 $results = $dal->get_models_by_make_name($make);
 echo "<h1>Models by $make</h1>";

 // check if there were any results
 if ($results){
  echo "<ul>";

 // cycle through results
 foreach ($results as $model){
   echo "<li>$model->make $model->name (Database ID: $model->id)</li>";
  }
  echo "</ul>";
 }
 else{
   // Display a message concerning lack of data
  echo "<p>Sorry, we have no information regarding that manufacturer.</p>";
 }
}
?>

As you can see, we now have results that we can call the field names as properties of a PHP object.


Step 7. Taking Things one Step Further

Often, it will be useful to convert the generic DALQueryResult into a more specific object. In this case, you can write business objects that accept a DALQueryResult as a constructor parameter. Then, you just use that to build the new object.

Here’s an example

<?php

class CarModel{
 private $_id;
 private $_make;
 private $_name;

 public function __construct(DALQueryResult $result){
   $this->_id = $result->id;
   $this->_make = $result->make;
   $this->_name = $result->name;
 }
 public function __get($var){
   switch ($var){
   case 'id':
     return $this->_id;
  break;
   case 'make':
     return $this->_make;
  break;
   case 'name':
     return $this->_name;
  break;
   default:
     return null;
     break;
 }
 }
 public function __toString(){
   return $this->_name;
 }
}

?>

Then, just write a query to return an array of these objects instead of an array of generic DALQueryResult objects.

Remember, always name your queries very specifically.

public function get_models_by_make_name_as_CarModel($name){
 // Reuse existing query
 $results = $this->get_models_by_make_name($sql);

 // check for results
 if (!$results){
   return $results;
 }
 else{
   // array to hold CarModel objects
   $object_results = array();
   // cycle through and convert to CarModel objects
   foreach ($results as $result){
     object_results[] = new CarModel($result);
   }
   // return array of CarModel objects
   return object_results;
 }
}

Building specific objects can become very useful when calculations are needed to extract meaningful data from fields.


Hope you all enjoyed the tutorial! Good luck.

"