渠道公函No.17 GetSignStrArray

1.GetSignStrArray说明

这个公函的主要是将分隔符的字符串,分割并保存到一个二维指针里去。

1.1先贴公函代码

#include "kernel/syspub.h"

int GetSignStrArray(char *s, char **m, const char *delimiters, char ***argvp)
{

    char   *t;
    char   *snew, *p, *q, *k;
    int    numtokens;
    int    i, j, l, f;

    snew = s;

    if ((t = (char *)calloc(strlen(snew) + 1, sizeof(char))) == NULL)
    {
       argvp = NULL;
       numtokens = -1;
    }
    else
    {
       strcpy(t, snew);
       numtokens = 0;   /*****参数个数*****/
       p = t;
       for(;;)
       {
          q = strpbrk(p, delimiters);
          if (q == NULL) break;
          l = q - p;
          for(j = 0, f = 0; j < l; j++)
             if (p[j] == '\\') f++;
          if (l > 0 && *(q - 1) == '\\' && f % 2 )
          {
             q++;
             p = q;
             continue;
          }
          q++;
          p = q;
          numtokens++;
       }

       if ((*argvp = calloc(numtokens + 1, sizeof(char * ))) == NULL)
       {
          free(t);
          numtokens = -1;
       }
       else
       {
          strcpy(t, snew);
          if (numtokens == 0) **argvp = t;
          else
          {
             k = t;
             *m = t;
             p = k;
             for(i = 0; i < numtokens;)
             {
                q = strpbrk(p, delimiters);
                l = q - p;
                for(j = 0, f = 0; j < l; j++)
                   if (p[j] == '\\') f++;
                if (l > 0 && *(q - 1) == '\\' && f % 2)
                {
                   q++;
                   p = q;
                   continue;
                3}
                *q = 0;
                *((*argvp) + i) = k;
                q++;
                p = q;
                k = p;
                i++;
             }
          }
       }
    }

    return numtokens;

}

1.2参数说明

  • 1.Buf参数,欲分割的字符串。
  • 2.&m参数, m为一个char 型的指针。将这个指针的地址传入,则可以给该指针赋值。
    该字段传入之后,需要在外部进行free。
  • 3."|"参数,分割符,是字符串型的。
  • 4.&arg参数,char **arg, 是一个二维指针。给二维指针和一维指针赋值。
    该字段传入之后,也需要在外部进行free。

1.3调用的例子程序

#include "kernel/syspub.h"

int main()
{
    char Buf[51]="bankAcc|Na\\|me|123.45|1|";

    char **arg=NULL;
    char *m=NULL;
    int num =0;
    
    num = GetSignStrArray( Buf, &m, "|",  &arg);

    printf("[%s] [%d]\n", m, num );
    printf("[%s] [%s] [%s] [%s]\n", arg[0], arg[1],arg[2], arg[3]);

    free(m);
    free(arg);

    return 0;
}
/***执行结果***
标准版渠道开发/stdcop/xip/src/kernel/kpublib.1.1/sample>GetSignStrArray_samp 
[bankAcc] [4] 
[bankAcc] [Na\|me] [123.45] [1]
***************/

2.看程序的时候的一些疑惑

问题 为什么要传入参数m,而且要free?

这个需要详细分析一下程序内容。程序内部calloc了一个和Buf大小相同的内存,将内存的地址赋值给了m。然后会把分隔符改写成'\0'。 然后提供给arg[n]来使用。所以如果直接使用Buf的内存,不能保证Buf的内存里的内容是否可以被修改。如果是常量字符串,直接修改静态区的常量会导致程序core。

问题 arg是什么,为什么在free(m)之后,还需要free(arg),arg的指针和m的指针是否有重复的地方?

程序中: if ((*argvp = calloc(numtokens + 1, sizeof(char * ))) == NULL)
*argvp,就是咱们的二维指针的第一维arg, 所以上面的代码,相当于为argvp分配了一段内存,在这段内存保存的数据,是*argvp类型的数据。
程序中 *(*argvp) + i) = k; 可以简化的看成给*(*argvp)赋值。就是给存储在*argvp里面的元素(实际是指针)进行赋值,让它们指向m申请的内存,前面问题分析了, m内存被分成不等额的连续的内存段,每一段的地址现在就是(*argvp)内的元素的值。
所以,free(arg)是释放 arg占用的内存,内部存储的是arg[0]到arg[n]这一个个指针元素。 free(m)是释放 arg[0]指向的内存, 可以理解为*arg[0]的内存。

在例子程序程序中,直接使用代码:
printf("[%s] [%s] [%s] [%s]\n", arg[0], arg[1],arg[2], arg[3]);
那么,这是数组的写法,arg到底是指针还是数组?

这个问题的详细内容在《C专家编程》用了两章的时间来讨论,参见《C专家编程》第九章再论数组和第十章再论指针。
在该程序里这样写的神奇之处在于,如果你不清晰的知道你是在使用一个指针,而认为arg是一个二维数组的话,你会发现,arg[0]和arg[1]的长度是不一样的。但是二维数组的每个一维数组的长度是不对等的,这在二维数组声明是无法实现的,因为你不知道arg[1]也就是arg+1的这个1是以多少位单位的。
所以,在这里,结论是arg是一个指针而不是数组,只是在arg写法上将 arg+1*sizeof(char*) 等价写成了arg[1]。而这种写法,从表象上得出了一个arg[][]类型的二维数组,与用小无相功模拟的72绝技有异曲同工之妙。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容