前言
早先接触编程的时候,虽然也接触不久(捂脸)一开始原生的sql还不熟练,做项目的时候开始直接使用,导致一开始就对整个sql认识有点偏差,实在是害人不浅,不过学到后面才慢慢开始脱离这个坑,很多东西,比如说框架什么的,我们如果只会用,对原理不理解而且也不想去理解,那么编程这一条会越走越窄的,回归正题,orm,相信大家应该平时开发项目的时候都有用过,比如php laravel框架的orm或者java spring的orm框架,对orm的有点总结的话,有以下几点:
- orm是一种工具,工具确实能解决一些重复,简单的劳动
- 隐藏了数据访问细节,使得我们的通用数据库交互变得简单易行,完全不用使用复杂的sql语句,只使用orm提供的面向对象的方法对数据进行增删查改
- 然而,对于复杂查询的话,orm还是远远不能满足的,最近在做一个报表项目,发现orm真的是没法很好的表达出来,还是写原生比较舒服点
- 效率,因为要执行各种匹配,肯定牺牲一定的效率,但目前有缓存和懒加载什么的,感觉效率无可挑剔
折腾(一)-SQL的拼凑
orm的主体就是字符串的拼凑,把用户输入的参数拼凑成sql语句,下面代码实现了常见的select, where order by 以及insert,其中where实现了多表查询
查询语句如下所示
$orm=new orm();
echo $orm->select(["name"=>"classid"],'uname','user')
->from([["name"=>"classid"], ["name_class"=>"id"]])
->from([["name"=>"classid"], ["name_class"=>"id"]]>orderby(["class"=>"id"],"desc");
$orm->insert([
["id"=>'123'],
["name"=>"seven"]
],"news",function ($orm) {
echo "hello world!";
});
拼凑效果如下所示:
其中hello world,是回调函数输出的,我们可以使用回调更方便的执行的业务代码,这里只是学习模拟我们js的回调函数,其实php也可以回调的。
拼凑代码如下:
<?php
ini_set('display_errors',1);
ini_set('error_reporting',E_ALL);
$map=function ($items) {
if (!is_array($items)) {
return $items;
} else {
$result="";
foreach ($items[1] as $item) {
if ($result!="")
$result.=",";
$result.=$item.$this->_aliastb($item);//类似news,news_class 添加到form后面
}
return $items[0].$result;
}
};
class orm
{
public $sql=array(
"select"=>"select ",
"from"=>array("from ",array()),
"where"=>" where ",
"orderby"=>" order by ",
"insertinto"=>"insert into ",//insert操作
"insertfields"=>"",
"values"=>" values"
);
public $sql_bak=[];
public function __construct()
{
$this->sql_bak=$this->sql;
}
public function select()
{
$fields=func_get_args();
foreach ($fields as $field) {
if (is_array($field)) {
$get_key=key($field);//获取数组的key,其实就是表的名字,还需要在字段前面加上表前缀
$this->__add(__FUNCTION__, $this->_aliastb($get_key).'.'.$field[$get_key]);
} else {
$this->__add(__FUNCTION__, $field);
}
}
return $this;
}
//插入数据
public function insert() {
$params=func_get_args();//获取可变参数
$fields=[];
$fields_values=[];
$callback=[];
foreach ($params as $param) {
if (is_array($param)) {//代表要拼接的是字段和值
foreach ($param as $item) {
//取出字段名
$field=key($item);
//取出字段的值
$field_value=$item[$field];
//把上面的字段和字段值插入相对应的数组
$fields[]=$field;
//拼凑字符串的时候,判断是否是字符串,如果是字符串则要加上单引号
if (is_string($field_value)) {
$field_value="'".$field_value."'";
}
$fields_values[]=$field_value;
}
//处理完参数之后,构造要插入字段的拼凑
$this->__add("insertfields", '('.implode($fields,',').')');
$this->__add("values", '('.implode($fields_values,',').')');
}
//如果是字符串说明传的是表名
if (is_string($param)) {
$this->__add("insertinto", $param);
}
if (is_callable($param)) {
$callback[]=$param;
}
}
foreach ($callback as $call) {
$call($this);
}
return $this;
}
//排序
function orderby($str, $order)
{
$order=' '.$order;
if (is_array($str)) {
$tb=key($str);
$this->__add(__FUNCTION__, $this->_aliastb($tb).'.'.$str[$tb].$order);
} else {
$this->__add(__FUNCTION__, $str.$order);
}
return $this;
}
//给表加别名
public function _aliastb($tbName) {
return ' _'.$tbName;
}
public function from($tableName)//处理from过程
{
if (is_array($tableName)) {//是数组的话就是代表多张表关联
if (count($tableName)!=2) return $this;//参数值类似[["name"=>classid""],["name_class"]=>"id"],小于2灭有意义
$tb1=current($tableName);
$tb2=next($tableName);
//第一步,把表的key值作为from参数进行处理
$tb1_key=key($tb1);
$tb1_value=$tb1[$tb1_key];
$tb2_key=key($tb2);
$tb2_value=$tb2[$tb2_key];
$this->__add(__FUNCTION__, $tb1_key);
$this->__add(__FUNCTION__, $tb2_key);
//第二步,拼凑where条件
$whereString=' _'.$tb1_key.'.'.$tb1_value.'='."_".$tb2_key.'.'.$tb2_value;
$this->where($whereString);
return $this;
} else { //字符串则表示单张表查询
$this->__add(__FUNCTION__, $tableName);
return $this;
}
}
public function where($str) {
$this->__add(__FUNCTION__, $str, " and ");
return $this;
}
public function __add($key, $str, $spliter=",")//实现字符串的累加
{
//调用不存在的方法,直接返回
if(!array_key_exists($key, $this->sql)) return ;
if (is_array($this->sql[$key])) {
//如果已经存在该项表明我们不做什么处理
if (!in_array($str, $this->sql[$key][1])) {
$this->sql[$key][1][]=$str;
}
} else {
//如果是字符串,直接进行字符串累加
// if (trim($this->sql[$key])!=$key)
if (preg_replace('/\s/', '', $this->sql[$key])!=$key && preg_replace("/\s/", '', $this->sql[$key])!="")
$this->sql[$key].=$spliter;
$this->sql[$key].=$str;
}
}
public function __toString()
{
global $map;
$map=Closure::bind($map ,$this);
//
$filter=function ($value, $key) {
// if(!is_string($value)) return true;
if(is_array($value)) {
if(count($value[1])>0) return true;
return false;
}
//如果是空字符串也不参与累加
if(preg_replace("/\s/", '', $value)==$key || preg_replace("/\s/", "", $value )=="")
return false;
return true;
};
$this->sql=array_filter($this->sql, $filter, ARRAY_FILTER_USE_BOTH );
$res=array_map($map, array_values($this->sql));//
$this->sql=$this->sql_bak;
return implode(array_values($res), ' ');//把新的数组重新构建成一个字符串
}
}
里面涉及到一些比较平时很少接触到的函数如array_map,array_filter,当时自从php7出来之后,php也不仅仅是只能做网站了吧,还是要懂一些函数的妙用。下一节,将介绍,拼凑字符串与pdo的组合,并做一个简单的小demo。