回调函数
所谓的回调函数是指:按照一定的形式由开发人员定义并编写实现内容。使用回调函数,实际上就是在调用某个函数(通常是API函数)时,将自己的一个函数(也就是回调函数)的地址作为参数传递给那个函数。而那个函数在需要的时候,也就是某种事情发生的时候,利用传递的函数地址调用回调函数,这时开发人员可以利用这个机会在回调函数中处理消息或完成一定的操作。回调函数只能是全局函数,或者是静态函数,因为这个函数只是在类中使用,所以为了维护类的完整性,我们用类的静态成员函数来做回调函数。
使用回调要完成以下工作:
1、获取函数地址:
获取函数地址很简单:只要使用函数名即可。比如test()是一个函数,则test就是该函数的地址。要将函数作为参数传递,只需要传递函数名即可。
2、声明一个函数指针:
声明一个函数指针,必须指定指针指向的函数类型,也就是声明应指定函数的返回类型以及函数的参数列表。如:
double cam(int);//函数原型
double (*fp)(int);//指针类型声明
因为cam为函数,所以(*fp)也为函数,所以,fp为函数指针
注:要声明指向特定类型的函数的指针,可以首先编写这种函数的原型,然后用(*fp)替换函数名,这样fp就是这类函数的指针。注意(*fp)必须用括号括起来。因为:
*fp(int)==>fp是一个返回指针的函数
(*fp)(int)==>fp是一个指向函数的指针
声明之后便可将相应函数的地址赋给它。如下所示:cam()的参数及返回类型必须与fp相同。
double cam(int);
double (*fp)(int);
fp=cam;
调用函数原型:
void test(int n,double(*fp)(int));第二个参数是一个函数指针,它指向的函数接受一个int参数,并返回一个double值。要让test使用cam()函数,则需将cam()的地址传给它,即:test(3,cam);
3、使用函数指针来调用函数
double cam(int);
double (*fp)(int);
fp=cam;
double x=cam(4);//使用函数名调用cam()
double y=(*fp)(5);//使用指针fp调用cam()
c++也允许像使用函数名那样使用fp:double y=fp(5);不过第一种方式更能显示出正在使用函数指针。
回调有两个主要缺点:1> 它们不是类型安全的。我们从来都不能确定处理函数使用了正确的参数来调用回调。2> 回调和处理函数是非常强有力的联系在一起的,因为处理函数必须要知道调用哪个回调。
c语言声明的优先级规则
规则:
A:声明从它的名字开始读取,然后按照优先级顺序依次读取;
B:优先级从高到低依次是:
B1、声明中被括号括起来的那部分;
B2、后缀操作符:括号()表示这是一个函数,而[]表示这是一个数组
B3、前缀操作符:*表示“指向...的指针”
C:如果const或volatile关键字的后面紧跟类型说明符(如int ,long等),那么它作用于类型说明符,在其他情况下,const或volatile关键字作用于它左边紧邻的指针星号。
分析:char *const *(*next)();
用优先级规则分析:
A:首先,看变量名“next”,并且它直接被括号括住;
B1:所以先把括号里面的东西作为一个整体,得出“next是一个指向...的指针”;
B:然后考虑括号外面的东西,在星号前缀与括号后缀之间作出选择;
B2:B2规则告诉我们优先级高的是右边的函数括号,所以得出“next是一个函数指针,指向一个返回...的函数”;
B3:然后处理前缀*,得出指针所指内容;
C:最后,把char *const解释为指向字符的常量指针。
即:next是一个指针,它指向一个函数,该函数返回另一个指针,该指针指向一个类型为char的常量指针。
Qt对sql数据库的访问
1、Qt连接数据库:
1). QT编译时已经编译了QtSql
2). 编译了ODBC插件。可以通过 configure -plugin-sql-odbc来保证,也可以单独编译~\src\plugins\sqldrivers\odbc
qmake -t vclib odbc.pro
qmake
nmake
编译后,在~\plugins\sqldrivers\下应该有qsqlodbcd4.dll(debug)或qsqlodbc4.dll
此时,可以用下面的程序,测试一下你的QT目前支持哪些数据库访问。
#include <QApplication> #include <QSqlDatabase> #include <QStringList> #include <QDebug> int main(int argc, char* argv[]) { QApplication app(argc, argv); qDebug() << "Available drivers:"; QStringList drivers = QSqlDatabase::drivers(); foreach(QString driver, drivers) qDebug() << "\t" << driver; QSqlDatabase db = QSqlDatabase::addDatabase("QODBC"); qDebug() << "ODBC driver valid?" << db.isValid(); }
如果输出中有
Available drivers: "QSQLITE" "QODBC4" "QODBC" ODBC driver valid? true
就说明已经可以成功支持ODBC了。
3). 要连接数据库,有3种方式:
要注意的就是连接数据库时使用的数据库名,和sqlite等是不同的,并不是直接写入数据库名称,而是DSN名。
如果你已经设置好了DSN,可以直接输入DSN名。 如果没有,可以采用DSN连接字符串直接连接ODBC数据库。
例如:
//下面例子连接到192.168.2.11上的sql server名为temp的数据库上。 QSqlDatabase db = QSqlDatabase::addDatabase("QODBC",); db.setHostName("192.168.2.11"); //如果dsn中已经含有SERVER,可以省略此句 QString dsn = QString("DRIVER={SQL SERVER};")+QString("SERVER=%1%2").arg(strHost).arg(";")+ QString("DATABASE=%1").arg(strDb); db.setDatabaseName(dsn); //即使dsn中已经设置了UID和PASSWD,仍然需要执行setUserName和setPassword的步骤 db.setUserName(strUser); db.setPassword(strPwd); if(!db.open()) { QMessageBox::critical(0, QObject::tr("Database Error"),db.lastError().text()); return false; }
如果dsn名设错,将会在db.open()时出现“[Microsoft][ODBC Driver Manager] Data source name not found and no default driver specified QODBC3: Unable to connect”
“[Microsoft][ODBC 驱动程序管理器] 未发现数据源名称并且未指定默认驱动程序 QODBC3: Unable to connect”错误。
其他类型数据库应该修改dsn字符串与其相适应:
下面是ODBC和OLEDB的连接字符串写法:
ODBC连接
适合数据库类型 连接方式
access "Driver={microsoft access driver(*.mdb)};dbq=*.mdb;uid=admin;pwd=pass;" dBase "Driver={microsoft dbase driver(*.dbf)};driverid=277;dbq=------------;" oracle "Driver={microsoft odbc for oracle};server=oraclesever.world;uid=admin;pwd=pass;" MSSQL server "Driver={sql server};server=servername;database=dbname;uid=sa;pwd=pass;" MS text "Driver={microsoft text driver(*.txt; *.csv)};dbq=-----;extensions=asc,csv,tab,txt;Persist SecurityInfo=false;" Visual Foxpro "Driver={microsoft Visual Foxpro driver};sourcetype=DBC;sourceDB=*.dbc;Exclusive=No;" MySQL "Driver={mysql};database=yourdatabase;uid=username;pwd=yourpassword;option=16386;" SQLite "Driver={SQLite3 ODBC Driver};Database=D:\SQLite\*.db" PostgreSQL "Driver={PostgreSQL ANSI};server=127.0.0.1;uid=admin;pwd=pass;database=databaseName"
OLEDB连接
适合的数据库类型 连接方式
access "Provider=microsoft.jet.oledb.4.0;data source=your_database_path;user id=admin;password=pass;" oracle "Provider=OraOLEDB.Oracle;data source=dbname;user id=admin;password=pass;" MS SQL Server "Provider=SQLOLEDB;data source=machinename;initial catalog=dbname;userid=sa;password=pass;" MS text "Provider=microsof.jet.oledb.4.0;data source=your_path;Extended Properties'text;FMT=Delimited'"
2、写入数据库
query.prepare("INSERT INTO test_c (sn,address,station,stime)" "VALUES (?,?,?,?)");//此种写法为ODBC风格 query.addBindValue(strSN); query.addBindValue(strName); query.addBindValue("zc"); query.addBindValue(stime); query.exec(); if (!query.execBatch())qDebug() << query.lastError();//此处给出出错信息。比如写入数据库失败时,就会在控制台打印出错误原因
3、将sql导入到access中
1)、从sql中读取数据
QSqlQuery query; QStringList m_imei; QStringList m_barcode; QStringList m_updata_data; if (rec==tr("全部导出")) { query.exec("SELECT * FROM test"); } else { query.exec("SELECT * FROM test WHERE (LAST_UPDATE_DATE > '"+num1+"') AND (LAST_UPDATE_DATE <='"+num2+"')"); } while (query.next()) { m_imei.append(query.value(1).toString()); m_barcode.append(query.value(2).toString()); m_uptate_date.append(query.value(3).toString().replace("T"," "));
2)、将读出的数据写入到access中
QSqlQuery query; for (int i=0;i<m_uptate_date.size();i++) { query.value(1)=m_imei.at(i); query.value(2)=m_barcode.at(i); query.value(3)=m_uptate_date.at(i); query.exec("INSERT INTO exp(LAST_UPDATE_DATE,IMEI,BARCODE)" "VALUES('"+m_uptate_date.at(i)+"','"+m_imei.at(i)+"','"+m_barcode.at(i)+"')"); }
4、在数据库中查询
void mainwindow::CheckDb() { QSqlQuery query; QString name; query.exec("SELECT ziduan FROM biaoming "); while (query.next()) { name = query.value(0).toString(); qDebug()<<name; } if (name=="aaa") { ...... } }
函数和数组
以前的,先贴出来,嘿嘿。
函数是处理复杂数据类型的关键,比如数组或者结构。编写特定函数处理特定的数组操作是有好处的,可靠性高,修改和调试方便,另外将存储属性与操作结合起来也是像oop思想迈进了一步。
首先函数需要知道对哪个数组进行操作,所以要把数组名作为参数传递给函数。为了使函数通用,不限定特定长度的数组,因此需要传递数组的长度。如:int sum(int arr[],int n);方括号为空表示可以把任意长度的数组传递给函数,但是arr实际上并不是数组,而是指针,但是在编写函数体的其余部分可将arr当做数组来处理。在c++中,当且仅当用于函数头或者函数原型时int *arr与int arr[]的含义才是相同的。都意味着arr是一个int型指针,因此不能在函数体中使用int p[]来声明指针。为将数组类型与元素数量告诉函数,需要两个不同的参数来传递,如:void fillArray(int arr[],int size);而不要试图通过方括号表示法来传递数组长度。如:
void fillArray(int arr[size]);编译器忽略为任何数组形参指定的长度。
#include <iostream> const int ArSize = 8; int sum_arr(int arr[], int n); // 原型 int main() { using namespace std; int cookies[ArSize] = {1,2,4,8,16,32,64,128}; int sum = sum_arr(cookies, ArSize);//调用 cout << "Total cookies eaten: " << sum << "\n"; return 0; } int sum_arr(int arr[], int n) { int total = 0; for (int i = 0; i < n; i++) total = total + arr[i]; return total; }
函数调用sum_arr(cookies,ArSize)将cookies数组第一个元素的地址和元素个数传递给sum_arr()函数,sum_arr()函数将cookies的地址赋给指针变量arr,将ArSize赋给int变量n。程序实际上并没有将数组内容传递给函数,而是将数组位置,元素类型及元素数目传递给函数。传递常规变量时,函数使用的是该变量的拷贝,但是传递数组时,函数将使用原来的数组。将数组地址作为参数可以节省复制整个数组所需的时间和内存。
c++将数组名解释为第一个元素的首地址,两种例外是:第一,数组声明使用数组名来标记存储位置;第二是对数组名使用sizeof将得到的是整个数组的长度。
函数和二维数组:
data[3][4]={{1,1,1,1},{2,2,2,2},{3,3,3,3}};
int total=sum(data,3);
data是一个数组名,该数组有3个元素,每一个元素本身又是一个数组,由4个int值组成。因此data的类型是指向由4个int组成的数组的指针。因此,原型如下:
int sum(int (*arr)[4],int size);其中的括号是不可缺的,因为int *arr[4]将声明一个由4个指向int的指针组成的数组,而不是由一个指向由4个int组成的数组的指针。另一种格式的原型:
int sum(int arr[][4],int size);//指向由4个int组成的数组。指针类型指定了列数
因此sum()函数只能接受由4列组成的数组,但长度变量size指定了行数。