怎么一步步编写简单的PHP的Framework(二十)

最后都变了- 提交于 2020-03-15 21:06:12

        上一次我把SqlParser这个类的主要结构讲了一下,这次我就将这个类的函数编写了。

        按照上节课的内容,主要的函数有下面几个:

        

private function _where($where){}
private function _field($field) {}
private function _distinct($distinct) {}
private function _table($table) {}
private function _order($order) {}
private function _group($group) {}


        首先我们说一下distinct函数,这个函数最终会返回字符串DISTINCT或者空串,我们可以想一下,这个函数传递的形参实际上是非常简单的,它只需要告诉框架我是否需要DISTINCT即可,所以在框架中只需做一个简单判定就好。

       现假设用户传递的参数为true或字符串'distinct'的时候代表用户想使用'DISTINCT',那么这个函数的实现就变成了这样:


private function _distinct($distinct) {
		return ((true === $distinct) || ('distinct' === $distinct)) ? 'DISTANCT' : '';
	}
      这里直接使用'distinct' === $distinct判定还是有点问题的,因为用户有时候还是会输入类似'Distinct'这种字符串,如果按照上面的写法,直接就返回空串了,但是用户本来的想法是返回'DISTINCT'。


      其实我们还可以把这个函数再简化一下,在一个SQL中,如果用户需要DISTINCT他才会使用distinct函数,所以我们可以通过判定传递的参数是否为空来判定:

    

private function _distinct($distinct) {
     return empty($distinct) ? '' : 'DISTINCT';
}


      如果这样这样写,那么还需要在execute方法中判定一下传递的$options中是否有key等于'distinct'的元素,如果有,那么调用_distintct方法的时候就传递一个非空的参数,否则直接传递一个null过来就行。

      说完了简单的distinct,我们再说一下field,它指明要查询的元素,比如:select uid,password from user;

      对这个函数,实际上在调用的时候是有点复杂的,因为可能用户是想所有的都抓取,即返回*,也有可能只返回部分数据,也有可能将某一个字符取别名。

      由于这个函数的形式是:

       

private function _field($field)


      传递的形参有可能是数组,有可能是字符串,也有可能传递一个null,那么我们需要一个个去判定。

      首先是为null的情况,代表的是查找所有的,即返回*。

      然后是字符串,这种的处理很简单,直接返回就好,如$this->field('uid,password'),传递的就是字符串'uid,password',这种实际上不太好。

     然后就是传递数组,首先是array('uid','password')这种方式,这种也需要foreach一下,然后进行字符串拼接即可,元素之间通过,分隔。

     如果传递的是array('uid' => 'test','password' => 'test') 那么解析之后就要变成'uid AS test,password AS test',所以拼接的时候需要传递AS子串。

     好了,分析完了之后,就直接亮代码吧。

     

private function _field($field) {
		if(is_array($field)) {
			$arr = array();
			foreach($field as $key => $val) {
				if(is_numeric($key)) {
					$arr[] = $val;
				} else {
					$arr[] = ($key . ' AS ' . $val);
				}
			}
			return implode(',',$arr);
		} elseif(is_string($field) && !empty($field)) {
			return $field;
		} else {
			return '*';
		}
	}
      我们再说一下table,这个函数指定要查询的SQL的表,它传递的可以是字符串,也可以是数组,如果是字符串,我们还是不进行处理,如果是数组,和上面的一样,也有可能有别名的情况,由于思路类似,我就不细讲了,直接贴代码了:



private function _table($table) {
		if(is_array($table)) {
			$arr = array();
			foreach($table as $key => $val) {
				if(is_numeric($key)) {
					$arr[] = $val;
				} else {
					$arr[] = ($key . ' AS ' . $val);
				}
			}
			return implode(',',$arr);
		} else {
			return $table;
		}
	}
      对于group,实际上很简单,它只是指定了按照哪个元素来分组而已,所以也可以一句话搞定:


 

private function _group($group) {
		return empty($group) ? '' : ' GROUP BY ' . $group;
	}


     对于order,它只是指定了排序的方式,如果传入的参数为空,那么框架也不处理,如果传递字符串,也不处理,只是将‘ORDER BY’字符串拼接上去即可,如果是数组,那么和上面几个函数一样,也需要判定是否key为数字,因为如果key不为数字也是可能的,比如array('test' => 'asc'),它代表的意思就是按照test递增排序:


private function _order($order) {
		if(empty($order)) {
			return '';
		}
		if(is_array($order)) {
			$arr = array();
			foreach($order as $key => $val) {
				if(is_numeric($key)) {
					$arr[] = $val;
				} else {
					$arr[] = ($key . ' ' . $val);
				}
			}
			return ' ORDER BY ' . implode(',',$arr);
		}
		return ' ORDER BY ' . $order;
	}
        


     指定了这些之后,我们在模型中就可以测试一下了,我刚才写的这几个函数就是在一个模型中测试的,使用如下:


<?php
class TestModel extends ModelBase {
	public function test() {
		$this->distinct()->where()->field(array(
			'id' => 'uid','user'
		))->table(array(
			'user' => 'TEST','limits'
		))->group()->order(array(
			'test','test2' => 'asc'
		))->select();
	}
}


         这样返回的SQL为:SELECT id AS uid,user FROM user AS TEST,limits WHERE where ORDER BY test,test2 asc,虽然这样的SQL还是错误的,但是距离正确已经不久了。

    由于我一边写这个,一边还要写代码(这些代码都是现场写的),所以消耗的时间比较多,而且也可能会有一些错误,希望大家见谅!!

    今天的代码点此下载


易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!