내가해야 할 수학 문제가 얼마나되는지 알려주세요!


36

선생님은 항상 숙제에 가장 복잡한 수학 문제를 제시합니다. 처럼 : pg. 546: 17-19, 22, 26, pg. 548: 35-67 odd, 79, 80-86 even. 숙제를 위해 얼마의 시간을 따로 두어야하는지 미리 알고 싶지만 그 모든 것을 알아 내고 싶지는 않습니다. 그렇기 때문에 당신의 임무는 저를 위해 그것을 프로그래밍하는 것입니다.

사양

  • args, stdio 등으로 완료 해야하는 문제를 자세히 설명하는 문자열이 표시됩니다.
  • 쉼표로 구분됩니다 (분리 될 수 있음 comma-space)
  • 그것은 단지 숫자의 형태로 하나의 문제가 포함됩니다 (예를 들어 79)
  • 그리고 형태의 범위 17-18(다시 말하면 선택적인 공백을 처리해야 함)
  • 범위는 양쪽 끝을 포함합니다
  • 범위는 선택적 접미사로한다 odd또는 even당신이 고려해야한다.
  • 일련의 범위 / 페이지 앞에는 pg. 545:선택적인 공백을 처리해야하는 형식의 페이지 번호가 앞에 붙습니다 . 모든 페이지에서 문제를 해결해야하므로 안전하게 무시할 수 있습니다.
  • 텍스트는 대문자 또는 소문자 일 수 있지만 둘다는 아닙니다.
  • 숙제를 위해해야 ​​할 문제의 수를 반환하십시오.
  • 이것은 이므로 바이트 단위의 가장 짧은 코드가 이깁니다!

테스트 사례

pg. 546: 17-19, 22, 26, pg. 548: 35-67 odd, 79, 80-86 even   ->    27
pg. 34: 1                                                    ->    1
PG. 565: 2-5,PG.345:7                                        ->    5
pg. 343: 5,8,13 - 56 even,pg. 345: 34 - 78,80                ->    70
pg.492: 2-4 odd,7-9 even                                     ->    2

12
교수는 당신에게 범위를 줄 수 2-4 odd있습니까? 더 간단한 접근법에는 몇 가지 문제가있는 것으로 보입니다.
Björn Lindqvist 2016 년

1
^ 현재 문제 진술에 따르면 이것이 테스트 사례라고 생각합니다.
mbomb007

2
이 테스트 사례가 있어야합니다.pg.492: 2-4 odd,7-9 even -> 2
mbomb007

2
범위가 겹칠 수 있습니까? 예를 들어 22-26,25-30?
Reto Koradi

1
@RetoKoradi 번호
Maltysen

답변:


15

CJam, 61 58 51 48 46 43 41 38 바이트

leuS-',/{':/W>:~_2*2<~z),>1f&\,5--~}%,

CJam 인터프리터 에서 테스트 사례를 확인하십시오 .

작동 원리

leuS-      e# Read a line from STDIN, convert to uppercase and remove spaces.
',/        e# Split at commas.
{          e# For each chunk:
  ':/W>    e#   Split at colons and only keep the last chunk.
  :~       e#   Evaluate the string inside the array.
  _2*      e#   Copy the array, repeated twice.
  2<       e#   Keep only the first two elements.

           e#   In all possible cases, the array now contains exactly two
           e#   integers, which are equal in case of a single problem.

  ~        e#   Dump the array on the stack.
  z),      e#   Push [0 ... -N], where N is the second integer.
  >        e#   Discard the first M elements, where M is the first integer.
  1f&      e#   Replace each element by its parity.
  \,       e#   Push L, the length of the original array.

           e#   EVEN and ODD all push elements, so L is 1 for single problems,
           e#   2 for simple ranges, 5 for odd ranges and 6 for even ranges.

  5--      e#   Discard all elements equal to L - 5 from the array of parities.

           e#   This removes 0's for odd ranges, 1's for even ranges, -3's for
           e#   other ranges and -4's for single problems, thus counting only
           e#   problems of the desired parities.

  ~        e#   Dump the resulting array on the stack.
}%         e# Collect the results in an array.
,          e# Compute its length.

최신 테스트 케이스에서 작동합니까?
mbomb007

그렇습니다. 최신 테스트 사례를 링크에 추가했습니다.
Dennis

10

펄-47 바이트

