[转]GCC系列: __attribute__((visibility("")))

在 objc-api.h 里面有很多关于__attribute__ 的定义.

例如

#if !defined(OBJC_VISIBLE)
#   if TARGET_OS_WIN32
#       if defined(BUILDING_OBJC)
#           define OBJC_VISIBLE __declspec(dllexport)
#       else
#           define OBJC_VISIBLE __declspec(dllimport)
#       endif
#   else
#       define OBJC_VISIBLE  __attribute__((visibility("default")))
#   endif
#endif

可以看到

 

OBJC_VISIBLE 的定义方式是 __attribute__((visibility("default"))).

那么究竟有什么作用呢? 下面举例说明.

 

GCC 有个visibility属性, 该属性是说, 启用这个属性:

1. 当-fvisibility=hidden时

动态库中的函数默认是被隐藏的即 hidden. 除非显示声明为__attribute__((visibility("default"))).

2. 当-fvisibility=default时

 动态库中的函数默认是可见的.除非显示声明为__attribute__((visibility("hidden"))).

 

特别说明: 这个特性是 GCC4.0 以后才有的.

 

基于GCC系列: 加载静态链接库的例子, 谈谈 visibility.

 

Car.c

#include<stdio.h>  
#include<stdlib.h> 
 
__attribute ((visibility("default"))) int drive()
{
    printf("Car driving...\n");
 
    return 0;
}
 
void stop()
{
    printf("Car stop.\n");
}

 

注意: attribute 的定义和使用方法.

 

使用 visibility 属性来编译动态链接库.

g++-4.9 -shared -o libCar.so -fvisibility=hidden Car.c

这样一来, drive 方法是可见的, 但是 stop 是不可见的.

 

main.c

#include<stdio.h>  
#include<stdlib.h> 
 
//
// 在 gcc 编译器下, 必须声明方法
// 在 clang 编译下, 只是给了警告
//
 
int drive();
void stop();
 
int main()
{
    drive();
    stop();
 
    return 0;
}

 

编译

g++-4.9 -o app main.c -L ./ -lCar

提示信息

Undefined symbols for architecture x86_64:
  "stop()", referenced from:
      _main in ccZnwENu.o
ld: symbol(s) not found for architecture x86_64
collect2: error: ld returned 1 exit status

这说明了上面提到的第一中情况.

 

 

重新编译 Car.c, 换个方式

g++-4.9 -shared -o libCar.so -fvisibility=default Car.c

注意: 这次使用了 default 属性.

 

 

编译 main.c

g++-4.9 -o app main.c -L ./ -lCar

编译成功, 运行 app

./app 
Car driving...
Car stop.


那么修改一下 Car.c 的代码

#include<stdio.h>  
#include<stdlib.h> 
 
__attribute ((visibility("default"))) int drive()
{
    printf("Car driving...\n");
 
    return 0;
}
 
__attribute ((visibility("hidden"))) void stop()
{
    printf("Car stop.\n");
}

使用 visibility=default 和 visibility=hidden 分别编译 Car.c.

可以知道, stop 函数都是隐藏的.

这个也很好的说明了上面两个问题.

注意: g++-4.9 改为 gcc-4.9也是一样的.

 

另外, 除了 default 和 hidden, 还有 internal, protected.请关注后续讨论.

posted @ 2018-09-30 19:05  ZYVV  阅读(2779)  评论(1编辑  收藏  举报