c, c++ 库调用相关知识

1。 重载是如何实现的?

几个同名的重载函数仍然是不同的函数,它们是如何区分的呢?我们自然想到函数接口的两个要素:参数与返回值。

如果同名函数的参数不同(包括类型、顺序不同),那么容易区别出它们是不同的函数。

如果同名函数仅仅是返回值类型不同,有时可以区分,有时却不能。例如:

void Function(void);

int Function (void);

上述两个函数,第一个没有返回值,第二个的返回值是int类型。如果这样调用函数:

int x = Function ();

则可以判断出Function是第二个函数。问题是在C++/C程序中,我们可以忽略函数的返回值。在这种情况下,编译器和程序员都不知道哪个Function函数被调用。

所以只能靠参数而不能靠返回值类型的不同来区分重载函数。编译器根据参数为每个重载函数产生不同的内部标识符。例如编译器为示例8-1-1中的三个Eat函数产生象_eat_beef、_eat_fish、_eat_chicken之类的内部标识符(不同的编译器可能产生不同风格的内部标识符)。


如果C++程序要调用已经被编译后的C函数,该怎么办?

假设某个C函数的声明如下:

void foo(int x, int y);

该函数被C编译器编译后在库中的名字为_foo,而C++编译器则会产生像_foo_int_int之类的名字用来支持函数重载和类型安全连接。由于编译后的名字不同,C++程序不能直接调用C函数。C++提供了一个C连接交换指定符号extern“C”来解决这个问题。例如:

extern “C”

{

void foo(int x, int y);

… // 其它函数

}

或者写成

extern “C”

{

#include “myheader.h”

… // 其它C头文件

}

这就告诉C++编译译器,函数foo是个C连接,应该到库中找名字_foo而不是找_foo_int_int。C++编译器开发商已经对C标准库的头文件作了extern“C”处理,所以我们可以用#include 直接引用这些头文件。


注意并不是两个函数的名字相同就能构成重载。全局函数和类的成员函数同名不算重载,因为函数的作用域不同。例如:

void Print(…); // 全局函数

class A

{…

void Print(…); // 成员函数

}

不论两个Print函数的参数是否不同,如果类的某个成员函数要调用全局函数Print,为了与成员函数Print区别,全局函数被调用时应加‘::’标志。如

::Print(…); // 表示Print是全局函数而非成员函数
一句话,extern "C" 用到的地方:
1). c 库相关的头文件中,如 c.h 中(使 C++ 能调用 c库中函
数):
#ifdef __cplusplus
extern "C"{
#endif
...函数声明...
#ifdef __cplusplus
}
#endif
2). 有时候在 C++ 中,要调用 c 库中的函数 funcA()
extern "C"
{
funcA();
}

extern "C"
{
#include "c.h"
}

2. c库和 C++ 库的调用关系。
简单的说就是 C++ 可以调用 c 的, c 不能调用 C++ 的。
如:
1)g++ -o exe main.cpp myc.o/myc.a/myc.so
可以。 //main.cpp 要调用 myc 中的函数。注意 extern "C"
2)gcc -o exe main.c myc++.o/myc++.a/myc++.so
不行。 //main.c 要调用 myc++ 中的函数。
3)A: mylibc.o/mylibc.a/mylibc.so 中一函数为
FunC()
{
FunCplus();
}
B: mylibc++.o/mylibc++.a/mylibc++.so 中一函数为
FunCplus();
此时, main.cpp 调用 FunC ();
g++ -o exe main.cpp A B 也编译不过,道理和 2)差不多。
4)A: mylibc.o/mylibc.a/mylibc.so 中一函数为
int funC(int);
B: mylibc++.o/mylibc++.a/mylibc++.so 中一函数为
int funCplus(int)
此时, main.cpp 中有:
funC(funCplus(para));
funCplus(funC(para));
g++ -o exe main.cpp A B 可以编译成功并运行。

3.编译到目标的内容
(静态库)在编译可执行程序时,只有程序中确实用到的函数才会被包括进去,虽然一个头文件里包含着函数库中的全体函数声明,但在可执行程序中使用 include 语句引用,它并不会把整个函数库的内容都包括到可执行文件中去。

4.从 so 库封装 so 库。
已经有 libqt-mt.so.3.3.4(base.so),若 g++midlib.cpp 中调用了
libqt-mt.so.3.3.4(base.so) 中的函数,且又要被编译为一个共享库(so), main.cpp 调用该生成的新共享库。

1)g++ -shared -o libg++midlib.so g++midlib.cpp -I/root/QT/qt-x11-free-3.3.4/include /root/QT/qt-x11-free-3.3.4/lib/libqt-mt.so.3.3.4
g++ -o exe libg++midlib.so main.cpp -I/root/QT/qt-x11-free-3.3.4/inlucde

2) g++ -shared -o libg++midlib.so g++midlib.cpp -I/root/QT/qt-x11-free-3.3.4/include
g++ -o exe libg++midlib.so main.cpp -I/root/QT/qt-x11-free-3.3.4/include /root/QT/qt-x11-free-3.3.4/lib/libqt-mt.so.3.3.4

在这两种情况下生成的可执行文件 exe,都仅只需要有新生成的库 libg++midlib.so 就可以运行了。

2---4 运行的环境(估计其他的也差不多):
Linux localhost.localdomain 2.4.20-8 #1 Thu Mar 13 17:54:28 EST 2003 i686 i686 i386 GNU/Linux

# gcc -v
Reading specs from /usr/lib/gcc-lib/i386-redhat-linux/3.2.2/specs
Configured with: ../configure --prefix=/usr --mandir=/usr/share/man --infodir=/usr/share/info --enable-shared --enable-threads=posix --disable-checking --with-system-zlib --enable-__cxa_atexit --host=i386-redhat-linux
Thread model: posix
gcc version 3.2.2 20030222 (Red Hat Linux 3.2.2-5)

前面一部分是从 林锐博士 的《高质量 c/c++ 编程指南》里抄的,后面的都是自己写的。
呵呵,以前总是有些概念搞不清楚,所以就自己做了下实验,正好现在在看 《x86汇编语言与计算机体系结构》, 《linkers and loaders》, 《80x86保护模式系列教程》, 想把相关方面的东西再搞清楚些,呵呵。希望和各位多多交流,呵呵。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值