Tips for Golfing in Brain-Flak

24

3

Brain-flak is a stack-based turing-tarpit language, written collaboratively between me, DJMcMayhem, and 1000000000.

Some users are very experienced in the mysterious ways of Brain-Flak. So I thought it a good idea to set up this question as a way for us, and hopefully others too, to share our knowledge with the community and lower the bar of entry to this language "designed to be painful to use". And perhaps even teach those of us with more experience some new tricks.

So what tips do you have for golfing in brain-flak? I'm looking for ideas that can be applied to code golf problems in general that are at least somewhat specific to brain-flak (e.g. "remove comments" is not an answer).

Please post one tip per answer.

Post Rock Garf Hunter

Posted 2016-10-05T17:31:09.063

Reputation: 55 382

Answers

22

Use the Third Stack

If you have read the title you might be a bit confused. Surely there are only two stacks in Brain-Flak? However I assure you that it exists and it is one of the most powerful if not the most powerful tool in writing and golfing Brain-Flak.


What is the "Third Stack"?

Every Brain-Flak program uses the third stack in one way or another but most of the use goes on behind the scenes and it is often useful to simply ignore the fact that it exists. Each parenthesis in the program either adds or removes one item from the stack. Three of the open braces ([< all add an item to the stack while their three conjugates )]> all remove an item from the stack. The value of the item on the stack is the value of the current scope of the program and using nilads will modify this value in certain ways. The close parenthesis ) has the unique function of moving an element from the Third Stack to the current stack; a push.

Hopefully this is becoming clear to you. The Third Stack is some sort of stack that remembers the return values of code that has already been executed. Let's walk through an example of a simple program keeping track of the two normal stacks and the Third Stack.

Example

We will walk through the following program. This program pushes -3, 1, -2 to the stack.

(([()()()])(()))

We start with three open braces, which all push a zero to the third stack.

Our stacks now look like this, the Third Stack is the one on the right and the active stack has a ^ under it:

        0
        0
  0  0  0
  ^

(([()()()])(()))
   ^

Now we have three () nilads. These do not do anything to the normal two stacks, however they do each add one to the top of the Third Stack making our stacks look like:

        3
        0
  0  0  0
  ^

(([()()()])(()))
         ^

Now we encounter a ] as stated before close braces remove an item from the Third Stack, but ] has the function of subtracting the element it removes from the top of the stack. Thus our new stacks will look like:

       -3
  0  0  0
  ^

(([()()()])(()))
          ^

This makes sense; [...] does negation so ] should subtract downwards.

Now we must execute a ). As you likely recall ) is the place in the program where stuff gets pushed to the stack so we will be moving the top of the Third Stack to the current stack, in addition we will be adding the -3 to the next element in the third stack.

 -3  0 -3
  ^

(([()()()])(()))
           ^

Once again we encounter one of our three open braces so we will add another element to our Third Stack.

        0
 -3  0 -3
  ^

(([()()()])(()))
            ^

As we said earlier () will increment the top of our third stack by one.

        1
 -3  0 -3
  ^

(([()()()])(()))
              ^

And ) will move the top of the Third Stack onto the active stack and add downwards

  1
 -3  0 -2
  ^

(([()()()])(()))
               ^

The last ) moves the Third Stack onto the active stack and since there are no elements left on the Third Stack for it to add, does nothing else.

 -2
  1
 -3  0
  ^

(([()()()])(()))
                ^

The program is over so we terminate and output.


This example is intended to give you a feel for what the Third Stack is and does. It does not include all of the operations, but hopefully you can figure out what each of them does on its own. If you are still struggling I have included a "cheatsheet" at the bottom of this answer to help you along.

Ok, So what?

Ok, now you understand the Third Stack, but "So what"? You were already using it even if you didn't call it the "Third Stack", how does thinking in terms of the Third Stack help you golf?

Lets look at a problem. You want to take the Triangle of a number. This is the sum of all numbers less than n.

One approach might be to create an accumulator on the offstack and add to it as you count down. This creates code that looks like this:

(<>)<>{(({}[()])()<>{})<>}{}<>({}<>)

Try it Online!

This code is pretty compact and one might think that it can't get much smaller. However if we approach it from a third stack point of view it become clear that this is grossly inefficient. Instead of putting our accumulator on the offstack we can put it on the third stack with a ( and retrieve it at the end we use ). We will once again loop through all the numbers, but this time we don't have to do much of anything to increment our Third Stack, the program does it for us. This looks like:

({()({}[()])}{})

Try it Online

This code is less than half the size of the pretty well golfed version we made before. In fact a computer search has proven that this program is the shortest possible program that can perform this task. This program can be explained using the "sum of all runs" approach, but I think it is a lot more intuitive and clear when explained using a Third Stack approach.

When do I use the Third Stack?

Ideally whenever you begin work on a new problem in Brain-Flak you should think to yourself how would I do this with the Third Stack in mind. However as a general rule of thumb whenever you have to keep track of some type of accumulator or have a running total it is a good idea to try putting that on your third stack instead of the two real stacks.

Another time that it might be a good idea to consider using your third stack is when you don't have any space to store some value on the other two stacks. This can be particularly useful when you are doing manipulations on two existing stacks and you want to save a value for later use without having to keep track of where it is.

Limitations of the Third Stack

The Third Stack is very powerful in a lot of ways but it comes with its own limitations and drawbacks.

Firstly, the maximum stack height for the Third Stack at any given point is determined at compile time. This means that if you want to use an amount of space on the stack you have to allocate that space when you are writing the program.

Secondly the Third Stack is not Random Access. This means that you cannot perform any operations on any value but the top most value. In addition you cannot move values around on the stack (say swap the first two elements).

Conclusion

The Third Stack is a powerful tool and I would consider it essential to every Brain-Flak user. It takes some getting used to and requires a shift in the way your think about programming in Brain-Flak, but when used properly it makes all the difference between a decent and an amazing when it comes to golfing.

Cheatsheet

Here is a list of operations and how they affect the Third Stack

 Operation  |                 Action
====================================================
   (,[,<    | Put a zero on top of the Third Stack
----------------------------------------------------
     )      | Add the top of the Third Stack to the
            | second element and move it to the 
            | active stack
----------------------------------------------------
     ]      | Subtract the top of the Third Stack
            | from the second element and pop it
----------------------------------------------------
     >      | Pop the top of the Third Stack
----------------------------------------------------
     ()     | Add one to the top of the Third Stack
----------------------------------------------------
     {}     | Pop the top of the active stack and
            | add it to the top of the Third Stack
----------------------------------------------------
     []     | Add the stack height to the Third
            | Stack
----------------------------------------------------
   <>,{,}   | Nothing

Post Rock Garf Hunter

Posted 2016-10-05T17:31:09.063

Reputation: 55 382

1Wow, fantastic tip! I was actually just coming here to write a similar answer when I saw this. +1! I prefer to think of it as The Accumulator, but The Third Stack metaphor is really interesting. – James – 2017-02-17T18:34:55.377

I always called it the "workspace" or "workstack". – sagiksp – 2017-03-30T05:09:11.530

On the TIO version of Brainflak, {...} returns the sum of all iterations. – CalculatorFeline – 2017-06-04T21:50:32.730

@CalculatorFeline Yes, this is true on nearly all versions of Brain-Flak except for some very early ones. However I'm not sure why you have commented that on this post in particular. – Post Rock Garf Hunter – 2017-06-04T23:10:26.513

<>,{,} | Nothing – CalculatorFeline – 2017-06-04T23:10:49.487

