当前位置: 首页 > news >正文

百度站长工具网站网络营销sem培训

百度站长工具网站,网络营销sem培训,源码如何做网站,中国建设银行网站进不去文章目录 sqlite - sqlite3_exec - c回调函数的处理概述笔记回调赋值实现用到的数据结构回调分发函数的实现具体的回调处理sqlite3_exe执行完后, 行集的具体处理END sqlite - sqlite3_exec - c回调函数的处理 概述 以前给客户写了个小程序, 处理sqlite执行sql时, 给定回调, 等…

文章目录

    • sqlite - sqlite3_exec - c++回调函数的处理
    • 概述
    • 笔记
    • 回调赋值实现
    • 用到的数据结构
    • 回调分发函数的实现
    • 具体的回调处理
    • sqlite3_exe执行完后, 行集的具体处理
    • END

sqlite - sqlite3_exec - c++回调函数的处理

概述

以前给客户写了个小程序, 处理sqlite执行sql时, 给定回调, 等sql执行完, 处理返回的行集.
那时候, 时间紧, 回调整的不太好, 只是能用而已.
这次给自己写账单分析程序, 关于用回调处理行集的部分, 想到了比以前好的方法.
可以只使用一个类静态回调函数(回调分发函数), 然后通过参数, 可以将具体干活的普通类成员函数, 赋值给回调参数. 程序简洁实用很多.
试过了, 好使.

笔记

demo就不单独写了, 从现有工程中, 摘取程序片段来说明.
关于回调这块, 因为有语法的细节, 即使知道这回事, 如果重新写一个新回调, 也会试错好多次.
让自己以后看, 能明白就行了.

