C Обновление файла и создание справки - PullRequest
2 голосов
/ 09 февраля 2011

У меня есть два файла .dat (ascii).Оба отсортированы.

1: файл клиента, содержащий;номер счета, имя, баланс 2: файл транзакции, содержащий;номер счета, дата, сумма продажи (сумма транзакции)

То, что я пытаюсь выполнить, - это создать новый обновленный файл клиентов, в котором обновлены сальдо для клиентов на основе сложения или вычитания суммы продажи соответствующей транзакции.

Пока мой код позволяет мне:

1: если для клиента не более одной транзакции, код выполняется отлично, записывает файл .dat с клиентами и их обновленными балансами.

2: если для клиента имеется более одной транзакции, мой код будет работать практически идеально, поскольку он выведет на экран обновленных клиентов и учетных записей, например:

1 james 540.00 2 john 762.00 3Пол 414,00 4 Сам 502,00

будет отображаться, но поскольку есть две транзакции для Джона созданный файл .dat, в то время как содержат

1 Джеймс 540,00 2 Джон 662,00 2 Джон 762,00 3 Пол 414,00 4 Сам502.00

Моя проблема здесь, мне нужно найти способ, чтобы созданный .dat содержал только одну строку для each клиент (номер счета)

Мой код прилагается, любая помощь будет принята с благодарностью.

#include <stdio.h>
#include <string.h>
int main(void)
{ 
    int account, matches=0;     /* account number */
    char date[ 30 ]; /* account Date */
    double balance, saleamount,total=0, temp;;  /* account SaleAmount */
    int transaccount;
    char name [ 30 ];
    char lastname[30];
    int lastaccount=-1;
    double lastbalance;



    FILE *cfPtr;     /* cfPtr = clients.dat file pointer */
    FILE *ctPtr;    /* cfPtr = transaction.dat file pointer */
    FILE *cfPtr2;    /* cfPtr2 = new client file */

    cfPtr2 = fopen( "clientupdate.dat", "w" );

    /* fopen opens file; exits program if file cannot be opened */ 
    if ( ( cfPtr = fopen( "clients.dat", "r" ) ) == NULL ) {
        printf( "clients could not be opened\n" );
        /*fflush(stdin);*/
    } /* end if */
    else
        if( ( ctPtr = fopen( "transactions.dat", "r" ) ) == NULL)
        {
            printf( "File could not be opened\n" );
            /*fflush(stdin);*/
        }
    else { /* read account, date,name, balance and SaleAmount from files */
        fscanf( cfPtr, "%d%s%lf", &account, &name, &balance );
        /*fflush(stdin);*/

        fscanf( ctPtr, "%d%s%lf", &transaccount, &date, &saleamount );
        /*fflush(stdin);*/

        printf( "%-13s%-10s%s\n", "  Account", "Name", "Balance" );
        printf("|----------------------------------|\n");
        while( !feof(ctPtr))
        {   

            while( !feof(cfPtr) &&matches==0 )
            {   
                if(account == transaccount)
                {
                    if (lastaccount != account) {
                    if (lastaccount != -1)
                    printf("  %-10d%-10s%.2lf\n", lastaccount, lastname, lastbalance);
                    lastaccount = account;
                    strcpy (lastname, name);
}

                    matches=1;
                    total=0;
                    temp = balance+saleamount;
                    total = total + temp;
                    balance = total; 
                    lastbalance = balance;
                }
                else
                {

                    fscanf( cfPtr, "%d%s%lf", &account, &name, &balance );
                    /*fflush(stdin);*/
                }   

            }
            fprintf( cfPtr2, "%d %s %.2lf\n", account, name, total );
            fscanf( ctPtr, "%d%s%lf", &transaccount, &date, &saleamount );
            /*fflush(stdin);*/
           matches=0;

        }
    }
       if (lastaccount != -1)
    printf("  %-10d%-10s%.2lf\n", lastaccount, lastname, lastbalance);
        fclose( cfPtr2 );
        getchar();
    return 0;
}

Ответы [ 3 ]

1 голос
/ 09 февраля 2011

Как и было обещано, вот краткий пример обновления «отец-сын». Я не потрудился проверить наличие ошибок открытия файлов, так как они уже есть в вашем коде, и реальное приложение должно проверять не только EOF на наличие fscanf, чтобы обнаружить неправильное форматирование файла.

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

#define bool int
#define false 0
#define true !false

#define MAX_ACCOUNT_NUMBER 99999

typedef struct
{
 int accountNumber;
 char accountName[30];
 float accountTotal;
} ClientRecordType;

typedef struct
{
 int accountNumber;
 char transactionDate[30];
 float transactionAmount;
} TransactionRecordType;

FILE *oldClientFile;
FILE *newClientFile;
FILE *transactionFile;


void getNextClient(FILE *p_clientFile, ClientRecordType *p_clientRecord)
{
 if (EOF == fscanf(p_clientFile, "%d%s%f", 
                   &p_clientRecord->accountNumber,
                   &p_clientRecord->accountName,
                   &p_clientRecord->accountTotal))
   p_clientRecord->accountNumber = MAX_ACCOUNT_NUMBER;
}


