Find the Smoothest Number

60

7

Your challenge is to find the smoothest number over a given range. In other words, find the number whose greatest prime factor is the smallest.

A smooth number is one whose largest prime factor is small. Numbers of this type are useful for the fast Fourier transform algorithm, cryptanalysis, and other applications.

For instance, over the range 5, 6, 7, 8, 9, 10, 8 is the smoothest number, because 8's greatest prime factor is 2, whereas all of the other numbers have a prime factor of 3 or greater.

Input: The input will be two positive integers, which define a range. The minimum allowable integer in the range is 2. You may choose whether the range is inclusive, exclusive, semi-exclusive, etc, as long as an arbitrary range can be specified within the bounds of your language. You may take the numbers via function input, stdin, command line argument, or any equivalent method for your language. No encoding extra information in the input.

Output: Return, print or equivalent one or more integers in the input range which are maximally smooth (minimal greatest factor). Returning multiple results is optional, but if you choose to do so the results must be clearly delimited. Native output format is fine for multiple results.

Please state in your answer how you are taking input and giving output.

Scoring: Code golf. Count by characters if written in ASCII, or 8*bytes/7 if not in ASCII.

Test cases:

Note: These are Python-style ranges, including the low end but not the high end. Change as appropriate to your program. Only one result is necessary.

smooth_range(5,11)
8
smooth_range(9,16)
9, 12
smooth_range(9,17)
16
smooth_range(157, 249)
162, 192, 216, 243
smooth_range(2001, 2014)
2002

isaacg

Posted 2014-08-18T23:30:24.447

Reputation: 39 268

3@isaacg you're encouraging the creation of pseudo-languages using smaller character sets. if we score 7-bit character sets different from 8-bit character sets, someone can pack most modern languages into 6 bits (64 characters gets us A-Z, 0-9, a handful of whitespace, 20 punctuation, and a few to spare). – Sparr – 2015-03-08T07:14:32.113

Are ranges specified as (start,length) instead of (start,end) acceptable? – CodesInChaos – 2014-08-21T16:26:11.057

1@CodesInChaos Sure. It's covered under the "or whatever" clause. – isaacg – 2014-08-21T17:09:10.467

3I don't see the point in penalizing non-ASCII answers. It would be simpler to just count bytes in all cases. – nyuszika7h – 2014-08-21T17:59:58.110

1@nyuszika7h Ascii is significantly smaller than a byte - it only uses 7 bits. Therefore, I denote one character by 7 bits, and scale other languages accordingly. However, if the language is non-ASCII but can pack all of its characters into 7 bits, I will not apply the surcharge. See J/K vs. APL. tl;dr Bytes is simpler, but gives APL et. al. a subtle but unfair advantage. – isaacg – 2014-08-21T18:31:53.757

Answers

100

CJam - 13

q~,>{mfW=}$0=

Try it at http://cjam.aditsu.net/

Example input: 2001 2014
Example output: 2002

Explanation:

q~ reads and evaluates the input, pushing the 2 numbers on the stack (say min and max)
, makes an array [0 1 ... max-1]
> slices the array starting at min, resulting in [min ... max-1]
{…}$ sorts the array using the block to calculate the sorting key
mf gets an array with all the prime factors of a number, in order
W= gets the last element of the array (W=-1), thus obtaining the largest prime factor to be used as a sorting key
0= gets the first element of the (sorted) array

aditsu quit because SE is EVIL

Posted 2014-08-18T23:30:24.447

Reputation: 22 326

Awesome! How does it work? – isaacg – 2014-08-19T00:23:52.790

38Well, I guess that's that. – Eric Tressler – 2014-08-19T00:23:55.177

Amazing. What a compact answer! Very nice. – Todd Lehman – 2014-08-19T00:42:23.580

5I need to add a factorize function to pyth. – isaacg – 2014-08-19T00:45:16.743

2As I would say in my games : gg – Zaenille – 2014-08-19T01:06:51.580

6This language is wizardry. – Brobin – 2014-08-19T03:52:00.580

8This is as close to just pulling some HQ9+ s**t as it can be without becoming a loophole. Awesome! – Ingo Bürk – 2014-08-19T19:45:57.717

25ヽ༼ຈل͜ຈ༽ノ mfW someone solved it in 13 chars. – teh internets is made of catz – 2014-08-19T22:48:02.663

2@Brobin It's not wizardry, it's just highly evolved for a very specific purpose - solving code golfs. Personally, I give more credit to a chef who can julienne with an 8" knife than some bloke on TV with a one-slap julienne machine. At least the knife has broader practical utility... – J... – 2014-08-19T22:49:56.437

1@J... I basically agree with you, but I created CJam partly as a reaction to GolfScript stealing the show every time while my 200+ char java answers barely got 1 or 2 upvotes – aditsu quit because SE is EVIL – 2014-08-20T07:09:05.757

1@aditsu It's still just a collection of one-char macros wrapped around a huge pile of code made to solve the CG of the day. Magic unicorn points will be the ruination of us all... – J... – 2014-08-20T09:09:13.517

3On the other hand, most languages are "wrappers" to serve a certain purpose. No one wants to write assembler/machine code. It's a blurry line to walk on. – Ingo Bürk – 2014-08-20T09:39:59.747

3@IngoBürk I... I like to write machine code ;D – J... – 2014-08-20T11:27:43.470

@J... My point still stands :) – Ingo Bürk – 2014-08-20T14:43:31.327

So when do we start submitting code golf answers via a single character that just maps to the real answer which is hard coded into the language? – Cory Klein – 2014-08-20T16:23:49.103

@CoryKlein never - it's a standard loophole to be avoided.

– aditsu quit because SE is EVIL – 2014-08-20T20:34:18.200

@EricTressler No one's done APL yet, so there's still a chance... – Izkata – 2014-08-20T21:44:53.867

Looks like a mix of Forth and APL... :-) – PhiLho – 2014-08-21T15:49:47.540

68

Regex (.NET PCRE flavour), 183 129 bytes

Don't try this at home!

This is not really a contender for the win. But Eric Tressler suggested solving this problem with nothing but a regex, and I couldn't resist giving it a go. This might be is possible in PCRE as well (and even shorter, see below), but I chose .NET because my solution needs arbitrary-length lookbehinds. Here we go:

(?<=^(1+),.*)(?=\1)(?=((11+)(?=.*(?=\3$)(?!(11+?)\4+$))(?=\3+$)|(?!(11+)\5+$)1+))(?!.+(?=\1)(?:(?!\2)|(?=((11+)(?=.*(?=\7$)(?!(11+?)\8+$))(?=\7+$)|(?!(11+)\9+$)1+)).*(?=\2$)(?=\6)))1+

The input is encoded as an inclusive comma-separated range, where both numbers are given in unary notation using 1s. The match will be the trailing S 1s where S is the smoothest number in the range. Ties are broken in favour of the smallest number.

So the second example from the question would be the following string (match underlined)

111111111,1111111111111111
                 =========

It is based on the (by now rather well-known) prime-checking regex, variations of which are embedded in there a whopping 6 times.

Here is a version using free-spacing and comments for those who want to know what's going on.

# Note that the beginning of the match we're looking for is somewhere
# in the second part of the input.
(?<=^(1+),.*)          # Pick up the minimum range MIN in group 1
(?=\1)                 # Make sure there are at least MIN 1s ahead

                       # Now there will be N 1s ahead of the cursor
                       # where MIN <= N <= MAX.


(?=(                   # Find the largest prime factor of this number
                       # store it in group 2.
  (11+)                # Capture a potential prime factor P in group 3
  (?=                  # Check that it's prime
    .*(?=\3$)          # Move to a position where there are exactly 
                       # P 1s ahead
    (?!(11+?)\4+$)     # Check that the remaining 1s are not composite
  )
  (?=\3+$)             # Now check that P is a divisor of N.
|                      # This does not work for prime N, so we need a 
                       # separate check
  (?!(11+)\5+$)        # Make sure that N is prime.
  1+                   # Match N
))

