Implement division

15

2

Implement a division algorithm in your favourite language which handles integer division. It need only handle positive numbers - but bonus points if it handles negative and mixed-sign division, too. Results are rounded down for fractional results.

The program may not contain the /, \, div or similar operators. It must be a routine which does not use native division capabilities of the language.

You only need to handle up to 32-bit division. Using repeated subtraction is not allowed.

Input

Take two inputs on stdin separated by new lines or spaces (your choice)

740 
2

Output

In this case, the output would be 370.

The solution which is the shortest wins.

Thomas O

Posted 2011-02-04T10:12:10.190

Reputation: 3 044

does golfscript's \ operator (swap two elements) count as forbidden? – John Dvorak – 2013-05-01T16:07:47.193

You should have waited to accept an answer. There's a shorter one now. – Pavel – 2016-12-08T04:07:23.160

What does your '' operator mean? – sergiol – 2018-06-08T23:05:37.970

2this is shortest-time but I haven't seen anyone time the code – phuclv – 2014-06-09T13:45:33.440

"Innovative solutions that do not use repeated subtraction" – so good old long division? Hardly innovative, that, though. – Joey – 2011-08-10T08:05:51.177

is 740,2 also permitted for the input? ie comma separated? – gnibbler – 2011-02-04T10:44:37.817

"Results are rounded down for fractional results" - ok, so apparently the input can also result in a non-integer number... But what about the divisor being larger than the divided (say, 5 and 10) - is that permitted or not? – Aurel Bílý – 2011-02-04T12:13:40.870

@gnibber That would be fine, but make it clear in the program description. – Thomas O – 2011-02-04T12:23:44.617

@Aurel300 It is not required to have fractional output. 1/7 may produce 0.142857... or 0. – Thomas O – 2011-02-04T12:24:36.300

2is using exponentials and other math functions really allowed? they use division behind the scenes, because many solutions are doing ab⁻¹ – Ming-Tang – 2011-02-05T07:12:36.000

@SHiNKiROU - yeah I would consider that cheating too... – Aurel Bílý – 2011-02-05T10:43:59.103

Answers

27

Python - 73 chars

Takes comma separated input, eg 740,2

from math import*
x,y=input()
z=int(exp(log(x)-log(y)))
print(z*y+y<=x)+z

gnibbler

Posted 2011-02-04T10:12:10.190

Reputation: 14 170

Output for "740,2" is 369. Is this correct? – Eelvex – 2011-02-17T23:23:29.363

@Eelvex, should have been <=, fixed it and made it shorter :) – gnibbler – 2011-02-17T23:43:09.953

5This, my friend, is CLEVER – Aamir – 2011-02-04T11:25:09.557

14

JavaScript, 61

A=Array,P=prompt,P((','+A(+P())).split(','+A(+P())).length-1)

This makes a string the length of the dividend ,,,,,, (6) and splits on the divisor ,,, (3), resulting in an array of length 3: ['', '', ''], whose length I then subtract one from. Definitely not the fastest, but hopefully interesting nonetheless!

Casey Chu

Posted 2011-02-04T10:12:10.190

Reputation: 1 661

2My favorite implementation here so far. Congrats for the cool code! – Thomas Eding – 2011-08-12T23:14:46.843

I tried to make it a little shorter. A=Array,P=prompt,P((''+A(+P())).split(','+A(+P())).length) – pimvdb – 2011-08-30T19:54:15.550

10

JavaScript - 36 characters

p=prompt;alert(p()*Math.pow(p(),-1))

PleaseStand

Posted 2011-02-04T10:12:10.190

Reputation: 5 369

5Replacing alert with p will net you some extra characters. :) – Casey Chu – 2011-08-09T03:00:40.870

9

Mathematica: 34 chars

Solves symbolically the equation (x a == b)

