通过修改数组

时间:2015-10-07 23:45:32

标签: c arrays pointers

我正在尝试将apache日志文件的块组织到一个数组中。例如,假设我的apache文件有这样一行:

[a] [b] [ab] [abc] file not found: /something

我想要实现的是一个数组(让我们将其命名为ext),以便:

ext[0] = a
ext[1] = b
ext[2] = ab
ext[3] = abc

然后,我通过以下方式为每个5000个字符的20个条目保留足够的空间:

char ext[20][5000];

然后我尝试按如下方式调用我的提取函数:

extract("[a] [b] [ab] [abc]",18,ext);

理想情况下,字符串被保存数据的变量替换,18被替换为显示实际字符串大小的变量,但我使用此数据作为示例。

提取函数无法编译。

抱怨说:

char s[20][5000]=*extr,*p,*l=longstring;

初始化程序无效。我猜这是s[20][5000]=*extr,但我正在尝试用索引值初始化一个字符数组,然后我想把它传递给函数调用者

然后抱怨:

warning: passing argument 3 of 'extract' from incompatible pointer type
  

我是否被迫严格使用指针和数学来计算偏移量,或者有没有办法传递实际的char数组,并且能够使用索引值修改它们,就像我试图做的那样?

long extract(char* longstring,long sz,char **extr){
    unsigned long sect=0,si=0,ssi=0;
    char s[20][5000]=*extr,*p,*l=longstring;
    while (sz-- > 0){
        if (*l=='['){sect=1;p=s[si++];if (si > 20){break;}}
        if (*l==']'){sect=0;}else{
        if (sect==1){*p++=*l;}
        }
        l++;
    }
}

更新

根据建议,我做了一些小改动,现在我的代码如下:

主线:

char ext[20][5000];
extract("[a] [b] [ab] [abc]",18,(char**)ext);
printf("%s\n",ext);
return 0;

功能:

long extract(char* longstring, long sz, char **extr) {
    unsigned long sect = 0, si = 0, ssi = 0;
    char **s = extr, *p, *l = longstring;
    while (sz-- > 0) {
        if (*l == '[') {
            sect = 1;
            p = s[si++];
            if (si > 20) {
                break;
            }
        }
        if (*l == ']') {
            sect = 0;
        } else {
            if (sect == 1) {
                *p++ = *l;
            }
        }
        l++;
    }
}

现在我收到了分段错误。我不确定为什么当我通过p=s[si++]设置一个字符串的偏移量然后在添加数据时递增它。我甚至将p=s[si++]更改为p=s[si++][0],试图特别想要特定索引的第一个字符的地址,但编译器显示“警告:赋值使得指针来自整数而没有强制转换”。

1 个答案:

答案 0 :(得分:1)

这使用扫描集%[]来解析字符串。扫描会跳过前导空格,然后扫描[。然后,扫描集读取的字符不是]。最后扫描]%n说明符报告处理的字符数,并将其添加到offset以通过字符串前进。 4999可以防止在字符串[5000]中写入太多字符。

#include <stdio.h>
#include <stdlib.h>

int extract ( char* longstring,char (*extr)[5000]) {
    int used = 0;
    int offset = 0;
    int si = 0;

    while ( ( sscanf ( longstring + offset, " [%4999[^]]]%n", extr[si], &used)) == 1) {
         //one item successfully scanned
        si++;
        offset += used;
        if ( si > 20) {
            break;
        }
    }
    return si;
}

int main( int argc, char *argv[])
{
    char ext[20][5000];
    int i = 0;
    int result = 0;

    result = extract("[a] [b] [ab] [abc]", ext);
    for ( i = 0; i < result; i++) {
        printf("ext[%d]  %s\n",i,ext[i]);
    }

    return 0;
}