#!perl -p054
{map$\+=($_%2x9^lc$')!~T,$&..$'*/\d+ ?-/}}{

새로운 테스트 사례를 통과하도록 수정되었습니다.


기발한

펄-36 바이트

#!perl -p054
$\+=/\d+ ?-/*($'-$&>>/o|e/i)+1}{

shebang을 4로 세면 입력은 stdin에서 가져옵니다.


샘플 사용법

$ echo pg. 546: 17-19, 22, 26, pg. 548: 35-67 odd, 79, 80-86 even | perl math-problems.pl
27

$ echo pg. 34: 1 | perl math-problems.pl
1

$ echo PG. 565: 2-5,PG.345:7 | perl math-problems.pl
5

$ echo pg. 343: 5,8,13 - 56 even,pg. 345: 34 - 78,80 | perl math-problems.pl
70

경고

짝수 / 홀수 범위의 경우 끝점 중 하나 이상이 범위의 패리티와 일치해야합니다. 예를 들어 11-19 odd, 11-20 odd10-19 odd모든 올바르게 5로 계산되지만, 10-20 odd6으로 과다 계산됩니다.


이것은 어떻게 작동 pg. 20: 13-15 even합니까? 또는 pg. 20: 13-14 even?
찰스

1
*한 글자보다 짧아서 &&쉽게 개선 할 수 있습니다.$\+=/\d+ ?-/*($'-$&>>/o|e/i)+1for@F}{
Grimmy

1
이봐, 이건 영리 해! 내가 빠진 것이 아니라면을 제거 할 수 있어야합니다 lc=~.
Dennis

1
나는 그 T^부분을 얻었 지만 어떻게 든 lc사건 이 바뀌었다 $'. 앞에 추가 lc$'할 것은 약간 짧은왔다. 즉 여전히 두 접근 방식에 대한 작동합니다 : lc$'!~(T^lc$_%2)($_%2x9^lc$')!~T
데니스

1
이전 버전의 @Dennis에는 괄호가 필요합니다. !~T그래도 천재입니다. 감사합니다!
primo

6

파이썬 2, 259 253 249 239 바이트

여기 사용해보십시오

아마 아직도 더 골프를 칠 수 있습니다.

편집 : 2-4 even예상대로 광산이 작동하지 않는 버그가 수정 되었습니다. 그런 다음 해당 수정 사항을 조정했습니다. 그 수정으로 4 바이트가 절약되었습니다!

편집 : 지금 사용 input()하고 2 바이트 사용자와 입력을 묶어야합니다 두 따옴표합니다.

import re
c=0
for x in re.sub(r'..\..*?:','',input()).replace(' ','').split(','):
 y=x.split('-')
 if len(y)<2:c+=1
 else:
    a,b=y[0],y[1];d=int(re.sub('[A-z]','',b))-int(a)+1
    if b[-1]>':':d=d/2+d%2*(int(a)%2==ord(b[-3])%2)
    c+=d
print c

덜 골프 (의견과 함께! : D) :

이 의견이 도움이 되길 바랍니다. 마지막 복잡한 줄을 올바르게 설명했는지 확실하지 않습니다.

import re
def f(s):
    c=0
    l=re.sub(r'..\..*?:','',s).replace(' ','').split(',')   # remove pg #'s and split
    for x in l:
        y=x.split('-')
        if len(y)<2:                                # if not a range of numbers
            c+=1
        else:
            a,b=y[0],y[1]                           # first and second numbers in range
            d=int(re.sub('[A-z]','',b))-int(a)+1    # number of pages
            if b[-1]>':':                           # if last character is not a digit
                # half the range
                # plus 1 if odd # of pages, but only if first and last numbers in the range
                #       are the same parity
                # ord(b[-3])%2 is 0 for even (v), 1 for odd (o)
                d=d/2+(d%2)*(int(a)%2==ord(b[-3])%2)
            c+=d
    print c

2
파이썬 2를 사용하고 있기 때문에 공백과 탭 (각 1 바이트)을 다른 들여 쓰기로 사용할 수 있습니다. 관련 팁
FryAmTheEggman

또한 처음에 카운트가 잘못되었다는 것을 알았습니다. 편집기에서 탭을 복사하면 공백으로 변환됩니다.
mbomb007

s=raw_input()들여 쓰기를 수행 하고 제거 하여 최소 4 바이트를 절약 할 수 있습니다 .

4

Pyth, 43 42 44 42 바이트

lsm-%R2}hKvMc-ecd\:G\-eK?}\ed}edGYc-rzZd\,

온라인으로 사용해보십시오 : 데모 또는 테스트 장치

나는 여전히 1 또는 2 바이트를자를 수 있다고 생각합니다.

설명

lsm-%R2}hKvMc-ecd\:G\-eK?}\ed}edGYc-rzZd\,  implicit: z = input string
                                    rzZ     convert z to lower-case
                                   -   d    remove all spaces from z
                                  c     \,  and split by ","
  m                                         map each part d to:
               cd\:                           split d by ":"
              e                               and only use the last part (removes page number)
             -     G                          remove all letters (removes odd/even)
            c       \-                        split by "-"
          vM                                  and evaluate all (one or two) numbers
         K                                    and store the result in K
       }hK            eK                      create the list [K[0], K[0]+1, ..., K[-1]]
    %R2                                       apply modulo 2 to each element
   -                                          and remove:
                         }\ed                   "e" in d (1 for in, 0 for not in)
                        ?    }edG               if d[-1] in "abcde...z" else
                                 Y              dummy value
 s                                            combine all the lists
l                                             print the length                                      

최신 테스트 케이스에서 작동합니까?
mbomb007

@ mbomb007 : 그렇습니다.
Dennis

3

자바 스크립트 (Spidermonkey console)-139

명령 줄에서 테스트하는 것이 더 쉽습니다.

for(r=/[:,] *(\d+)[- ]*(\d+)? *(o|e)?/gi,m=readline(z=0);f=r.exec(m);z+=!b||((p=b-a)%2||!c|a%2^/e/i.test(c))+p/(2-!c)|0)[,a,b,c]=f
print(z)

언 골프 드 :

// any number set after "pg:" or a comma
// \1 is FROM, \2 is TO, \3 is odd/even 
r=/[:,] *(\d+)[- ]*(\d+)? *(o|e)?/gi;
m=readline();
z=0; // this is the sum.
while(f=r.exec(m)){
    [,from,to,oddEven]=f;
    if(!to) {
        z++;
    } else {
        if((to-from)%2) {
            // if to and from are not the same parity, add 1
            z++;
        } else {
            // if to and from are not the same parity...
            if(!oddEven) {
                // and we don't have a parity marker, add one
                z++;
            } else if(a%2 != /e/i.test(c)) {
                // if we do have a parity marker,
                // AND the parity of from and to matches the 
                // parity of the oddEven sign, then add 1
                z++;
            }
        }
        // then add the difference between to-from and
        // if oddEven exists, divide by two and round down
        z+=(to-from)/(oddEven?2:1)|0;
    }

}
print(z);

[,from,to]단지 수 [from,to]?
Yytsi 2016 년

1
@TuukkaX no는 배열의 첫 번째 값을 버리는 것이므로 r.exec일치하는 전체 문자열을 포함합니다.
Patrick Roberts

3

요소-488 바이트 :

USING: arrays ascii kernel math math.parser math.ranges pcre sequences ;
IN: examples.golf.homework

: c ( a -- b )
    >lower "(?:[,:]|^) *(\\d+) *(?:- *(\\d+) *(e|o)?)?" findall [
        rest [ second dup string>number swap or ] map
        dup length 1 = [ drop 1 ] [
            dup length 2 = [ first2 swap - 1 + ] [
                first3 "o" = [ [a,b] [ odd? ] count ] [
                    [a,b] [ even? ] count
                ] if
            ] if
        ] if
    ] map sum ;

2

배시 344 315 306 294 262 252 242 240

IFS=,
o(){ R=0;for ((i=$1;i<=$2;R+=i++%2));do :
done
}
e(){ q=$R;o $*;((R=q-R))
}
for c in ${1,,};do
c=${c#*:}
m=${c##* }
l=${c%-*}
l=${l// }
h=${c#*-}
[[ a< $m ]]&&h=${h% *}
h=${h// }
((R=h-l+1))
eval "${m::1} $l $h"
((t+=R))
done;echo $t

나는 이것이 가능한 한 많이 골프를 쳤다고 생각하지 않지만 첫 번째 제출에는 나쁘지 않습니다. 아래는 주석이 달린 버전입니다.

IFS=, # Setup IFS for the for loops, We want to be able to split on commas

o(){ # Odd
    R=0  # Reset the R variable

    # Increments R for each odd element in the range
    # $1-$2 inclusive
    for ((i=$1;i<=$2;R+=i++%2));do
        : # Noop command
    done
}

e(){ # Even
    # Save R, it contains the total number of elements between low
    # and high
    q=$R
    # Call Odd, This will set R
    o $*
    # Set R = total number of elements in sequence - number of odd elements.
    ((R=q-R))
}

# This lowercases the firs arg. IFS causes it to split on commas.
for c in ${1,,};do
    c=${c#*:}  # Strips the page prefix if it exists
    m=${c##* }  # Capture the odd/even suffix if it exists
    l=${c%-*}  # Capture low end of a range, Strips hyphen and anything after it
    l=${l// }  # Strips spaces
    h=${c#*-}  # Capture high end of a range, Strips up to and including hyphen

    # If we have captured odd/even in m this will trigger and strip
    # it from the high range variable.
    [[ a< $m ]]&&h=${h% *}
    h=${h// }  # Strip Spaces

    # Give R a value.
    # If we are in a range it will be the number of elements in that range.
    # If we arent l and h will be equal are R will be 1
    ((R=h-l+1))

    # Call the odd or even functions if we captured one of them in m.
    # If we didnt m will contain a number and this will cause a warning
    # to stderr but it still works.
    eval "${m::1} $l $h"

    # Add R to total
    ((t+=R))
done

# Print result
echo $t

테스트 케이스를 실행하십시오.

bash math.sh "pg. 546: 17-19, 22, 26, pg. 548: 35-67 odd, 79, 80-86 even"
bash math.sh "pg. 34: 1"
bash math.sh "PG. 565: 2-5,PG.345:7"
bash math.sh "pg. 343: 5,8,13 - 56 even,pg. 345: 34 - 78,80"
bash math.sh "pg.492: 2-4 odd,7-9 even"

규칙을 읽는 방법에 따라 다른 4 바이트를 저장할 수 있습니다. 경우에도 / 홀수 항상 소문자를 ${1,,}변경할 수 있습니다$1


최신 테스트 케이스에서 작동합니까?
mbomb007

방금 테스트했지만 그렇습니다.
다니엘 웨이크 필드

1

자바 스크립트 ( ES6 ), 149

Firefox에서 스 니펫을 실행하여 테스트

F=s=>s.replace(/([-,.:]) *(\d+) *(o?)(e?)/ig,(_,c,v,o,e)=>
  c=='-'?(t+=1+(o?(v-(r|1))>>1:e?(v-(-~r&~1))>>1:v-r),r=0)
  :c!='.'&&(t+=!!r,r=v)
,r=t=0)&&t+!!r

// Less golfed

U=s=>{
  var r = 0, // value, maybe range start
  t = 0; // total
  s.replace(/([-,.:]) *(\d+) *(o?)(e?)/ig, // execute function for each match
    (_ // full match, not used
     , c // separator char, match -:,.
     , v // numeric value
     , o // match first letter of ODD if present
     , e // match first letter of EVEN if present
    )=>
    {
      if (c == '-') // range end
      {
        t++; // in general, count range values as end - start + 1
        if (o) // found 'odd'
        {
          r = r | 1; // if range start is even, increment to next odd
          t += (v-r)>>1; // end - start / 2
        }
        else if (e) // found 'even'
        {
          r = (r+1) & ~1; // if range start is odd, increment to next even
          t += (v-r)>>1; // end - start / 2
        }
        else
        {
          t += v-r; // end - start
        }
        r = 0; // range start value was used
      }
      else if (c != '.') // ignore page numbers starting with '.'
      { 
        // if a range start was already saved, then it was a single value, count it
        if (r != 0) ++t;
        r = v; // save value as it counld be a range start
      }
    }
  )            
  if (r != 0) ++t; // unused, pending range start, was a single value
  return t
}

// TEST

out=x=>O.innerHTML+=x+'\n';

test=["pg. 546: 17-19, 22, 26, pg. 548: 35-67 odd, 79, 80-86 even",
"pg. 34: 1", "PG. 565: 2-5,PG.345:7",
"pg. 343: 5,8,13 - 56 even,pg. 345: 34 - 78,80"];

test.forEach(t=>out(t + ' --> ' + F(t)))
<pre id=O></pre>


1

C ++ 226 224 222

나는 파티에 좀 늦었다는 것을 알고 있지만 이것은 재미있는 문제처럼 보였고 C 계열 언어를 사용하는 항목이 부족하여 나를 괴롭 혔습니다.

다음은 정규 표현식이나 문자열 대체를 사용하지 않고 간단한 수학을 사용하는 C ++ 함수입니다.

void f(){char c;int o=0,i,j;while(cin>>c)c=='p'||c==80?cin.ignore(9,58):cin.unget(),cin>>i>>c&&c==45?cin>>j>>c&&(c=='e'||c=='o')?cin.ignore(9,44),c=='e'?i+=i&1,j+=!(j&1):(i+=!(i&1),j+=j&1),o+=(j-i)/2:o+=j-i:0,++o;cout<<o;}

언 골프 :

void f()
{
  char c;
  int o=0,i,j;
  while(cin>>c)
    c=='p'||c==80?cin.ignore(9,58):cin.unget(),
    cin>>i>>c&&c==45?
      cin>>j>>c&&(c=='e'||c=='o')?
        cin.ignore(9,44),
        c=='e'?
          i+=i&1,j+=!(j&1)
        :(i+=!(i&1),j+=j&1),
        o+=(j-i)/2
      :o+=j-i
    :0,
    ++o;
  cout<<o;
}

나는 읽을 수 있다고 말하지 않았다. :) 삼항 연산자는 지옥입니다. 그래도 최선을 다하여 (다양한) 포맷을 시도 했으므로 적어도 조금 도움이되기를 바랍니다.

사용법 :

#include <iostream>
using namespace std;

void f(){char c;int o=0,i,j;while(cin>>c)c=='p'||c==80?cin.ignore(9,58):cin.unget(),cin>>i>>c&&c==45?cin>>j>>c&&(c=='e'||c=='o')?cin.ignore(9,44),c=='e'?i+=i&1,j+=!(j&1):(i+=!(i&1),j+=j&1),o+=(j-i)/2:o+=j-i:0,++o;cout<<o;}

int main()
{
  f();
}


0

파이썬 2-163 바이트 :

여기 사용해보십시오

따옴표 안에 입력해야합니다

import re
print len(eval(re.sub('([^,]+:|) *(\d+) *-? *(\d*)(?=.(.)).*?,',r'[x for x in range(\2,\3+1 if \3.0 else \2+1)if x%2!="oe".find("\4")]+',input()+',[]')))

설명:

일반적인 접근 방식은 기존 입력을 유효한 파이썬으로 변환 한 다음이를 평가하는 것입니다. 쉼표로 구분 된 각 값은 배열로 변환 된 다음 모두 함께 추가되고 길이는 최종 결과를 제공합니다.

예를 들어 input을 사용하면 12-15 odd,19평가 전에 정규식 대체가 다음을 생성합니다.

[x for x in range(12,15+1 if 15.0 else 12+1)if x%2!="oe".find("o")]
+[x for x in range(19,+1 if .0 else 19+1)if x%2!="oe".find("[")]
+[]

이것을 더 세분화하려면 :

  • 15+1 if 15.0 else 12+1 이 비트는 범위 또는 단일 값이 있는지 여부에 따라 range ()의 두 번째 인수가 올바른지 확인합니다 (\ 3이 비어 있으면 \ 3.0은 false로 평가됨).
  • if x%2!="oe".find("o")범위의 마지막 숫자에서 두 문자가 떨어진 값에 따라 ( (?=.(.))정규식에서 두 문자를 사용하지 않고 두 문자를 보임) 세 가지 가능한 결과가 있습니다.

    • x%2!="oe".find("o")x % 2 != 0(홀수 일치 만)으로 평가
    • x%2!="oe".find("e")x % 2 != 1일치 (만 일치)
    • x%2!="oe".find("[")평가 x % 2 != -1(이 문자는 마지막 숫자에서 두 문자 떨어져 있기 때문에 여러 가지 일 수 있지만 홀수 / 짝수를 의도하면 o 또는 e 만됩니다)
  • 마지막에 무작위로 보이는 + []는 쉼표로 구분 된 목록의 마지막 토큰이 마지막 숫자에서 2 자리 떨어진 문자를 갖도록하는 것입니다. 그렇지 않으면 그렇지 않을 것입니다.
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.