Solve[x#[[1]]==#[[2]],x]&@Input[]

Dr. belisarius

Posted 2011-02-04T10:12:10.190

Reputation: 5 345

223 chars, Solve[x#==#2]&@@Input[] – chyanog – 2013-07-21T10:42:49.977

8

Python, 37

Step 1. Convert to unary.

Step 2. Unary division algorithm.

print('1'*input()).count('1'*input())

boothby

Posted 2011-02-04T10:12:10.190

Reputation: 9 038

8

Python - 72 chars

Takes comma separated input, eg 740,2

x,y=input();z=0
for i in range(32)[::-1]:z+=(1<<i)*(y<<i<=x-z*y)
print z

gnibbler

Posted 2011-02-04T10:12:10.190

Reputation: 14 170

7

Python - 41 chars

Takes comma separated input, eg 740,2

x,y=input();z=x
while y*z>x:z-=1 
print z

gnibbler

Posted 2011-02-04T10:12:10.190

Reputation: 14 170

1This, in some cases, is worse than continuously subtracting. e.g., input is 5,4. while loop will run 4 times while in case of subtraction, we will only have to subtract once. – Aamir – 2011-02-04T11:20:59.733

6

Python, 70

Something crazy I just thought (using comma separated input):

from cmath import*
x,y=input()
print round(tan(polar(y+x*1j)[1]).real)

If you accept small float precision errors, the round function can be dropped.

JBernardo

Posted 2011-02-04T10:12:10.190

Reputation: 1 659

4

Yabasic - 17 characters

input a,b:?a*b^-1

PleaseStand

Posted 2011-02-04T10:12:10.190

Reputation: 5 369

3

Ruby 1.9, 28 characters

(?a*a+?b).split(?a*b).size-1

Rest of division, 21 characters

?a*a=~/(#{?a*b})\1*$/  

Sample:

a = 756
b = 20
print (?a*a+?b).split(?a*b).size-1  # => 37
print ?a*a=~/(#{?a*b})\1*$/         # => 16

For Ruby 1.8:

a = 756
b = 20
print ('a'*a+'b').split('a'*b).size-1  # => 37
print 'a'*a=~/(#{'a'*b})\1*$/          # => 16

LBg

Posted 2011-02-04T10:12:10.190

Reputation: 130

NoMethodError: private method `split' called for 69938:Fixnum – rkj – 2011-08-26T08:27:21.507

@rkj, Sorry, Ruby 1.9 only. To run on Ruby 1.8 you must do ('a'*a+'b').split('a'*b).size-1, 3 characters bigger. – LBg – 2011-08-29T00:30:30.830

3

PHP - 82 characters (buggy)

$i=fgets(STDIN);$j=fgets(STDIN);$k=1;while(($a=$j*$k)<$i)$k++;echo($a>$i?--$k:$k);

This is a very simple solution, however - it doesn't handle fractions or different signs (would jump into an infinite loop). I won't go into detail in this one, it is fairly simple.

Input is in stdin, separated by a new line.

PHP - 141 characters (full)

$i*=$r=($i=fgets(STDIN))<0?-1:1;$j*=$s=($j=fgets(STDIN))<0?-1:1;$k=0;$l=1;while(($a=$j*$k)!=$i){if($a>$i)$k-=($l>>=2)*2;$k+=$l;}echo$k*$r*$s;

Input and output same as the previous one.

Yes, this is almost twice the size of the previous one, but it:

  • handles fractions correctly
  • handles signs correctly
  • won't ever go into an infinite loop, UNLESS the second parameter is 0 - but that is division by zero - invalid input

Re-format and explanation:

$i *= $r = ($i = fgets(STDIN)) < 0 ? -1 : 1;
$j *= $s = ($j = fgets(STDIN)) < 0 ? -1 : 1;
                                    // First, in the parentheses, $i is set to
                                    // GET variable i, then $r is set to -1 or
                                    // 1, depending whether $i is negative or
                                    // not - finally, $i multiplied by $r ef-
                                    // fectively resulting in $i being the ab-
                                    // solute value of itself, but keeping the
                                    // sign in $r.
                                    // The same is then done to $j, the sign
                                    // is kept in $s.

$k = 0;                             // $k will be the result in the end.

$l = 1;                             // $l is used in the loop - it is added to
                                    // $k as long as $j*$k (the divisor times
                                    // the result so far) is less than $i (the
                                    // divided number).

while(($a = $j * $k) != $i){        // Main loop - it is executed until $j*$k
                                    // equals $i - that is, until a result is
                                    // found. Because a/b=c, c*b=a.
                                    // At the same time, $a is set to $j*$k,
                                    // to conserve space and time.

    if($a > $i)                     // If $a is greater than $i, last step
        $k -= ($l >>= 2) * 2;       // (add $l) is undone by subtracting $l
                                    // from $k, and then dividing $l by two
                                    // (by a bitwise right shift by 1) for
                                    // handling fractional results.
                                    // It might seem that using ($l>>=2)*2 here
                                    // is unnecessary - but by compressing the
                                    // two commands ($k-=$l and $l>>=2) into 1
                                    // means that curly braces are not needed:
                                    //
                                    // if($a>$i)$k-=($l>>=2)*2;
                                    //
                                    // vs.
                                    //
                                    // if($a>$i){$k-=$l;$l>>=2;}

    $k += $l;                       // Finally, $k is incremented by $l and
                                    // the while loop loops again.
}

echo $k * $r * $s;                  // To get the correct result, $k has to be
                                    // multiplied by $r and $s, keeping signs
                                    // that were removed in the beginning.

Aurel Bílý

Posted 2011-02-04T10:12:10.190

Reputation: 1 083

You used a division operator in this one, you might get away with a bit shift though. ;) – Thomas O – 2011-02-04T18:12:19.907

@Thomas O yeah... I noticed it now... I was actually thinking about a bit shift (when I changed it to /=2 instead of /=10) - but it was one more char... Guess I'll have to use it anyway... Btw that is not division at all :D. – Aurel Bílý – 2011-02-04T18:52:28.663

The question says you need to use stdin, which PHP does have support for. – Kevin Brown – 2011-02-05T02:08:49.597

@Bass5098 Aaahhh... Oh well, gained 4 chars... Fixed. – Aurel Bílý – 2011-02-05T17:07:10.857

3

APL (6)

⌊*-/⍟⎕

/ is not division here, but foldr. i.e, F/a b c is a F (b F c). If I can't use foldr because it's called /, it can be done in 9 characters:

⌊*(⍟⎕)-⍟⎕

Explanation:

  • : input()
  • ⍟⎕: map(log, input())
  • -/⍟⎕: foldr1(sub, map(log, input()))
  • *-/⍟⎕: exp(foldr1(sub, map(log, input())))
  • ⌊*-/⍟⎕: floor(exp(foldr1(sub, map(log, input()))))

marinus

Posted 2011-02-04T10:12:10.190

Reputation: 30 224

2

Bash, 72 64 characters

read x y;yes ''|head -n$x>f;ls -l --block-size=$y f|cut -d\  -f5

Output an infinite number of newlines, take the first x, put them all into a file called f, then get the size of f in blocks the size of y. Took manatwork's advice to shave off eight characters.

Hovercouch

Posted 2011-02-04T10:12:10.190

Reputation: 659

As “Take two inputs on stdin separated by new lines or spaces (your choice)”, better choose the later, the space separated values. In which case you can write read x y. With a few more spaces removed can be reduced to 64 characters: http://pastebin.com/Y3SfSXWk

– manatwork – 2013-09-20T08:20:22.580

2

Haskell, 96 characters

main=getLine>>=print.d.map read.words
d[x,y]=pred.snd.head.filter((>x).fst)$map(\n->(n*y,n))[0..]

Input is on a single line.

The code just searches for the answer by taking the divisor d and multiplying it against all integers n >= 0. Let m be the dividend. The largest n such that n * d <= m is picked to be the answer. The code actually picks the least n such that n * d > m and subtracts 1 from it because I can take the first element from such a list. In the other case, I would have to take the last, but it's hard work to take the last element from an infinite list. Well, the list can be proven to be finite, but Haskell does not know better when performing the filter, so it continues to filter indefinitately.

Thomas Eding

Posted 2011-02-04T10:12:10.190

Reputation: 796

2

Scala 77

def d(a:Int,b:Int,c:Int=0):Int=if(b<=a)d(a-b,b,c+1)else c
d(readInt,readInt)

user unknown

Posted 2011-02-04T10:12:10.190

Reputation: 4 210

2

PHP, 55 characters

<?$a=explode(" ",fgets(STDIN));echo$a[0]*pow($a[1],-1);

Output (740/2): http://codepad.viper-7.com/ucTlcq

Kevin Brown

Posted 2011-02-04T10:12:10.190

Reputation: 5 756

44 chars: <?$a=fgetcsv(STDIN);echo$a[0]*pow($a[1],-1); Just use a comma instead of a space to separate numbers. – jdstankosky – 2013-05-10T17:28:28.153

2

Common Lisp, 42 charaacters

(1-(loop as x to(read)by(read)counting t))

Accepts space or line-separated input

Strigoides

Posted 2011-02-04T10:12:10.190

Reputation: 1 025

1

Python - 45 chars

Takes comma separated input, eg 740,2

x,y=input()
print-1+len((x*'.').split('.'*y))

gnibbler

Posted 2011-02-04T10:12:10.190

Reputation: 14 170

1

Python, 94 characters

A recursive binary search:

a,b=input()
def r(m,n):return r(m,m+n>>1)if n*b>a else n if n*b+b>a else r(n,2*n)
print r(0,1)

memo

Posted 2011-02-04T10:12:10.190

Reputation: 21

1

Python, 148

Other solutions may be short, but are they web scale?

Here's an elegant, constant-time solution that leverages the power of the CLOUD.

from urllib import*
print eval(urlopen('http://tryhaskell.org/haskell.json?method=eval&expr=div%20'+raw_input()+'%20'+raw_input()).read())['result']

Did I mention it also uses Haskell?

Lambda Fairy

Posted 2011-02-04T10:12:10.190

Reputation: 311

0

C, 43 bytes

Implements a/b in a linear search. Handles only positive arguments.

i;f(a,b){i=0;while(1)if(++i*b>a)return--i;}

Ungolfed:

i;
f(a, b){
  i = 0;              
  while (1)           
    if (++i * b > a)  
      return --i;     
}

Karl Napf

Posted 2011-02-04T10:12:10.190

Reputation: 4 131

0

Smalltalk, Squeak 4.x flavour

define this binary message in Integer:

% d 
    | i |
    d <= self or: [^0].
    i := self highBit - d highBit.
    d << i <= self or: [i := i - 1].
    ^1 << i + (self - (d << i) % d)

Once golfed, this quotient is still long (88 chars):

%d|i n|d<=(n:=self)or:[^0].i:=n highBit-d highBit.d<<i<=n or:[i:=i-1].^1<<i+(n-(d<<i)%d)

But it's reasonnably fast:

[0 to: 1000 do: [:n |
    1 to: 1000 do: [:d |
        self assert: (n//d) = (n%d)]].
] timeToRun.

-> 127 ms on my modest mac mini (8 MOp/s)

Compared to regular division:

[0 to: 1000 do: [:n |
    1 to: 1000 do: [:d |
        self assert: (n//d) = (n//d)]].
] timeToRun.

-> 31 ms, it's just 4 times slower

I don't count the chars to read stdin or write stdout, Squeak was not designed for scripting.

FileStream stdout nextPutAll:
    FileStream stdin nextLine asNumber%FileStream stdin nextLine asNumber;
    cr

Of course, more stupid repeated subtraction

%d self>d and:[^0].^self-d%d+1

or plain stupid enumeration

%d^(0to:self)findLast:[:q|q*d<=self]

could work too, but are not really interesting

aka.nice

Posted 2011-02-04T10:12:10.190

Reputation: 411

0

#include <stdio.h>
#include <string.h>
#include <math.h>


main()
{
   int i,j,ans;
   i=740;
   j=2;

   ans = pow(10,log10(i) - log10(j));
   printf("\nThe answer is %d",ans);
}

jithin

Posted 2011-02-04T10:12:10.190

Reputation: 1

0

DC: 26 characters

?so?se0[1+dle*lo>i]dsix1-p

I do admit that it is not the fastest solution around.

Fors

Posted 2011-02-04T10:12:10.190

Reputation: 3 020

0

Python 54

Takes comma delimited input.

  1. Makes a string of dots of length x
  2. Replaces segments of dots of length y with a single comma
  3. Counts commas.

Words because markdown dies with a list followed by code?:

x,y=input()
print("."*x).replace("."*y,',').count(',')

user8777

Posted 2011-02-04T10:12:10.190

Reputation:

0

Q, 46

{-1+(#){x-y}[;y]\[{s[x-y]<>(s:signum)x}[y];x]}

.

q){-1+(#){x-y}[;y]\[{s[x-y]<>(s:signum)x}[y];x]}[740;2]
370
q){-1+(#){x-y}[;y]\[{s[x-y]<>(s:signum)x}[y];x]}[740;3]
246

tmartin

Posted 2011-02-04T10:12:10.190

Reputation: 3 917

0

int divide (int num, int divideBy)
{
    int sum = 0;
    int shift = divideBy -1;
    while (num > divideBy) {
        sum += num >> shift ;
        num = (num >> shift ) + (num & divideBy);
    }
    if (num == divideBy) ++sum;
    return sum; 
}

Reference: http://www.forums.hscripts.com/viewtopic.php?f=13&t=1358

Parvej Solkar

Posted 2011-02-04T10:12:10.190

Reputation: 1

0

Python, 40 characters

print(float(input())*float(input())**-1)

fdvfcges

Posted 2011-02-04T10:12:10.190

Reputation: 101

0

Python, 37

x,y=input()
print len(('0'*x)[y-1::y])

Constructs a string of length x ('0'*x) and uses extended slicing to pick every yth character, starting from the index y-1. Prints the length of the resulting string.

Like Gnibbler, this takes comma separated input. Removing it costs 9 chars:

i=input
x,y=i(),i()
print len(('0'*x)[y-1::y])

Justin

Posted 2011-02-04T10:12:10.190

Reputation: 19 757

0

Python, 46 bytes

Nobody had posted the boring subtraction solution, so I could not resist doing it.

a,b=input()
i=0
while a>=b:a-=b;i+=1
print i

cemper93

Posted 2011-02-04T10:12:10.190

Reputation: 670

0

Retina 0.7.3, 33 bytes (not competing)

The language is newer than the challenge. Takes space-separated input with the divisor first. Dividing by zero is undefined.

\d+
$*
^(.+) (\1)+.*$
$#+
.+ .*
0

Try it online

mbomb007

Posted 2011-02-04T10:12:10.190

Reputation: 21 944

How do you count this as 25 bytes? If you expect unary I/O you should say so (and I think then it's 24 bytes). Not sure why you treat the 0 case separately though: http://retina.tryitonline.net/#code=LisKJCoKKC4rKcK2KFwxKSouKgokIys&input=Mgo3NDA

– Martin Ender – 2016-12-07T22:24:07.290

It was mis-copied – mbomb007 – 2016-12-07T23:05:26.707