Código C - preciso esclarecer a eficácia
Pergunta
Olá, escrevi um código com base em um requisito.
(Field1_6)(Field2_30)(Field3_16)(Field4_16)(Field5_1)(Field6_6)(Field7_2)(Field8_1)..... este é um balde (8 campos) de dados. Receberemos 20 baldes de cada vez significa totalmente 160 campos. Preciso tomar os valores de campo3, Field7 e Fields8 com base em condições predefinidas. Se o argumento de entrada for n, pegue os três campos do 1º balde e, se for Y, preciso pegar os três campos de qualquer outro balde que não seja o 1º. Se o Argumet estiver y, então eu preciso digitalizar todos os 20 baldes um após o outro e verificar o primeiro campo do balde não é igual a 0 e, se for verdadeiro, pegue os três campos desse balde e saída. Eu escrevi o código e também está funcionando bem ... mas não tão confiante de que ele é eficaz. Tenho medo de um acidente algum tempo. Por favor, sugestão abaixo o código.
int CMI9_auxc_parse_balance_info(char *i_balance_info,char *i_use_balance_ind,char *o_balance,char *o_balance_change,char *o_balance_sign
)
{
char *pch = NULL;
char *balance_id[MAX_BUCKETS] = {NULL};
char balance_info[BALANCE_INFO_FIELD_MAX_LENTH] = {0};
char *str[160] = {NULL};
int i=0,j=0,b_id=0,b_ind=0,bc_ind=0,bs_ind=0,rc;
int total_bukets ;
memset(balance_info,' ',BALANCE_INFO_FIELD_MAX_LENTH);
memcpy(balance_info,i_balance_info,BALANCE_INFO_FIELD_MAX_LENTH);
//balance_info[BALANCE_INFO_FIELD_MAX_LENTH]='\0';
pch = strtok (balance_info,"*");
while (pch != NULL && i < 160)
{
str[i]=(char*)malloc(strlen(pch) + 1);
strcpy(str[i],pch);
pch = strtok (NULL, "*");
i++;
}
total_bukets = i/8 ;
for (j=0;str[b_id]!=NULL,j<total_bukets;j++)
{
balance_id[j]=str[b_id];
b_id=b_id+8;
}
if (!memcmp(i_use_balance_ind,"Y",1))
{
if (atoi(balance_id[0])==1)
{
memcpy(o_balance,str[2],16);
memcpy(o_balance_change,str[3],16);
memcpy(o_balance_sign,str[7],1);
for(i=0;i<160;i++)
free(str[i]);
return 1;
}
else
{
for(i=0;i<160;i++)
free(str[i]);
return 0;
}
}
else if (!memcmp(i_use_balance_ind,"N",1))
{
for (j=1;balance_id[j]!=NULL,j<MAX_BUCKETS;j++)
{
b_ind=(j*8)+2;
bc_ind=(j*8)+3;
bs_ind=(j*8)+7;
if (atoi(balance_id[j])!=1 && atoi( str[bc_ind] )!=0)
{
memcpy(o_balance,str[b_ind],16);
memcpy(o_balance_change,str[bc_ind],16);
memcpy(o_balance_sign,str[bs_ind],1);
for(i=0;i<160;i++)
free(str[i]);
return 1;
}
}
for(i=0;i<160;i++)
free(str[i]);
return 0;
}
for(i=0;i<160;i++)
free(str[i]);
return 0;
}
Solução
Eu tive dificuldade em ler seu código, mas FWIW adicionei alguns comentários, HTH:
// do shorter functions, long functions are harder to follow and make errors harder to spot
// document all your variables, at the very least your function parameters
// also what the function is suppose to do and what it expects as input
int CMI9_auxc_parse_balance_info
(
char *i_balance_info,
char *i_use_balance_ind,
char *o_balance,
char *o_balance_change,
char *o_balance_sign
)
{
char *balance_id[MAX_BUCKETS] = {NULL};
char balance_info[BALANCE_INFO_FIELD_MAX_LENTH] = {0};
char *str[160] = {NULL};
int i=0,j=0,b_id=0,b_ind=0,bc_ind=0,bs_ind=0,rc;
int total_bukets=0; // good practice to initialize all variables
//
// check for null pointers in your arguments, and do sanity checks for any
// calculations
// also move variable declarations to just before they are needed
//
memset(balance_info,' ',BALANCE_INFO_FIELD_MAX_LENTH);
memcpy(balance_info,i_balance_info,BALANCE_INFO_FIELD_MAX_LENTH);
//balance_info[BALANCE_INFO_FIELD_MAX_LENTH]='\0'; // should be BALANCE_INFO_FIELD_MAX_LENTH-1
char *pch = strtok (balance_info,"*"); // this will potentially crash since no ending \0
while (pch != NULL && i < 160)
{
str[i]=(char*)malloc(strlen(pch) + 1);
strcpy(str[i],pch);
pch = strtok (NULL, "*");
i++;
}
total_bukets = i/8 ;
// you have declared char*str[160] check if enough b_id < 160
// asserts are helpful if nothing else assert( b_id < 160 );
for (j=0;str[b_id]!=NULL,j<total_bukets;j++)
{
balance_id[j]=str[b_id];
b_id=b_id+8;
}
// don't use memcmp, if ('y'==i_use_balance_ind[0]) is better
if (!memcmp(i_use_balance_ind,"Y",1))
{
// atoi needs balance_id str to end with \0 has it?
if (atoi(balance_id[0])==1)
{
// length assumptions and memcpy when its only one byte
memcpy(o_balance,str[2],16);
memcpy(o_balance_change,str[3],16);
memcpy(o_balance_sign,str[7],1);
for(i=0;i<160;i++)
free(str[i]);
return 1;
}
else
{
for(i=0;i<160;i++)
free(str[i]);
return 0;
}
}
// if ('N'==i_use_balance_ind[0])
else if (!memcmp(i_use_balance_ind,"N",1))
{
// here I get a headache, this looks just at first glance risky.
for (j=1;balance_id[j]!=NULL,j<MAX_BUCKETS;j++)
{
b_ind=(j*8)+2;
bc_ind=(j*8)+3;
bs_ind=(j*8)+7;
if (atoi(balance_id[j])!=1 && atoi( str[bc_ind] )!=0)
{
// length assumptions and memcpy when its only one byte
// here u assume strlen(str[b_ind])>15 including \0
memcpy(o_balance,str[b_ind],16);
// here u assume strlen(str[bc_ind])>15 including \0
memcpy(o_balance_change,str[bc_ind],16);
// here, besides length assumption you could use a simple assignment
// since its one byte
memcpy(o_balance_sign,str[bs_ind],1);
// a common practice is to set pointers that are freed to NULL.
// maybe not necessary here since u return
for(i=0;i<160;i++)
free(str[i]);
return 1;
}
}
// suggestion do one function that frees your pointers to avoid dupl
for(i=0;i<160;i++)
free(str[i]);
return 0;
}
for(i=0;i<160;i++)
free(str[i]);
return 0;
}
Uma técnica útil quando você deseja acessar compensações em uma matriz é criar uma estrutura que mapeie o layout da memória. Então você lança seu ponteiro para um ponteiro da estrutura e usa os membros da estrutura para extrair informações em vez de seus vários memcpy
Eu também sugiro que você reconsidere seus parâmetros à função em geral, se você colocar todos eles em uma estrutura, você tem melhor controle e torna a função mais legível por exemplo
int foo( input* inbalance, output* outbalance )
(ou o que quer que você esteja tentando fazer)
Outras dicas
Meu sentimento é que esse código é muito quebradiço. Pode muito bem funcionar quando tiver uma boa entrada (não proponho a mesa, verifique a coisa para você), mas se receber algumas entradas incorretas, ele travará e queimará ou fornecerá resultados enganosos.
Você já testou para insumos inesperados? Por exemplo:
- Suponha que i_balance_info seja nulo?
- Suponha que i_balance_info seja ""?
Suponha que haja menos de 8 itens na sequência de entrada, o que essa linha de código fará?
memcpy(o_balance_sign,str[7],1);
Suponha que o item em STR [3] tenha menos de 16 chars de comprimento, o que essa linha de código fará?
memcpy(o_balance_change,str[3],16);
Minha abordagem para escrever esse código seria proteger contra todas essas eventualidades. No mínimo, eu adicionaria declarações assert (), normalmente escreveria a validação explícita e os erros de retorno quando for ruim. O problema aqui é que a interface não parece permitir nenhuma possibilidade de que haja uma entrada ruim.