python实现人脸签到系统

本文实例为大家分享了python实现人脸签到系统的具体代码,供大家参考,具体内容如下

简易版人脸签到/签退系统

管理员可进行录入人脸操作,以及导出各类签到情况表;

普通学生只可人脸识别进行签到签退操作。

python实现人脸签到系统_第1张图片

face_Recognizer.py

import cv2
import os
import numpy as np
from PIL import Image #pillow
import pyttsx3
import sys
import test.student_sql as connect #导入py
import test.recognizer_sql as baseConnect #导入py
import time
import json
 
def makeDir(engine):
  flag= 0
  if not os.path.exists("face_trainer"):
    print("创建预训练环境")
    engine.say('检测到第一次启动,未检测到环境,正在创建环境')
    engine.say('正在创建预训练环境')
    os.mkdir("face_trainer")
    engine.say('创建成功')
    engine.runAndWait()
    flag=1
  if not os.path.exists("Facedata"):
    print("创建训练环境")
    engine.say('正在创建训练环境')
    os.mkdir("Facedata")
    engine.say('创建成功')
    engine.runAndWait()
    flag=1
  return flag
 
def getFace(cap,path_id):
  # 调用笔记本内置摄像头,所以参数为0,如果有其他的摄像头可以调整参数为1,2
  #cap = cv2.VideoCapture(0)
  #xml文件为自己的文件路径
  face_detector = cv2.CascadeClassifier(r'F:\npyWorkspace\venv\Lib\site-packages\cv2\data\haarcascade_frontalface_default.xml')
  #face_id = input('\n enter user id:')
  print('\n Initializing face capture. Look at the camera and wait ...')
  count = 0
  while True:
    # 从摄像头读取图片
    sucess, img = cap.read()
    # 转为灰度图片
    gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)
    # 检测人脸
    faces = face_detector.detectMultiScale(gray, 1.3, 5)
    for (x, y, w, h) in faces:
      cv2.rectangle(img, (x, y), (x+w, y+w), (255, 0, 0))
      count += 1
      # 保存图像
      cv2.imwrite("Facedata/User." + str(path_id) + '.' + str(count) + '.jpg', gray[y: y + h, x: x + w])
      cv2.imshow('image', img)
    # 保持画面的持续。
    k = cv2.waitKey(1)
    if k == 27:  # 通过esc键退出摄像
      break
    elif count >= 100: # 得到1000个样本后退出摄像
      break
  cv2.destroyAllWindows()
 
def getImagesAndLabels(path, detector):
  imagePaths = [os.path.join(path, f) for f in os.listdir(path)] # join函数的作用
  faceSamples = []
  ids = []
  for imagePath in imagePaths:
    PIL_img = Image.open(imagePath).convert('L') # convert it to grayscale
    img_numpy = np.array(PIL_img, 'uint8')
    id = int(os.path.split(imagePath)[-1].split(".")[1])
    faces = detector.detectMultiScale(img_numpy)
    for (x, y, w, h) in faces:
      faceSamples.append(img_numpy[y:y + h, x: x + w])
      ids.append(id)
  return faceSamples, ids
 
 
def trainFace():
  # 人脸数据路径
  path = 'Facedata'
  recognizer = cv2.face.LBPHFaceRecognizer_create()
  detector = cv2.CascadeClassifier(r'F:\npyWorkspace\venv\Lib\site-packages\cv2\data\haarcascade_frontalface_default.xml')
  print('Training faces. It will take a few seconds. Wait ...')
  faces, ids = getImagesAndLabels(path, detector)
  recognizer.train(faces, np.array(ids))
  recognizer.write(r'face_trainer\trainer.yml')
  print("{0} faces trained. Exiting Program".format(len(np.unique(ids))))
 
