নিরাপদে দুটি সংখ্যা যুক্ত করুন Add


24

সকলেই জানেন যে সি একটি সুন্দর, নিরাপদ, উচ্চ স্তরের প্রোগ্রামিং ভাষা। তবে আপনি, কোডার হিসাবে নিম্নলিখিত কাজটি সেট করেছেন।

দুটি সংখ্যা যুক্ত করতে একটি প্রোগ্রাম লিখুন।

  • ইনপুট: দুটি স্থান পৃথক পূর্ণসংখ্যা।
  • আউটপুট: ইনপুটটিতে দুটি সংখ্যার যোগফল।

মোড় म्हणजे আপনার কোডটি অবশ্যই 100% নিরাপদ থাকতে হবে। অন্য কথায়, ইনপুটটি যাই হোক না কেন এটি অবশ্যই সঠিকভাবে আচরণ করবে। যদি ইনপুটটি দুটি স্থান পৃথক পূর্ণসংখ্যা হয়, উভয়ই 100 অঙ্কের চেয়ে কম দীর্ঘ হয় তবে এটি অবশ্যই যোগফলকে আউটপুট করে। অন্যথায়, এটি একটি ত্রুটি বার্তা আউটপুট এবং নিরাপদে প্রস্থান করতে হবে।

সর্বোপরি এটি কতটা কঠিন হতে পারে?

জেনারেল কুদোগুলি প্যাথলজিকাল ইনপুট কেস দেওয়া হবে যা অন্যান্য লোকের উত্তরকে ভঙ্গ করে :)

উবুন্টুতে জিসিসি -ওয়াল-ওয়েক্সট্রা ব্যবহার করে কোডটি অবশ্যই সতর্কতা ছাড়াই সংকলন করতে হবে।


ব্যাখ্যা।

  • ইনপুট স্টিডিনের।
  • অনুভূমিক সাদা স্থান কেবলমাত্র একটি একক স্থানের অক্ষর। প্রথম সংখ্যার আগে কিছুই থাকা উচিত নয় এবং ইনপুটটি নতুন লাইন + ইওএফ বা কেবল ইওএফ দিয়ে শেষ করা উচিত।
  • একমাত্র বৈধ ইনপুট, অগমেন্টেড ব্যাকাস-নওর ফর্মটিতে বর্ণিত , হ'ল:
    NONZERODIGIT = "1" / "2" / "3" / "4" / ​​"5" / "6" / "7" / "8" / "9"
    পজিটিভেনদার = ননজারডিজিট * 98 ডিজিট
    NEGATIVENumber = "-" পজিটিভেনদার
    সংখ্যা = নেগাটিভেন্দ্র / পজিটিভেনদার / "0"
    VALIDINPUT = নম্বর এসপি নম্বর * 1LF EOF
  • ত্রুটি বার্তাটি হ'ল একক বর্ণ 'ই', এর পরে একটি নতুন লাইন।
  • কোডটি অবশ্যই ইনপুট কী তা নির্বিশেষে 0.5s এর কমের মধ্যে পরিষ্কারভাবে শেষ করতে হবে।
code-golf  c 

2
আমি পছন্দ করি "অন্যথায়" শব্দটির একক ব্যবহারের সাথে আপনার স্পেকটি কতটা স্পষ্ট হয়ে উঠেছে। খুব ভাল লেখা (আমার মতে) খুব খারাপ চ্যালেঞ্জটি কেবলমাত্র সি তে, তবে আমি দেখতে পাচ্ছি কেন এটি কেন এমন হতে হবে।
রেইনবোল্ট

3
সতর্কতা ছাড়াই + কোড-গল্ফ + সি? এটি আকর্ষণীয় হতে বাধ্য!
Fors

1
স্টিডিন না কমান্ড লাইন?
চিকিত্সা অস্পষ্টতা

2
আমরা কি কঠোরভাবে গ্রহণযোগ্য ইনপুট ফর্ম্যাট পেতে পারি? যেমন প্রথম সংখ্যার আগে কোনও অতিরিক্ত শ্বেত স্পেস (বা কোনও সংখ্যা নয় কিছু নয়), দুই সংখ্যার মধ্যে যে পরিমাণ আনুভূমিক শ্বেতস্পেস (যেমন অনুভূমিক ট্যাব এবং স্পেস) এবং লাইনফিড বা ক্যারিজ ছাড়া আর কিছুই লাইনফিডের সাথে দ্বিতীয়টির পরে ফিরে আসে না নম্বর?
Fors