(?!                    # Now we need to make sure that here is not 
                       # another (smaller) number M with a smaller 
                       # largest prime factor

  .+                   # Backtrack through all remaining positions
  (?=\1)               # Make sure there are still MIN 1s ahead

  (?:
    (?!\2)             # If M is itself less than P we fail 
                       # unconditionally.
  |                    # Else we compare the largest prime factors.
    (?=(               # This is the same as above, but it puts the
                       # prime factor Q in group 6.
      (11+)
      (?=
        .*(?=\7$)
        (?!(11+?)\8+$)
      )
      (?=\7+$)
    |
      (?!(11+)\9+$)
      1+
    ))
    .*(?=\2$)          # Move to a position where there are exactly 
                       # P 1s ahead
    (?=\6)             # Try to still match Q (which means that Q is
                       # less than P)
  )
)
1+                     # Grab all digits for the match

You can test it online over here. Don't try too large inputs though, I make no guarantees about the performance of this monster.

Edit:

I ended up porting this to PCRE (which only requires two steps), and shortening the regex by almost a third. Here is the new version:

^(1+),.*?\K(?=\1)(?=((11+)(?=.*(?=\3$)(?!(11+?)\4+$))(?=\3+$)|(?!(11+)\5+$)1+))(?!.+(?=\1)(?:(?!\2)|(?=((?2))).*(?=\2$)(?=\6)))1+

This is essentially the same, with two changes:

  • PCRE does not support arbitrary-length lookbehind (which I used to get the MIN into group 1). However, PCRE does support \K which resets the beginning of the match to the current cursor position. Hence (?<=^(1+),.*) becomes ^(1+),.*?\K, which already saves two bytes.
  • The real savings come from PCRE's recursion feature. I'm not actually using recursion, but you can use (?n) to match group n again, similar to a subroutine call. Since the original regex contained the code for finding a number's largest prime factor twice, I was able to replace the whole bulk of the second one with a simple (?2).

Martin Ender

Posted 2014-08-18T23:30:24.447

Reputation: 184 808

You write “This does not work for prime N, so we need a separate check”. However, could you not make it work for prime N simply by changing (=\3+$) to (=\3*$) (and also (=\7+$) to (=\7*$)) and then save yourself the extra primality check? This would save 32 characters. – Timwi – 2015-09-04T19:36:59.227

1@Timwi I need to check that the largest prime factor (group 3 or 7) is actually prime. This requires that there is another copy of the factor after first capturing it, which wouldn't be the case for primes. While I work around that in .NET by putting a lookbehind somewhere in there such that I could move back a bit for the check, this wouldn't be possible in the shorter PCRE version due to the lack of variable-length lookbehinds. It probably is possible to shorten that bit, but I don't think just changing + to * works. – Martin Ender – 2015-09-04T21:15:12.573

2@MartinEnder Hi! I imagine you're long since over this challenge, but I just surfed in, saw a regex solution, and couldn't help but totally ignore your warning at the top of this post :) I find it hard to golf other people's code, so after looking at your regex and getting confused, I attempted it from scratch and came up with this:

(.*),.*?\K(?=(..+)((?=((?(R)\6|\2))*$).*(?=\4$)(?!(..+)\5+$)))(?!.+(?=\1)(?=(..+)(?3)).*(?!\2)\6).+

99 bytes in PCRE. Also, I've come across a lot of your work on this site and I'm a big fan :D Looking forward to a regex battle in the future! – jaytea – 2017-10-20T06:14:17.797

1I was playing code golf with that comment so I'll just put the addendum here: you can knock off 4b by taking \4$ out of the lookahead and sticking it after the negative lookahead, but this severely impacts performance (every subgroup of digits <= \4 is checked for compositeness rather than just \4 itself) and fails on longer inputs. – jaytea – 2017-10-20T06:19:34.037

1@jaytea Sorry for taking forever to get back to you on this. Since you wrote the thing from scratch, I think you should post a separate answer. That's a great score and you deserve the credit for it. :) – Martin Ender – 2017-11-19T11:04:43.237

Very kind of you to say, but I feel it's too similar in the context of a challenge that wasn't actually supposed to be solved with regex! Haha. – jaytea – 2017-11-19T13:02:10.650

1

I tackled this myself and came up with a 66 byte PCRE solution: ((.+).*),(?!.*(?=\1)(((?=(..+)(\5+$))\6)*)(?!\2)).*(?=\1)\K(?3)\2$

– Deadcode – 2019-01-18T04:48:27.840

@jaytea Ran your regex through an automated test and it fails on (and only on) singleton ranges [P,P] where P is any prime number, returning P-1 instead of P – and on [2,2], [2,3], and [3,3], it returns a non-match. Other than that it passes perfectly. The after-the-negative-lookahead length optimization results in a 10.3× slowdown in my engine when tested on all ranges [A,B] where A≤B and A+B ≤ 255, and a 4.38× slowdown for A+B ≤ 127. – Deadcode – 2019-01-19T00:58:08.683

37Holy mother of god – Newb – 2014-08-21T00:51:47.060

17

Regex (PCRE flavour), 66 (65) bytes

Inspired by seeing that both Martin Ender and jaytea, two regex geniuses, wrote regex solutions to this code golf, I wrote my own from scratch. The famous prime-checking regex does not appear anywhere in my solution.

