免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
楼主: duanjigang
打印 上一主题 下一主题

源码阅读第一期:axel和wget [复制链接]

论坛徽章:
0
121 [报告]
发表于 2011-10-06 10:48 |只看该作者
回复  duanjigang

哦,我说的是第五行
假设no_proxy被axel处理后内容为:
,bbs.chinaunix.net,t ...
seufy88 发表于 2011-10-05 21:31



    不好意思,我看到你那个问题怎么都是斜体字,就想编辑下。。不想到数据弄丢了。。
对不起啊,有空的话,你把那个帖子重新编辑下,补充上完整内容,实在不好意思

论坛徽章:
0
122 [报告]
发表于 2011-10-06 10:54 |只看该作者
本帖最后由 duanjigang 于 2011-10-06 11:08 编辑

首先,配置文件中可以这么写

  1. no_proxy=www.126.com,bbs.chinaunix.net,www.test.com
复制代码
然后,在文件
conf.c中的函数conf_init中能看到对no_proxy进行加工的代码

  1. /* Convert no_proxy to a 0-separated-and-00-terminated list..        */
  2.         for( i = 0; conf->no_proxy[i]; i ++ )
  3.                 if( conf->no_proxy[i] == ',' )
  4.                         conf->no_proxy[i] = 0;
  5.         conf->no_proxy[i+1] = 0;
复制代码
no_proxy中的url是用0分割,00结束的字符串
conn_init函数中

  1. //注意,host指向第一个字符串
  2.         char *proxy = conn->conf->http_proxy, *host = conn->conf->no_proxy;
  3.         int i;
  4.        
  5.         //如果本身就没设置代理服务器,那么就无所谓no_proxy了,因为没有任何代理
  6.         if( *conn->conf->http_proxy == 0 )
  7.         {
  8.                 proxy = NULL;
  9.         }//如果设置了代理的话
  10.         else if( *conn->conf->no_proxy != 0 )//no_proxy不是空串,也就是说no_proxy中是有内容的,需要对某些url不进行代理访问
  11.         {
  12.                 //遍历no_proxy
  13.                 //no_proxy
  14.                 for( i = 0; ; i ++ )
  15.                         //找到一个字符串的结束
  16.                         if( conn->conf->no_proxy[i] == 0 )
  17.                         {
  18.                                 //www.126.com\0bbs.chinaunix.net\0www.test.com\0\0
  19.                                 //如果host在conn->host中出现了, www.126.com如果在url中出现了的话,
  20.                                 //针对于conn->host的代理就为空
  21.                                 if( strstr( conn->host, host ) != NULL )
  22.                                         proxy = NULL;
  23.                                 //host指向下一个url的开始
  24.                                 host = &conn->conf->no_proxy[i+1];
  25.                                 //如果紧邻两个字符都是0,则表明结束了,退出
  26.                                 if( conn->conf->no_proxy[i+1] == 0 )
  27.                                         break;
  28.                         }
  29.         }
  30.        
复制代码
不知道这下说清楚没?呵呵,可能我理解问题能力比较弱。。
对了,发现host后,proxy置空,应该当时就break,这样比找到结尾再退出更合理。
因此,我觉得这段代码

  1. if( strstr( conn->host, host ) != NULL )
  2. proxy = NULL;
复制代码
是不是应该改为

  1. if( strstr( conn->host, host ) != NULL )
  2. {
  3.     proxy = NULL;
  4.     break;
  5. }
复制代码
更合理些?

论坛徽章:
1
双子座
日期:2013-11-06 17:18:01
123 [报告]
发表于 2011-10-06 14:34 |只看该作者
本帖最后由 seufy88 于 2011-10-06 14:36 编辑

回复 122# duanjigang


    很感谢.
no_proxy中各url是以\0分隔的话就说的通了.还没看到conf.c这个文件呢.不好意思.
另外,conn->host中只能存放一个url,不能像no_proxy一样存放多个.这点应该可以明确了.:wink:

论坛徽章:
0
124 [报告]
发表于 2011-10-06 15:49 |只看该作者
本帖最后由 wangzhen11aaa 于 2011-10-09 16:53 编辑

  1. init_switch(opt->type)
  2. {
  3. . . . . . .
  4.   case OPT_VALUE:   /*一般是指短选项含有参数,此类型是含有参数为 1.
  5.   longopt->has_arg = required_argumet;
  6.   if (opt->short_name)  /*如果短选项不为空*/
  7.     *p++ = ';';   /*将":"添入char 类型short_options数组中,因为还有有参将会出现这样的形式: 比如:'A'  ':'*/
  8.     break;           /*其他返回*/
  9. case  OPT_BOOLEN:  /*这里是指的是长选项*/
  10.    longopt->has_arg = optional_argument;   /* # define optional_argument      2
  11. 此值定义为2,表示理想参数为 2*/
  12.    longopt = &long_options[o++];  /*操作 long_options数组,因为是BOOLEN类型,所以要有不包含即“no-"的选项*/
  13.    longopt->name = no_prefix (opt->long_name);  /1、___---->main.c添入long_options数组的元素的域,no_prefix()表示加上"no-"表示没有*/
  14.    longopt->has_arg = no_argument;    /*没有参数 为 0*/
  15.   longopt->val = i | BOOLEAN_NEG_MARKER;  /*longopt->val的值为 一个大于1024的数作为不包含的条件*/
  16.           break;  
  17.        default:
  18.            assert (opt->argtype != -1);  /*如果argtype不是 -1*/
  19.            longopt->has_arg = opt->argtype;   /*has_arg指向opt->argtype指向的地方*/
  20.           if (opt->short_name)  /*如果短选项不为 0*;
  21.             {
  22.                if (longopt->has_arg == required_argument)
  23.                  *p++ = ':';  这里因为有short_name且不为0,并且有参数1,格式为 “X" ":";
  24. 84   
  25. 385             }
  26. 386         }
  27. 387     }
  28. 388   /* Terminate short_options. */
  29. 389   *p = '\0';  /*在short_options数组中存放了一个"最后放入终结字符"\0""/
  30. 392   assert (o <= countof (long_options));  /*循环的大小当然是小于总个数了*/
  31. 393 }
复制代码
1、_________________------------------>

  1. 302 no_prefix (const char *s)
  2. 303 {
  3. 304   static char buffer[1024];   
  4. 305   static char *p = buffer;
  5. 306
  6. 307   char *cp = p;   
  7. 308   int size = 3 + strlen (s) + 1;  /*看样子no- 和string不是在一起的。在options_tables[]中既可以看出 "no-STRING\0" 加上前面的"no-"  和 后面的 \0字符*/
  8. 309   if (p + size >= buffer + sizeof (buffer))    /*如果字符串地址大于了buffer数组的地址那么就abort()*/
  9. 310     abort ();
  10. 311
  11. 312   cp[0] = 'n', cp[1] = 'o', cp[2] = '-';  /*给前面这几个字符赋值*/
  12. 313   strcpy (cp + 3, s);  /*然后copy*/
  13. 314   p += size;  /*用p记录当前使用的最后位置即为\0后面第一个字符位置*/
  14. 315   return cp;  /*返回字符初始地址*/
  15. 316 }
复制代码

论坛徽章:
0
125 [报告]
发表于 2011-10-06 19:52 |只看该作者
本帖最后由 wangzhen11aaa 于 2011-10-09 16:55 编辑