3
এছাড়াও, স্থান-বিচ্ছিন্ন মানে কি 1) ASCII মান 32, 2 এর সাথে একটি অক্ষর দ্বারা পৃথক) ASCII মান 32, 3 সহ একটি স্বতন্ত্র সংখ্যার দ্বারা পৃথক করা হয় যে কোনও একটি ASCII হোয়াইটস্পেস অক্ষর দ্বারা পৃথক (32, 10, 9, সম্ভবত 13) ?), ৪) এএসসিআইআই হোয়াইটস্পেস অক্ষরগুলির সেট থেকে একক শ্বেত স্পেস অক্ষরের একটি স্বেচ্ছাসেবী সংখ্যার দ্বারা পৃথক করা, বা ৫) এএসসিআইআই হোয়াইটস্পেসের অক্ষরের সেটটির যথেচ্ছ সংখ্যার দ্বারা বিচ্ছিন্ন?
ব্যবহারকারী 12205

উত্তর:


4

6610 বাইট (স্বাক্ষরিত)

"গুড বয়" সি প্রোগ্রাম যা সমস্ত চ্যালেঞ্জের মানদণ্ডগুলি পূরণ করে। Negativeণাত্মক সংখ্যার জন্য 10 এর পরিপূরক ব্যবহার করে। এছাড়াও একটি পরীক্ষার জোতা এবং পরীক্ষার ক্ষেত্রে অন্তর্ভুক্ত।

/*
Read input from STDIN. The input must conform to VALIDINPUT:

    NONZERODIGIT = "1" / "2" / "3" / "4" / "5" / "6" / "7" / "8" / "9"
    POSITIVENUMBER = NONZERODIGIT *98DIGIT
    NEGATIVENUMBER = "-" POSITIVENUMBER
    NUMBER = NEGATIVENUMBER / POSITIVENUMBER / "0"
    VALIDINPUT = NUMBER SP NUMBER *1LF EOF

Check syntax of input. If input is correct, add the two numbers and print
to STDOUT.

LSB => least significant byte
MSB => most significant byte
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <stdbool.h>
#include <assert.h>
#define NUL ('\0')
/*
    maximum characters in VALIDINPUT:
        '-'     1
        POSITIVENUMBER  MAXDIGITS
        ' '     1
        '-'     1
        POSITIVENUMBER  MAXDIGITS
        LF      1
*/
#define MAXDIGITS (99)
#define MAXVALIDINPUT (2*MAXDIGITS+4)

void die() { printf("E\n"); exit(1); }

/*
    Add two NUMBERs and print the result to STDOUT.  The NUMBERS have
been separated into POSITIVENUMBERS and sign information.

Arguments:
    first       - pointer to LSB of 1st POSITIVENUMBER
    firstSize   - size of 1st POSITIVENUMBER
    firstNegative   - is 1st # negative?
    second      - pointer to LSB of 2nd POSITIVENUMBER
    secondSize  - size of 2nd POSITIVENUMBER
    secondNegative  - is 2nd # negative?
    carry       - carry from previous place?

Returns:
    sum[]       - sum
    addNUMBERs()    - carry to next place?

- Don't use complementDigit(popDigit(p,s),n). Side-effects generate two pops.
*/
#define popDigit(p,s) ((s)--,(*(p++)-'0'))
#define complementDigit(c,n) ((n) ? 9-(c) : (c))

#define pushSum(c) (*(--sumPointer)=(c))
#define openSum() (pushSum(NUL))
#define closeSum() ;
char    sum[MAXVALIDINPUT];
char    *sumPointer = sum+sizeof(sum);
int addNUMBERs(char *first, int firstSize, bool firstNegative,
        char *second, int secondSize, bool secondNegative,
        int previousCarry) {
    int firstDigit, secondDigit;
    int mySum;
    int myCarry;

    /*
        1st half of the problem.

        Build a stack of digits for "first" and "second"
    numbers. Each recursion of addNUMBERs() contains one digit
    of each number for that place. I.e., the 1st call holds
    the MSBs, the last call holds the LSBs.

        If negative, convert to 10s complement.
    */
    assert((firstSize > 0) && (secondSize > 0));
    if (firstSize > secondSize) {
        firstDigit = popDigit(first, firstSize);
        firstDigit = complementDigit(firstDigit, firstNegative);
        secondDigit = 0;
    } else if (secondSize > firstSize) {
        firstDigit = 0;
        secondDigit = popDigit(second, secondSize);
        secondDigit = complementDigit(secondDigit, secondNegative);
    } else {
        //  same size
        firstDigit = popDigit(first, firstSize);
        firstDigit = complementDigit(firstDigit, firstNegative);
        secondDigit = popDigit(second, secondSize);
        secondDigit = complementDigit(secondDigit, secondNegative);
    }

    //  recursion ends at LSB
    if ((firstSize == 0) && (secondSize == 0)) {
        //  if negative, add 1 to complemented LSB
        if (firstNegative) {
            firstDigit++;
        }
        if (secondNegative) {
            secondDigit++;
        }
        myCarry = previousCarry;
    } else {
        myCarry = addNUMBERs(first, firstSize, firstNegative,
            second, secondSize, secondNegative,
            previousCarry);
    }

    /*
        2nd half of the problem.

        Sum the digits and save them in first[].
    */
    mySum = firstDigit + secondDigit + ((myCarry) ? 1 : 0);
    if ((myCarry = (mySum > 9))) {
        mySum -= 10;
    }
    pushSum(mySum + '0');
    return(myCarry);
}

//  Handle the printing logic.
void addAndPrint(char *first, int firstSize, bool firstNegative,
        char *second, int secondSize, bool secondNegative,
        int previousCarry) {

    openSum();
    addNUMBERs(first, firstSize, firstNegative,
        second, secondSize, secondNegative,
        previousCarry)
    closeSum();
    if (*sumPointer<'5') {
        //  it's positive
        for (; *sumPointer=='0'; sumPointer++) {} // discard leading 0s
        //  if all zeros (sumPointer @ NUL), back up one
        sumPointer -= (*sumPointer == NUL) ? 1 : 0;
        printf("%s\n", sumPointer);
    } else {
        //  it's negative
        char    *p;

        //  discard leading 0s (9s in 10s complement)
        for (; *sumPointer=='9' && *sumPointer; sumPointer++) {}
        //  if -1 (sumPointer @ EOS), back up one
        sumPointer -= (*sumPointer == NUL) ? 1 : 0;
        for (p=sumPointer; *p; p++) {
            *p = '0' + ('9' - *p); // uncomplement
            //  special handling, +1 for last digit
            *p += (*(p+1)) ? 0 : 1;
        }
        printf("-%s\n", sumPointer);

    }
    return;
}

/*
    Lex a number from STDIN.

Arguments:
    bufferPointer - pointer to a pointer to a buffer, use as
            **buffer = c;   // put "c" in the buffer
            *buffer += 1;   // increment the buffer pointer
            (*buffer)++;    // also increments the buffer pointer

All sorts of side-effects:
    - getc(stdin)
    - ungetc(...,stdin)
    - modifies value of **bufferPointer
    - modifies value of *bufferPointer

Returns:
    lexNUMBER() - number of bytes added to *bufferPointer,
            *1 if POSITIVENUMBER,
            *-1 if NEGATIVENUMBER
    *bufferPointer - points to the LSB of the number parsed + 1
*/
#define pushc(c) (*((*bufferPointer)++)=c)
bool lexNUMBER(char **bufferPointer) {
    char    c;
    int size = 0;
    bool    sign = false;

    /* lex a NUMBER */
    if ((c=getchar()) == '0') {
        pushc(c);
        c = getchar();
        size++;
    } else {
        if (c == '-') {
            sign = true;
            c = getchar();
            // "-" isn't a digit, don't add to size
        }
        if (c == '0') {
            die();
        }
        for (size=0; isdigit(c); size++) {
            if (size >= MAXDIGITS) {
                die();
            }
            pushc(c);
            c = getchar();
        }
    }
    if (size < 1) {
        die();
    }
    ungetc(c,stdin);        // give back unmatched character
    return (sign);
}

int main() {
    int c;
    char    buffer[MAXVALIDINPUT];
    char    *bufferPointer;
    char    *first, *second;
    int firstSize, secondSize;
    bool    firstNegative, secondNegative;

    bufferPointer = buffer + 1; // hack, space for leading digit
    //  parse 1st number
    first = bufferPointer;
    firstNegative = lexNUMBER(&bufferPointer);
    firstSize = bufferPointer - first;
    *(bufferPointer++) = NUL;   // hack, space for EOS
    bufferPointer++;        // hack, space for leading digit
    //  parse separating blank
    if ((c=getchar()) != ' ') {
        die();
    }
    //  parse 2nd number
    second = bufferPointer;
    secondNegative = lexNUMBER(&bufferPointer);
    secondSize = bufferPointer - second;
    *(bufferPointer++) = NUL;   // hack, space for EOS
    //  parse end of input
    c = getchar();
    if (! ((c == EOF) || ((c == '\n') && ((c=getchar()) == EOF))) ) {
        die();
    }
    //  Some very implementation-specific massaging.
    *(--first) = '0';       // prefix with leading 0
    *(first+(++firstSize)) = NUL;   // add EOS
    *(--second) = '0';      // prefix with leading 0
    *(second+(++secondSize)) = NUL; // add EOS
    //  add and print two arbitrary precision numbers
    addAndPrint(first, firstSize, firstNegative,
        second, secondSize, secondNegative, false);
    return(0);
}

আপনাকে শুরু করার জন্য এখানে কয়েকটি পরীক্ষার জোতা এবং কয়েকটি পরীক্ষার কেস। পার্লের অতিরিক্ত ব্যবহার ছিঁড়ে ফ্রি মনে করুন Fe এটি যে সিস্টেমে বিকাশ করা হয়েছিল তাতে আধুনিক বাশ ছিল না।

#!/bin/bash
#
#   testharness.sh
#
# Use as: bash testharness.sh program_to_be_tested < test_data
#
# Each line in the test data file should be formatted as:
#
#   INPUT = bash printf string, must not contain '"'
#   OUTPUT = perl string, must not contain '"'
#           (inserted into the regex below, use wisely)
#   TESTNAME = string, must not contain '"'
#   GARBAGE = comments or whatever you like
#   INPUTQUOTED = DQUOTE INPUT DQUOTE
#   OUTPUTQUOTED = DQUOTE OUTPUT DQUOTE
#   TESTQUOTED = DQUOTE TESTNAME DQUOTE
#   TESTLINE = INPUTQUOTED *WSP OUTPUTQUOTED *WSP TESTQUOTED GARBAGE
TESTPROGRAM=$1
TMPFILE=testharness.$$
trap "rm $TMPFILE" EXIT
N=0         # line number in the test file
while read -r line; do
    N=$((N+1))
    fields=$(perl -e 'split("\"",$ARGV[0]);print "$#_";' "$line")
    if [[ $fields -lt 5 ]]; then
        echo "skipped@$N"
        continue
    fi
    INPUT=$(perl -e 'split("\"",$ARGV[0]);print "$_[1]";' "$line")
    OUTPUT=$(perl -e 'split("\"",$ARGV[0]);print "$_[3]";' "$line")
    TESTNAME=$(perl -e 'split("\"",$ARGV[0]);print "$_[5]";' "$line")
    printf -- "$INPUT" | $TESTPROGRAM > $TMPFILE
    perl -e "\$t='^\\\s*$OUTPUT\\\s*\$'; exit (<> =~ \$t);" < $TMPFILE
    if [[ $? -ne 0 ]]; then     # perl -e "exit(0==0)" => 1
        echo "ok $TESTNAME"
    else
        echo -n "failed@$N $TESTNAME," \
            "given: \"$INPUT\" expected: \"$OUTPUT\" received: "
        cat $TMPFILE
        echo
    fi
done


পরীক্ষার মামলার একটি ছোট সেট:

"0 0"       "0" "simple, 0+0=0"
"1 1"       "2" "simple, 1+1=2"

""      "E" "error, no numbers"
"0"     "E" "error, one number"
"0  0"      "E" "error, two/too much white space"
"0 0 "      "E" "error, trailing characters"
"01 0"      "E" "error, leading zeros not allowed"

"-0 0"      "E" "error, negative zero not allowed
"0 -0"      "E" "error, negative zero not allowed here either

"1 1\n"     "2" "LF only allowed trailing character

"\0001 1\n"    "E" "error, try to confuse C string routines #1"
"1\00 1\n"  "E" "error, try to confuse C string routines #2"
"1 \0001\n"    "E" "error, try to confuse C string routines #3"
"1 1\000\n"    "E" "error, try to confuse C string routines #4"
"1 1\n\000"    "E" "error, try to confuse C string routines #5"

"-1 -1"     "-2"    "add all combinations of -1..1 #1"
"-1 0"      "-1"    "add all combinations of -1..1 #2"
"-1 1"      "0" "add all combinations of -1..1 #3"
"0 -1"      "-1"    "add all combinations of -1..1 #4"
"0 0"       "0" "add all combinations of -1..1 #5"
"0 1"       "1" "add all combinations of -1..1 #6"
"1 -1"      "0" "add all combinations of -1..1 #7"
"1 0"       "1" "add all combinations of -1..1 #8"
"1 1"       "2" "add all combinations of -1..1 #9"

"0 123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789" "123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789" "0+99 digits should work"

"100000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 100000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000" "200000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000" "99 digits+99 digits should work"

"500000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 500000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000" "1000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000" "test for accumulator overflow"

"-123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789 0" "-123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789" "0+negative 99 digits work"

"-100000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 -100000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000" "-200000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000" "99 digits+99 digits (both negative) should work"

"-500000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 -500000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000" "-1000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000" "test for negative accumulator overflow"

"1234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890 0" "E" "error, 100 digits"

শেষ দুটি বড় নেতিবাচক সংখ্যা পরীক্ষা স্থির করে। তারা নেতিবাচক সংখ্যা পরীক্ষা করছিল না।
স্কট লিডলি

আপনার কোড থেকে -ওয়াল-ওয়েক্সট্রা দিয়ে আমি প্রচুর সতর্কতা পেয়েছি।

@ ল্যাম্বিক অন্যান্য পোস্টারগুলি লেক্সারের অনুলিপি এবং সংকলক সতর্কতাগুলি নির্ধারণ করতে দ্বিধা বোধ করতে পারে। এটি একটি নিরাপদ লেক্সার বাস্তবায়ন করে এবং স্বেচ্ছাচারিত নির্ভুলতা পাটিগণিত (সাজানো) করে। আমি ম্যাল-কোডিংয়ে কিছু প্রচেষ্টা ব্যয় করেছি (কোডে মন্তব্য দেখুন), যাতে আপনার প্রচুর সংকলক সতর্কতা পাওয়া উচিত। লেক্সারের জন্য সোনার তারা এবং অন্যান্য মানদণ্ড উপেক্ষা করার জন্য পিছনে একটি সুইফ্ট বুট সম্পর্কে কীভাবে?
স্কট লিডলি

এটা খুব যুক্তিসঙ্গত বলে মনে হচ্ছে :) এটি সম্পন্ন বিবেচনা করুন।

1
@ লেম্বিক 10 ম পরিপূরক পাটিগণিত ব্যবহার করে সমস্ত সি সমাধান দিয়ে প্রতিস্থাপন করেছেন। bcপৃথক উত্তর ব্যবহার করে "খারাপ উদাহরণ" সমাধান সরানো হয়েছে ।
স্কট লিডলি

3

289

সম্পাদনা : এই কোডটি কেবল ধনাত্মক পূর্ণসংখ্যার জন্য কাজ করে। আমি এই উত্তরটি পোস্ট করার পরে নিয়মগুলি পরিবর্তন হয়েছে।

#include <stdio.h>
#include <stdlib.h>
int r[101],s;void x(int i){r[i]|=32;if(r[i]>41)r[i+1]++,r[i]-=10,x(i+1);}void f(int b){int c=getchar();if(c!=b){if(s>99||c<48||c>57)exit(puts("E"));f(b);r[s]+=c-48;x(s++);}}int main(){f(32);s=0;f(10);for(s=100;s--;)if(r[s])putchar(r[s]+16);return 0;}

অসম্পূর্ণ এবং মন্তব্য করা সংস্করণ:

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

//global variables are automatically init to zero
int result[101]; //101 because 2 numbers of 100 digits can give a 101-digits result
int currentNumber;

void reportAddition(int i) {
    result[i]|=0x20; //flag "active" value, 6th bit
    if(result[i]>9+0x20) {
        result[i+1]++;
        result[i]-=10;
        reportAddition(i+1);
    }
}

void addNumber(int endingChar) {
    int c=getchar();
    if(c!=endingChar) {
        if(currentNumber>99||c<'0'||c>'9') //error
            exit(puts("Error"));
        addNumber(endingChar);
        result[currentNumber]+=c-'0';
        reportAddition(currentNumber); //handle case when addition give a value greater than 9
        currentNumber++;
    }
}

int main() {

    addNumber(' '); //add first number
    currentNumber=0;
    addNumber('\n'); //add second

    for(currentNumber=100;currentNumber--;)
        if(result[currentNumber])
            putchar(result[currentNumber]+'0'-0x20); //display char
    return 0;
}

আপনি কি 99 ডিজিটের চেয়ে বেশি ইনপুট প্রত্যাখ্যান করেন? এটা আমার মনে হয় আপনার উচিত।
জন ডিভোরাক

@ জনডভোরাক, হ্যাঁ আমি করি।
মাইকেল এম

আমি বেশ কয়েকটি সংকলনের সতর্কতা ./tmp.c: In function ‘f’: ./tmp.c:3:1: warning: suggest parentheses around comparison in operand of ‘|’ [-Wparentheses] ./tmp.c:3:1: warning: suggest parentheses around comparison in operand of ‘|’ [-Wparentheses] ./tmp.c: In function ‘main’: ./tmp.c:3:1: warning: control reaches end of non-void function [-Wreturn-type]
পেয়েছি

আমি সংখ্যাটি ইতিবাচক ছিল তা নির্দিষ্ট করে বলতে চাইনি। দুঃক্ষিত।

@ আমার এখানে এই সতর্কতা নেই, ঠিক করে প্রতিস্থাপন (s>99|c<48|c>57)করা (s>99||c<48||c>57)হচ্ছে?
মাইকেল এম

2

442

এটি বেশ দীর্ঘ, তাই আমি সপ্তাহান্তে আরও গল্ফ করতে পারি। ধরে নিই ইনপুট স্টিডিনের, ইওএফ-টার্মিনেটেড (নিউলাইন ছাড়াই), বিভাজকটি ASCII মান 32 এর একমাত্র অক্ষর (এটি, ' 'অক্ষর)।

#include<stdio.h>
char a[102],b[102],c[102],*p=a,d;int i,j=101,l,L,k,F=1;int main(){while(~(k=getchar())){if(47<k&&k<58){p[i++]=k;if(i==101)goto f;}else if(k==32&&F)p=b,l=i,F=0,i=0;else goto f;}L=i;for(i=(l<L?l:L)-1;i+1;i--){c[j]=(L<l?b[i]-48+a[i+l-L]:a[i]-48+b[i+L-l]);if(c[j--]>57)c[j]++,c[j+1]-=10;}for(i=(L<l?l-L-1:L-l-1);i+1;i--)c[j--]=(L<l?a[i]:b[i]);for(i=0;i<102;i++)if(c[i]&&(c[i]-48||d))d=putchar(c[i]);return 0;f:return puts("E");}

ত্রুটি বার্তাটি একটি একক অক্ষর 'ই' হবে, তার পরে একটি নতুন লাইন থাকবে।

নতুন লাইনের সাথে এবং সামান্য খানিকটা ইনডেন্টেশন যুক্ত: (একটি পঠনযোগ্য সংস্করণ অনুসরণ করা হয়েছে, তাই এখানে এড়িয়ে চলা নির্দ্বিধায়)

#include<stdio.h>
char a[102],b[102],c[102],*p=a,d;
int i,j=101,l,L,k,F=1;
int main(){
    while(~(k=getchar())){
        if(47<k&&k<58){
            p[i++]=k;
            if(i==101)goto f;
        }else if(k==32&&F)p=b,l=i,F=0,i=0;
        else goto f;
    }
    L=i;
    for(i=(l<L?l:L)-1;i+1;i--){
        c[j]=(L<l?b[i]-48+a[i+l-L]:a[i]-48+b[i+L-l]);
        if(c[j--]>57)c[j]++,c[j+1]-=10;
    }
    for(i=(L<l?l-L-1:L-l-1);i+1;i--)c[j--]=(L<l?a[i]:b[i]);
    for(i=0;i<102;i++)if(c[i]&&(c[i]-48||d))d=putchar(c[i]);
    return 0;
    f:return puts("E");
}

পঠনযোগ্য সংস্করণ (কিছু বিবৃতি এটিকে আরও পঠনযোগ্য করে তুলতে কিছুটা পরিবর্তন করা হয়েছে, তবে তারা যা করে তা একই হওয়া উচিত):

#include <stdio.h>
char num1[102],num2[102],sum[102]; //globals initialised to 0
char *p=num1;
int outputZero=0, noExtraSpace=1;
int i=0, j=101, len1, len2, ch;
#define min(x,y) (x<y?x:y)
int main(){
    while((ch=getchar())!=-1) { //assumes EOF is -1
        if('0'<=ch && ch<='9') {
            p[i]=ch;
            i++;
            if(i==101) goto fail; //if input too long
        } else if(ch==' ' && noExtraSpace) {
            p=num2;
            len1=i;
            noExtraSpace=0;
            i=0;
        } else goto fail; //invalid character
    }
    len2=i;
    for(i=min(len1, len2)-1; i>=0; i--) {
        //add each digit when both numbers have that digit
        sum[j]=(len2<len1?num2[i]-'0'+num1[i+len1-len2]:num1[i]-'0'+num2[i+len2-len1]);
        if(sum[j]>'9') { //deal with carries
            sum[j-1]++;
            sum[j]-=10;
        }
        j--;
    }
    for(i=(len2<len1?len1-len2-1:len2-len1-1); i>=0; i--) {
        //copy extra digits when one number is longer than the other
        sum[j]=(len2<len1?num1[i]:num2[i]);
        j--;
    }
    for(i=0; i<102; i++) {
        if(sum[i] && (sum[i]-'0' || outputZero)) {
            putchar(sum[i]);
            outputZero=1;
            //if a digit has been output, the remaining zeroes must not be leading
        }
    }
    return 0;
    fail:
    puts("Error");
    return 0;
}

goto fail;জিনিস উপহাস অ্যাপল করার জন্য এখানে হয়।

আমি ব্যবহৃত জিসিসিটির সংস্করণটি হ'ল gcc (Ubuntu/Linaro 4.6.3-1ubuntu5) 4.6.3এবং কোনও সতর্কতা নেই।


ইনপুট হিসাবে -1 1 সম্পর্কে কি।

@ ল্যাম্বিক ত্রুটি বার্তা প্রিন্ট করে কারণ ইনপুটটি "দুটি স্থান পৃথক ধনাত্মক পূর্ণসংখ্যার " হওয়া উচিত ।
ব্যবহারকারী 12205

ওহ সরি, আমি এটাকে ছড়িয়েছি। আমি ভেবেছিলাম যে আমি এটি পরিবর্তন করেছি।

বর্তমানে আপনার কোডটি সর্বদা আমার জন্য ই আউটপুট করে। উদাহরণস্বরূপ আমি "1 1" চেষ্টা করেছি। আপনি কি স্ট্যান্ডিন থেকে ইনপুটটি আনতে পারেন?

@ ল্যাম্বিক ইনপুটটি স্টিডিনের ( getchar()সর্বদা স্টিডিনের কাছ থেকে আসে)। এটি নতুন লাইন ছাড়া ইওএফ-সমাপ্ত বলে ধরে নেওয়া হয় । আপনি এটি পরীক্ষা করতে পারেন, হয় হয় [1] [স্পেস] [1] [Ctrl + D] [Ctrl + D], অথবাecho -n '1 1' | program
ব্যবহারকারী 12205

0

633 বাইট

অর্ধেক চ্যালেঞ্জ পূরণ করে "খারাপ ছেলে" সি প্রোগ্রাম আপত্তিজনক সি, প্রচুর সতর্কতা নিক্ষেপ করে, কিন্তু কাজ করে ... সাজানো। নির্বিচারে নির্ভুলতা গণিত আসলে দ্বারা সম্পন্ন হয় bc

#include<stdio.h>
#include<string.h>
#include<ctype.h>
#define g() getc(stdin)
#define z(c) b[i++]=c,b[i]='\0'
x(){puts("E\n");exit(1);}main(){int c,y=0,i=0;char b[232]="",*h="echo ",*t=" | bc -q | tr -d '\\\\\\012'";strcat(b,h);i=strlen(h);if((c=g())=='0'){z(c);y=1;c=g();}else{if(c=='-'){z(c);c=g();}c!='0'||x();ungetc(c,stdin);for(y=0;isdigit(c=g());y++){z(c);y<99||x();}}y>0||x();c==' '||x();z('+');y=0;if((c=g())=='0'){z(c);y=1;c=g();}else{if(c=='-'){z(c);c=g();}c!='0'||x();do{if(!isdigit(c))break;z(c);y++;y<=99||x();}while(c=g());}y>0||x();strcat(b+i,t);i+=strlen(t);if(!((c==-1)||((c=='\n')&&((c=g())==-1))))x();system(b);}

আনমিনাইড সংস্করণ

/*
Read input from STDIN. The input must conform to VALIDINPUT:

    NONZERODIGIT = "1" / "2" / "3" / "4" / "5" / "6" / "7" / "8" / "9"
    POSITIVENUMBER = NONZERODIGIT *98DIGIT
    NEGATIVENUMBER = "-" POSITIVENUMBER
    NUMBER = NEGATIVENUMBER / POSITIVENUMBER / "0"
    VALIDINPUT = NUMBER SP NUMBER *1LF EOF

Check syntax of input. If input is correct, use the shell command
"echo NUMBER+NUMBER | bc -q" to do the arbitrary precision integer arithmetic.

NB, this solution requires that the "normal" bc be 1st in the PATH.


    Fun C language features used:
    - ignore arguments to main()
    - preprocessor macros
    - pointer arithmetic
    - , operator
    - ?: operator
    - do-while loop
    - for loop test does input
    - implicit return/exit
    - short is still a type!
    - ungetc()
    - use int as bool and 0 & 1 as magic numbers
    - implicit type coersion

5/5/14, Stop fighting the syntax graph. Get rid of "positive" flag.
*/
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <ctype.h>
#define MAXDIGITS (99)
#define pushc(c) (buffer[index++]=c,buffer[index]='\0')
#define pushs(s) (strcat(buffer+index,s),index+=strlen(s))

int die() { printf("E\n"); exit(1); }

int main () {
    int c;
    /*
    buffer budget:
    5               "echo "
    1+MAXDIGITS (include "-")   NUMBER
    1               "+"
    1+MAXDIGITS (include "-")   NUMBER
    25              " | bc -q | tr -d '\\\012'"
    1               NUL
    */
    char    buffer[5+1+MAXDIGITS+1+1+MAXDIGITS+25+1] = "";
    short   index = 0;
    short   digits;

    pushs("echo ");
    // parse 1st number
    digits = 0;
    if ((c=getchar()) == '0') {
        pushc(c);
        digits = 1;
        c = getchar();
    } else {
        if (c == '-') {
            // "-" doesn't count against digits total
            pushc(c);
            c = getchar();
        }
        (c != '0') || die();
        ungetc(c,stdin);
        for (digits=0; isdigit(c=getchar()); digits++) {
            pushc(c);
            (digits<MAXDIGITS) || die();
        }
    }
    (digits>=1) || die();
    // parse separating blank
    (c == ' ') || die();
    pushc('+');
    // parse 2nd number
    digits = 0;
    if ((c=getchar()) == '0') {
        pushc(c);
        digits = 1;
        c = getchar();
    } else {
        if (c == '-') {
            // "-" doesn't count against digits total
            pushc(c);
            c = getchar();
        }
        (c != '0') || die();
        do {
            if (!isdigit(c)) {
                break;
            }
            pushc(c);
            digits++;
            (digits<=MAXDIGITS) || die();
        } while(c=getchar());
    }
    (digits>=1) || die();
    pushs(" | bc -q | tr -d '\\\\\\012'");
    // parse end of input
    if (! ((c == EOF) || ((c == '\n') && ((c=getchar()) == EOF))) ) {
        die();
    }
    // add two arbitrary precision numbers and print the result to STDOUT
    system(buffer);
}
আমাদের সাইট ব্যবহার করে, আপনি স্বীকার করেছেন যে আপনি আমাদের কুকি নীতি এবং গোপনীয়তা নীতিটি পড়েছেন এবং বুঝতে পেরেছেন ।
Licensed under cc by-sa 3.0 with attribution required.