基于指纹考勤机的真实的PHP操作Access数据库成功案例(最终实现)

本文涉及的产品
RDS SQL Server Serverless,2-4RCU 50GB 3个月
推荐场景:
云数据库 RDS SQL Server,基础系列 2核4GB
简介:     听了我的建议,我们单位的食堂准备使用一台指纹考勤机统计吃饭人次,这样院里好给食堂的承包人以相应饭补。以前买过一台彩屏指纹机,数据库是access的,今儿又买了一台准备放到食堂里,而且考虑到停电,还特地配备了电池盒。

    听了我的建议,我们单位的食堂准备使用一台指纹考勤机统计吃饭人次,这样院里好给食堂的承包人以相应饭补。以前买过一台彩屏指纹机,数据库是access的,今儿又买了一台准备放到食堂里,而且考虑到停电,还特地配备了电池盒。

    这台不是彩屏的,但更加灵敏,反应速度比原来买的那台快,而且快不少。看看参数把:




    但有一样,使用U盘下载考勤数据时,是加密的.txt文档,用他们公司的软件才能把数据导入为Access表中。当然也可以自己弄.txt到.mdb的过程,但极为痛苦,极度难受,还很耗时。对于本周就上线运行的食堂管理系统而言,这是显然不合适的。
    但是是可以实现的,这里是ocx,说下实现思路:
    首先只能使用IE浏览器,通过调用ocx实现相关功能,就和一卡通里的IC卡模块一样。接着实例化一个类,使用GetGeneralLogData()方法获取记录,麻烦的就在这了,得到的结果不是想当然的数组或是比较容易处理的类型。这块很耗时。
    可以看一下函数的原型:
  1. long GetGeneralLogData(long *apnEnrollNumber, long *apnVerifyMode,long *apnInOutMode, DATA *apnDateTime)

    执行成功返回1,失败返回对应的错误代码。要先用LoadGeneralLogData或USBLoadDataFromFile指令,不然该指令会执行失败。
    不多说了,附件里是ocx,呵呵,有兴趣的自己研究去吧。这个思路都是浮云,在规定的时间里完不成的。涉及到调试ocx插件等等潜在问题,而且不是一个OCX或DLL的FKAttend就能搞定。我的方案是,使用他们给的工具导成mdb文档,再导入到我这个一卡通的大数据库里,这样所有记录都能得到妥善保存。而且领导也能一个页面,足不出户,查看到食堂消费情况。

    所以要通过PHP,读取mdb并写入mysql。
    这次用的不是ThinkPHP,而是叫做DoitPHP的框架,也是头一次使用,还以为比TP好用,看官网说数据处理特别快,但是不太容易掌握,总体还是略显青涩。
    这是MDB文件上传模块:
  1. //数据库文件上传
  2.     public function uploadAction(){
  3.         //保存目录检测
  4.         $dirurl = MDBDIR;
  5.         $dir = $this->instance('file_list');
  6.         $dir->make_dir($dirurl);
  7.         
  8.         
  9.         $fileupload = $this->instance('file_upload');
  10.         
  11.         $new_file = $dirurl.MDBFILE;
  12.         
  13.         $result = $fileupload->upload($_FILES['upload'], $new_file);

  14.         if(!$result){
  15.             $this->assign('msg','文件上传失败!');
  16.             $this->saveAction();
  17.         }
  18.         
  19.         if(!$this->inputDBAction()){
  20.             $this->assign('msg','文件导入数据库失败!');
  21.         }else{
  22.             $this->assign('msg',"文件已成功导入至数据库,请您在)."' >这里查询记录!");
  23.         }
  24.         
  25.         $this->set_layout('usual');
  26.         $this->assign('load_js_name','canteen');
  27.         
  28.         //display page
  29.         $this->display('save');
  30.     }

    上传成功,就要自动填入Mysql中,这是PHP操作Access数据库的模块:
  1. //access数据库信息导入
  2.     public function inputDBAction(){
  3.         if(file_exists(MDBDIR.MDBFILE)){
  4.                     
  5.             $access = $this->module('access');
  6.     //注意mdb是有密码,但没有用户名
  7.             $access->connect(MDBDIR.MDBFILE, '','2002', '', 0);
  8.             
  9.             $execarr = array();
  10.             $result = $access->query("select * from ".mb_convert_encoding("考勤流水表", "GBK", "UTF-8")." where Used = No");
  11.             while($arr = $access->fetch_array($result))
  12.             {
  13.              $execarr[] = $arr;
  14.             }
  15.             
  16.             //现在写入数据库
  17.              $att = $this->model('att_attendance');
  18.              $flag = TRUE;
  19.              foreach($execarr as $value){
  20.                  $data = $att->createRow();
  21.                  $data->userid = trim($value['UserID']);
  22.                  $data->flowtime = strtr(trim($value['FlowTime']),'/','-');
  23.                 
  24.                  $result = $att->save($data);

  25.                  if($result 0){$flag = FALSE;break;}
  26.              }
  27.             
  28.             return $flag;

  29.         }
  30.     }
    
    现在做查询分组统计:
  1. public function queryAction(){
  2.         //权限验证
  3.         $checkauth = $this->module('Checkauth');
  4.         $checkauth->checkAuth(CANTEEN);
  5.         
  6.         //设定日期
  7.         if(!isset($_POST['start']) && !isset($_POST['end'])){
  8.             //默认返回最近1周的记录
  9.             $now = getdate();
  10.             //********根据今天来确定本周的日期,开始************
  11.             if($now['wday']==0)$foo = -7;
  12.             else $foo = 1-$now['wday'];
  13.             $start = $now['year'].'-'.$now['mon'].'-'.($now['mday']+$foo);
  14.             
  15.             if($now['wday']==0)$foo = 0;
  16.             else $foo = 7-$now['wday'];
  17.             $end = $now['year'].'-'.$now['mon'].'-'.($now['mday']+$foo);
  18.              //********根据今天来确定本周的日期,结束************

  19.             $result = $this->model('att_attendance');
  20.             //求出分组统计
  21.             $att = $result->execute("select att_attendance.userid,att_member.name,count(att_attendance.userid) as total from att_attendance,att_member WHERE att_attendance.userid = att_member.userid GROUP BY userid");
  22.             
  23.             //求得总数,使用Mysql嵌套查询
  24.             $sum = $result->execute("select sum(total) from (select att_attendance.userid,att_member.name,count(att_attendance.userid) as total from att_attendance,att_member WHERE att_attendance.userid = att_member.userid GROUP BY userid) as sum");
  25.             //在这里继续,显示出分组结果和总数结果,只是还没有把时间增加进去
  26.         }else{
  27.             //对提交上来的start和end作出新的查询
  28.         }
  29.         
  30.         $this->set_layout('usual');
  31.         $this->assign('load_js_name','canteen');
  32.         exit();
  33.         //display page
  34.         $this->display();
  35.     }


    有几个技术点需要记录下来,以备忘:
    1、不能直接使用PhpMyAdmin构造的SQL,在这个例子中会失败,问题出要出在FlowTime身上;
    2、格式问题,为了安全起见,要使用strtr把2011/11/11,替换为2011-11-11,为的是保持数据库通用日期时间格式;
    3、在DoitPHP这个框架里使用insert()或save()保存数据时,面对object,而非array类型,尽量不要使用$data['UserID']的数组形式,会出现错误,而要采用$data->UserID;
    4、特别好玩的是,Access里Boolean型的值有三种True/False,On/Off和Yes/No,我这里mdb设定的就是Yes/No;
    5、注意mdb文件是加密了的,当然也必须加密,否则食堂的人弄到mdb怎么办?当然mdb很容易就被破解了,不过领导看到的是我这个系统里读出的记录,我不给他权限,他哪里能把自己修改了的mdb文件上传给领导看呢。哈哈。领导,你要给我涨工资哦~~~


    最后上传并且自动导入成功:










    当然这里还用了一个access的PHP类,大家可以到我的这篇文章去看:

    五岳之巅原创,转载请注明出处。谢谢。

    附表:

SQL  语法元素
Microsoft 
Access
Microsoft SQL 
Server
标识符
限制不超过 64 个字符。
允许使用关键字和特殊字符。
可以用任何字符开头。
SQL Server 6.5:
  • 限制不超过 30 个字符。
  • 不允许使用关键字和特殊字符。
  • 必须用字母字符开头。
SQL Server 7.0 的标识符与 Access 完全兼容。
输出字段
允许多个输出字段具有相同名称。
在视图中不支持多个相同输出字段名。
日期分隔符号
英镑符(#)
撇号(')
Boolean 常量
True、False;On、Off;Yes、No。
整数:1(真)、0(假)
字符串连接
和号(&)
加号(+)
通配符
星号(*)与零个或更多字符匹配。
问号(?)与单个字符匹配。
叹号(!)意味着不在列表中。
英镑符(#)意味着单个数字。
百分号(%)与零个或更多字符匹配。
下划线(_)与单个字符匹配。
上插入符(^)意味着不在列表中。
没有与英镑符(#)对应的字符。
TOP
如果有一个 ORDER BY 子句,自动包含层次。
SQL Server 6.5 不支持。
SQL Server 7.0 需要一个明确的 WITH TIES 子句。
CREATE INDEX
允许创建升序和降序索引。
允许声明主键,没有 Null 值,并且忽略 Null 值。
DROP INDEX
语法是:
Drop Index ON
语法是:
Drop Index ,
DISTINCTROW
支持(允许选择单个记录)。
不支持。
OWNERACCESS
支持(在执行时控制许可权)。
不支持。
Table in UNION
支持(允许使用下列语法指定表:
TABLE
不支持。
ORDER BY in Unions
支持。允许通过联合查询中的子句实现多种排序。
支持。允许通过语句末尾的子句实现一种排序。
TRANSFORM
支持。用于交叉表查询。
不支持。
PARAMETERS
支持(在 SQL 中记录)。
不支持。


阅读(11457) | 评论(5) | 转发(2) |
0

上一篇:3年的奋斗目标

下一篇:应该如何做笔记

给主人留下些什么吧!~~

phpstar2011-12-20 12:45:09

项目做完了吗?我们很感兴趣,私聊下。

五岳之巅2011-11-28 22:31:02

huhuegg: 吃饱了撑的.....
呵呵,您一定是没吃饱,告您一个秘密:食堂还有剩饭呢。
就是别忘了单位进出人多,先到我这录指纹,然后打卡才能吃饭,月末有详单统计,人次、消费、补助一目了然。但就有一点,别怪我没提醒,千万别吃太饱,否则就真撑着了。

huhuegg2011-11-28 09:48:14

吃饱了撑的

十七岁的回忆2011-11-23 00:39:18

好熟悉的仪器呀!~
给力的文章,顶一下!

2011-11-21 17:36:25

必须支持!

评论热议
请登录后评论。

登录 注册

相关实践学习
使用SQL语句管理索引
本次实验主要介绍如何在RDS-SQLServer数据库中,使用SQL语句管理索引。
SQL Server on Linux入门教程
SQL Server数据库一直只提供Windows下的版本。2016年微软宣布推出可运行在Linux系统下的SQL Server数据库,该版本目前还是早期预览版本。本课程主要介绍SQLServer On Linux的基本知识。 相关的阿里云产品:云数据库RDS SQL Server版 RDS SQL Server不仅拥有高可用架构和任意时间点的数据恢复功能,强力支撑各种企业应用,同时也包含了微软的License费用,减少额外支出。 了解产品详情: https://www.aliyun.com/product/rds/sqlserver
相关文章
|
3月前
|
存储 SQL 关系型数据库
PHP与数据库交互:从基础到进阶
【10月更文挑战第9天】在编程的世界里,数据是流动的血液,而数据库则是存储这些珍贵资源的心脏。PHP作为一门流行的服务器端脚本语言,其与数据库的交互能力至关重要。本文将带你从PHP与数据库的基本连接开始,逐步深入到复杂查询的编写和优化,以及如何使用PHP处理数据库结果。无论你是初学者还是有一定经验的开发者,这篇文章都将为你提供宝贵的知识和技巧,让你在PHP和数据库交互的道路上更加从容不迫。
|
16天前
|
SQL Oracle 数据库
使用访问指导(SQL Access Advisor)优化数据库业务负载
本文介绍了Oracle的SQL访问指导(SQL Access Advisor)的应用场景及其使用方法。访问指导通过分析给定的工作负载,提供索引、物化视图和分区等方面的优化建议,帮助DBA提升数据库性能。具体步骤包括创建访问指导任务、创建工作负载、连接工作负载至访问指导、设置任务参数、运行访问指导、查看和应用优化建议。访问指导不仅针对单条SQL语句,还能综合考虑多条SQL语句的优化效果,为DBA提供全面的决策支持。
44 11
|
26天前
|
关系型数据库 MySQL 数据库
数据库数据恢复—MYSQL数据库文件损坏的数据恢复案例
mysql数据库文件ibdata1、MYI、MYD损坏。 故障表现:1、数据库无法进行查询等操作;2、使用mysqlcheck和myisamchk无法修复数据库。
|
2月前
|
SQL 数据库
GBase8a 数据库集群v953扩容案例问题分享
GBase8a 数据库集群v953扩容案例问题分享
|
2月前
|
SQL 数据库
gbase 8a 数据库 SQL优化案例-关联顺序优化
gbase 8a 数据库 SQL优化案例-关联顺序优化
|
2月前
|
存储 SQL 数据库
Sybase数据恢复—Sybase数据库常见问题之数据库无法启动的恢复案例
Sybase数据库数据恢复环境: Sybase数据库版本:SQL Anywhere 8.0。 Sybase数据库故障&分析: Sybase数据库无法启动。 使用Sybase Central连接报错。 数据库数据恢复工程师经过检测,发现Sybase数据库出现故障的原因是:异常断电造成Sybase数据库无法回写正常数据,导致多个存储页数据不一致,系统表描述和存储表不一致,部分存储页底层数据完全杂乱。
|
2月前
|
Oracle 关系型数据库 数据库
Oracle数据恢复—Oracle数据库文件有坏快损坏的数据恢复案例
一台Oracle数据库打开报错,报错信息: “system01.dbf需要更多的恢复来保持一致性,数据库无法打开”。管理员联系我们数据恢复中心寻求帮助,并提供了Oracle_Home目录的所有文件。用户方要求恢复zxfg用户下的数据。 由于数据库没有备份,无法通过备份去恢复数据库。
|
2月前
|
存储 Oracle 关系型数据库
oracle数据恢复—Oracle数据库文件大小变为0kb的数据恢复案例
存储掉盘超过上限,lun无法识别。管理员重组存储的位图信息并导出lun,发现linux操作系统上部署的oracle数据库中有上百个数据文件的大小变为0kb。数据库的大小缩水了80%以上。 取出&并分析oracle数据库的控制文件。重组存储位图信息,重新导出控制文件中记录的数据文件,发现这些文件的大小依然为0kb。
|
1月前
|
存储 Oracle 关系型数据库
服务器数据恢复—华为S5300存储Oracle数据库恢复案例
服务器存储数据恢复环境: 华为S5300存储中有12块FC硬盘,其中11块硬盘作为数据盘组建了一组RAID5阵列,剩下的1块硬盘作为热备盘使用。基于RAID的LUN分配给linux操作系统使用,存放的数据主要是Oracle数据库。 服务器存储故障: RAID5阵列中1块硬盘出现故障离线,热备盘自动激活开始同步数据,在同步数据的过程中又一块硬盘离线,RAID5阵列瘫痪,上层LUN无法使用。
|
2月前
|
存储 Java 关系型数据库
在Java开发中,数据库连接是应用与数据交互的关键环节。本文通过案例分析,深入探讨Java连接池的原理与最佳实践
在Java开发中,数据库连接是应用与数据交互的关键环节。本文通过案例分析,深入探讨Java连接池的原理与最佳实践,包括连接创建、分配、复用和释放等操作,并通过电商应用实例展示了如何选择合适的连接池库(如HikariCP)和配置参数,实现高效、稳定的数据库连接管理。
72 2