C++和C混合编程
C++比C晚出现,C++代码如果能够调用C语言的代码,那么会更好的利用现有的成果,但是事实上C++代码是无法直接调用C代码的,这是因为C++编译器在编译.cpp文件时生成的函数名与C编译器在编译.c文件时生成的函数名是不一样的。
C++为了支持重载,其编译器在编译完成后会对原有的函数名进行修改,比如
test(int i)和
test(int i, int j)
这两个函数在编译完成后可能就会被C++编译器修改成:
_ZDtesti
_ZDtestii
这种样式
但是C编译器却不会修改函数名,这样问题就就来了,如果在一个C++代码中包含一个声明C函数的头文件时,那么很可能在编译完成后,头文件中声明的这个C函数名会被修改!这样在C++代码中使用这个C函数的时候就会发生找不到函数名的问题,事实上这个函数在C代码中是存在的,只不过C++编译器一厢情愿的把函数名修改了。那么如何解决呢?
2.解决办法
很简单,显式的告诉C++编译器,这段代码是用C语言编译的函数,你就不要把函数名转化为C++的格式了。
extern "C" { int socket_send(); // 明确的告诉C++编译器,这是一个用C语言编译的函数 }
这样C++编译器在执行这段代码时,识别到extern "C"关键字,就会以C编译器的方式来编译括号内的代码。
3.由此引发的问题
这样虽然在C++编译下没有问题了,但是如果一个.c文件再去包含这个头文件时,又会发生问题,因为extern "C"不是C语言的关键字,这样.c文件又不能包含这个头文件了。如何能够既让.cpp文件能够包含这个头文件,又能让.c文件能够包含这个头文件呢,于是下面的写法就产生了:
使用条件编译的方式,如果判断是C++的编译器,就带上extern "C",如果是C的编译器就不带extern "C",由此,问题得到妥善解决。需要注意的是:__cplusplus是C++编译器内置的宏。
#ifdef __cplusplus extern "C" { #endif ....... #ifdef __cplusplus } #endif
%02x与%2x 之间的区别
输出最小宽度 用十进制整数来表示输出的最少位数。若实际位数多于定义的宽度,则按实际位数输出,若实际位数少于定义的宽度则补以空格或0(当最小宽度数值以0开头时)。 X 表示以十六进制形式输出 02 表示不足两位,前面补0输出;如果超过两位,则实际输出 举例: printf("%02X", 0x345); //打印出:345 printf("%02X", 0x6); //打印出:06 而如果直接写为 %2x,数据不足两位时,实际输出,即不额外补0输出; 如果超过两位,则实际输出。 printf("%2X", 0x345); //打印出:345 printf("%2X", 0x6); //打印出:6
for example
如:
Scala
-
object Test {
-
def main(args: Array[String]): Unit = {
-
println("%02X".format(12)) // 0c
-
println("%02X".format(2)) // 02
-
-
println("%2X".format(12)) // c
-
println("%2X".format(2)) // 2
-
}
-
}
Java
-
public class Test {
-
public static void main(String[] args) {
-
System.out.println(String.format("%02X",12 )); // 0c
-
System.out.println(String.format("%02X",2 )); // 02
-
-
System.out.println(String.format("%2X",12 )); // c
-
System.out.println(String.format("%2X",2 )); // 2
-
}
-
}