getopt_long()函数应该是最重要的解析函数

  1.     main(int argc, char **argv)
  2. {
  3. 923   longindex = -1;
  4. 924   int retconf;
  5. 925   bool use_userconfig = false;
  6. 926
  7. 927   while ((retconf = getopt_long (argc, argv,
  8. 928                                 short_options, long_options, &longindex)) !=   -1) 1、____-------> /lib/getoptl.c*/
  9. 929     {
  10. 930       int confval;
  11. 931       bool userrc_ret = true;
  12. 932       struct cmdline_option *config_opt;
  13. 933       confval = long_options[longindex].val;  /*这里是控制数组大小在1024以内,因为1024以外的是那些no-形式的*/
  14. 934       config_opt = &option_data[confval & ~BOOLEAN_NEG_MARKER];
  15. 935       if (strcmp (config_opt->long_name, "config") == 0)  /*看是否此长类型名已经注册*/
  16. 936         {
  17. 937           userrc_ret &= run_wgetrc (optarg);  /*如果注册,那么就读刚刚到配置文件*/
  18. 938           use_userconfig = true;  /*use_userconfig 表示可用*/
  19. 939         }
  20. 940       if (!userrc_ret)
  21. 941         {
  22. 942           printf ("Exiting due to error in %s\n", optarg);/*如果没有,那么就返回错误*/
  23. 943           exit (2);
  24. 944         }
  25. 945       else
  26. 946         break;
  27. 947     }
复制代码
1、_____-------------->

  1. 39 int
  2. 40 getopt_long (int argc, char *__getopt_argv_const *argv, const char *options,   
  3. 41              const struct option *long_options, int *opt_index) /*+++++=====>/lib/getopt.in.h*/
  4. 42 {     
  5. 43   return _getopt_internal (argc, (char **) argv, options, long_options,
  6. 44                            opt_index, 0, 0);/*2、________---------->lib/getopt.c*/
  7. 45 }
  8. +++++=========>
复制代码
/* 如果独立的程序获得正确的getopt_long()和getopt_long_only()函数;声明char **argv。但是libc用char * const * argv这是不正确的,因为getopt_long()和getopt_long_only()可以改变argv的值*/
95 #if !defined __need_getopt   /*这原来是#if defined __GETOPT_PREFIX && !defined __need_getopt,但是这会引起重定义警告,如果unistd.h和getopt.h被包含, 因为unistd.h包含了已经原来定义__need_getopt的getopt.h。*/
96 # if defined __GETOPT_PREFIX  /*如果包含了getopt.h,那么就使用使用char **argv。
97 #  define __getopt_argv_const /* empty */
98 # else  /*如果没有包含,就定义为 char *const *argv*/
99 #  define __getopt_argv_const const
100 # endif
101 #endif

  1. 2、_____________---------------->
  2. 1127 int
  3. 1128 _getopt_internal (int argc, char **argv, const char *optstring,
  4. 1129                   const struct option *longopts, int *longind, int long_only,        
  5. 1130                   int posixly_correct)
  6. 1131 {
  7. 1132   int result;
  8. 1133
  9. 1134   getopt_data.optind = optind;  /*初始化为 1。++++++++======>getopt_data[]数组 类型为*/
  10. 1135   getopt_data.opterr = opterr;  /*初始化为1*/
  11. 1136
  12. 1137   result = _getopt_internal_r (argc, argv, optstring, longopts,
  13. 1138                                longind, long_only, &getopt_data,
  14. 1139                                posixly_correct); /*3、_______------->初始化getopt_data/
  15. 1140
  16. 1141   optind = getopt_data.optind;   /*表示下一个选项字符的索引值*/
  17. 1142   optarg = getopt_data.optarg;  /*下一个选项字符*/
  18. 1143   optopt = getopt_data.optopt;  /*
  19. 1144
  20. 1145   return result;
  21. 1146 }
复制代码
++++=============>

  1. 97 static struct _getopt_data getopt_data;
  2. /*见前面*/
  3. [code]
  4. 3、____________------------->
  5. 345 int
  6. 346 _getopt_internal_r (int argc, char **argv, const char *optstring,
  7. 347                     const struct option *longopts, int *longind,
  8. 348                     int long_only, struct _getopt_data *d, int posixly_correct)
  9. 349 {
  10. 350   int print_errors = d->opterr;
  11. 351
  12. 352   if (argc < 1)  /*argc小于0是不可能的*/
  13. 353     return -1;
  14. 354
  15. 355   d->optarg = NULL;  /*是否带有参数*/
  16. 356
  17. 357   if (d->optind == 0 || !d->__initialized)  /*如果没有初始化,或者d->optind == 0*/
  18. 358     {
  19. 359       if (d->optind == 0)
  20. 360         d->optind = 1;  /* Don't scan ARGV[0], the program name.  */
  21. 361       optstring = _getopt_initialize (argc, argv, optstring, d,
  22. 362                                       posixly_correct);   /*4、________-------->lib/getopt.c*/
  23. 363       d->__initialized = 1;   /*已经初始化*/
  24. 364     }
  25. 365   else if (optstring[0] == '-' || optstring[0] == '+')
  26. 366     optstring++;
  27. 367   if (optstring[0] == ':')  /*表示带有参数*/
  28. 368     print_errors = 0;  
  29. 374 #if defined _LIBC && defined USE_NONOPTION_FLAGS  如果使用零选项标记 例子:- 就是一种非选项的格式。
  30. 375 # define NONOPTION_P (argv[d->optind][0] != '-' || argv[d->optind][1] == '\0' \
  31. 376                       || (d->optind < d->__nonoption_flags_len                     \
  32. 377                           && __getopt_nonoption_flags[d->optind] == '1'))
  33. 378 #else
  34. 379 # define NONOPTION_P (argv[d->optind][0] != '-' || argv[d->optind][1] == '\0')
  35. 380 #endif
  36. 381
  37. 382   if (d->__nextchar == NULL || *d->__nextchar == '\0')   /*如果下一个选项字符为 NULL*/
  38. 388       if (d->__last_nonopt > d->optind)   /*d->optind表示下一个选项字符的索引*/
  39. 389         d->__last_nonopt = d->optind;   
  40. 390       if (d->__first_nonopt > d->optind)   
  41. 391         d->__first_nonopt = d->optind;
  42.      画个图                      |-A|  | B 234|(这个是非选项| |- Q|  
  43.                                  这种情况就会让|-A| |- Q| 他们放在一起,后面才是非选项。这个就是后面从新排序的一个例子*/
  44. 392
  45. 393       if (d->__ordering == PERMUTE)  /*如果是PERMUTE处理方法*/
  46. 394         {
  47. 395      
  48. 398           if (d->__first_nonopt != d->__last_nonopt
  49. 399               && d->__last_nonopt != d->optind)
  50. 400             exchange ((char **) argv, d);  /*交换空选项和非空选项的位置5、______----->*/
  51. 401           else if (d->__last_nonopt != d->optind)
  52. 402             d->__first_nonopt = d->optind;   /*交换后,地址互换*/
  53. 403
  54. 404         
  55. 405            
  56. 406
  57. 407           while (d->optind < argc && NONOPTION_P)  /*如果下个要取的索引小于参数总数和如果还是非选项,那么依次加,为了计算最后那个非选项的位置。.*/
  58. 408             d->optind++;
  59. 409           d->__last_nonopt = d->optind;
  60. 410         }
  61.    /*这里处理的就是长选项*/
  62. 417       if (d->optind != argc && !strcmp (argv[d->optind], "--"))
  63. 418         {
  64. 419           d->optind++;
  65. 420
  66. 421           if (d->__first_nonopt != d->__last_nonopt
  67. 422               && d->__last_nonopt != d->optind)
  68. 423             exchange ((char **) argv, d);   /*将 -- 也看作一个空选项,依然要交换*/
  69. 424           else if (d->__first_nonopt == d->__last_nonopt)
  70. 425             d->__first_nonopt = d->optind; /*如果两者相等说明没有空选项*/
  71. 426           d->__last_nonopt = argc;
  72. 427
  73. 428           d->optind = argc;
  74. 429         }
  75. 434       if (d->optind == argc)
  76. 435         {
  77. 436         
  78. 438           if (d->__first_nonopt != d->__last_nonopt)
  79. 439             d->optind = d->__first_nonopt;
  80. 440           return -1;  /*这里交换完毕,记住返回的是- 1*/
  81. 441         }
  82. 442
复制代码

  1. 第二种情况
  2. 446       if (NONOPTION_P)  /*如果是-- 开头,根据REQUIRE_ORDER规则就直接返回 -1*/
  3. 447         {
  4. 448           if (d->__ordering == REQUIRE_ORDER)
  5. 449             return -1;
  6. 450           d->optarg = argv[d->optind++];
  7. 451           return 1;
  8. 452         }
  9.                         /*要是其他就计算后面的那个字符*/
  10. 457         d->__nextchar = (argv[d->optind] + 1
  11. 458                   + (longopts != NULL && argv[d->optind][1] == '-'));  如果下个是 -那么就返回 argv[d->optind] + 1 + 1的那个字符。
  12. 459     }
  13. 476   if (longopts != NULL     /*判断是否为长选项 必须是 - - 格式,后面再判断是否还有字符,如果有一个字符,并且规定了long_only这个数值*/
  14. 477       && (argv[d->optind][1] == '-'
  15. 478           || (long_only && (argv[d->optind][2]
  16. 479                             || !strchr (optstring, argv[d->optind][1]))))) /*optring就是短选项数组*/
  17. 480     {
  18. 481       char *nameend;
  19. 482       unsigned int namelen;  
  20. 483       const struct option *p;
  21. 484       const struct option *pfound = NULL;
  22. 485       struct option_list
  23. 486       {
  24. 487         const struct option *p;
  25. 488         struct option_list *next;
  26. 489       } *ambig_list = NULL;
  27. 490       int exact = 0;
  28. 491       int indfound = -1;
  29. 492       int option_index;
  30. 493
  31. 494       for (nameend = d->__nextchar; *nameend && *nameend != '='; nameend++)
  32. 495                     ;
  33. 496       namelen = nameend - d->__nextchar;  /*计算长选项的长度*/
  34. 500       for (p = longopts, option_index = 0; p->name; p++, option_index++) /*整个测试,来检测到底是哪个长选项*/
  35. 501         if (!strncmp (p->name, d->__nextchar, namelen))  
  36. 502           {
  37. 503             if (namelen == (unsigned int) strlen (p->name))/*这是判断字符串大小的函数,如果相等就返回 0 */
  38.                 {
  39. 505              
  40. 506                 pfound = p;   /*如果找到了,对应选项*/
  41. 507                 indfound = option_index; /*索引项赋给给indfound*/
  42. 508                 exact = 1;
  43. 509                 break;
  44. 510               }
  45.               /*第一次找到了前n个字符相等的,但是长度不相等*/
  46. 511             else if (pfound == NULL)
  47. 512               {
  48. 513                 pfound = p;
  49. 515                 indfound = option_index;  
  50. 516               }          /*后面如果再次前n个字符相等,但是长度不同的,那么如下面作*/
  51. 517             else if (long_only  
  52. 518                      || pfound->has_arg != p->has_arg /*如果包含参数也不同*/
  53. 519                      || pfound->flag != p->flag
  54. 520                      || pfound->val != p->val)
  55. 521               {
  56. 523                 struct option_list *newp = malloc (sizeof (*newp));  /*就链入链表*/
  57. 524                 newp->p = p;  
  58. 525                 newp->next = ambig_list;   
  59. 526                 ambig_list = newp;
  60. 527               }
  61. 528           }
  62. 530       if (ambig_list != NULL && !exact)
  63. 531         {
  64. 532           if (print_errors)   如果不为0,而且exact 为 0,就是没有找到!?
  65. 533             {
  66. 534               struct option_list first;   
  67. 535               first.p = pfound;      /*记录第一个那个前n个相同,长度不同的结构,将其链入链表*/
  68. 536               first.next = ambig_list;
  69. 537               ambig_list = &first;  
  70. 539 #if defined _LIBC && defined USE_IN_LIBIO
  71. 540               char *buf = NULL;  
  72. 541               size_t buflen = 0;
  73. 542
  74. 543               FILE *fp = open_memstream (&buf, &buflen); /*打开

  75. }
复制代码
4、_________________----------->

  1. 224 static const char *
  2. 225 _getopt_initialize (int argc _GL_UNUSED,
  3. 226                     char **argv _GL_UNUSED, const char *optstring,
  4. 227                     struct _getopt_data *d, int posixly_correct)
  5. 228 {
  6. 229  
  7. 232
  8. 233   d->__first_nonopt = d->__last_nonopt = d->optind; /*初始化为1*/
  9. 234
  10. 235   d->__nextchar = NULL; /*下一个选项字符*/
  11. 236
  12. 237   d->__posixly_correct = posixly_correct || !!getenv ("POSIXLY_CORRECT"); /*返回getenv()是返回环境的常量值*/
  13. 238
  14. 239  
  15. 240
  16. 241   if (optstring[0] == '-')
  17. 242     {
  18. 243       d->__ordering = RETURN_IN_ORDER;  /*根据开头的字符来判断检测选项类型的方法,见前面的分析*/
  19. 244       ++optstring;
  20. 245     }
  21. 246   else if (optstring[0] == '+')
  22. 247     {
  23. 248       d->__ordering = REQUIRE_ORDER;
  24. 249       ++optstring;
  25. 250     }
  26. 251   else if (d->__posixly_correct)
复制代码

  1. 5、_________--------->
  2. 143 static void
  3. 144 exchange (char **argv, struct _getopt_data *d)
  4. 145 {
  5. 146   int bottom = d->__first_nonopt;  /*第一个空选项索引*/
  6. 147   int middle = d->__last_nonopt;  /*空选项后的第一个字符索引*/
  7. 148   int top = d->optind;  /*下一次要去访问的字符选项的索引*/
  8. 149   char *tem;
  9. 160   if (d->__nonoption_flags_len > 0 && top >= d->__nonoption_flags_max_len)
  10. 161     {

  11. 162       /* We must extend the array.  如果下一次要访问的索引大于整个大小,所以要拓展到top+1*/
  12. 163          presents new arguments.  */
  13. 164       char *new_str = malloc (top + 1);
  14. 165       if (new_str == NULL)
  15. 166         d->__nonoption_flags_len = d->__nonoption_flags_max_len = 0;
  16. 167       else
  17. 168         {
  18. 169           memset (__mempcpy (new_str, __getopt_nonoption_flags,
  19. 170                              d->__nonoption_flags_max_len),
  20. 171                   '\0', top + 1 - d->__nonoption_flags_max_len);  /*__mempcpy()函数是memcpy()的类似函数,不同的是它返回指向最后一个元素\0的指针,然后初始化为\0*/
  21. 172           d->__nonoption_flags_max_len = top + 1;
  22. 173           __getopt_nonoption_flags = new_str;  /*
  23. 174         }
  24. 175     }
  25. 176 #endif
  26. 177
  27. 178   while (top > middle && middle > bottom)
  28. 179     {
  29. 180       if (top - middle > middle - bottom)
  30. 181         {
  31. 182           /* Bottom segment is the short one.  */
  32. 183           int len = middle - bottom;  /*如果空选项所占字符小于非空选项大小,那么只移动空字符大小的那部分*/
  33. 184           register int i;
  34. 187           for (i = 0; i < len; i++)
  35. 188             {
  36. 189               tem = argv[bottom + i];
  37. 190               argv[bottom + i] = argv[top - (middle - bottom) + i];
  38. 191               argv[top - (middle - bottom) + i] = tem;
  39. 192               SWAP_FLAGS (bottom + i, top - (middle - bottom) + i);
  40. 193             }
  41. 194           /* Exclude the moved bottom segment from further swapping.  */
  42. 195           top -= len;  /*top空选项首位置*/
  43. 196         }
  44. 197       else
  45. 198         {
  46. 199           /* Top segment is the short one.  */
  47. 200           int len = top - middle;
  48. 201           register int i;
  49. 202
  50. 203           /* Swap it with the bottom part of the bottom segment.  */
  51. 204           for (i = 0; i < len; i++)
  52. 205             {                              /*典型的交换算法*/
  53. 206               tem = argv[bottom + i];
  54. 207               argv[bottom + i] = argv[middle + i];
  55. 208               argv[middle + i] = tem;
  56. 209               SWAP_FLAGS (bottom + i, middle + i);
  57. 210             }
  58. 212           bottom += len;    /*空选项的开始所在索引*/
  59. 213         }
  60. 214     }
  61. 215
  62. 216   /* Update records for the slots the non-options now occupy.  */
  63. 217
  64. 218   d->__first_nonopt += (d->optind - d->__last_nonopt);  /*更新位置,*/
  65. 219   d->__last_nonopt = d->optind;
  66. 220
复制代码

论坛徽章:
0
126 [报告]
发表于 2011-10-08 18:45 |只看该作者
duanjigang, 你用啥看代码?你分析code,格式看起来挺舒服的。是咋弄出来的?我还没具体看code。。。

论坛徽章:
0
127 [报告]
发表于 2011-10-09 08:55 |只看该作者
回复 122# duanjigang


    版主你好,我看了看你06年写的那个关于原始套接字的帖子,
   不明白为什么你printf MAC地址和IP地址的时候都要一个8位
   的全1相与?

论坛徽章:
0
128 [报告]
发表于 2011-10-09 09:35 |只看该作者
duanjigang, 你用啥看代码?你分析code,格式看起来挺舒服的。是咋弄出来的?我还没具体看code。。。
janock 发表于 2011-10-08 18:45



    测试用vi+gcc啊
然后分析代码基本上是在windows下用VC6+VC助手

论坛徽章:
0
129 [报告]
发表于 2011-10-09 09:45 |只看该作者
回复  duanjigang


    版主你好,我看了看你06年写的那个关于原始套接字的帖子,
   不明白为什么你 ...
inknk 发表于 2011-10-09 08:55



    是为了把要打印的值格式化为一个8bit的整数

论坛徽章:
0
130 [报告]
发表于 2011-10-09 09:50 |只看该作者
回复 129# duanjigang


    但是不与的话不也是8位的吗?而且和1相与还是没变呀?
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP