写作日期:2016.08.31

修改日期:2016.09.01 、2016.09.02。

交流qq:992591601

用了几天时间复习了下C语言。对于C语言的字符串操作有些不习惯,于是作为练习,写下了下面这样错误的程序:

  1. #include <stdio.h>
  2. #define MAX_STR_SIZE 100
  3. void copy_string(char* from, char* to) {
  4. while (*to++ = *from++);
  5. }
  6. void swap_strs(char* str1, char* str2) {
  7. char tmp[MAX_STR_SIZE];
  8. copy_string(str1, tmp);
  9. copy_string(str2, str1);
  10. copy_string(tmp, str2);
  11. }
  12. void sort_strs_by_ascii(char* arr[3]) {
  13. if (strcmp(arr[0], arr[1]) < 0)
  14. swap_strs(arr[0], arr[1]);
  15. if (strcmp(arr[1], arr[2]) < 0)
  16. swap_strs(arr[1], arr[2]);
  17. if (strcmp(arr[0], arr[1]) < 0)
  18. swap_strs(arr[0], arr[1]);
  19. }
  20. int main() {
  21. char* arr[3] = {"dd", "aa", "cc"};
  22. sort_strs_by_ascii(arr);
  23. for (int i = 0; i < 3; i++)
  24. printf("%s\n", arr[i]);
  25. return 0;
  26. }

这段程序写得当然不怎么样,主要是为了练习C语言,特意去使用一些东西,例如C字符串指针的,指针数组。

程序中的数组的中存放三个字符串指针。将该数组作为参数来通过sort_strs_by_ascii方法排序。但在字符串操作过程中报错。

之后我才了解,char* arr这样的字符串指针,指向的值是存放在常量区的,不可改写。但该指针可以随意指向其它的地址空间。

而char arr[]这样的数组指针虽然本质也是指针,指针指向的内容却是固定在数组的内存空间的。但该数组空间里的内容是可以改写的。

接下来,我也大言不惭地用反汇编技术来解释下char* arr和char arr[]的区别。

截图部分第一个白色框是char arr[]初始化对应的汇编语句,可以清晰看到其过程。首先字符串本身是在静态区。白框的语句是三个赋值过程,将静态区0x00D95858地址的内容赋给arr数组地址。所以并不是仅仅让arr指针指向0x00D95858。而是实实在在地将该地址的内容赋过来。因为字符串比较短,所以编译器编译的汇编语言是三个赋值语句,如果是很长的字符串,这个过程一定是循环。

第二个白框是,char* arr的过程,很简单,就是指向静态区对应的字符串地址就好了。

所以可以知道,不管哪种方式,xxx = "abcd",便意味着字符串赋值操作,C语言就会首先在静态区分配内存存储该字符串。接下来,采用char arr[] 或char* arr,则各有各的方式。

为了解决上面那个程序,使用了C语言的二级指针。

基本思路就是:

先建立用指针数组,里面每一个指针指向一个字符串数组。再使用二级指针来指向这个指针数组。

这样就构造了一种自制的“二维数组”。达到了我程序想达到的效果。当然这程序只是为了练习,不值一提~

  1. ##include <stdio.h>
  2. #define MAX_STR_SIZE 100
  3. void copy_string(char* from, char* to) {
  4. while (*to++ = *from++);
  5. }
  6. void swap_strs(char* str1, char* str2) {
  7. char tmp[MAX_STR_SIZE];
  8. copy_string(str1, tmp);
  9. copy_string(str2, str1);
  10. copy_string(tmp, str2);
  11. }
  12. void sort_strs_by_ascii(char **p) {
  13. if (strcmp(*p, *(p + 1)) < 0)
  14. swap_strs(*p, *(p + 1));
  15. if (strcmp(*(p + 1), *(p + 2)) < 0)
  16. swap_strs(*(p + 1), *(p + 2));
  17. if (strcmp(*p, *(p + 1)) < 0)
  18. swap_strs(*p, *(p + 1));
  19. }
  20. int main() {
  21. char arr0[] = "aaaaaa";
  22. char arr1[] = "cccccc";
  23. char arr2[] = "bbbbbb";
  24. char* p0 = arr0;
  25. char* p1 = arr1;
  26. char* p2 = arr2;
  27. char* final_arr[3];
  28. final_arr[0] = p0;
  29. final_arr[1] = p1;
  30. final_arr[2] = p2;
  31. char **p3 = final_arr;
  32. sort_strs_by_ascii(p3);
  33. for (int i = 0; i < 3; i++) {
  34. printf("%s\n", final_arr[i]);
  35. }
  36. return 0;
  37. }

C/C++ char* arr与char arr[]的区别(反汇编解析)的更多相关文章

  1. char* strcpy( char* dest, const char* src ), int binary_search(int *arr, int key, int n), 可能的实现

    #include <stdio.h> char* stringCopy( char* dest, const char* src ) { size_t i = 0; while (dest ...

  2. char[]数组与char *指针的区别

    char[]数组与char *指针的区别 问题描述 虽然很久之前有看过关于char指针和char数组的区别,但是当时没有系统的整理,到现在频繁遇到,在string,char[], char *中迷失了 ...

  3. no suitable ctr exists to convert from 'int' to 'std::basic_string<char,std::char_traits<char>,std::allocator<char> >

    int xfun(int *a,int n) { int x = *a;//a的类型是int *,a+1跳动一个int的长度 ; pa < a + n; pa++)//指向同一个类型的指针比较大 ...

  4. C语言执行时报错“表达式必须是可修改的左值,无法从“const char [3]”转换为“char [120]” ”,原因:字符串不能直接赋值

    解决该问题的方法:使用strcpy函数进行字符串拷贝   原型声明:char *strcpy(char* dest, const char *src); 头文件:#include <string ...

  5. 【转】深入理解const char*p,char const*p,char *const p,const char **p,char const**p,char *const*p,char**const p

    一.可能的组合: (1)const char*p (2)char const*p (3)char *const p(4)const char **p (5)char const**p (6)char ...

  6. 初始化char指针--赋值和strcpy() 本质区别【转】

    原文地址:http://hi.baidu.com/todaygoodhj/item/0500b341bf2832e3bdf45180 使用常量字符串初始化char指针,或者使用strcpy复制,从语法 ...

  7. char *p 与char p[] 比较

    看看下面的程序的输出: #include <stdio.h>char *returnStr(){    char *p="hello world!";    retur ...

  8. char str[]和char *str的区别

    1.http://blog.csdn.net/szchtx/article/details/10396149 char ss[]="C++";  ss[0]='c';        ...

  9. char *c和char c[]区别

    char *c和char c[]区别 问题引入:在实习过程中发现了一个以前一直默认的错误,同样char *c = "abc"和char c[]="abc",前者 ...

随机推荐

  1. 判断Window在哪个屏幕

    最近在做窗口最大化时需要一个功能,如果是多个显示器的话,需要在当前显示器最大化,由于是根据屏幕长宽进行设置Window大小,没有使用WindowState.Maximized,window.Left不 ...

  2. 爱上WPF,努力才会有希望!

    从WinForm转向WPF开发已经有两个多月了,通过不断深入地学习与运用,现在是越来越爱它了.它实在是太强大了.运用WPF,你不仅可以做Win界面,也可以很快转向Web开发,因为Silverlight ...

  3. Unity3D将来时:WebGL

    作者:小玉链接:https://zhuanlan.zhihu.com/p/19974794来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注明出处. 随着Unity5.0的发布,W ...

  4. 『TCP/IP详解——卷一:协议』读书笔记——11

    2013-08-23 20:00:18 第4章 ARP:地址解析协议 4.1 引言 ARP(Address Resolution Protocol,地址解析协议)是获取物理地址的一个TCP/IP协议. ...

  5. Codeforces Zip-line 650D 345Div1D(LIS)

    传送门 大意:给出一个序列,求修改一个数过后的最长上升子序列. 思路:可以用主席树在线搞,也可以用树状数组离线搞,明显后者好写得多.我们首先读取所有的询问,然后就把询问绑在给出的位置,然后我们正向做一 ...

  6. video标签的属性和方法总结

    最近想做一个弹幕插件,查了很多video标签的属性和方法 error属性 在正常读取时候,使用媒体数据的过程中,video元素或audio元素的error属性为null,但是任何时候只要出现错误,er ...

  7. 用sql取出来的list需要处理成map的两种情况

    1. 原生sql: select a.id,a.name from a SQLQuery sqlQuery=this.getSession().createSQLQuery(sb.toString() ...

  8. oracl 创建用户

    -- Create the user create user ADMIN  default tablespace SYSTEM  temporary tablespace TEMP  profile ...

  9. 决策树 -- C4.5算法

    C4.5是另一个分类决策树算法,是基于ID3算法的改进,改进点如下: 1.分离信息   解释:数据集通过条件属性A的分离信息,其实和ID3中的熵:   2.信息增益率   解释:Gain(A)为获的A ...

  10. 在浏览器的JavaScript里new Date().toUTCString()后,传递给C# DateTime().TryParse()会发生什么?

    Format 1. Sun, 09 Oct 2016 13:24:35 GMT Format 2. Sun, 9 Oct 2016 13:36:09 UTC Format 1 是在IE里面产生的(Wi ...