def checkFace(cam,names,engine,sign_flag):
  sex = {"female":"女士","male":"先生"}
  recognizer = cv2.face.LBPHFaceRecognizer_create()
  recognizer.read('face_trainer/trainer.yml')
  cascadePath = r"F:\npyWorkspace\venv\Lib\site-packages\cv2\data\haarcascade_frontalface_default.xml"
  faceCascade = cv2.CascadeClassifier(cascadePath)
  font = cv2.FONT_HERSHEY_SIMPLEX
  idnum = 0
  names = ['yumengzhen', 'dujuanjuan','litingting','kangming','wangyizhe']
  #cam = cv2.VideoCapture(0)
  minW = 0.1 * cam.get(3)
  minH = 0.1 * cam.get(4)
  while True:
    ret, img = cam.read()
    gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)
    faces = faceCascade.detectMultiScale(
      gray,
      scaleFactor=1.2,
      minNeighbors=5,
      minSize=(int(minW), int(minH))
    )
    for (x, y, w, h) in faces:
      cv2.rectangle(img, (x, y), (x + w, y + h), (0, 255, 0), 2)
      idnum, confidence = recognizer.predict(gray[y:y + h, x:x + w])
      if confidence < 100:
        Name =connect.readName(idnum) #connect 传入ID 学生信息表找到 返回 name
        Sex =connect.readSex(idnum) #connect ID 学生信息表找到 返回 Sex
        StudentID =connect.readStudentID(idnum) #connect ID 学生信息表找到 返回 studentID
        #idnum = names[idnum]  #利用数据库 读取学生信息表 该id 对应的name
        confidence = "{0}%".format(round(100 - confidence))
        if sign_flag=='0': #签到
          say(engine, "欢迎     "+Name+ sex[Sex]+"     签到成功 ")
          baseConnect.insertd(idnum,Name,StudentID,Sex) #签到表中 插入签到信息
          print("欢迎   "+Name+ sex[Sex] + "签到成功 ")
        else :
          say(engine, "欢迎     "+Name+ sex[Sex]+"     签退成功 ")
          baseConnect.insertt(idnum,Name,StudentID,Sex) #签到表中 插入签退信息
          print("欢迎   "+Name+ sex[Sex] + "签退成功 ")
 
        cv2.imshow("img",img)
        os.system("pause")
        return
      else:
        idnum = "unknown"
        confidence = "{0}%".format(round(100 - confidence))
      cv2.putText(img, str(idnum), (x + 5, y - 5), font, 1, (0, 0, 255), 1)
      cv2.putText(img, str(confidence), (x + 5, y + h - 5), font, 1, (0, 0, 0), 1)
    cv2.imshow('camera', img)
    k = cv2.waitKey(10)
    if k == 27:
      break
  cam.release()
  cv2.destroyAllWindows()
 
 
def say(engine,str):
  engine.say(str)
  engine.runAndWait()
 
def admission(): #录入信息模块
  #names = {"yumengzhen":0,"dujuanjuan":1,"litingting":2}
  say(engine, "请输入您的学号      ")
  StudentID = input("请输入学号:")
  # 读取数据库信息表 取出Name 对应ID
  ID=connect.readIDbaseStudentID(StudentID) #connect 传入name 学生信息表找到 返回 ID
  if ID==-1:#没有找到该学生插入学生信息
    while True:
      say(engine,"没有找到该学生信息 输人 0 注册 1重新输入")
      op=input("\n 没有找到该学生信息 输人数字 0 注册学生信息 1重新输入")
      if op=='0':
        Name,studentID,Sex=input("输入学生信息: Name studentID Sex").split()
        connect.insert(Name,studentID,Sex) #插入学生信息信息
      else:
        StudentID = input("请输入学号:")
      ID=connect.readIDbaseStudentID(StudentID) #connect 传入name 学生信息表找到 返回 ID
      if ID!=-1 :
        break
  say(engine, "正在打开摄像头")
  cam = cv2.VideoCapture(0)
  say(engine, "注视摄像头,开始采集人脸数据")
  getFace(cam, ID) # 实际传入的是id
  cam.release()
 
 
if __name__ == '__main__':
  names = {"yumengzhen":0,"dujuanjuan":1,"litingting":  2}
  password="123456" #密码
  engine = pyttsx3.init()
  rate = engine.getProperty('rate')
  engine.setProperty('rate', rate - 20)
  flag=makeDir(engine)
  #trainFace()
  while True:
    if flag==1 :
      flag = 0
      say(engine, "首次使用 没有人脸信息 ")
      say(engine, "是否要录入新的人脸信息   ")
      say(engine, "输入0 代表是 输入其他表示退出")
      value = input("0:是 or other:否")
      if value=='0':
        while True:
          admission()
          say(engine, "是否要继续录入新的人脸信息   ")
          say(engine, "输入0 代表是 输入其他表示退出")
          firstflag = input("0:是  其他:退出")
          if firstflag != '0':
            break
        say(engine, "采集完毕,开始训练")
        trainFace()
        say(engine, "训练完毕     ")
 
    #say(engine, "请选择登录方式  ")
    say(engine, "输入 0管理人员模式    1 进入签到/签退模式   2 退出学生签到系统 ")
    user=input("\n0:管理人员模式    1:进入签到/签退模式  2:退出学生签到系统\n")
 
    if user=='0':
      say(engine, "输入管理员密码  ")
      pd=input("\n输入管理员密码 :\n")
      count=1
      while True:
        if count==3:
          say(engine," 输入密码错误超过3次 强制退出输入   ")
          break
 
        if password == pd:
          say(engine, "管理员模式  ")
          #say(engine, "输入数字 0 导出签到表   1 导出个人签到表   2 导出时长表    3 导出信息表    4 录入人脸信息  5 退出")
          op = input("\n0:导出所有同学签到表  1:导出个人签到表 2:导出所有人员时长表  3:导出学生信息表 4 录入人脸信息  5 退出\n")
          if op == '0':
            baseConnect.sign()#导出签到表
            say(engine, "导出签到表成功  ")
            pass
          elif op == '1':
            say(engine,"输入导出学生的学号")
            StudentID=input("输入导出学生的学号")
            ID=connect.readIDbaseStudentID(StudentID)
            if ID==-1:
              say(engine, "没有该学生信息 ")
            else:
              baseConnect.peoson_sign(StudentID)#导出个人签到表
              Name =connect.readName(ID) #connect 传入ID 学生信息表找到 返回 name
              say(engine, "导出 "+Name+"  信息成功")
 
          elif op == '2':
            baseConnect.total_time()#导出时长表
            say(engine,"导出时长表成功   ")
          elif op == '3':
            #导出学生信息表
            connect.find_student_all()
            say(engine, "导出学生信息成功   ")
          elif op == '4':
            while True:
              admission()
              say(engine, "是否要继续录入新的人脸信息   ")
              say(engine, "输入0 代表是 输入其他表示退出")
              secondflag = input("0:是  其他:退出")
              if secondflag != '0':
                break
            say(engine, "采集完毕,开始训练")
            trainFace()
            say(engine, "训练完毕     ")
          elif op == '5':
            say(engine, "已退出 管理员模式  ")
            break
          else:
            say(engine, "输入形式错误 请重新输入   ")
        else:
          say(engine, "输入密码错误   请重新输入 ")
          pd = input("\n输入管理员密码 :\n")
          count += 1;
 
    elif user=='1':
      say(engine, "欢迎进入学生系统签到/签退模式   ")
      sign_flag=0;
      while True:
        say(engine, "输入数字 0 签到   1 签退")
        sign_flag = input("\n0: 签到   1 签退\n")
        if sign_flag=='1' or sign_flag=='0' :
          break
        else :
          say(engine,"  请输入正确的输入形式")
      say(engine, "开始人脸识别")
      say(engine, "正在打开摄像头")
      cam = cv2.VideoCapture(0)
      checkFace(cam, names, engine,sign_flag)
 
    elif user=='2':
      say(engine, "信息已保存")
      say(engine, "再见")
      sys.exit(0)
    else:
      say(engine, "输入错误请重新输入   ")

student_sql.py  

学生

