在Php项目中构建答题数据库需要综合考虑业务需求、数据关系和性能优化,以下是详细的建库方案,包含表结构设计、字段说明及关联逻辑。
核心表结构设计
答题系统通常涉及用户、题目、试卷、答题记录等核心模块,需建立以下数据表:
用户表(users)
存储用户基础信息,区分不同角色(如学生、管理员)。
| 字段名 | 类型 | 约束 | 说明 |
|——–|——|——|——|
| id | int(11) | PRIMARY KEY, AUTO_INCREMENT | 用户ID |
| username | varchar(50) | NOT NULL, UNIQUE | 用户名 |
| password | varchar(255) | NOT NULL | 密码(建议加密存储) |
| email | varchar(100) | UNIQUE | 邮箱 |
| role | tinyint(1) | DEFAULT 0 | 角色(0学生/1管理员) |
| created_at | timestamp | DEFAULT CURRENT_TIMESTAMP | 创建时间 |
| updated_at | timestamp | DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP | 更新时间 |

题目表(questions)及分类信息,支持多种题型。
| 字段名 | 类型 | 约束 | 说明 |
|---|---|---|---|
| id | int(11) | PRIMARY KEY, AUTO_INCREMENT | 题目ID |
| type | tinyint(1) | NOT NULL | 题型(1单选/2多选/3判断/4填空) |
| options | text | JSON格式存储选项(单选/多选/判断题有效) | |
| answer | text | NOT NULL | 标准答案(填空题为逗号分隔值) |
| explanation | text | 答案解析 | |
| category_id | int(11) | FOREIGN KEY | 题目分类ID |
| difficulty | tinyint(1) | DEFAULT 1 | 难度等级(1-5) |
| created_at | timestamp | DEFAULT CURRENT_TIMESTAMP | 创建时间 |
试卷表(papers)
管理试卷基本信息,支持随机组卷或固定组卷。
| 字段名 | 类型 | 约束 | 说明 |
|——–|——|——|——|
| id | int(11) | PRIMARY KEY, AUTO_INCREMENT | 试卷ID || varchar(100) | NOT NULL | 试卷名称 |
| description | text | 试卷描述 |
| total_score | decimal(5,2) | NOT NULL | 试卷总分 |
| time_limit | int(11) | 答题限制时间(分钟) |
| question_ids | text | JSON格式存储题目ID列表 |
| created_by | int(11) | FOREIGN KEY (users.id) | 创建者ID |
| status | tinyint(1) | DEFAULT 1 | 状态(0禁用/1启用) |
| created_at | timestamp | DEFAULT CURRENT_TIMESTAMP | 创建时间 |
答题记录表(records)
记录用户答题过程及结果。
| 字段名 | 类型 | 约束 | 说明 |
|——–|——|——|——|
| id | int(11) | PRIMARY KEY, AUTO_INCREMENT | 记录ID |
| user_id | int(11) | FOREIGN KEY (users.id) | 用户ID |
| paper_id | int(11) | FOREIGN KEY (papers.id) | 试卷ID |
| user_answers | text | JSON格式存储用户答案(键为题目ID,值为答案内容) |
| score | decimal(5,2) | 得分 |
| start_time | datetime | 答题开始时间 |
| end_time | datetime | 答题结束时间 |
| duration | int(11) | 答题耗时(秒) |
| status | tinyint(1) | DEFAULT 0 | 状态(0未完成/1已完成) |

题目分类表(categories)分类管理,便于组织题目。
| 字段名 | 类型 | 约束 | 说明 |
|---|---|---|---|
| id | int(11) | PRIMARY KEY, AUTO_INCREMENT | 分类ID |
| name | varchar(50) | NOT NULL | 分类名称 |
| parent_id | int(11) | DEFAULT 0 | 父分类ID(支持多级分类) |
| created_at | timestamp | DEFAULT CURRENT_TIMESTAMP | 创建时间 |
数据库优化建议
- 索引设计:在常用查询字段(如user_id、paper_id、created_at)上建立索引,提高查询效率。
- JSON字段应用:questions表的options和records表的user_answers使用JSON类型,便于存储和解析复杂结构数据。
- 分表策略:若答题记录数据量较大,可按时间或用户ID分表,避免单表数据膨胀。
- 数据加密:用户密码建议使用password_hash()函数加密存储,敏感信息(如邮箱)可考虑AES加密。
PHP操作示例
使用PDO连接数据库并插入答题记录:
$pdo = new PDO('mysql:host=localhost;dbname=exam_db', 'username', 'password');
$stmt = $pdo->prepare("INSERT INTO records (user_id, paper_id, user_answers, start_time)
VALUES (:user_id, :paper_id, :answers, NOW())");
$stmt->execute([
'user_id' => 1,
'paper_id' => 5,
'answers' => json_encode(['1' => 'A', '2' => 'B,C'])
]);
相关问答FAQs
Q1: 如何实现试卷的随机组卷功能?
A1: 可通过以下步骤实现:

- 在questions表中按题型和难度筛选题目;
- 使用PHP的array_rand()函数随机抽取指定数量的题目;
- 将选中的题目ID列表以JSON格式存入papers表的question_ids字段;
- 用户答题时解析该字段获取题目列表。
Q2: 答题记录中的用户答案如何进行批改?
A2: 批改逻辑需分题型处理:
- 单选题/判断题:直接对比用户答案与questions表的answer字段;
- 多选题:将用户答案按逗号分割后排序,与标准答案排序后对比;
- 填空题:支持模糊匹配,可使用explode()分割用户答案后逐个对比关键词。
批改完成后更新records表的score和status字段。
来源互联网整合,作者:小编,如若转载,请注明出处:https://www.aiboce.com/ask/247171.html