1
完善资料让更多小伙伴认识你,还能领取20积分哦, 立即完善>
ODBC(Open Database Connectivity, 开 放 式 数 据 库 连 接), 是 一 种 用 来 在 相 关 或 不 相 关 的 数 据 库 管 理 系 统(DBMS)(Database Manage System) 中 存 取 数 据 的 标 准 应 用 程 序 接 口(API)。
Visual C++ 的MFC 基 类 库 定 义 了 几 个 数 据 库 类。 在 利 用ODBC 编 程 时, 经 常 要 使 用 到 CDatabase( 数 据 库 类),CRecordSet( 记 录 集 类) 和CRecordView( 可 视 记 录 集 类)。 CDatabase 类 对 象 提 供 了 对 数 据 源 的 连 接, 通 过 它 你 可 以 对 数 据 源 进 行 操 作。 CRecordSet 类 对 象 提 供 了 从 数 据 源 中 提 取 出 的 记 录 集。CRecordSet 对 象 通 常 用 于 两 种 形 式: 动 态 行 集(dynasets) 和 快 照 集(snapshots)。 动 态 行 集 能 保 持 与 其 他 用 户 所 做 的 更 改 保 持 同 步。 快 照 集 则 是 数 据 的 一 个 静 态 视 图。 每 一 种 形 式 在 记 录 集 被 打 开 时 都 提 供 一 组 记 录, 所 不 同 的 是, 当 你 在 一 个 动 态 行 集 里 滚 动 到 一 条 记 录 时, 由 其 他 用 户 或 是 你 应 用 程 序 中 的 其 他 记 录 集 对 该 记 录 所 做 的 更 改 会 相 应 地 显 示 出 来。 CRecordView 类 对 象 能 以 控 制 的 形 式 显 示 数 据 库 记 录。 这 个 视 图 是 直 接 连 到 一 个CRecordSet 对 象 的 表 视 图。 应 用Visual C++ 的AppWizard 可 以 自 动 生 成 一 个ODBC 应 用 程 序 框 架。 方 法 是: 打 开File 菜 单 的New 选 项, 选 取Projects, 填 入 工 程 名, 选 择MFC AppWizard (exe), 然 后 按AppWizard 的 提 示 进 行 操 作。 当AppWizard 询 问 是 否 包 含 数 据 库 支 持 时, 如 果 你 想 读 写 数 据 库, 那 么 选 定Database view with file support; 而 如 果 你 想 访 问 数 据 库 的 信 息 而 不 想 回 写 所 做 的 改 变, 那 么 选 定Database view without file support 选 项 就 比 较 合 适 了。 选 择 了 数 据 库 支 持 之 后Database Source 按 钮 会 激 活, 选 中 它 去 调 用Data Options 对 话 框。 在Database Options 对 话 框 中 会 显 示 已 向ODBC 注 册 的 数 据 库 资 源, 选 定 你 所 要 操 作 的 数 据 库, 如:Super_ES, 单 击OK 后 会 出 现Select Database Tables 对 话 框, 其 中 列 举 了 你 所 选 中 的 数 据 库 中 包 含 的 全 部 表, 选 择 你 希 望 操 作 的 表 后, 单 击OK。 在 选 定 了 数 据 库 和 数 据 表 之 后, 你 可 以 按 照 惯 例 继 续 进 行AppWizard 操 作。 特 别 需 要 指 出 的 是: 在 生 成 的 应 用 程 序 框 架View 类( 如:CMyXXView) 中 包 含 一 个 指 向CMyXXSet 对 象 的 指 针m_pSet(建立时编译器自动为你生成了), 该 指 针 由AppWizard 建 立, 目 的 是 在 视 表 单 和 记 录 集 之 间 建 立 联 系, 使 得 记 录 集 中 的 查 询 结 果 能 够 很 容 易 地 在 视 表 单 上 显 示 出 来。 有 关m_pSet 的 详 细 用 法 可 以 参 见Visual C++ Online Book。 程 序 与 数 据 语 言 建 立 联 系, 使 用CDatebase::OpenEx() 或CDatabase::Open() 函 数 来 进 行 初 始 化。 数 据 库 对 象 必 须 在 你 使 用 它 构 造 一 个 记 录 集 对 象 之 前 被 初 始 化。 ---- 1 . 查 询 记 录 ---- 查 询 记 录 使 用CRecordSet::Open() 和CRecordSet::Requery() 成 员 函 数。 在 使 用CRecordSet 类 对 象 之 前, 必 须 使 用CRecordSet::Open() 函 数 来 获 得 有 效 的 记 录 集。 一 旦 已 经 使 用 过CRecordSet::Open() 函 数, 再 次 查 询 时 就 可 以 应 用CRecordSet::Requery() 函 数。 在 调 用CRecordSet::Open() 函 数 时, 如 果 已 经 将 一 个 已 经 打 开 的CDatabase 对 象 指 针 传 给CRecordSet 类 对 象 的m_pDatabase 成 员 变 量, 则 使 用 该 数 据 库 对 象 建 立ODBC 连 接; 否 则 如 果m_pDatabase 为 空 指 针, 就 新 建 一 个CDatabase 类 对 象 并 使 其 与 缺 省 的 数 据 源 相 连, 然 后 进 行CRecordSet 类 对 象 的 初 始 化。 缺 省 数 据 源 由GetDefaultConnect() 函 数 获 得。 你 也 可 以 提 供 你 所 需 要 的SQL 语 句, 并 以 它 来 调 用CRecordSet::Open() 函 数, 例 如: MyXXSet.Open(AFX_DATABASE_USE_DEFAULT,strSQL); ---- 如 果 没 有 指 定 参 数, 程 序 则 使 用 缺 省 的SQL 语 句, 即 对 在GetDefaultSQL() 函 数 中 指 定 的SQL 语 句 进 行 操 作: CString CSuper_ESSet::GetDefaultSQL() {return _T("[BasicData],[MainSize]");} ---- 对 于GetDefaultSQL() 函 数 返 回 的 表 名, 对 应 的 缺 省 操 作 是SELECT 语 句, 即: SELECT * FROM BasicData,MainSize ---- 查 询 过 程 中 也 可 以 利 用CRecordSet 的 成 员 变 量m_strFilter 和m_strSort 来 执 行 条 件 查 询 和 结 果 排 序。m_strFilter 为 过 滤 字 符 串, 存 放 着SQL 语 句 中WHERE 后 的 条 件 串;m_strSort 为 排 序 字 符 串, 存 放 着SQL 语 句 中ORDER BY 后 的 字 符 串。 如: Super_ESSet.m_strFilter="TYPE='电动机'"; Super_ESSet.m_strSor; Super_ESSet.Requery(); 对应的SQL语句为: SELECT * FROM BasicData,MainSize WHERE TYPE='电动机' ORDER BY VOLTAGE ---- 除 了 直 接 赋 值 给m_strFilter 以 外, 还 可 以 使 用 参 数 化。 利 用 参 数 化 可 以 更 直 观, 更 方 便 地 完 成 条 件 查 询 任 务。 使 用 参 数 化 的 步 骤 如 下: ---- (1) . 声 明 参 变 量: CString p1; float p2; ---- (2) . 在 构 造 函 数 中 初 始 化 参 变 量 p1=_T(""); p2=0.0f; m_nParams=2; ---- (3) . 将 参 变 量 与 对 应 列 绑 定 pFX- >SetFieldType(CFieldExchange::param) RFX_Text(pFX,_T("P1"),p1); RFX_Single(pFX,_T("P2"),p2); ---- 完 成 以 上 步 骤 之 后 就 可 以 利 用 参 变 量 进 行 条 件 查 询 了: m_pSet- >m_strFilter="TYPE=? AND VOLTAGE=?"; m_pSet- >pfont-family: 宋体; mso-ascii-font-family: 'Times New Roman'; mso-hansi-font-family: 'Times New Roman';">电 动 机"; m_pSet- >p2=60.0; m_pSet- >Requery(); ---- 参 变 量 的 值 按 绑 定 的 顺 序 替 换 查 询 字 串 中 的"?" 适 配 符。 ---- 如 果 查 询 的 结 果 是 多 条 记 录 的 话, 可 以 用CRecordSet 类 的 函 数Move(),MoveNext(),MovePrev(),MoveFirst() 和MoveLast() 来 移 动 光 标。 ---- 2 . 增 加 记 录 ---- 增 加 记 录 使 用AddNew() 函 数, 要 求 数 据 库 必 须 是 以 允 许 增 加 的 方 式 打 开: m_pSet- >AddNew(); //在表的末尾增加新记录 m_pSet- >SetFieldNull(&(m_pSet- >m_type), FALSE); m_pSet- >m_typfont-family: 宋体; mso-ascii-font-family: 'Times New Roman'; mso-hansi-font-family: 'Times New Roman';">电 动 机"; ... //输入新的字段值 m_pSet- > Update(); //将新记录存入数据库 m_pSet- >Requery(); //重建记录集 ---- 3 . 删 除 记 录 ---- 直 接 使 用Delete() 函 数, 并 且 在 调 用Delete() 函 数 之 后 不 需 调 用Update() 函 数: m_pSet- >Delete(); if (!m_pSet- >IsEOF()) m_pSet- >MoveNext(); else m_pSet- >MoveLast(); ---- 4 . 修 改 记 录 ---- 修 改 记 录 使 用Edit() 函 数: m_pSet- >Edit(); //修改当前记录 m_pSet- >m_typcolor: red; font-family: 宋体; mso-ascii-font-family: 'Times New Roman'; mso-hansi-font-family: 'Times New Roman';">发电机"; //修改当前记录字段值 ... m_pSet- >Update(); //将修改结果存入数据库 m_pSet- >Requery(); ---- 5 . 撤 消 操 作 ---- 如 果 用 户 选 择 了 增 加 或 者 修 改 记 录 后 希 望 放 弃 当 前 操 作, 可 以 在 调 用Update() 函 数 之 前 调 用: CRecordSet::Move(AFX_MOVE_REFRESH); ---- 来 撤 消 增 加 或 修 改 模 式, 并 恢 复 在 增 加 或 修 改 模 式 之 前 的 当 前 记 录。 其 中 的 参 数AFX_MOVE_REFRESH 的 值 为 零。 ---- 6 . 数 据 库 连 接 的 复 用 ---- 在CRecordSet 类 中 定 义 了 一 个 成 员 变 量m_pDatabase: CDatabase* m_pDatabase; ---- 它 是 指 向 对 象 数 据 库 类 的 指 针。 如 果 在CRecordSet 类 对 象 调 用Open() 函 数 之 前, 将 一 个 已 经 打 开 的CDatabase 类 对 象 指 针 传 给m_pDatabase, 就 能 共 享 相 同 的CDatabase 类 对 象。 如: CDatabase m_db; CRecordSet m_set1,m_set2; m_db.Open(_T("Super_ES"));//建立ODBC连接 m_set1.m_pDatabase=&m_db; //m_set1复用m_db对象 m_set2.m_pDatabse=&m_db; // m_set2复用m_db对象 ---- 7 .SQL 语 句 的 直 接 执 行 ---- 虽 然 通 过CRecordSet 类, 我 们 可 以 完 成 大 多 数 的 查 询 操 作, 而 且 在CRecordSet::Open() 函 数 中 也 可 以 提 供SQL 语 句, 但 是 有 的 时 候 我 们 还 想 进 行 一 些 其 他 操 作, 例 如 建 立 新 表, 删 除 表, 建 立 新 的 字 段 等 等, 这 时 就 需 要 使 用 到CDatabase 类 的 直 接 执 行SQL 语 句 的 机 制。 通 过 调 用CDatabase::ExecuteSQL() 函 数 来 完 成SQL 语 句 的 直 接 执 行: BOOL CDB::ExecuteSQLAndReportFailure(const CString& strSQL) { TRY { m_pdb- >ExecuteSQL(strSQL);//直接执行SQL语句 } CATCH (CDBException,e) { CString strMsg; strMsg.LoadString(IDS_EXECUTE_SQL_FAILED); strMsg+=strSQL; return FALSE; } END_CATCH return TRUE; } ---- 应 当 指 出 的 是, 由 于 不 同DBMS 提 供 的 数 据 操 作 语 句 不 尽 相 同, 直 接 执 行SQL 语 句 可 能 会 破 坏 软 件 的DBMS 无 关 性, 因 此 在 应 用 中 应 当 慎 用 此 类 操 作。 ---- 8 . 动 态 连 接 表 ---- 表 的 动 态 连 接 可 以 利 用 在 调 用CRecordSet::Open() 函 数 时 指 定SQL 语 句 来 实 现。 同 一 个 记 录 集 对 象 只 能 访 问 具 有 相 同 结 构 的 表, 否 则 查 询 结 果 将 无 法 与 变 量 相 对 应。 void CDB::ChangeTable() { if (m_pSet- >IsOpen()) m_pSet- >Close(); switch (m_id) { case 0: m_pSet- >Open(AFX_DB_USE_DEFAULT_TYPE, "SELECT * FROM SLOT0"); //连接表SLOT0 m_id=1; break; case 1: m_pSet- >Open(AFX_DB_USE_DEFAULT_TYPE, "SELECT * FROM SLOT1"); //连接表SLOT1 m_id=0; break; } } ---- 9 . 动 态 连 接 数 据 库 ---- 由 于 与 数 据 库 的 连 接 是 通 过CDatabase 类 对 象 来 实 现 的, 所 以 我 们 可 以 通 过 赋 与CRecordSet 类 对 象 参 数m_pDatabase 以 连 接 不 同 数 据 库 的CDatabase 对 象 指 针, 就 可 以 动 态 连 接 数 据 库。 void CDB::ChangeConnect() { CDatabase* pdb=m_pSet- >m_pDatabase; pdb- >Close(); switch (m_id) { case 0: if (!pdb- >Open(_T("Super_ES"))) //连接数据源Super_ES { AfxMessageBox("数据源Super_ES打开失败," "请检查相应的ODBC连接", MB_OK|MB_ICONWARNING); exit(0); } m_id=1; break; case 1: if (!pdb- >Open(_T("Motor"))) //连接数据源Motor { AfxMessageBox("数据源Motor打开失败," "请检查相应的ODBC连接", MB_OK|MB_ICONWARNING); exit(0); } m_id=0; break; } } |
|
|
|
只有小组成员才能发言,加入小组>>
2445 浏览 0 评论
9172 浏览 4 评论
36850 浏览 19 评论
5042 浏览 0 评论
24837 浏览 34 评论
1557浏览 2评论
1792浏览 1评论
2232浏览 1评论
1585浏览 0评论
558浏览 0评论
小黑屋| 手机版| Archiver| 德赢Vwin官网 ( 湘ICP备2023018690号 )
GMT+8, 2024-12-31 02:03 , Processed in 1.098912 second(s), Total 46, Slave 40 queries .
Powered by 德赢Vwin官网 网
© 2015 bbs.elecfans.com
关注我们的微信
下载发烧友APP
德赢Vwin官网 观察
版权所有 © 湖南华秋数字科技有限公司
德赢Vwin官网 (电路图) 湘公网安备 43011202000918 号 电信与信息服务业务经营许可证:合字B2-20210191 工商网监 湘ICP备2023018690号