Do not read this if you don't want some unary regex magic spoiled for you. If you do want to take a shot at figuring out this magic yourself, I highly recommend starting by solving some problems in ECMAScript regex:

  1. Match prime numbers (if you aren't already familiar with doing this in regex)
  2. Match powers of 2 (if you haven't already done so). Or just work your way through Regex Golf, which includes Prime and Powers. Make sure to do both the Classic and Teukon problem sets.
  3. Find the shortest way to match powers of N where N is some constant (i.e. specified in the regex, not the input) which can be composite (but is not required to be). For example, match powers of 6.

  4. Find a way of matching Nth powers, where N is some constant >=2. For example, match perfect squares. (For a warmup, match prime powers.)

  5. Match correct multiplication statements. Match triangular numbers.

  6. Match Fibonacci numbers (if you're as crazy as I am), or if you want to stick to something shorter, match correct statements of exponentiation (for a warmup, return as a match the logarithm in base 2 of a power of 2 – bonus, do the same for any number, rounding it however you like), or factorial numbers (for a warmup, match primorial numbers).

  7. Match abundant numbers (if you're as crazy as I am)

  8. Calculate an irrational number to requested precision (e.g. divide the input by the square root of 2, returning the rounded result as a match)

(The regex engine I wrote may be of help, as it is very fast at unary math regexes and includes a unary numerical mode which can test ranges of natural numbers (but also has a strings mode which can evaluate non-unary regexes, or unary with delimiters). By default it is ECMAScript compatible, but has optional extensions (which can selectively add subsets of PCRE, or even molecular lookahead, something that no other regex engine has).)

Otherwise, read on, and also read this GitHub Gist (warning, many spoilers) which chronicles the journey of pushing ECMAScript regex to tackle natural number functions of increasing difficulty (starting with teukon's set of puzzles, not all of them mathematical, which sparked this journey).

As with the other regex solutions to this problem, the input is given as two numbers in bijective unary, separated by a comma, representing an inclusive range. Only one number is returned. The regex could be modified to return all of the numbers that share the same smallest largest prime factor, as separate matches, but that would require variable-length lookbehind and either putting \K in a lookahead or returning the result as a capture instead of a match.

The technique used here of repeated implicit division by smallest prime factor is identical to that used in the Match strings whose length is a fourth power answer I posted a while back.

With no further ado: ((.+).*),(?!.*(?=\1)(((?=(..+)(\5+$))\6)*)(?!\2)).*(?=\1)\K(?3)\2$

You can try it out here.

And the free-spacing version, with comments:

                        # No ^ anchor needed, because this algorithm always returns a
                        # match for valid input (in which the first number is less than
                        # or equal to the second number), and even in /g mode only one
                        # match can be returned. You can add an anchor to make it reject
                        # invalid ranges.

((.+).*),               # \1 = low end of range; \2 = conjectured number that is the
                        # smallest number in the set of the largest prime factor of each
                        # number in the range; note, it is only in subsequent tests that
                        # this is implicitly confined to being prime.
                        # We shall do the rest of our work inside the "high end of range"
                        # number.

(?!                     # Assert that there is no number in the range whose largest prime
                        # factor is smaller than \2.
  .*(?=\1)              # Cycle tail through all numbers in the range, starting with \1.

  (                     # Subroutine (?3):
                        # Find the largest prime factor of tail, and leave it in tail.
                        # It will both be evaluated here as-is, and later as an atomic
                        # subroutine call. As used here, it is not wrapped in an atomic
                        # group. Thus after the return from group 3, backtracking back
                        # into it can increase the value of tail – but this won't mess
                        # with the final result, because only making tail smaller could
                        # change a non-match into a match.

    (                   # Repeatedly divide tail by its smallest prime factor, leaving
                        # only the largest prime factor at the end.

      (?=(..+)(\5+$))   # \6 = tool to make tail = \5 = largest nontrivial factor of
                        # current tail, which is implicitly the result of dividing it
                        # by its smallest prime factor.
      \6                # tail = \5
    )*
  )
  (?!\2)                # matches iff tail < \ 2
)

# now, pick a number in the range whose largest prime factor is \2
.*(?=\1)                # Cycle tail through all numbers in the range, starting with \1.
\K                      # Set us up to return tail as the match.
(?3)                    # tail = largest prime factor of tail
\2$                     # Match iff tail == \2, then return the number whose largest
                        # prime factor is \2 as the match.

The algorithm can be easily ported to ECMAScript by replacing the subroutine call with a copy of the subroutine, and returning the match as a capture group instead of using \K. The result is 80 bytes in length:

((x+)x*),(?!.*(?=\1)((?=(xx+)(\4+$))\5)*(?!\2)).*(?=\1)(((?=(xx+)(\8+$))\9)*\2$)

Try it online!

Note that ((.+).*) can be changed to ((.+)+), dropping the size by 1 byte (from 66 to 65 bytes) with no loss of correct functionality – but the regex exponentially explodes in slowness.

Try it online! (79 byte ECMAScript exponential-slowdown version)

Deadcode

Posted 2014-08-18T23:30:24.447

Reputation: 3 022

11

Python 2, 95

i=input()
for a in range(*i):
 s=a;p=2
 while~-a:b=a%p<1;p+=1-b;a/=p**b
 if p<i:i=p;j=s                                        
print j

Finds the smoothness of the the numbers by trial division until the number is 1. i stores the smallest smoothness so far, j stores the number that gave that smoothness.

Thanks to @xnor for the golfs.

isaacg

Posted 2014-08-18T23:30:24.447

Reputation: 39 268

1That if/else has got to be shortenable. My first thought is b=a%p<1;p+=1-b;a/=p**b. Or an exec that runs one of the two in an interleaved string. Also, maybe while~-a works. – xnor – 2014-08-19T01:24:00.183

isaacg — I love this answer! What a brilliant way you found to search for the largest prime factor! I have updated my answer to borrow your method, with credit to you on the method. – Todd Lehman – 2014-08-19T04:03:16.393

Great solution! Using s,p=a,2, i,j=p,s, @xnor's ideas, removing redundant indentation and putting the while block into one line yields 95 characters. Not sure how you came up with 98... – Falko – 2014-08-19T05:30:03.997

this code is full of emoticons, :) – Rosenthal – 2014-08-19T06:32:51.943

@Falko those two changes save no characters. 7->7. – isaacg – 2014-08-19T07:55:18.587

@isaacg: Oh, you're totally right! – Falko – 2014-08-19T07:57:32.840

10

J, 22 20 19 chars

({.@/:{:@q:)@(}.i.)

E.g.

   2001 ({.@/: {:@q:)@(}. i.) 2014
2002

(Functions taking two arguments are infix in J.)

FireFly

Posted 2014-08-18T23:30:24.447

Reputation: 7 107

Here {: is the same as >./ and saves 1 byte. – randomra – 2015-03-08T01:34:05.093

@randomra You're right—good call! – FireFly – 2015-03-08T17:14:22.270

Beautiful. TIO if you'd like to add it: Try it online!

– Jonah – 2019-01-19T02:41:09.980

I also had a crack at it, didn't get it as short as this answer. Still: (#~ (= <./)@:(i:"1&1)@:*@:(_&q:))@:([ + i.@-~) – ɐɔıʇǝɥʇuʎs – 2014-08-20T14:33:59.337

9

Mathematica, 61 45 39 characters

Range@##~MinimalBy~Last@*FactorInteger&

Very straightforward implementation of the spec as an unnamed function.

  • Get the range (inclusive).
  • Factor all integers.
  • Find the minimum, sorted by largest prime factor.

Martin Ender

Posted 2014-08-18T23:30:24.447

Reputation: 184 808

9

Haskell, 96 94 93 86 80 characters

x%y|x<2=y|mod x y<1=div x y%y|0<1=x%(y+1)
a#b=snd$minimum$map(\x->(x%2,x))[a..b]

usage via GHCi (a Haskell shell):

>5 # 9
8
>9 # 15
9

EDIT: now a much simpler algorithm.

this solution includes both numbers in the range (so 8 # 9 and 7 # 8 are both 8)

explanation:

the (%) function takes two parameters, x and y. when y is 2, the function returns the smoothness of x.

the algorithm from here is simple - get the combined list of all smoothnesses of numbers in the input with each smoothness storing a reference to it's original number, sort then to get the smallest, and return it's referenced number.


here is an ungolfed javascript version with the same algorithm:

function smoothness(n,p)
{
    p = p || 2
    if (x == 1)
        return p
    if (x % p == 0)
        return smoothness(x/p, p)
    else
        return smoothness(x,p+1);
}
function smoothnessRange(a, b)
{
    var minSmoothness = smoothness(a);
    var min=a;
    for(var i=a+1;i <= b;i++)
        if(minSmoothness > smoothness(i))
        {
            minSmoothness = smoothness(i)
            min = i
        }
    return min;
}

proud haskeller

Posted 2014-08-18T23:30:24.447

Reputation: 5 866

Would it be possible to alias minimum to something shorter? That looks like it would save some characters. – isaacg – 2014-08-19T19:43:35.617

I tried it, but because of the monomorphism restriction it actually costs one character – proud haskeller – 2014-08-19T19:44:26.217

You can't just do m=minimum? Haskell is still a mystery. – isaacg – 2014-08-19T19:45:07.077

No, because of the monomorphism restriction. It is something to do with type classes – proud haskeller – 2014-08-19T19:46:08.437

1@isaacg To bypass the monomorphism restriction one would have to write m l=minimum l – proud haskeller – 2014-08-19T19:54:00.380

2I was going to post a Haskell solution, until I saw yours which beats even my incomplete version... +1 – nyuszika7h – 2014-08-21T18:31:46.893

8

Bash+coreutils, 56 bytes

seq $@|factor|sed 's/:.* / /'|sort -nk2|sed '1s/ .*//;q'

Input is from from exactly two command-line arguments (Thanks @nyuszika7h !!!). Output is a singular result printed to STDOUT.

  • seq provides the range of numbers, one per line, from the command-line arguments.
  • factor reads those numbers and outputs each number followed by a colon and the sorted list of prime factors of that number. So the largest prime factor is at the end of each line.
  • The first sed removes the colon and all but the last/largest prime factor, so leaves a list of each number (column 1) and its largest prime factor (column 2).
  • sort numerically in increasing order by the column 2.
  • The final sed matches line 1 (number whose largest prime factor is the smallest in the list), removes everything including and after the first space, then quits. sed automatically prints the result of this substitution before quitting.

Output:

$ ./smooth.sh 9 15
12
$ ./smooth.sh 9 16
16
$ ./smooth.sh 157 249
162
$ ./smooth.sh 2001 2014
2002
$ 

Note ranges in this context are inclusive of both endpoints.

Digital Trauma

Posted 2014-08-18T23:30:24.447

Reputation: 64 644

1seq $@ is 3 bytes shorter, if you can assume that there are only two arguments. – nyuszika7h – 2014-08-21T18:36:56.900

@nyuszika7h Nice idea - thanks! – Digital Trauma – 2014-08-21T18:48:13.540

8

Lua - 166 chars

I don'tdidn't have (yet!) enough reputation to comment on AndoDaan's solution, but here are some improvements on his code

a,b=io.read("*n","*n")s=b for i=a,b do f={}n=i d=2 while n>1 do while n%d<1 do f[#f+1]=d n=n/d end d=d+1 end p=math.max(unpack(f))if p<s then s=p c=i end end print(c)

Changes :

  • The n%d==0 by n%d<1 which is equivalent in this case
  • Removed a space
  • Replaced table.insert(f,d) by f[#f+1]=d (#f is the number of elements of f)

Adriweb

Posted 2014-08-18T23:30:24.447

Reputation: 333

Ah, glad I glanced here. Ah, the first two i should have checked and caught, but your third improvement is new to me (I mean just different than what I'm used to). That's going to help me a lot here and over at golf.shinh.com. Thanks! – AndoDaan – 2014-08-19T21:50:12.503

5

Python 2, 67

f=lambda R,F=1,i=2:[n for n in range(*R)if F**n%n<1]or f(R,F*i,i+1)

Thinking about another golf gave me an idea for a new algorithm to check smoothness, hence the late answer.

The prime factorization of the factorial i! includes exactly the primes at most i. So, if n is a product of distinct primes, its smoothness (largest prime factor) is the smallest i for which n is a divisor of i!. To account for repeated prime factors in n, we can instead use a sufficiently high power of i!. In particular, (i!)**n suffices.

The code tries increasing factorials F=i!, updated recursively. We filter for the divisors of F in the input range, and output them if there are any, and otherwise move on to (i+1)!.

Test case:

>> f([157, 249])
[162, 192, 216, 243]

xnor

Posted 2014-08-18T23:30:24.447

Reputation: 115 687

4

C,  149   95

Edited answer:

I cannot claim credit for this solution. This updated answer borrows the beautiful method used by isaacg in his Python solution. I wanted to see if it was possible to write it in C as a nested for/while loop with no curly braces, and it is!

R(a,b,n,q,p,m){for(;a<b;m=p<q?a:m,q=p<q?p:q,n=++a,p=2)while(n>1)if(n%p)p++;else n/=p;return m;}

Explanation:

  • Function R(a,b,n,q,p,m) scans the range a to b-1 and returns the first smoothest number found. Invocation requires adherence to the following form: R(a,b,a,b,2,0) so that variables inside the function are effectively initialized as follows: n=a;q=b;p=2;m=0;.

Original answer:

This was my original answer...

P(n,f,p){for(;++f<n;)p=p&&n%f;return p;}
G(n,f){for(;--f>1;)if(n%f==0&&P(f,1,1))return f;}
R(a,b,p,n){for(;++p;)for(n=a;n<b;n++)if(G(n,n)==p)return n;}

Explanation:

  • Function P(n,f,p) tests value n for primality and returns true (nonzero) if n is prime or false (zero) if n is non-prime. f and p must both be passed as 1.
  • Function G(n,f) returns the greatest prime factor of n. f must be passed as n.
  • Function R(a,b,p,n) scans the range a to b-1 and returns the first smoothest number found. p must be passed as 1. n can be any value.

Test driver:

test(a,b){printf("smooth_range(%d, %d)\n%d\n",a,b,S(a,b,1,0));}
main(){test(5,11);test(9,16);test(9,17);test(157,249);test(2001,2014);}

Output:

smooth_range(5, 11)
8
smooth_range(9, 16)
9
smooth_range(9, 17)
16
smooth_range(157, 249)
162
smooth_range(2001, 2014)
2002

Todd Lehman

Posted 2014-08-18T23:30:24.447

Reputation: 1 723

I would argue this falls foul of "No encoding extra information in the input" clause. – Alchymist – 2014-08-21T23:02:06.037

@Alchymist — You may be right...but I don't think there's any actual extra information in the pseudo-arguments. At least not any information that is any kind of clue as to the answer. – Todd Lehman – 2014-08-21T23:29:22.400

4

Haskell - 120

import Data.List
import Data.Ord
x!y=(minimumBy(comparing(%2)))[x..y]
x%y|x<y=y|x`mod`y==0=(x`div`y)%y|otherwise=x%(y+1)

Example usage:

> 5 ! 10
8
> 9 ! 15
9
> 9 ! 16
16
> 157 ! 248
162
> 2001 ! 2013
2002

Taylor Fausak

Posted 2014-08-18T23:30:24.447

Reputation: 761

1Couldn't you use <1 instead of ==0? – dfeuer – 2019-03-05T05:44:45.510

Yeah, that would be a nice improvement. There's lots of little things that could be done better. Fortunately this answer already does all of them: https://codegolf.stackexchange.com/a/36461

– Taylor Fausak – 2019-03-12T19:41:28.573

4

Q, 91 characters K, 78 characters

{(x+{where x=min x}{(-2#{x div 2+(where 0=x mod 2_til x)@0}\[{x>0};x])@0}'[(x)_til y+1])@0}

k would probably shave a dozen characters

edit: indeed, treating the upper bound as non inclusive this time

{*:x+{&:x=min x}{*:-2#{6h$x%2+*:&:x={y*6h$x%y}[x]'[2_!x]}\[{x>0};x]}'[(x)_!y]}

Arthur B

Posted 2014-08-18T23:30:24.447

Reputation: 310

4

Note: This answer is not allowable.

This answer uses multiple features of Pyth added after the challenge was asked.

I added another new feature, calling unary range on a 2 element tuple, which shortens the solution by two characters, to this:

Pyth, 7

hoePNUQ

Input is now taken comma separated. The rest is the same.


This answer uses a feature of Pyth that was added after this question was asked, specifically after seeing @aditsu's wonderful CJam solution. That being said, I wanted to demonstrate what adding that feature has made possible. The feature is P, which is an arity-1 function which on integer input returns a list of all prime factors of the input, sorted smallest to largest.

Pyth, 9

hoePNrQvw

Uses Python-style ranges, newline separated on STDIN. Outputs smallest solution to STDOUT.

Explanation:

      Q = eval(input())                         Implicit, because Q is present.
h     head(                                     First element of
 o         order_by(                            Sort, using lambda expression as key.
                    lambda N:                   Implicit in o
  e                          end(               Last element of
   PN                            pfact(N)),     List containing all prime factors of N.
  r                 range(                      Python-style range, lower inc, upper exc.
   Q                      Q,                    A variable, initialized as shown above.
   vw                     eval(input()))))      The second entry of the range, same way.

Tests:

$ newline='
'

$ echo "9${newline}16" | ./pyth.py -c 'hoePNrQvw'
9

$ echo "9${newline}17" | ./pyth.py -c 'hoePNrQvw'
16

$ echo "157${newline}249" | ./pyth.py -c 'hoePNrQvw'
162

$ echo "2001${newline}2014" | ./pyth.py -c 'hoePNrQvw'
2002

isaacg

Posted 2014-08-18T23:30:24.447

Reputation: 39 268

@MartinBüttner Yep, as suggested by his comment on the CJam solution

– Adriweb – 2014-08-21T12:22:02.993

@MartinBüttner Yeah, P, is the new feature. I'll put that in the answer. – isaacg – 2014-08-21T12:32:19.120

1Allowable or not, not only I like it, but I also think that those short "macros" are readable if you pay attention - they convert to straightforward Python, after all. Something has to be said for a golf language that is good for golf but not necessarily obfuscating. – Reinstate Monica – 2014-08-21T14:52:36.350

@KubaOber Thanks, Kuba. That's always been my intention in writing Pyth, to make it as golfed and as readable as possible. I'm glad it's working. – isaacg – 2014-08-21T20:47:51.847

3

Lua - 176 characters

a,b=io.read("*n","*n")s=b for i=a,b do f={}n=i d=2 while n>1 do while n%d==0 do table.insert(f, d)n=n/d end d=d+1 end p=math.max(unpack(f))if p<s then s=p c=i end end print(c)

I really should stop golfing in Lua. There's no point.

AndoDaan

Posted 2014-08-18T23:30:24.447

Reputation: 2 232

14IMHO, code golfing is like boxing: there are weight-classes. A given language may not win outright, but it is fun and illuminating to golf within that class/language. – Michael Easter – 2014-08-19T01:26:20.210

3

Clojure - 173 170 chars

I'm a Clojure newbie. Golfed:

(defn g[x,d](if(and(= 0(mod x d))(.isProbablePrime(biginteger d) 1))d 0))(defn f[i](apply max-key(partial g i)(range 2(inc i))))(defn s[a,b](first(sort-by f(range a b))))

Sample runs:

Ranges include low-end, exclude high-end: [a,b) Only prints one of the smoothest numbers, if multiple occur.

(println (s 5 11))
(println (s 9 16))
(println (s 9 17))
(println (s 157, 249))
(println (s 2001, 2014))

yields:

bash$ java -jar clojure-1.6.0.jar range.clj
8
9
16
192
2002

Ungolfed:

(defn g [x,d] (if (and (= 0(mod x d)) (.isProbablePrime (biginteger d) 1)) d 0))
(defn f [i] (apply max-key (partial g i) (range 2 (inc i))))
(defn s [a,b] (first (sort-by f (range a b))))

Michael Easter

Posted 2014-08-18T23:30:24.447

Reputation: 585

1A range that includes the low end and excludes the high end is usually written [a,b). – murgatroid99 – 2014-08-19T17:12:21.833

yep, thanks for the note – Michael Easter – 2014-08-19T17:23:54.460

3

C# LINQ: 317 303 289 262

using System.Linq;class P{static void Main(string[]a){System.Console.Write(Enumerable.Range(int.Parse(a[0]),int.Parse(a[1])).Select(i=>new{i,F=F(i)}).Aggregate((i,j)=>i.F<j.F?i:j).i);}static int F(int a){int b=1;for(;a>1;)if(a%++b<1)while(a%b<1)a/=b;return b;}}

Ungolfed:

using System.Linq;

class P
{
  static void Main(string[]a)
  {
    System.Console.Write(
      Enumerable.Range(int.Parse(a[0]), int.Parse(a[1])) //create an enumerable of numbers containing our range (start, length)
        .Select(i => new { i, F = F(i) }) //make a sort of key value pair, with the key (i) being the number in question and the value (F) being the lowest prime factor
        .Aggregate((i, j) => i.F < j.F ? i : j).i); //somehow sort the array, I'm still not entirely sure how this works
  }
  static int F(int a)
  {
    int b=1;
    for(;a>1;)
      if(a%++b<1)
        while(a%b<1)
          a/=b;
    return b;
  }
}

It takes in the start and the length from the command line and will return the largest smooth number.

I used answers from here and here to make my answer.

Thanks to VisualMelon for tweaking it and shaving 12 bytes off! I also got rid of the braces in the if saving 2 bytes, and CodeInChaos pointed out some obvious stuff I missed (thanks again).

ldam

Posted 2014-08-18T23:30:24.447

Reputation: 190

Couple of general purpose small things, you can save 4 bytes in F by defining int b next to m. In a couple of places you perform the comparison a%b==0, and a and b are always positive you can cut a byte for each by checking if it's less than 1 a%b<1. You can also save a byte by incrementing b in the if's condition a%++b<0 rather than in the for by initializing it to 1. I also think in this case it's cheaper to just fully qualify System.Console.WriteLine and avoid the namespace clause. – VisualMelon – 2014-08-20T18:57:43.210

@VisualMelon Thanks, updated with your ideas :) – ldam – 2014-08-21T06:49:33.037

The m=...:m; thingy falls outside the while loop. Therefore, you can drop the m=0, and replace return m; with return m=b>m?b:m;. Then, you can drop the m=...:m; entirely. – tomsmeding – 2014-08-21T11:54:38.693

It may sound weird, but this is - to me- less redable than CJam and J. I guess C# was designed to be verbose, and attempts at making it less so make it unreadable? Hmm.... – Reinstate Monica – 2014-08-21T14:49:35.293

No I agree, LINQ looks like a demon when you just see it here and there and never actually play with it yourself. Once you grasp it though, it's really cool :) With that said, I still don't fully understand how Aggregate works, I just tried it after seeing it in another answer to get to my new object instead of just one field within it, and it just happened to work perfectly :) – ldam – 2014-08-21T15:08:59.430

>

  • OP clarified that -int.Parse(a[0]) is not required -16 2) Why namespace instead of using? -5 3) WriteLine to Write-4 4) TurnFinto an instance method -1 5) replaceI = ibyi` -2
  • < – CodesInChaos – 2014-08-21T17:55:26.887

    >

  • Didn't see that, thanks 2) Good point. 3) will change 4) How would an instance method help? 5) will change.
  • < – ldam – 2014-08-22T11:06:56.600

    3

    Ruby, 65 62

    require'prime'
    s=->a,b{(a..b).min_by{|x|x.prime_division[-1]}}
    

    With apologies to https://codegolf.stackexchange.com/a/36484/6828, this is the golfed (and slightly simplified) version of that. Uses an inclusive range since it's a character shorter.

    1.9.3-p327 :004 > s[157,249]
     => 192 
    1.9.3-p327 :005 > s[5,11]
     => 8 
    1.9.3-p327 :006 > s[9,15]
     => 12 
    1.9.3-p327 :007 > s[9,16]
     => 16 
    

    And thanks to YenTheFirst for saving three characters.

    histocrat

    Posted 2014-08-18T23:30:24.447

    Reputation: 20 600

    1You can actually get away without the [0], since array comparison will prioritize the first element anyway. This will give different, but still correct, results. – YenTheFirst – 2014-08-20T22:39:56.900

    2

    Pyth, 7 bytes

    .mePbrF
    

    Try it here!

    Outputs all smooth integers in the range \$[a, b)\$. For the smooth integers in \$[a, b]\$, just use } in place of r.

    .mePbrF – Full program with arguments a and b.
         rF – Fold by half-inclusive range. Yields the integers in [a, b).
    .m      – Values b in that list which give minimal results when applied f.
      ePb   – function / block f. 
       Pb   – Prime factors of b.
      e     – Last element. This is guaranteed to yield the largest, as they're sorted.
    

    Mr. Xcoder

    Posted 2014-08-18T23:30:24.447

    Reputation: 39 774

    2

    R, 83

    library(gmp)
    n=a:b
    n[which.min(lapply(lapply(lapply(n,factorize),max),as.numeric))]
    

    where the bottom of the input range is assigned to a and the top (inclusive) is assigned to b.

    gmp is a package that is available on CRAN. It felt dirty until I saw that absurd mf function in CJam. Install by typing install.packages("gmp") in the console.

    shadowtalker

    Posted 2014-08-18T23:30:24.447

    Reputation: 461

    1If you're using lapply 3 times you might want to alias it (i. e. l=lapply and then use l(...). Similarly since factorize is the only function you use from package gmp you can use gmp::factorize instead of loading the library and then using factorize. Your code thus would become l=lapply;n=a:b;n[which.min(l(l(l(n,gmp::factorize),max),as.numeric))] which is 69 bytes. – plannapus – 2016-05-10T07:07:19.620

    2

    PowerShell - 85

    ($args[0]..$args[1]|sort{$d=2
    while($_-gt1){while(!($_%$d)){$m=$d;$_/=$d}$d++}$m})[0]
    

    This will sort a range of numbers (inclusive) based on each number's max prime factor. It returns the lowest sorted element.

    > smooth 5 10
    8
    > smooth 9 15
    12
    > smooth 9 16
    16
    > smooth 157 248
    243
    > smooth 2001 2013
    2002
    

    Rynant

    Posted 2014-08-18T23:30:24.447

    Reputation: 2 353

    2

    J - 16 char

    Using the (start, length) style of range, as allowed by the comments.

    (0{+/:{:@q:@+)i.
    

    To be used as a dyadic verb: left argument is start, right is length.

       5 (+)i. 6              NB. range
    5 6 7 8 9 10
       5 (q:@+)i. 6           NB. prime factorizations
    5 0 0
    2 3 0
    7 0 0
    2 2 2
    3 3 0
    2 5 0
       5 ({:@q:@+)i. 6        NB. largest prime factors
    5 3 7 2 3 5
       5 (+/:{:@q:@+)i. 6     NB. sort range by smallest factors
    8 6 9 5 10 7
       5 (0{+/:{:@q:@+)i. 6   NB. take first entry
    8
       f=:(0{+/:{:@q:@+)i.    NB. can also be named
       2001 f 13
    2002
    

    A (start, end) solution is +2 chars, and excludes the end; including the end is +2 more. But on the bright side, it looks rather nice since we match up all the {braces}.

    (0{}./:{:@q:@}.)i.    NB. excluding
    (0{}./:{:@q:@}.)1+i.  NB. including
    

    algorithmshark

    Posted 2014-08-18T23:30:24.447

    Reputation: 8 144

    2

    Seriously, 8*14/7 = 16 (non-competitive)

    ,x;`yM`M;m@í@E
    

    Seriously was created after this challenge, but I wanted to post this answer because it exemplifies the type of challenges Seriously is good at.

    Try it online!

    Explanation:

    ,x;`yM`M;m@í@E
    ,x;             make two copies of range(a,b) (a,b = input())
       `  `M;       make two copies of the result of the map:
        yM            push maximum prime factor
             m@í    push index of minimum element from prime factors
                @E  push element from range with given index
    

    Mego

    Posted 2014-08-18T23:30:24.447

    Reputation: 32 998

    1

    Jelly, 7 bytes, score = 7÷7×8 = 8, language postdates challenge

    rÆfṀ$ÐṂ
    

    Try it online!

    Takes the lower and upper range endpoints as two separate arguments. Outputs a list of all the smoothest numbers in the range. (This can be viewed as a function, in which case the output is a Jelly list, or as a full program, in which case the output happens to use the same list representation that JSON does.)

    Explanation

    Those times when your Jelly program is just a literal translation of the spec…

    rÆfṀ$ÐṂ
    r        Range from {first argument} to {second argument}
         ÐṂ  Return the elements which have the minimum
       Ṁ$      largest
     Æf          prime factor
    

    user62131

    Posted 2014-08-18T23:30:24.447

    Reputation:

    1

    Whispers v2, 183 bytes

    > Input
    > Input
    >> 1…2
    >> ∤L
    >> L’
    >> Each 4 3
    >> Select∧ 5 L
    >> Each 7 6
    >> Lⁿ10
    > -1
    >> Each 9 8
    >> L‖R
    >> Each 12 11 3
    >> 13ᴺ
    >> Each 9 14
    > 0
    >> 15ⁿ16
    >> Output 17
    

    Try it online!

    Aside from revamping the algorithm, the only real golfing opportunity can be found on lines 9 and 10, but swapping them. When they're the other way round, the code is 1 byte longer, due to the repeated use of 9 or 10.

    The structure tree for this program is:

    tree

    Red represents nilad lines, green represents function arguments. Arrows from \$a \to b\$ show the use of line \$a\$ as an argument for line \$b\$.

    How it works.

    We start at lines 1, 2 and 3. The first two simply take and store the inputs \$\alpha\$ and \$\beta\$ on lines 1 and 2 respectively. There two values are then passed to line 3, 1…2, which form the range

    $$R := [\alpha, \alpha+1, ..., \beta-1, \beta]$$

    As you can see from the tree above, \$R\$ is used in two places: line 6 and line 13. As line 6 eventually leads to line 13, we'll take the path down 6.

    Line 6 is an Each statement, that maps a function (the first line reference) over an array (the second line reference). Here, the function is defined on line 4 as \$f(x) = \{i \: | \: (i | x), (i \le x), i \in \mathbb{N}\}\$ i.e. the array of \$x\$'s divisors. The array is \$R\$, so line 6 iterates over \$R\$, returning the divisors of each integer, forming a new array

    $$D := [f(\alpha), f(\alpha+1), ..., f(\beta-1), f(\beta)]$$

    We then get more complicated as we skip down to line 8. Line 8 is also an Each statement, but its function argument, line 7 is split over two lines, 7 and 5:

    >> L’
    >> Select∧ 5 L
    

    Mathematically, this is the function \$g(A) = \{i \: | \: (i \in \mathbb{P}), (i \in A)\}\$, that takes a set \$A\$ and returns all the primes in \$A\$. Line 8 is iterating over \$D\$, essentially filtering all composite numbers from each subarray in \$D\$, to form

    $$A := [g(f(\alpha)), g(f(\alpha+1)), ..., g(f(\beta-1)), g(f(\beta))]$$

    Then, we encounter yet another Each statement on line 11, which iterates line 9 over \$A\$. Line 9 takes an array and returns the final element of that array. As \$g(A)\$ returns the array sorted, this is equivalent to return the largest value in that array. At this point, we remember that \$g(f(x))\$ returns the list of prime factors of \$x\$, so line 9 maps each \$x\$ to its largest prime factor, returning

    $$B := [\max(g(f(\alpha))), \max(g(f(\alpha+1))), ..., \max(g(f(\beta-1))), \max(g(f(\beta)))]$$

    Once we have \$B\$, we now have an array we can sort \$R\$ by, e.g. we sort each value \$x\$ in \$R\$ by the resulting value of \$\max(g(f(x)))\$. However, we do not have a sort-by command in Whispers, only the simple Python sort(list). Luckily, due to how sort sorts lists of lists, we can concatenate each \$x\$ with \$\max(g(f(x))\$ and sort by the first element, what Python does naturally.

    Lines 12 and 13 perform this concatenation, by iterating a concatenate function (line 12) over \$B\$ and \$R\$. This forms an array of pairs of numbers, in the form \$[\max(g(f(x))), x]\$. Next, we sort this array on line 14, which, due to the way Python compares lists, places the numbers with the lowest largest prime factor towards the end.

    Almost finished. The only things left to do are to extract \$R\$, sorted by \$B\$ (we'll call this array \$R_B\$. This is done simply by taking the last element of each subarray (line 9 already does this for us, so we simply need Each 9 14 on line 15, rather than defining a new function). Once that's done, we simply take the first element in \$R_B\$, which will be (one of) the number with the lowest largest prime factor i.e. the smoothest number.

    caird coinheringaahing

    Posted 2014-08-18T23:30:24.447

    Reputation: 13 702

    1

    Brachylog, 5 bytes

    ⟦₃ḋᵒh
    

    Try it online!

    Takes input as a list of two (or more, actually) values in no particular order, and outputs a single value through the output variable of the predicate (if you try to run it as a full program, it'll just print true. at you, unless you stick a w on the end). The input is interpreted as a Python-style range because to feed multiple values I need to give it a subscript no matter whether it's , , or so I may as well choose the one that matches the test cases.

        h The first item of
    ⟦₃    the range represented by the input
       ᵒ  after it's been sorted by
      ḋ   prime factorization.
    

    Saves three bytes over the more obvious ⟦₃{ḋh}ᵒh by taking advantage of that Brachylog sorts its lists by elements first rather than lengths, such that the first step of sorting by prime factorizations is sorting by largest prime factors.

    Unrelated String

    Posted 2014-08-18T23:30:24.447

    Reputation: 5 300

    1

    Cobra - 150

    def f(r as vari int)
        x,y=r
        c,o=y,0
        for n in x:y,for m in n:0:-1
            p=1
            for l in 2:m,if m%l<1,p=0
            if n%m<=0<p
                if m<c,c,o=m,n
                break
        print o
    

    Not even sure why I bothered, cobra just can't compete here.

    Οurous

    Posted 2014-08-18T23:30:24.447

    Reputation: 7 916

    1Cobra looks identical to python... What are the differences? – Beta Decay – 2014-08-20T09:29:31.513

    @BetaDecay Cobra is what happens when you give C# the syntax of Python. The Cobra Website

    – Οurous – 2014-08-20T11:35:46.740

    1

    Ruby - 113 chars

    Using the stdlib. Returns one result. Tested on ruby 2.1.2.

    require 'prime'
    def smooth_range(a,b)
      (a...b).sort_by{|e|e.prime_division.flat_map{|f,p|[f]*p}.uniq.max}[0]
    end
    

    John P. Terry

    Posted 2014-08-18T23:30:24.447

    Reputation: 11

    1

    Welcome to Programing Puzzles and Code Golf Stack Exchange. Thanks for posting your result. Since this is a code-golf question, please include your character count in your answer. You can use a tool such as this one: http://www.javascriptkit.com/script/script2/charcount.shtml

    – isaacg – 2014-08-20T00:34:18.827

    1

    Perl (5.10+), 83

    for(<>..<>){$n=$_;$p=2;$_%$p&&$p++or$_/=$p while$_>1;$m=$p,$r=$n if$p<$m||!$m}
    say$r
    

    (linebreak can be removed). Takes two endpoints of an inclusive range on two lines of stdin (because <> is cheaper than accessing ARGV) and outputs the smoothest to stdout. If there's a tie for smoothest, prints the smallest. Could print the biggest at the cost of one character.

    The algorithm is basically isaacg's way of finding the greatest prime factor, although we came up with it independently. That part golfs down beautifully to a single statement in perl, the rest has more overhead than I'd like though.

    Should be run under perl -E or with a use 5.012 preamble. If you can't do that, replace say$r with print$r,$/.

    hobbs

    Posted 2014-08-18T23:30:24.447

    Reputation: 2 403

    1

    Python 2 (84)

    f=lambda n,p=2:n>1and f(n/p**(n%p<1),p+(n%p>0))or p
    print min(range(*input()),key=f)
    

    @isaacg's solution, but with a min by function key in place of explicit min-finding, and a recursive function playing the role of the iteration.

    Run in Stackless Python to avoid recursion limits.

    It looks wasteful to use the paranthesized condition (n%p<1), then repeat its negation also in parantheses (n%p>0), but that was the best I got. I tried things a bunch of things, but they turned out worse.

    f(n/p**(n%p<1),p+(n%p>0))     # Current for comparison
    f(*[n/p,n,p,p+1][n%p>0::2])
    n%p and f(n,p+1)or f(n/p,p)
    f(*n%p and[n,p+1]or[n/p,p])
    

    I welcome any improvements you can think of.

    xnor

    Posted 2014-08-18T23:30:24.447

    Reputation: 115 687

    1

    Java 8 - 422 454 chars

    I'm learning Java 8, and wanted to give this a shot relative to Java (or even Java 8 streams).

    Compared to other languages, this is brutal but an interesting exercise.

    Golfed:

    import java.util.stream.*;import java.math.*;
    class F{int v;int i;public int getV() { return v; }
    F(int i){this.i = i;v=IntStream.range(2,i+1).map(j->((i%j==0)&&new BigInteger(""+j).isProbablePrime(1))?j:0).max().getAsInt();}}
    public class T{
    int s(int a, int b){return IntStream.range(a,b+1).boxed().map(F::new).sorted(java.util.Comparator.comparingInt(F::getV)).collect(java.util.stream.Collectors.toList()).get(0).i;}}
    

    Ungolfed:

    import java.util.stream.*;
    import java.math.*;
    
    class F {
        int v;
        int i;
        public int getV() { return v; }
        F (int i) { 
            this.i = i;
            v = IntStream.range(2,i+1)
                         .map( j -> ((i%j==0) && 
                               new BigInteger(""+j).isProbablePrime(1))?j:0)
                         .max()
                         .getAsInt();
        }
    }
    
    public class T {
        int s(int a, int b) {
            return IntStream.range(a,b+1)
                        .boxed()
                        .map(F::new)
                        .sorted(java.util.Comparator.comparingInt(F::getV))
                        .collect(java.util.stream.Collectors.toList())
                        .get(0).i;
        }
    }
    

    example run using:

    public static void main(String[] s) {
        System.out.println(new T().s(157,249));
    }
    
    192
    

    Michael Easter

    Posted 2014-08-18T23:30:24.447

    Reputation: 585

    1

    MATL (non-competitive), 20 bytes

    This language was designed after the challenge

    Range is inclusive at both ends. The numbers are taken as two separate inputs.

    2$:t[]w"@YfX>v]4#X<)
    

    Try it online!

    Explanation

    2$:          % implicitly input two numbers. Inclusive range
    t            % duplicate                      
    []           % empty array
    w            % swap elements in stack         
    "            % for each                  
      @          %   push loop variable
      Yf         %   prime factors                  
      X>         %   maximum value
      v          %   vertical concatenation         
    ]            % end for each                         
    4#X<         % arg min 
    )            % index with this arg min into initial range of numbers
    

    Luis Mendo

    Posted 2014-08-18T23:30:24.447

    Reputation: 87 464

    I guess today this would be 17 bytes &:[]y"@YfX>h]&X<) or perhaps 16 :[]y"@YfX>h]&X<). The & really was a great idea (and I'm guessing y wasn't available back then?).

    – sundar - Reinstate Monica – 2018-07-10T13:53:32.363

    And it looks like broadcast Yf with prefixed 1's would have been useful here too, but that's probably not enough to decide it's a good idea in general. :) – sundar - Reinstate Monica – 2018-07-10T13:55:28.133

    Yes, this was the very beginning, so no y or &. Credit to Suever for the very useful semantics of the latter (my initial idea was make it mean "one input more than default"). If we see more instances where Yf with adding ones would be useful, it may really be worth adding that feature. Problem is, there are about 34 answers that use Yf (according to this script), so it's hard to tell

    – Luis Mendo – 2018-07-10T15:13:55.973

    0

    Japt -g, 8 bytes

    õV ñ_k Ì
    

    Try it


    Explanation

                 :Implicit input of integers U & V
    õV           :Range [U,V]
       ñ_        :Sort by passing each through a function
         k       :  Prime factors
           Ì     :  Last element
                 :Implicitly output the first element in that array
    

    Shaggy

    Posted 2014-08-18T23:30:24.447

    Reputation: 24 623

    0

    Julia, 58 bytes

    using Primes
    !r=[r;][indmin(max(factor(Set,n)...)for n=r)]
    

    Takes a Range object of the form 5:9 as input, returns the smallest of the smoothest numbers in the range. (No TIO link, because TIO doesn't seem to have the Primes package.)

    sundar - Reinstate Monica

    Posted 2014-08-18T23:30:24.447

    Reputation: 5 296

    0

    05AB1E, 6 bytes

    Input as a list of two items, outputs just one found integer in the [A,B] range (inclusive on both sides):

    ŸΣfθ}н
    

    Try it online or verify all test cases.

    Explanation:

    Ÿ          # Create a list in the inclusive range of the (implicit) input
               #  i.e. [9,15] → [9,10,11,12,13,14,15]
     Σ  }      # Sort it by:
      f        #  Get the prime factors
               #   i.e. 9 → [3]
               #   i.e. 10 → [2,5]
       θ       #  Pop and push the last item
               #   i.e. [3] → 3
               #   i.e. [2,5] → 5
               #  i.e. [9,10,11,12,13,14,15] → [9,12,10,15,14,11,13]
         н     # After sorting, pop and push the first item
               #  i.e. [9,12,10,15,14,11,13] → 9
               # (and output the result implicitly as result)
    

    8 bytes alternative with same input-format, but outputs all found integers (as string) instead of one:

    ŸDf€θWQÏ
    

    Try it online or verify all test cases.

    Explanation:

    Ÿ          # Create a list in the inclusive range of the (implicit) input
               #  i.e. [9,15] → [9,10,11,12,13,14,15]
     D         # Duplicate this list
      f        # Get the prime factors of each
               #  i.e. [9,10,11,12,13,14,15] → [[3],[2,5],[11],[2,3],[13],[2,7],[3,5]]
       €θ      # Only leave the last value of each inner list
               #  i.e. [[3],[2,5],[11],[2,3],[13],[2,7],[3,5]] → [3,5,11,3,13,7,5]
         W     # Get the minimum of this list (without popping the list itself)
               #  i.e. [3,5,11,3,13,7,5] → 3
          Q    # Check for each value in the list if it is equal to this minimum
               #  i.e. [3,5,11,3,13,7,5] and 3 → [1,0,0,1,0,0,0]
           Ï   # Leave only the values of the duplicated list at the truthy indices
               #  i.e. [9,10,11,12,13,14,15] and [1,0,0,1,0,0,0] → ["9","12"]
               # (and output the result implicitly as result)
    

    Kevin Cruijssen

    Posted 2014-08-18T23:30:24.447

    Reputation: 67 575

    0

    Charcoal, 44 bytes

    Nθ≔…θNη≔¹ζW¬№η¹«≦⊕ζW⊙η¬﹪λζUMη⎇﹪λζλ÷λζ»I⁺θ⌕η¹
    

    Try it online! Link is to verbose version of code. Takes input as a half-open range and returns the lowest smoothest number from that range. Works by trial division of all the numbers in the range until at least one has become completely factorised (in which case it is the smoothest). Explanation:

    Nθ
    

    Input the lower end of the range.

    ≔…θNη
    

    Create an array covering the whole range.

    ≔¹ζ
    

    Set the initial factor to 1.

    W¬№η¹«
    

    Repeat until there is at least one 1 in the array.

    ≦⊕ζ
    

    Increment the factor.

    W⊙η¬﹪λζ
    

    Repeat while at least one element is divisible by the factor.

    UMη⎇﹪λζλ÷λζ»
    

    Divide all divisible elements by the factor.

    I⁺θ⌕η¹
    

    Look up the (first) index of the 1, add on the lower end of the range, and print the result as a string.

    Neil

    Posted 2014-08-18T23:30:24.447

    Reputation: 95 035

    0

    ECMAScript 2016 - 100 characters

    m=>M=>(r=_=>M-m?[m,...r(++m)]:[])().sort((x,y)=>(s=(n,p=2)=>n>p?n%p?s(n,p+1):s(n/p,p):p)(x)-s(y))[0]
    

    Definitely gets hurt by the lack of a built-in range.

    This is a pretty standard anonymous function. It takes in a minimum (inclusive) and a maximum (exclusive), and then spits out the smallest smoothest number in that range.

    M Dirr

    Posted 2014-08-18T23:30:24.447

    Reputation: 41

    0

    MathGolf, 9 bytes

    ↨áæ─g¶╙0§
    

    Try it online!

    Explanation

    Could be 1-2 bytes shorter with a "get prime factorization" operator.

    ↨           pop a, b, loop from a to b (inclusive)
     á          sort by comparator
      æ         start block of length 4
       ─        get divisors
        g       pop a, (b), pop operator from code, filter
         ¶      is_prime(n)
          ╙     maximum of two elements, max of list, maximum by filter
           0    push 0
            §   get from array
    

    maxb

    Posted 2014-08-18T23:30:24.447

    Reputation: 5 754

    0

    C#, 240 234 226 161 characters


    int S(int i,int b){int s=2<<29,n=0,k,j,p;for(k=i;i<b;k=++i){for(;k>1;--k){if(i%k<1){p=1;for(j=2;j<k;++j)if(k%j<1)p=0;if(p>0)break;}}if(k<s){s=k;n=i;}}return n;}}
    


    Acknowledgements & Reduction Explanations
    - Thanks to CodesInChaos for reductions to 226, see comments for specifics
    - Reduced to 161 after re-reading the rules and seeing that a mere function is allowed (i.e., a free-standing program is not required.)

    Usage
    function:
    - input: two integers
    - output: if multiple valid answers exist, returns the smallest


    Explanation (with comments and runnable test-wrapper)

    class Test
    {
        int S(int i, int b)
        {
            //s = lowest gpf found so far
            //n = best answer so far
            //i = range iterator
            //p = prime indicator (1=true, 0=false)
            int s = 2 << 29, n = 0, k, j, p;
            for(k=i; i<b; k=++i)
            {
                //iterate through the factors of i
                for(;k>1;--k)
                {
                    if (i % k < 1)
                    {
                        p=1;
                        //determine if k is prime
                        for (j = 2; j < k; ++j)
                            if (k % j < 1)
                                p = 0;
                        if (p>0) break;
                    }
                }
                if (k < s)
                {
                    s = k;
                    n = i;
                }
            }
            return n;
        }
    
        public static void Main(string[] args)
        {
            var d = new D();
            System.Diagnostics.Debug.Assert(d.S(5, 11) == 8);
            System.Diagnostics.Debug.Assert(d.S(9, 16) == 9);
            System.Diagnostics.Debug.Assert(d.S(9, 17) == 16);
            System.Diagnostics.Debug.Assert(d.S(157, 249) == 162);
            System.Diagnostics.Debug.Assert(d.S(2001, 2014) == 2002);
        }
    }
    

    Richard II

    Posted 2014-08-18T23:30:24.447

    Reputation: 151

    Why using System? As far as I can tell, you're only using it once, a qualified name System.Console.Write should be shorter. – CodesInChaos – 2014-08-21T15:55:33.607

    thanks @CodesInChaos. In my early iterations, I used System for 3 things, then forgot to cleanup after eliminating the other 2. Code updated. – Richard II – 2014-08-21T16:01:16.327

    1Turing p into an int should save 6 chars. – CodesInChaos – 2014-08-21T16:20:14.543

    yes @CodesInChaos, that thought also occurred to me over the weekend, and just hadn't gotten around to posting an update. – Richard II – 2014-08-25T15:36:05.640

    reduced to 161 by changing from a command-line app to a simple method (after re-reading the rules) – Richard II – 2014-08-28T22:17:27.273