void getNextTransaction(FILE *p_transactionFile,
                        TransactionRecordType *p_transactionRecord)
{
 if (EOF == fscanf(p_transactionFile, "%d%s%f", 
                   &p_transactionRecord->accountNumber,
                   &p_transactionRecord->transactionDate,
                   &p_transactionRecord->transactionAmount))
   p_transactionRecord->accountNumber = MAX_ACCOUNT_NUMBER;
}


void writeUpdatedClientRecord(FILE *p_newClientFile,
                              ClientRecordType *p_clientRecord)
{
 fprintf(p_newClientFile, "%d %s %.2f ",
         p_clientRecord->accountNumber,
         p_clientRecord->accountName,
         p_clientRecord->accountTotal);
} 


bool performTransactionUpdate(FILE *p_oldClientFile,
                              FILE *p_newClientFile,
                              FILE *p_transactionFile)
{
 ClientRecordType clientRecord;
 TransactionRecordType transactionRecord;

 getNextClient(p_oldClientFile, &clientRecord);
 getNextTransaction(p_transactionFile, &transactionRecord);
 while (MAX_ACCOUNT_NUMBER != clientRecord.accountNumber)
 {
  if (clientRecord.accountNumber == transactionRecord.accountNumber)
  {
   clientRecord.accountTotal += transactionRecord.transactionAmount;
   getNextTransaction(p_transactionFile, &transactionRecord);
  }
  else if (clientRecord.accountNumber < transactionRecord.accountNumber)
  {
   writeUpdatedClientRecord(p_newClientFile, &clientRecord);
   getNextClient(p_oldClientFile, &clientRecord);
  }
  else
  {
   return false;
  }
 }
 if (MAX_ACCOUNT_NUMBER != transactionRecord.accountNumber) return false;
 return true;
}


int main(int argc, char *argv[])
{
 oldClientFile = fopen("clients.dat", "r");
 newClientFile = fopen("newclients.dat", "w");
 transactionFile = fopen("transactions.dat", "r");

 if (performTransactionUpdate(oldClientFile,
                              newClientFile,
                              transactionFile))
   printf("\nUpdate completed without errors!\n");
 else
   printf("\nUnsorted files or invalid transactions encountered\n");

 close(newClientFile);
 close(oldClientFile);
 close(transactionFile);
 return 0;
}
0 голосов
/ 10 февраля 2011

Я закончил с помощью опубликованного ответа, и один из учеников придумал этот рабочий код. Спасибо, что помогли мне понять, как заставить это работать.

#include <stdio.h>
#include <string.h>
int main(void)
{ 
    int account;     /* account number */
    char date[ 30 ]; /* account Date */
    double balance, saleamount; /* account SaleAmount */
    int transaccount;
    char name [ 30 ];
    int newaccount;
    char newname[ 30];
    double newbalance;
    int match=0;



    FILE *cfPtr;     /* cfPtr = clients.dat file pointer */
    FILE *ctPtr;    /* cfPtr = transaction.dat file pointer */
    FILE *cfPtr2;    /* cfPtr2 = new client file */

    cfPtr2 = fopen( "clientupdate.dat", "w" );

    /* fopen opens file; exits program if file cannot be opened */ 
    if ( ( cfPtr = fopen( "clients.dat", "r" ) ) == NULL ) {
        printf( "clients could not be opened\n" );

    } /* end if */
    else
        if( ( ctPtr = fopen( "transactions.dat", "r" ) ) == NULL)
        {
            printf( "File could not be opened\n" );

        }
    else { /* read account, date,name, balance and SaleAmount from files */
        fscanf( cfPtr, "%d%s%lf", &account, &name, &balance );


        fscanf( ctPtr, "%d%s%lf", &transaccount, &date, &saleamount );
        fscanf( cfPtr2, "%d%s%lf", &newaccount, &newname, &newbalance );
        while ( !feof( cfPtr ) )
        {
            transaccount=1;
            newbalance=balance;
            while ( !feof( ctPtr ) )
            {
                if(transaccount==account&&match==0)
                {
                 newbalance = balance + saleamount;
                 match++;
                }
                else
                    if(transaccount==account&&match>0)
                     newbalance = newbalance + saleamount;

                fscanf( ctPtr, "%d%s%lf", &transaccount, date, &saleamount );
            }
            rewind( ctPtr);
            match=0;

        fprintf( cfPtr2, "%d %s %.2f\n", account, name, newbalance );
        fscanf( cfPtr, "%d%s%lf", &account, name, &balance );
        }

        rewind( cfPtr2);
    }
        fclose( cfPtr2 );
        fclose( cfPtr );
        fclose( ctPtr );
        getchar();
    return 0;
}
0 голосов
/ 09 февраля 2011

Типичным подходом было бы загрузить файл клиента в память, обработать файл транзакции для каждого клиента и затем сбросить данные клиента обратно на диск.

Если вы предпочитаете делать все по-старому, вы, конечно, можете отсортировать файл транзакции в клиентский заказ до обработки.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...