免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 2035 | 回复: 4
打印 上一主题 下一主题

[C] C如何用宏区别常量资源和变量资源? [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2013-07-03 11:34 |只看该作者 |倒序浏览
本帖最后由 true_casey 于 2013-07-03 11:35 编辑
  1. static
  2. void testB (char *src)
  3. {
  4.         /* 判断传入的src 是属于 @1/2/3 ??? */
  5.         do_somthing ();
  6. }

  7. static
  8. void testA (char *src)
  9. {
  10.         char *a = "hello world";
  11.         char b[100] = "hello world";

  12.         testB (a);                                        /* @1 */
  13.         testB (b);                                        /* @2 */
  14.         testB ("hello world" );                                /* @3 */
  15. }
复制代码
@1 有名常量
@2 缓冲资源
@3 未名常量

怎么用宏去区别啊???

论坛徽章:
17
处女座
日期:2013-08-27 09:59:352015亚冠之柏太阳神
日期:2015-07-30 10:16:402015亚冠之萨济拖拉机
日期:2015-07-29 18:58:182015年亚洲杯之巴勒斯坦
日期:2015-03-06 17:38:17摩羯座
日期:2014-12-11 21:31:34戌狗
日期:2014-07-20 20:57:32子鼠
日期:2014-05-15 16:25:21亥猪
日期:2014-02-11 17:32:05丑牛
日期:2014-01-20 15:45:51丑牛
日期:2013-10-22 11:12:56双子座
日期:2013-10-18 16:28:17白羊座
日期:2013-10-18 10:50:45
2 [报告]
发表于 2013-07-03 14:40 |只看该作者
本帖最后由 myworkstation 于 2013-07-03 14:42 编辑

编译器和语言并没有直观的提供什么功能去实现这个处理,可以换个思路,在C中有四个存储类型static,automatic,register,dynamic。每种类型的存储的地址都是可识别的,通过对存储地址的判断可以识别实事的变量或常量变型。
char *a = "hello world";testB (a); 和 testB ("hello world" );  这两个调用实际上是一样的。编译器在处理的时候会把对hello world的变引指向相同的地方(编译器基本都这么进行优化处理,不属于标准规定)。根据上述说法那下面的公式成立:编译器对常量变量的内存处理策略+操作系统的内存布局=可明确定位的内存识别。由于操作系统的内存布局因系统而定,编译器的实现也各有不同,所以就算得出结论要实现相关处理的代码也是很难进行移植的。下面是完成相关功能的代码在linux下测试通过。
  1. #include <stdlib.h>
  2. #include <stdio.h>
  3. #include <string.h>

  4.     int global_dummy = 0;
  5.     static
  6.     void testB (char *src)
  7.     {
  8.             /* 判断传入的src 是属于 @1/2/3/4*/
  9.             int local_dummy = 0;
  10.             if( (unsigned long)src < (unsigned long )&local_dummy ){
  11.                 //if( src > testB ){
  12.                     if(  (unsigned long )src <  (unsigned long )&global_dummy ){
  13.                         printf("string literal\n\n");
  14.                     }
  15.                     else if (  (unsigned long ) src > (unsigned long ) &global_dummy){
  16.                         printf("malloc string\n\n");
  17.                     }
  18.                 //}
  19.             }
  20.             else
  21.             {
  22.                 printf("array: stack\n\n");
  23.             }
  24.     }
  25.     static
  26.     void testA ()
  27.     {
  28.             char *a = "hello world";
  29.             char b[100] = "hello world";
  30.             char *c = malloc(100);
  31.             strcpy(c,a);
  32.             printf("char *a = \"hello world\";\n");
  33.             testB (a);                                        /* @1 */
  34.             printf("char b[100] = \"hello world\";\n");
  35.             testB (b);                                        /* @2 */
  36.             printf(" (\"hello world\" )\n");
  37.             testB ("hello world" );                                /* @3 */
  38.             printf("char *c = malloc(100);\n");
  39.             testB (c );                                /* @4 */
  40.             free(c);
  41.     }

  42. int main(int argc,const char** argv)
  43. {
  44.     testA();
  45.     return 0;
  46. }
复制代码
程序的运行结果如下:
char *a = "hello world";
string literal

char b[100] = "hello world";
array: stack

("hello world" )
string literal

char *c = malloc(100);
malloc string


下面贴个linux下程序运行的内存布局图,可以加深对上述代码的理解

论坛徽章:
14
巨蟹座
日期:2013-11-19 14:09:4615-16赛季CBA联赛之青岛
日期:2016-07-05 12:36:0515-16赛季CBA联赛之广东
日期:2016-06-29 11:45:542015亚冠之全北现代
日期:2015-07-22 08:09:472015年辞旧岁徽章
日期:2015-03-03 16:54:15巨蟹座
日期:2014-12-29 08:22:29射手座
日期:2014-12-05 08:20:39狮子座
日期:2014-11-05 12:33:52寅虎
日期:2014-08-13 09:01:31巳蛇
日期:2014-06-16 16:29:52技术图书徽章
日期:2014-04-15 08:44:01天蝎座
日期:2014-03-11 13:06:45
3 [报告]
发表于 2013-07-03 15:51 |只看该作者
char *a = "hello world";
之所以能编译通过,是因为编译器网开一面,以照顾旧有的海量代码。但不等于说 a[] 就是可写的,码农得自己保证不修改a[]值。
换种说法,看到以前代码中 char *a = "hello world"; 就不用改了,能编译通过;但正在写的代码,可不能这样,得写成正确的 const char *a = "hello world";

论坛徽章:
14
水瓶座
日期:2014-06-10 09:51:0215-16赛季CBA联赛之江苏
日期:2017-11-27 11:42:3515-16赛季CBA联赛之八一
日期:2017-04-12 14:26:2815-16赛季CBA联赛之吉林
日期:2016-08-20 10:43:1215-16赛季CBA联赛之广夏
日期:2016-06-23 09:53:58程序设计版块每日发帖之星
日期:2016-02-11 06:20:00程序设计版块每日发帖之星
日期:2016-02-09 06:20:0015-16赛季CBA联赛之上海
日期:2015-12-25 16:40:3515-16赛季CBA联赛之广夏
日期:2015-12-22 09:39:36程序设计版块每日发帖之星
日期:2015-08-24 06:20:002015亚冠之德黑兰石油
日期:2015-08-07 09:57:302015年辞旧岁徽章
日期:2015-03-03 16:54:15
4 [报告]
发表于 2013-07-03 17:44 |只看该作者
屌爆了,完全不懂{:3_188:}

论坛徽章:
17
处女座
日期:2013-08-27 09:59:352015亚冠之柏太阳神
日期:2015-07-30 10:16:402015亚冠之萨济拖拉机
日期:2015-07-29 18:58:182015年亚洲杯之巴勒斯坦
日期:2015-03-06 17:38:17摩羯座
日期:2014-12-11 21:31:34戌狗
日期:2014-07-20 20:57:32子鼠
日期:2014-05-15 16:25:21亥猪
日期:2014-02-11 17:32:05丑牛
日期:2014-01-20 15:45:51丑牛
日期:2013-10-22 11:12:56双子座
日期:2013-10-18 16:28:17白羊座
日期:2013-10-18 10:50:45
5 [报告]
发表于 2013-07-03 18:28 |只看该作者
回复 3# bruceteen


    你的提醒很有必要,这个行为在不同的c和c++的编译器上有会不同的行为,应该说char *a = "hello world";这种写法不能算错误的写法,所以c编译器不报错也不警告,就此问题稍微展开一下,说说这个做具体会有什么样的行为。早期的K&C认为这是完全合法的,但从K&C第二版标准开始就明确指出“修改string literal的结果是未定义的,字符串并不总是可修改的,可能会放在只读内存区域“。从C89开始,这个问题有了更明确的规定。char *a = "hello world";定义a是一种字符指针类并使用指定的字符数组(类型为array of char)“hello world"初始化,如果试图修改数组的内容行为将是未定义的。关键的解释就在"未定义“这个解释上,实际上未定义不是错误,它在这里表示代码不可移值。所以在c89和c99规范的”公共扩展“部分有如下规则:String literals是可修改的,此时指向另外一个内容完全相同的对象。在c++中则有区别,c++2003标准规定string literal是由const字符数组构成的(array of n const char),所以C++明确了string literal是不可修改的。C++没有规定所有内容相同的string literal会指向同一个对象(相同的内容指向唯一的对象),这是由编译器实现决定的,同时指出了试图修改string literal的行为是未定义的。为了兼容C,所以不加const 的char*依然是可用的,但通常编译器会给出警告。最后C++标准明确规定不赞成这样做(deprecated)。 修饰为deprecated表示将来不再支持这个特性。


这个问题要解释清楚就得看标准了。下面是相关标准中的信息摘要:

C99
char *p = "abc";
defines p with type ‘‘pointer to char’’ and initializes it to point to an object with type ‘‘array of char’’
with length 4 whose elements are initialized with a character string literal. If an attempt is made to use p to
modify the contents of the array, the behavior is undefined.

J.5.5 Writable string literals
1 String literals are modifiable (in which case, identical string literals should denote distinct
objects) (6.4.5).

C++2003
An ordinary string literal has type “array of n
const char” and static storage duration.

Whether all string literals are distinct (that is, are stored in nonoverlapping objects) is implementationdefined.
The effect of attempting to modify a string literal is undefined.

D.4 Implicit conversion from const strings [depr.string]
1 The implicit conversion from const to non-const qualification for string literals (4.2) is deprecated.
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP