
本文旨在解决在使用OpenCV进行人脸识别考勤时,由于代码逻辑问题导致考勤信息重复写入CSV文件的问题。通过分析问题代码,找出重复写入的原因,并提供修改后的代码示例,确保考勤记录的准确性和唯一性。文章还讨论了进一步优化方案,例如将已记录的名字列表保存在内存中,避免重复读取文件,提高程序效率。
在人脸识别考勤系统中,实时从摄像头捕获图像,识别出人脸并记录考勤信息是常见的需求。然而,在实际应用中,可能会遇到重复记录考勤信息的问题。本文将针对这个问题,提供详细的解决方案和优化建议。
原代码的问题在于 markAttendance 函数中,每次检测到人脸时,都会打开 Attendance.csv 文件,读取所有行,然后检查当前识别到的人名是否已存在于列表中。由于 if name not in nameList: 语句位于 for 循环内部,因此每次循环都会进行一次判断,导致在同一帧图像中多次识别到同一张脸时,会重复写入考勤信息。
核心思路是将判断人名是否已存在于列表中的逻辑,从 for 循环内部移到循环外部,确保只有在读取完所有已存在的人名后,才进行判断和写入操作。
以下是修改后的 markAttendance 函数代码:
def markAttendance(name):
with open('Attendance.csv','r+') as f:
myDataList = f.readlines()
nameList = []
for line in myDataList:
entry = line.split(',')
nameList.append(entry[0])
# 循环结束后再进行判断
if name not in nameList:
now = datetime.now()
dtString = now.strftime('%H:%M:%S')
f.writelines(f'\n{name},{dtString}')代码解释:
每次调用 markAttendance 函数都读取整个 CSV 文件效率较低。更优化的方案是在程序启动时一次性读取所有已记录的名字,并将其保存在内存中。后续的考勤记录直接与内存中的列表进行比较,只有在发现新的人名时才写入文件,并更新内存中的列表。
以下是优化后的代码示例:
def readNames():
with open('Attendance.csv', 'r') as f:
nameList = []
for line in f:
entry = line.split(',')
nameList.append(entry[0])
return nameList
def markAttendance(name, nameList):
if name not in nameList:
nameList.append(name) # add directly to list
# write to file
with open('Attendance.csv', 'a') as f:
dt = datetime.now().strftime('%H:%M:%S')
f.writelines(f'\n{name},{dt}')
# --- 程序启动时 ---
nameList = readNames() # read only once at start
# --- 在主循环中 ---
while True:
# ...
if matches[matchIndex]:
# ...
markAttendance(name, nameList)代码解释:
注意事项:
本文针对人脸识别考勤系统中重复写入 CSV 文件的问题,提供了两种解决方案:
选择哪种方案取决于具体的应用场景和性能需求。在实际应用中,可以根据需要进行调整和优化,以达到最佳的考勤效果。
以上就是解决人脸识别考勤系统中重复写入CSV文件的问题的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号