iolist跟list有什么区别?(经典)

看到erlang-china.org上有个帖子在问这个问题
引用
一直不太明白iolist,跟list到底有什么区别?请各位大侠指教。。


我翻看了半天erlang的文档也没写的太明白,所以就看看源码:
erts/emulator/beam/utils.c

Java代码 复制代码  收藏代码
  1. 3015int io_list_len(Eterm obj)   
  2. 3016{   
  3. 3017    Eterm* objp;   
  4. 3018    Sint len = 0;   
  5. 3019    DECLARE_ESTACK(s);   
  6. 3020    goto L_again;   
  7. 3021  
  8. 3022    while (!ESTACK_ISEMPTY(s)) {   
  9. 3023        obj = ESTACK_POP(s);   
  10. 3024    L_again:   
  11. 3025        if (is_list(obj)) {   
  12. 3026        L_iter_list:   
  13. 3027            objp = list_val(obj);   
  14. 3028            /* Head */  
  15. 3029            obj = CAR(objp);   
  16. 3030            if (is_byte(obj)) {   
  17. 3031                len++;   
  18. 3032            } else if (is_binary(obj) && binary_bitsize(obj) == 0) {   
  19. 3033                len += binary_size(obj);   
  20. 3034            } else if (is_list(obj)) {   
  21. 3035                ESTACK_PUSH(s, CDR(objp));   
  22. 3036                goto L_iter_list; /* on head */  
  23. 3037            } else if (is_not_nil(obj)) {   
  24. 3038                goto L_type_error;   
  25. 3039            }   
  26. 3040            /* Tail */  
  27. 3041            obj = CDR(objp);   
  28. 3042            if (is_list(obj))   
  29. 3043                goto L_iter_list; /* on tail */  
  30. 3044            else if (is_binary(obj) && binary_bitsize(obj) == 0) {   
  31. 3045                len += binary_size(obj);   
  32. 3046            } else if (is_not_nil(obj)) {   
  33. 3047                goto L_type_error;   
  34. 3048            }   
  35. 3049        } else if (is_binary(obj) && binary_bitsize(obj) == 0) { /* Tail was binary */  
  36. 3050            len += binary_size(obj);   
  37. 3051        } else if (is_not_nil(obj)) {   
  38. 3052            goto L_type_error;   
  39. 3053        }   
  40. 3054    }   
  41. 3055  
  42. 3056    DESTROY_ESTACK(s);   
  43. 3057    return len;   
  44. 3058  
  45. 3059 L_type_error:   
  46. 3060    DESTROY_ESTACK(s);   
  47. 3061    return -1;   
  48. 3062}  


从源码可以看出来iolist是这样的定义的:
1. []
2. binary
3. 列表, 每个元素是int(0-255)或者binary或者iolist.
其中binary是指 bitsize % 8 == 0 .
int 是0-255


root@ubuntu:/usr/src/otp# erl
Erlang R13B04 (erts-5.7.5) [source] [smp:2:2] [rq:2] [async-threads:0] [hipe] [kernel-poll:false]

Eshell V5.7.5  (abort with ^G)
2> iolist_size(<<>>).
0
3> iolist_size(<<1:1>>).
** exception error: bad argument
in function  iolist_size/1
called as iolist_size(<<1:1>>)
4> iolist_size(<<1:8>>).
1
5> iolist_size([]).
0
6> iolist_size(<<1,2>>).
2
7> iolist_size([1,2]).
2
8> iolist_size([1,2, <<1,2>>]).
4
9> iolist_size([1,2, <<1,2>>, [2]]).
5
10> iolist_size([1,2, <<1,2>>, [2]]).
5
11> iolist_size([<<1:1>>]).
** exception error: bad argument
in function  iolist_size/1
called as iolist_size([<<1:1>>])
12>


Iolist的作用是用于往port送数据的时候.由于底层的系统调用如writev支持向量写, 就避免了无谓的iolist_to_binary这样的扁平话操作, 避免了内存拷贝,极大的提高了效率.
建议多用.
转载: http://mryufeng.iteye.com/blog/634867

你可能感兴趣的:(iolist跟list有什么区别?(经典))