在C语言怎样对数据和 *** 作的封装

在C语言怎样对数据和 *** 作的封装,第1张

以下仅为个人理解:

数据封装就是使用类似结构体的形式,将多个相关数据合并到一个结构体中,在程序中作为一个整体进行付值和调用 *** 作。

*** 作封装就是对多个重复使用且具有相同功能的语句进行整合,打包成一个实现固定功能的函数

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)

}

}


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/sjk/9943118.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-03
下一篇 2023-05-03

发表评论

登录后才能评论

评论列表(0条)

保存