table.class.php 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429
  1. <?php
  2. /**
  3. * 数据库表抽象类
  4. *
  5. * 定义必须实现的抽象方法
  6. * 定义可以继承的方法,提高table层的书写效率
  7. *
  8. * @createtime 2018/03/01
  9. * @author 空竹
  10. * @copyright 芝麻开发(http://www.zhimawork.com)
  11. */
  12. abstract class Table {
  13. protected $table_name = '';//表名,不带前缀,前缀在config中定义
  14. protected $table_id = '';//指定ID字段名称,必须
  15. protected $table_status = '';//指定状态字段名称,如果有
  16. protected $table_order = '';//指定排序字段名称,如果有
  17. protected $pdo;
  18. protected $table_fullname;
  19. protected $table_struct;
  20. function __construct(){
  21. //初始数据库连接PDO
  22. global $mypdo;
  23. $this->pdo = $mypdo;
  24. $table_prefix = $this->pdo->prefix;
  25. //数据库全名
  26. $this->table_fullname = $table_prefix . $this->table_name;
  27. //数据库结构
  28. $this->table_struct = $this->struct();
  29. }
  30. ////////////////////////////////////////////////////
  31. /**************抽象方法,子类必须实现**************/
  32. ////////////////////////////////////////////////////
  33. //数据库结构
  34. //完整展示数据库所有字段及其含义
  35. //映射字段名称提供给其他层使用,不对外暴露数据库字段
  36. abstract protected function struct();
  37. //增
  38. //@param $attr array -- 键值同struct()返回的数组
  39. abstract public function add($attr);
  40. ///////////////////////////////////////////////////////////////
  41. /**************非抽象方法,用于继承,也可以重载***************/
  42. ///////////////////////////////////////////////////////////////
  43. //获取详情
  44. public function getInfoById($id){
  45. //查询语句必须用sql_check_input检查参数
  46. $id = $this->pdo->sql_check_input(array('number', $id));
  47. $sql = "select * from ". $this->table_fullname ." where ". $this->table_id ." = $id limit 1";
  48. $rs = $this->pdo->sqlQuery($sql);
  49. $r = array();
  50. if($rs){
  51. foreach($rs as $key => $val){
  52. $r[$key] = $this->dataToAttr($val);
  53. }
  54. return $r[0];
  55. }else{
  56. return $r;
  57. }
  58. }
  59. //此项目每个表都有code字段,所以放在基类里面
  60. public function getInfoByCode($code){
  61. //查询语句必须用sql_check_input检查参数
  62. $code = trim($code);
  63. $code = $this->pdo->sql_check_input(array('string', $code));
  64. $sql = "select * from ". $this->table_fullname ." where ".$this->table_name."_code = $code limit 1";
  65. $rs = $this->pdo->sqlQuery($sql);
  66. $r = array();
  67. if($rs){
  68. foreach($rs as $key => $val){
  69. $r[$key] = $this->dataToAttr($val);
  70. }
  71. return $r[0];
  72. }else{
  73. return $r;
  74. }
  75. }
  76. //此项目每个表都有code字段,大部分表有date字段,放在基类里面统一处理
  77. public function getInfoByCodeAndDate($code, $date){
  78. //查询语句必须用sql_check_input检查参数
  79. $code = trim($code);
  80. $code = $this->pdo->sql_check_input(array('string', $code));
  81. //查询语句必须用sql_check_input检查参数
  82. $date = trim($date);
  83. $date = $this->pdo->sql_check_input(array('number', $date));
  84. $sql = "select * from ". $this->table_fullname ." where ".$this->table_name."_code = $code and ".$this->table_name."_date = $date limit 1";
  85. $rs = $this->pdo->sqlQuery($sql);
  86. $r = array();
  87. if($rs){
  88. foreach($rs as $key => $val){
  89. $r[$key] = $this->dataToAttr($val);
  90. }
  91. return $r[0];
  92. }else{
  93. return $r;
  94. }
  95. }
  96. //获取列表(分页)
  97. //$count、$page和$pagesize都为0时,返回全部结果(适用于无需分页的情况)
  98. //
  99. //@param $filter array -- 过滤条件,格式见Table::filterToWhere
  100. //@param $count -- 0:返回列表 1:返回结果数量
  101. //@param $page -- 当前第几页
  102. //@param $pagesize -- 每页数量
  103. // $page 参数 和 $limit 不能同时使用,只能使用一个对limit进行限制
  104. public function getList($filter = array(), $count = 0, $page = 0, $pagesize = 0, $order = 0){
  105. $where = $this->filterToWhere($filter);
  106. $orderByStr = (empty($order) ? 'asc' : 'desc');
  107. if($count == 0){//列表
  108. $sql = "select * from ". $this->table_fullname ." $where order by ".$this->table_id." ".$orderByStr;
  109. if($page > 0){//分页
  110. $startrow = ($page - 1) * $pagesize;
  111. $sql_limit = " limit $startrow, $pagesize";
  112. $sql .= $sql_limit;
  113. }
  114. $rs = $this->pdo->sqlQuery($sql);
  115. $r = array();
  116. if($rs){
  117. foreach($rs as $key => $val){
  118. $r[$key] = $this->dataToAttr($val);
  119. }
  120. return $r;
  121. }else{
  122. return $r;
  123. }
  124. }else{//统计
  125. $sql = "select count(*) as c from ". $this->table_fullname . " $where ";
  126. $rs = $this->pdo->sqlQuery($sql);
  127. if($rs){
  128. return $rs[0]['c'];
  129. }else{
  130. return 0;
  131. }
  132. }
  133. }
  134. //获取列表(分页)
  135. //$count、$page和$pagesize都为0时,返回全部结果(适用于无需分页的情况)
  136. //
  137. //@param $filter array -- 过滤条件,格式见Table::filterToWhere
  138. //@param $count -- 0:返回列表 1:返回结果数量
  139. //@param $page -- 当前第几页
  140. //@param $pagesize -- 每页数量
  141. // $page 参数 和 $limit 不能同时使用,只能使用一个对limit进行限制
  142. //本函数与 getList 区别在于,使用date字段进行排序,没有date字段的表不能使用本函数
  143. public function getListOrderByDate($filter = array(), $count = 0, $page = 0, $pagesize = 0, $order = 0){
  144. $where = $this->filterToWhere($filter);
  145. $orderByStr = (empty($order) ? 'asc' : 'desc');
  146. if($count == 0){//列表
  147. $sql = "select * from ". $this->table_fullname ." $where order by ".$this->table_name."_date ".$orderByStr;
  148. if($page > 0){//分页
  149. $startrow = ($page - 1) * $pagesize;
  150. $sql_limit = " limit $startrow, $pagesize";
  151. $sql .= $sql_limit;
  152. }
  153. $rs = $this->pdo->sqlQuery($sql);
  154. $r = array();
  155. if($rs){
  156. foreach($rs as $key => $val){
  157. $r[$key] = $this->dataToAttr($val);
  158. }
  159. return $r;
  160. }else{
  161. return $r;
  162. }
  163. }else{//统计
  164. $sql = "select count(*) as c from ". $this->table_fullname . " $where ";
  165. $rs = $this->pdo->sqlQuery($sql);
  166. if($rs){
  167. return $rs[0]['c'];
  168. }else{
  169. return 0;
  170. }
  171. }
  172. }
  173. //删
  174. public function del($id){
  175. $where = array(
  176. $this->table_id => array('number', $id)
  177. );
  178. return $this->pdo->sqldelete($this->table_fullname, $where);
  179. }
  180. //--------------------------------------------------
  181. //------------未实现的常见方法,便于重载------------
  182. //--------------------------------------------------
  183. //改
  184. //@param $attr 数组,键值参考add()
  185. public function edit($id, $attr){}
  186. //修改指定字段
  187. public function update($id, $attrs)
  188. {
  189. $params = array();
  190. foreach ($attrs as $key => $value) {
  191. //$type = self::getTypeByAttr($key);
  192. $params[$this->table_name.'_'.$key] = array('string', $value);
  193. }
  194. //where条件
  195. $where = array( $this->table_id => array("number", $id));
  196. //返回结果
  197. $r = $this->pdo->sqlupdate($this->table_fullname, $params, $where);
  198. return $r;
  199. }
  200. //code不存在则add,否则update
  201. public function addOrUpdateByCode($attrs)
  202. {
  203. if (empty($attrs) || empty($attrs['code'])) {
  204. return 0;
  205. }
  206. $rs = $this->getInfoByCode($attrs['code']);
  207. if (empty($rs)) {
  208. $id = $this->add($attrs);
  209. } else {
  210. $id = $rs['id'];
  211. $this->update($id, $attrs);
  212. }
  213. return $id;
  214. }
  215. //(code,date)为唯一标识,不存在则add,否则update
  216. public function addOrUpdateByCodeDate($attrs)
  217. {
  218. if (empty($attrs) || empty($attrs['code']) || empty($attrs['date'])) {
  219. return 0;
  220. }
  221. $rs = $this->getInfoByCodeAndDate($attrs['code'], $attrs['date']);
  222. if (empty($rs)) {
  223. $id = $this->add($attrs);
  224. } else {
  225. $id = $rs['id'];
  226. $this->update($id, $attrs);
  227. }
  228. return $id;
  229. }
  230. //--------------------------------------------------
  231. //------------依赖于字段设置的常见方法,可以重载----
  232. //--------------------------------------------------
  233. //单独修改状态
  234. public function updateStatus($id, $status){
  235. $where = array(
  236. $this->table_id => array('number', $id)
  237. );
  238. $param = array(
  239. $this->table_status => array('number', $status)
  240. );
  241. return $this->pdo->sqlupdate($this->table_fullname, $param, $where);
  242. }
  243. //单独修改排序
  244. public function updateOrder($id, $order){
  245. $where = array(
  246. $this->table_id => array('number', $id)
  247. );
  248. $param = array(
  249. $this->table_order => array('number', $order)
  250. );
  251. return $this->pdo->sqlupdate($this->table_fullname, $param, $where);
  252. }
  253. //--------------------------------------------------
  254. //------------其他方法------------------------------
  255. //--------------------------------------------------
  256. //从数据库取出的数据转化键值后输出
  257. protected function dataToAttr($data){
  258. $r = array();
  259. foreach($this->table_struct as $k => $v){
  260. $r[$k] = $data[$v];
  261. }
  262. return $r;
  263. }
  264. //把filter转化为where子句
  265. //仅支持最常见的=和like,不支持其他符号
  266. //@param $filter array 键值要符合struct()中的定义
  267. //----参数示例----
  268. # $filter = array(
  269. # 'account' => array('abc', '=s'),//精确搜索
  270. # 'title' => array('xyz', '%s'),//模糊搜索
  271. # 'desc' => 'test',//不指定则为精确搜索
  272. # 'type' => 1
  273. # )
  274. //----参数示例结束----
  275. protected function filterToWhere($filter){
  276. $struct = $this->table_struct;
  277. $where = ' where 1=1 ';
  278. foreach($filter as $k => $v){
  279. if(is_array($v)){
  280. $val = $v[0];
  281. $operator = $v[1];
  282. $extraParam = $v[2]; //仅自定义的特殊方法使用
  283. }else{
  284. $val = $v;
  285. $operator = '=s';
  286. }
  287. $field_name = $struct[$k];
  288. if($operator == '=n'){//数字
  289. $val = $this->pdo->sql_check_input(array('number', $val));
  290. $where .= " and $field_name = $val ";
  291. }
  292. if($operator == '=s'){//字符串,精确搜索
  293. $val = $this->pdo->sql_check_input(array('string', $val));
  294. $where .= " and $field_name = $val ";
  295. }
  296. if($operator == '>n'){//大于数字
  297. $val = $this->pdo->sql_check_input(array('number', $val));
  298. $where .= " and $field_name > $val ";
  299. }
  300. if($operator == '>=n'){//大于等于数字
  301. $val = $this->pdo->sql_check_input(array('number', $val));
  302. $where .= " and $field_name >= $val ";
  303. }
  304. if($operator == '<n'){//小于数字
  305. $val = $this->pdo->sql_check_input(array('number', $val));
  306. $where .= " and $field_name < $val ";
  307. }
  308. if($operator == '<=n'){//小于等于数字
  309. $val = $this->pdo->sql_check_input(array('number', $val));
  310. $where .= " and $field_name <= $val ";
  311. }
  312. if($operator == '%s'){//字符串,模糊搜索
  313. $val = '%'.$val.'%';
  314. $val = $this->pdo->sql_check_input(array('string', $val));
  315. $where .= " and $field_name like $val " ;
  316. }
  317. if($operator == '=n_arr'){//or 连接的数字数组
  318. $where .= " and (";
  319. $first = true;
  320. foreach ($val as $item) {
  321. if ($first) {
  322. $where .= " $field_name = $item ";
  323. $first = false;
  324. } else {
  325. $where .= " or $field_name = $item ";
  326. }
  327. }
  328. $where .= ") ";
  329. }
  330. if($operator == '=date'){//时间戳在当天
  331. $val = $this->pdo->sql_check_input(array('number', $val));
  332. if (empty($val)) {
  333. break;
  334. }
  335. $begin = $val; //不分时区
  336. $end = $val + 24*60*60;
  337. $where .= " and ($field_name >= $begin and $field_name < $end) ";
  338. }
  339. if($operator == 'date2date'){//时间戳在两天之间
  340. if (!empty($val[0])) {
  341. $begin = $val[0] - 8*60*60; //东8时区
  342. $where .= " and $field_name >= $begin ";
  343. }
  344. if (!empty($val[1])) {
  345. $end = $val[1] + 16*60*60;
  346. $where .= " and $field_name < $end ";
  347. }
  348. }
  349. if($operator == 'id_in_arr'){//id在数组中
  350. $where .= " and (1=0 ";
  351. foreach ($val as $tempId) {
  352. $tempId = $this->pdo->sql_check_input(array('number', $tempId));
  353. $where .= " or $field_name=$tempId ";
  354. }
  355. $where .= ") ";
  356. }
  357. //以下是本项目特殊处理方法
  358. if($operator == 'date_between'){//date在两个日期之间
  359. $val = $this->pdo->sql_check_input(array('number', $val));
  360. $extraParam = $this->pdo->sql_check_input(array('number', $extraParam));
  361. $where .= " and $field_name > $extraParam and $field_name <= $val ";
  362. }
  363. }
  364. return $where;
  365. }
  366. }
  367. ?>