import pymssql as py
import pandas as pd
# 连接数据库,创建学生表,进行表查询,表录入
server = "DESKTOP-XXX"# 连接自己数据库的服务器地址
user = "sa"# 连接帐号
password = "123"# 连接密码
conn = py.connect(server, user, password, "student_message") #获取连接
cursor = conn.cursor() # 获取光标
 
# 创建表
cursor.execute("""
IF OBJECT_ID('students', 'U') IS NOT NULL
  DROP TABLE students
CREATE TABLE students (
  ID INT NOT NULL,
  name VARCHAR(100),
  StudentID INT,
  Sex VARCHAR(100)
  )
""")
conn.commit()
#第一次运行时建立表,之后再运行无需再建
 
def insert(Name, studentID, Sex):
    count_students = 0
    try:
      conn = py.connect(server, user, password, "student_message") # 获取连接
      cursor =conn.cursor()
      cursor.execute(' select count(ID) from students')
      for row in cursor:
        count_students = row[0]
        print(row[0])
      cursor.executemany(
        "INSERT INTO students VALUES (%d, %s, %d,%s)",
        [(count_students+1, Name, studentID, Sex)])
    # 你必须调用 commit() 来保持你数据的提交如果你没有将自动提交设置为true
      conn.commit()
      conn.close()
    except py.InterfaceError:
      print("数据库连接出错")
    except py.ProgrammingError:
      print("数据错误,请检查输入的数据")
    except py.OperationalError:
      print("数据错误,请检查输入的数据")
 
 
# 导出学生信息表
def find_student_all():
  try:
    conn = py.connect(server, user, password, "student_message") # 获取连接
    cursor =conn.cursor()
    sql = "select * from students"
    df = pd.read_sql(sql, conn)
    df.to_excel('all.xlsx',index=False)
    print('ok')
    conn.commit()
    conn.close()
  except py.InterfaceError:
    print("数据库连接出错")
  except py.ProgrammingError:
    print("数据错误,请检查输入的数据")
 
#find_student_all()
 
def readName(idnum):
  Name = -1
  try:
    conn = py.connect(server, user, password, "student_message") # 获取连接
    cursor =conn.cursor()
    cursor.execute(' select Name from students where ID='+str(idnum))
    for row in cursor:
      if row[0]!=[]:
        Name = row[0]
    conn.commit()
    conn.close()
  except py.InterfaceError:
    print("数据库连接出错")
  except py.ProgrammingError:
    print("数据错误,请检查输入的数据")
  return Name
 
def readIDbaseStudentID(StudentID):
  ID = -1
  try:
    conn = py.connect(server, user, password, "student_message") # 获取连接
    cursor =conn.cursor()
    cursor.execute(' select ID from students where StudentID='+str(StudentID))
    for row in cursor:
      if row[0]!=[]:
        ID = row[0]
    conn.commit()
    conn.close()
  except py.InterfaceError:
    print("数据库连接出错")
  except py.ProgrammingError:
    print("数据错误,请检查输入的数据")
  return ID
 
def readSex(idnum):
  Sex = -1
  try:
    conn = py.connect(server, user, password, "student_message") # 获取连接
    cursor =conn.cursor()
    cursor.execute(' select Sex from students where ID='+str(idnum))
    for row in cursor:
       if row[0]!=[]:
          Sex = row[0]
    conn.commit()
    conn.close()
  except py.InterfaceError:
    print("数据库连接出错")
  except py.ProgrammingError:
    print("数据错误,请检查输入的数据")
  return Sex
 
 
def readID(name):
  # 多个id
  ID = -1
  try:
    conn = py.connect(server, user, password, "student_message") # 获取连接
    cursor =conn.cursor()
    cursor.execute(' select ID from students where name='+'\''+str(name)+'\'')
    for row in cursor:
      if row[0]!=[]:
        ID = row[0]
    conn.commit()
    conn.close()
  except py.InterfaceError:
    print("数据库连接出错")
  except py.ProgrammingError:
    print("数据错误,请检查输入的数据")
  return ID
 
 