@CalculatorFeline Ah, I see. What I said is correct, {...} does do nothing. Since { does not create a new scope on the third stack all of its contents are added to any earlier scope. I could explain this better if I didn't have the character restriction, If you would like to talk further we can move over to the third stack.

– Post Rock Garf Hunter – 2017-06-04T23:22:26.767

Also, the Third Stack interpretation lowers the requirement of everything matching to only {} matching. Also, (<[ are all the same, except for close bracket type. – CalculatorFeline – 2017-06-05T04:31:25.340

@DJMcMayhem Accumulator usually implies one integer though... – Erik the Outgolfer – 2017-06-23T08:47:13.967

21

Finding modulus/remainder

Finding n modulo m is one of the basic arithmetic operations, important for many challenges. For cases m > 0 and n >= 0, the following 46-byte snippet may be used. It assumes that n is on top of the active stack with m the next one down, and replaces them with n mod m. It leaves the rest of the stacks intact.

({}(<>))<>{(({})){({}[()])<>}{}}{}<>([{}()]{})

This annotated version shows the stack contents at some points in the program. ; separates the two stacks and . marks the active stack.

. n m;
({}(<>))<>
{   . m; r 0   (r = n - km)
    (({}))
    . m m; r 0
    {({}[()])<>}
    {}
}
m-n%m-1 m; . 0
{}<>([{}()]{})
. n%m;

feersum

Posted 2016-10-05T17:31:09.063

Reputation: 29 566

It took me a while to understand what the unannotated part did ({({}[()])<>}), but once I figured it out... Genius :-) – ETHproductions – 2017-05-15T22:01:38.873

11

Push-Pop Redundancy

This is a big one. It is also a little bit of a nuanced one.

The idea is that if you push something and then pop it without doing anything you should not have pushed it at all.

For example, if you want to move something to the offstack and then add one to it you might write the following code:

({}<>)({}())

This can be simpler like this:

({}<>())

The first program picks up the item moves it, picks it up again and adds one, while the second one does both in one fell swoop.

That example was simple however it can get a lot more complex. Take for instance:

({}<({}<>)><>)(<((()()()){}[((){}{})])>)

The reduction is less clear here but the 4th pop in the program can be reduced with the 2nd push, like so:

(<((()()()){}[((){}<({}<>)><>{})])>)

This is the most powerful tool in my golfing repertoire but it takes some practice to get good at it. Once you have been doing these for a while you will be able to spot these almost instantly.

Post Rock Garf Hunter

Posted 2016-10-05T17:31:09.063

Reputation: 55 382

In the latter example, isn't the part in the first pair of parentheses equivalent to ({}<{}>)? – feersum – 2016-11-03T09:41:24.630

@feersum No it is not. It moves a copy of the second item on the stack to the off stack while ({}<{}>) destroys the item entirely. However these programs were not really meant to be optimal just to highlight the principle at work here. – Post Rock Garf Hunter – 2016-11-28T04:46:23.073

6

Optimise your integers

Integers are tedious to represent in Brain-Flak. Fortunately we have a question that will help Golf a Brain-Flak Integer for you. (Note that the question is designed to push the integer to the stack, so push-pop redundancy probably applies to more realistic programs.)

Neil

Posted 2016-10-05T17:31:09.063

Reputation: 95 035

Note that we also have brain-flak.github.io/integer/, which runs one of those algorithms online for convenience.

– James – 2016-11-28T03:29:44.847

@DrMcMoylex All we need now as in implementation of the integer metagolfer in Brain-Flak itself! – Neil – 2016-11-28T09:37:23.850

1

We have that. http://codegolf.stackexchange.com/a/98054/31716 :D

– James – 2016-11-28T16:31:37.073

5

Push extra loop counters

Frequently, you'll want to do something like

Perform X operation on every element in the stack

or

Perform X operation on every pair of adjacent elements in the stack

When the input may contain '0's, or the result of operation X may give a 0, this is really inconvenient. Because you'll need to do:

([])
{

  {}

  ({}...<>)
  ([])

}

To do X to each element, and then later

<>
([])
{
  {}
  ({}<>)
  <>
  ([])
}
<>

To reverse the array again.

Even worse, if we're operatiing on pairs of adjacent elements, we'll need to do ([][()]) in place of ([]). This is really inconvenient. Here's the trick: While you're doing X to each element, push a 1 over onto the alternate stack right above the X(element). Then, while you're reversing it, you can simply do

<>
{
  {}
  ({}<>)
  <>
}
<>

This is 8 bytes shorter, so when you factor in the extra code to push 1, you'll end up saving 4-6 bytes. For a more concrete example, compare the task of getting deltas of an array. Without this trick, you'd need:

([][()]){

    {}

    ([{}]({})<>)<>

    ([][()])

}{}{}<>

([])
{{}({}<>)<>([])}<>

For 62. With this trick, you'll have

([][()]){

    {}

    ([{}]({})<>)(())<>

    ([][()])

}{}{}<>

{{}({}<>)<>}<>

For 58. If used the right way (for example, reversing first, and removing two ([][()]) from later), this could save even more in speficic scenarios.

James

Posted 2016-10-05T17:31:09.063

Reputation: 54 537

3

Take advantage of the 'Stack-Height' nilad

Especially in challenges, or challenges where you always know the size of the input, you can take advantage of the 'Stack-Height' nilad [] to create integers.

Let's work through this with a hypothetical challenge: output CAT. The non-golfy way is to use the online integer golfer to push 67, 65, and 84. This gives:

(((((()()()()){}){}){}()){}())

(((((()()()()){}){}){}){}())

((((((()()()){}()){}){})){}{})

(Newlines for clarity). This is 88 bytes, and not that great. If we instead push the consecutive differences between values, we can save a lot. So we wrap the first number in a push call, and subtract 2:

(   (((((()()()()){}){}){}()){}())  [()()] )

Then, we take this code, wrap it in a push call, and add 19 to the end:

(  ((((((()()()()){}){}){}()){}())[()()])   ((((()()()){})){}{}()) )

This is 62 bytes, for a whopping 26 byte golf!

Now here is where we get to take advantage of the stack-height nilad. By the time we start pushing 19, we know that there are already 2 items on the stack, so [] will evaluate to 2. We can use this to create a 19 in fewer bytes. The obvious way is to change the inner ()()() to ()[]. This only saves two bytes though. With some more tinkering, it turns out we can push 19 with

((([][]){})[]{})

This saves us 6 bytes. Now we are down to 56.

You can see this tip being used very effectively on these answers:

James

Posted 2016-10-05T17:31:09.063

Reputation: 54 537

Your CAT program actually pushes TAC :P – Post Rock Garf Hunter – 2017-04-03T19:38:41.993

Also, 52 bytes

– Post Rock Garf Hunter – 2017-04-03T19:42:51.813

2

I know this is a tip but I can't help myself, 50 bytes.

– Post Rock Garf Hunter – 2017-04-03T19:51:30.280

1

Another weird but sometimes effective tip to help abuse []: prepending 0s with (<>)s before your code. This only really works if you're planning to push the code in reverse anyway, but if you come across the right number you can reduce your code quite a bit. A rather extreme example where I append 6 0s, which lets me use just as many []s as I use ()

– Jo King – 2018-04-06T14:24:47.570

2

Better looping

Here's an easy one:

A fairly common construct is:

(({})<{({}[()]<...>)}{}>)

Where you want to loop n times but still keep the n. However this can be written as:

({<({}[()]<...>)>()}{})

to save 2 bytes.

Another fairly common paradigm is

([])({<{}>...<([])>}{})

Which will loop and accumulate the entire stack. Due to some fancy maths this is the same as:

(([]){[{}]...([])}{})

Post Rock Garf Hunter

Posted 2016-10-05T17:31:09.063

Reputation: 55 382

2

Use the wiki

We have a wiki! It has some short comings, but it is helpful resource. It has lists of useful, well golfed, stack clean programs that you can paste into your code. If you want to do something you think someone might have done before there is a good chance it is on the wiki.

Post Rock Garf Hunter

Posted 2016-10-05T17:31:09.063

Reputation: 55 382

1

Check your negatives

Sometimes you can golf a few bytes by strategically choosing what to negate with the [...] monad.

A simple example is in nested [...]s. For example [()()()[()()]] could just be [()()()]()()

Say you want to check whether a value is any of the start brackets (<{[. The initial attempt is to push the difference between each character and loop over subtracting it

({}<(((((       #Push the differences between start bracket characters
(((()()()()){}){}){})   #Push 32
[()])                   #Push 31
[((()()())()){}{}])     #Push 20
){})><>)                #Push 40
<>({<(([{}]<>{}))>(){[()](<{}>)}<>})

However, you can save 4bytes by pushing the negative versions of the differences instead!

({}<(((((       #Push the differences between start bracket characters
((([()()()()]){}){}){}) #Push -32
())                     #Push -31
((()()())()){}{})       #Push -20
){})><>)                #Push -40
<>({<(({}<>{}))>(){[()](<{}>)}<>})

Generally, this doesn't save you very much, but it also costs very little effort to change around what the [...] is surrounding. Be on the lookout for situations where you can push the negative of a counter instead of the positive to save on incrementing multiple times instead of decrementing later. Or swapping out (a[b]) with ([a]b) to make the difference between two numbers negative instead of positive.

Jo King

Posted 2016-10-05T17:31:09.063

Reputation: 38 234

1Similar things can be done with zeros <a<b>c> -> <abc> and <a[b]c> -> <abc>. – Post Rock Garf Hunter – 2018-04-06T15:12:48.967