bool COrderProcBase::db_proc_excel_to_db()
{bool b_rc = false;int ArySize = 0;CStringArray strAry;int i = 0;int id = 0;CString strOrgFile;CString strCpyEnFile;CFileOperation file_opt;CString cs_drive;CString cs_dir;CString cs_name;CString cs_ext;CString csCsvToExcel;CString csTmp;do {// 建立数据库中的表// !! 在建立表之前, 要先检查表是否存在, 如果不存在, 才去建立表// 否则在表存在的情况下, 再建立表, 就会返回错误, 错误原因是表已经存在// 检查表是否存在是一段sql, 返回的行集是2行(列名称, 列值),1列if (!is_db_tbl_exist()) // 这里要执行sql, 就会返回一个行集, 就用到了回调.{if (!db_create_tbl()){break;}}
bool COrderProcJH::is_db_tbl_exist()
{return COrderProcBase::is_db_tbl_exist(_T("TBL_FILE_JH"));
}

回调赋值实现

bool COrderProcBase::is_db_tbl_exist(TCHAR* pszTblName)
{// 表在数据中是否存在?bool b_rc = false;string str_utf8_sql;string str_sql;char* zErrMsg = NULL;	//错误信息const char* sql = NULL;int rc = 0;CString csMsg;char szBuf[4096];TAG_PARAM_CALLBACK_FOR_SQLITE3_EXEC param;do {_ASSERT(NULL != pszTblName);if (NULL == pszTblName){break;}memset(szBuf, 0, sizeof(szBuf));sprintf(szBuf, "SELECT EXISTS(SELECT name FROM sqlite_schema WHERE type='table' AND name='%s');", getLogicProxy()->my_W2A(pszTblName).c_str());str_sql = szBuf;str_utf8_sql = getLogicProxy()->UnicodeToUTF8(getLogicProxy()->my_A2W(str_sql));param.pThis = this;param.pFn = &COrderProcBase::callback_for_sqlite3_exec_proc_tbl_exist;param.row_set.clear();// 回调的赋值 为参数3, 参数4// 这样填写回调, 回调分发函数(类静态成员函数)只有一个, 不用变.// 具体的回调处理函数(类普通成员函数), 赋值在参数中. // 如果具体的回调处理函数不是这个类的this, 也可以在参数中指定. 这样就可以用任何类实例的任何普通成员函数来处理回调了, 很方便.// 这个参数中要赋值的类实例指针, 可以是一个基类的指针, 让处理回调的类都是这个基类的子类, 有基类中规定的处理回调的纯虚接口就好.rc = sqlite3_exec(get_db_handle(), str_utf8_sql.c_str(), COrderProcBase::callback_for_sqlite3_exec_dispatch, &param, &zErrMsg);if (SQLITE_OK != rc) {csMsg.Format(_T("错误 : %s"), getLogicProxy()->UTF8ToUnicode(zErrMsg).c_str());getLogicProxy()->addTips(csMsg);break;}// 等sql执行完, 只要执行的对, 就可以用另外一个函数来处理返回的结果集// 针对不同类型的sql, 实现不同的结果集处理函数b_rc = procResultSet_isTblExist(param.row_set, pszTblName);} while (false);if (NULL != zErrMsg) {sqlite3_free(zErrMsg);zErrMsg = NULL;}return b_rc;
}

用到的数据结构

// 类成员函数指针类型的定义
class COrderProcBase;
// PFN_CALLBACK_FOR_SQLITE3_EXEC是sqlite3_exec回调要求的函数样式
typedef int (COrderProcBase::* PFN_CALLBACK_FOR_SQLITE3_EXEC)(void* data, int argc, char** argv, char** azColName);
class COrderProcBase
{
public:
// ...// 这个结构在类中直接定义的typedef struct _tag_param_callback_for_sqlite3_exec{COrderProcBase* pThis; // 本类的指针, 用来从回调分发函数(类静态成员函数) 执行到类的普通成员函数PFN_CALLBACK_FOR_SQLITE3_EXEC pFn; // 本类中实际的回调处理函数指针(类的普通成员函数)C_my_row_set row_set; // 执行sql后, 返回的行集 这里是装在返回行集的类}TAG_PARAM_CALLBACK_FOR_SQLITE3_EXEC;

回调分发函数的实现

int  COrderProcBase::callback_for_sqlite3_exec_dispatch(void* data, int argc, char** argv, char** azColName)
{int i_rc = 0;TAG_PARAM_CALLBACK_FOR_SQLITE3_EXEC* param;do {if (NULL == data){break;}param = (TAG_PARAM_CALLBACK_FOR_SQLITE3_EXEC*)data;if ((NULL != param->pThis) && (NULL != param->pFn)){// 这里就可以执行sqlite3_exec指定的回调函数了(任意类, 任意类的回调实现(类的普通成员函数))// 这个回调分发函数基本不用再改.i_rc = ((param->pThis)->*(param->pFn))(data, argc, argv, azColName);}} while (false);return i_rc;
}

具体的回调处理

这个根据具体要执行的sql具体处理, 最普通的就是将行集先存起来.
等sqlite3_exec完事了, 再拿存的行集来判断结果.

以判断库中表是否存在为例
如果在回调中, 只是将行集存起来, 那么下面的回调就是一个通用的实现.

int COrderProcBase::callback_for_sqlite3_exec_proc_tbl_exist(void* data, int argc, char** argv, char** azColName)
{int i_rc = 0;TAG_PARAM_CALLBACK_FOR_SQLITE3_EXEC* param;int i = 0;do {if (NULL == data){break;}param = (TAG_PARAM_CALLBACK_FOR_SQLITE3_EXEC*)data;C_my_row row;C_my_key_val key_val;for (i = 0; i < argc; i++){key_val.m_cs_key = azColName[i];key_val.m_cs_val = argv[i];row.m_list.AddTail(key_val);}// 回调进来一次, 就是一行param->row_set.m_list.AddTail(row);} while (false);return i_rc;
}

sqlite3_exe执行完后, 行集的具体处理

以判断库中表是否存在为例
这个实现应该和每种不同的sql相关, 拿回调中存好的行集做不同的判断.

bool COrderProcBase::procResultSet_isTblExist(C_my_row_set& row_set, const TCHAR* pszTblName)
{bool b_rc = false;POSITION pos;POSITION pos1;C_my_row row;C_my_row row1;C_my_key_val key_val;TCHAR szBuf[4096];bool b_stop = false;do {if (NULL == pszTblName){break;}if (1 != row_set.m_list.GetSize()){break;}for (pos = row_set.m_list.GetHeadPosition(); (NULL != pos); row_set.m_list.GetNext(pos)){row = row_set.m_list.GetAt(pos);if (1 != row.m_list.GetSize()){break;}for (pos1 = row.m_list.GetHeadPosition(); (NULL != pos1); row.m_list.GetNext(pos1)){b_stop = true;key_val = row.m_list.GetAt(pos1);memset(szBuf, 0, sizeof(szBuf));_stprintf(szBuf, _T("EXISTS(SELECT name FROM sqlite_schema WHERE type='table' AND name='%s')"), pszTblName);if (key_val.m_cs_key != szBuf){break;}if (key_val.m_cs_val != _T("1")){break;}b_rc = true;break;}if (b_stop){break;}}} while (false);return b_rc;
}

END

http://www.ds6.com.cn/news/22175.html

相关文章:

  • 网站建设需求确认书上海b2b网络推广外包
  • phpstudy怎么做网站百度账号注册申请
  • 公司网站建设应符合哪些法规网站免费推广网站
  • 广州网站开发招聘简述网站制作的步骤
  • WordPress影视站源码无锡seo公司哪家好
  • 深圳企业100强seo优化培训多少钱
  • 广州十大网站建设最好用的手机优化软件
  • 美国服务器网站推荐成人计算机速成培训班
  • 点个赞科技 网站制作信息流广告公司排名
  • ppt做会动彩字网站方象科技的企业愿景
  • 做网站大概多少钱关键字排名查询工具
  • 网站上写个招贤纳士怎么做搜索引擎简称seo
  • 龙岩网站建设一般多少钱重庆森林影评
  • 重庆市门户网站制作广州疫情已经达峰
  • 建网站 做淘宝客app定制开发
  • 擦边球做网站挣钱网站百度不收录的原因
  • 营销型企业网站建设应遵守的原则指数基金定投技巧
  • 湖南交通建设监理协会网站杭州百度快照推广
  • 森东网站建设天津seo推广优化
  • 国内设计品牌优化的含义是什么
  • 乌鲁木齐网站设计找哪家com网站域名注册
  • 西安建设局官方网站免费自助建站平台
  • 河南平台网站建设制作谷歌推广开户
  • 外贸网站建设广州网络营销的四大特点
  • web开发需要学什么百度快速排名优化工具
  • 做网站被攻击谁的责任sem培训学校
  • 百度做网站投广告竞价排名服务
  • 好的网站特点win7优化大师官网
  • 在谷歌上做外贸网站有用吗百度排名怎么做
  • 怎么用sharepoint做网站深圳百度推广关键词推广