def readStudentID(idnum):
  StudentID = -1
  try:
    conn = py.connect(server, user, password, "student_message") # 获取连接
    cursor =conn.cursor()
    cursor.execute(' select StudentID from students where ID='+str(idnum))
    for row in cursor:
      if row[0]!=[]:
        StudentID = row[0]
    conn.commit()
    conn.close()
  except py.InterfaceError:
    print("数据库连接出错")
  except py.ProgrammingError:
    print("数据错误,请检查输入的数据")
  return StudentID
 
 
# 关闭连接
# conn.close()
#
# # 注:在任何时候,在一个连接下,一次正在执行的数据库操作只会出现一个cursor对象

recognizer_sql.py

import pymssql as py
import time
import pandas as pd
 
server = "DESKTOP-XXXX"# 连接服务器地址
user = "sa" # 连接帐号
password = "123" # 连接密码
conn = py.connect(server, user, password, "student_message") #获取连接
cursor = conn.cursor() # 获取光标
 
def insertd(idnum,Name,StudentID,Sex):  # 签到
  conn = py.connect(server, user, password, "student_message") # 获取连接
  timenow = time.strftime("%Y-%m-%d %H:%M:%S", time.localtime())
  cursor = conn.cursor()
  cursor.execute("INSERT INTO qiandao VALUES (%d, %s,%d,%s, %s, %s, %s,%d )", (idnum,Name,StudentID,Sex,timenow, '0', '0', 0))
  conn.commit()
  # 必须调用 commit() 来保持数据的提交
def insertt(idnum,Name,StudentID,Sex):  # 签退
  conn = py.connect(server, user, password, "student_message") # 获取连接
  cursor = conn.cursor()
  cursor.execute("SELECT starttime FROM qiandao WHERE ID=%s and flag=%d",(idnum,0))
  starttimeget = cursor.fetchone()
  sat = str(tuple(starttimeget))
  timenow = time.strftime("%Y-%m-%d %H:%M:%S", time.localtime())
  timeArray = time.strptime(sat, "('%Y-%m-%d %H:%M:%S',)")
  timeStamp = int(time.mktime(timeArray))
  timecout = time.time() - timeStamp
  m, s = divmod(timecout, 60)
  h, m = divmod(m, 60)
  timepass = str(h) + '小时 ' + str(m) + '分钟 ' + str(s) + '秒'
  print(timepass)
  cursor.executemany("INSERT INTO qiandao VALUES (%d, %s,%d,%s, %s, %s, %s,%d )",
    [(idnum,Name,StudentID,Sex,'0', timenow, timepass, 1)])
  conn.commit()
 
def peoson_sign(StudentID):# 导出学生信息表_按照学号
  conn = py.connect(server, user, password, "student_message") # 获取连接
  cursor = conn.cursor()
  sql = "select * from qiandao where StudentID=" + str(StudentID)
  df = pd.read_sql(sql, conn)
  df.to_excel(r'E:\01STUDY\20190701\work\openVersion\excel\studentID_sign.xlsx',index=False)
  print('ok')
  conn.commit()
 
#peoson_sign(2016002105)
 
def sign():# 导出签到表
  conn = py.connect(server, user, password, "student_message") # 获取连接
  cursor = conn.cursor()
  sql = "select * from qiandao"
  df = pd.read_sql(sql, conn)
  df.to_excel(r'E:\01STUDY\20190701\work\openVersion\excel\sign_all.xlsx', index=False)
  print('ok')
  conn.commit()
#sign()
 
def total_time():# 导出时长表#sign()
  conn = py.connect(server, user, password, "student_message") # 获取连接
  cursor = conn.cursor()
  sql = "select * from qiandao where convert(nvarchar(max),count) != convert(nvarchar(max),0)"
  df = pd.read_sql(sql, conn)
  df.to_excel(r'E:\01STUDY\20190701\work\openVersion\excel\total_time.xlsx', index=False)
  print('ok')
  conn.commit()
 
#
# if __name__=='__main':
#   sign()
#   #peoson_sign(2016002105)
 
conn.close()

更多学习资料请关注专题《管理系统开发》。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

你可能感兴趣的:(python实现人脸签到系统)