package fj; import static fj.P.weakMemo; /** * A product-4. * * @version %build.number% */ public abstract class P4<A, B, C, D> { /** * Access the first element of the product. * * @return The first element of the product. */ public abstract A _1(); /** * Access the second element of the product. * * @return The second element of the product. */ public abstract B _2(); /** * Access the third element of the product. * * @return The third element of the product. */ public abstract C _3(); /** * Access the fourth element of the product. * * @return The fourth element of the product. */ public abstract D _4(); /** * Map the first element of the product. * * @param f The function to map with. * @return A product with the given function applied. */ public final <X> P4<X, B, C, D> map1(final F<A, X> f) { return new P4<X, B, C, D>() { public X _1() { return f.f(P4.this._1()); } public B _2() { return P4.this._2(); } public C _3() { return P4.this._3(); } public D _4() { return P4.this._4(); } }; } /** * Map the second element of the product. * * @param f The function to map with. * @return A product with the given function applied. */ public final <X> P4<A, X, C, D> map2(final F<B, X> f) { return new P4<A, X, C, D>() { public A _1() { return P4.this._1(); } public X _2() { return f.f(P4.this._2()); } public C _3() { return P4.this._3(); } public D _4() { return P4.this._4(); } }; } /** * Map the third element of the product. * * @param f The function to map with. * @return A product with the given function applied. */ public final <X> P4<A, B, X, D> map3(final F<C, X> f) { return new P4<A, B, X, D>() { public A _1() { return P4.this._1(); } public B _2() { return P4.this._2(); } public X _3() { return f.f(P4.this._3()); } public D _4() { return P4.this._4(); } }; } /** * Map the fourth element of the product. * * @param f The function to map with. * @return A product with the given function applied. */ public final <X> P4<A, B, C, X> map4(final F<D, X> f) { return new P4<A, B, C, X>() { public A _1() { return P4.this._1(); } public B _2() { return P4.this._2(); } public C _3() { return P4.this._3(); } public X _4() { return f.f(P4.this._4()); } }; } /** * Returns the 1-product projection over the first element. * * @return the 1-product projection over the first element. */ public final P1<A> _1_() { return F1Functions.lazy(P4.<A, B, C, D>__1()).f(this); } /** * Returns the 1-product projection over the second element. * * @return the 1-product projection over the second element. */ public final P1<B> _2_() { return F1Functions.lazy(P4.<A, B, C, D>__2()).f(this); } /** * Returns the 1-product projection over the third element. * * @return the 1-product projection over the third element. */ public final P1<C> _3_() { return F1Functions.lazy(P4.<A, B, C, D>__3()).f(this); } /** * Returns the 1-product projection over the fourth element. * * @return the 1-product projection over the fourth element. */ public final P1<D> _4_() { return F1Functions.lazy(P4.<A, B, C, D>__4()).f(this); } /** * Provides a memoising P4 that remembers its values. * * @return A P4 that calls this P4 once for any given element and remembers the value for subsequent calls. */ public final P4<A, B, C, D> memo() { P4<A, B, C, D> self = this; return new P4<A, B, C, D>() { private final P1<A> a = weakMemo(self::_1); private final P1<B> b = weakMemo(self::_2); private final P1<C> c = weakMemo(self::_3); private final P1<D> d = weakMemo(self::_4); public A _1() { return a._1(); } public B _2() { return b._1(); } public C _3() { return c._1(); } public D _4() { return d._1(); } }; } /** * Returns a function that returns the first element of a product. * * @return A function that returns the first element of a product. */ public static <A, B, C, D> F<P4<A, B, C, D>, A> __1() { return P4::_1; } /** * Returns a function that returns the second element of a product. * * @return A function that returns the second element of a product. */ public static <A, B, C, D> F<P4<A, B, C, D>, B> __2() { return P4::_2; } /** * Returns a function that returns the third element of a product. * * @return A function that returns the third element of a product. */ public static <A, B, C, D> F<P4<A, B, C, D>, C> __3() { return P4::_3; } /** * Returns a function that returns the fourth element of a product. * * @return A function that returns the fourth element of a product. */ public static <A, B, C, D> F<P4<A, B, C, D>, D> __4() { return P4::_4; } @Override public final String toString() { return Show.p4Show(Show.<A>anyShow(), Show.<B>anyShow(), Show.<C>anyShow(), Show.<D>anyShow()).showS(this); } @Override public final boolean equals(Object other) { return Equal.equals0(P4.class, this, other, () -> Equal.p4Equal(Equal.anyEqual(), Equal.anyEqual(), Equal.anyEqual(), Equal.anyEqual())); } @Override public final int hashCode() { return Hash.p4Hash(Hash.<A>anyHash(), Hash.<B>anyHash(), Hash.<C>anyHash(), Hash.<D>anyHash()).hash(this); } }