数据封装就是使用类似结构体的形式,将多个相关数据合并到一个结构体中,在程序中作为一个整体进行付值和调用 *** 作。
*** 作封装就是对多个重复使用且具有相同功能的语句进行整合,打包成一个实现固定功能的函数。
C一般不能直接调用C++函数库,需要将C++库封装成C接口后,才可以使用C调用。下面举例,说明一个封装策略:
//code in add.cxx
#include "add.h"
int sample::method()
{
cout<<"method is called!\n"
}
//code in add.h
#include
using namespace std
class sample
{
public:
int method()
}
将上面的两个文件生成动态库libadd.so放到 /usr/lib目录下,编译命令如下:
sudo g++ -fpic -shared -g -o /usr/lib/libadd.so add.cxx -I ./
由于在C中不能识别类,所以要将上面类的成员函数,要封装成C接口函数才能被调用。下面进行封装,将输出接口转换成C接口。
//code in mylib.cxx
#include "add.h"
#ifndef _cplusplus
#define _cplusplus
#include "mylib.h"
#endif
int myfunc()
{
sample ss
ss.method()
return 0
}
//code in mylib.h
#ifdef _cplusplus
extern "C"
{
#endif
int myfunc()
#ifdef _cplusplus
}
#endif
在linux下,gcc编译器并没用变量_cplusplus来区分是C代码还是C++ 代码(没有宏定义),如果使用gcc编译器,这里我们可以自己定义一个变量_cplusplus用于区分C和C++代码,所以在mylib.cxx中定义 了一个变量_cplusplus用于识别是否需要“extern "C"”将函数接口封装成C接口。但是如果使用g++编译器则不需要专门定义_cplusplus,编译命令如下:
g++ -fpic -shared -g -o mylib.so mylib.cxx -la -I ./
main.c
#include
#include
#include "mylib.h"
int
main()
{
int (*dlfunc)()
void *handle//定义一个句柄
handle = dlopen("./mylib.so", RTLD_LAZY)//获得库句柄
dlfunc = dlsym(handle, "myfunc")//获得函数入口
(*dlfunc)()
dlclose(handle)
return 0
}
编译命令如下:
gcc -o main main.c ./mylib.so -ldl
下面就可以执行了。
需要说明的是,由于main.c 和 mylib.cxx都需要包含mylib.h,并且要将函数myfunc封装成C接口函数输出需要“extern "C"”,而C又不识别“extern "C"”,所以需要定义_cplusplus来区别处理mylib.h中的函数myfunc。
在main.c的main函数中直接调用myfunc()函数也能执行,这里介绍的是常规调用库函数的方法。
可以参考一下这个:使用CDatabase类来读取Microsoft
Access数据库,主要实现了以下功能:
从Microsoft
Access
数据库读取数据
不使用
ODBC数据源
进行数据库连接
在List
view控件中显示数据
//
以下是部分
代码
void
CReadDBDlg::OnRead()
{
//
TODO:
Add
your
control
notification
handler
code
here
CDatabase
database
CString
sSql
CString
sCatID,
sCategory
CString
sDriver
=
"MICROSOFT
ACCESS
DRIVER
(*.mdb)"
CString
sDsn
CString
sFile
=
"c:\\works\\ReadDB\\Test.mdb"
//
Change
path
here
int
iRecord
=
0
//
Create
a
pseudo
DSN
including
the
name
of
the
Driver
and
the
Excel
file
//
so
we
donhave
to
have
an
explicit
DSN
installed
in
our
ODBC
admin
sDsn.Format("ODBCDRIVER={%s}DSN=''DBQ=%s",sDriver,sFile)
TRY
{
//
Open
the
database
database.Open(NULL,false
,false
,sDsn)
//
the
recordset
CRecordset
recset(
&database
)
//
Build
the
SQL
query
string
sSql
=
"SELECT
CatID,
Category
"
"FROM
Categories"
//
Execute
the
query)
recset.Open(CRecordset::forwardOnly,sSql,CRecordset::readOnly)
ResetListControl()
//
Column
heading
m_ListControl.InsertColumn(1,"Cat
ID",LVCFMT_LEFT,30,0)
m_ListControl.InsertColumn(1,"Category",LVCFMT_LEFT,30,1)
//
Browse
the
result
while
(
!recset.IsEOF()
)
{
//
Read
one
record
recset.GetFieldValue("CatID",sCatID)
recset.GetFieldValue("Category",sCategory)
//
Insert
values
into
the
list
control
m_ListControl.InsertItem(0,sCatID,0)
m_ListControl.SetItemText(0,1,sCategory)
//
goto
next
record
recset.MoveNext()
}
//
Close
the
database
database.Close()
}
CATCH(CDBException,
e)
{
//
If
a
database
exception
occured,
pop
up
error
msg
AfxMessageBox("Database
error:
"+e->m_strError)
}
END_CATCH
}
void
CReadDBDlg::ResetListControl()
{
m_ListControl.DeleteAllItems()
int
iNbrOfColumns
CHeaderCtrl*
pHeader
=
(CHeaderCtrl*)m_ListControl.GetDlgItem(0)
if
(pHeader)
{
iNbrOfColumns
=
pHeader->GetItemCount()
}
for
(int
i
=
iNbrOfColumns
i
>=
0
i--)
{
m_ListControl.
DeleteColumn
(i)
}
}
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)