001/*
002 * Java Genetic Algorithm Library (jenetics-7.2.0).
003 * Copyright (c) 2007-2023 Franz Wilhelmstötter
004 *
005 * Licensed under the Apache License, Version 2.0 (the "License");
006 * you may not use this file except in compliance with the License.
007 * You may obtain a copy of the License at
008 *
009 *      http://www.apache.org/licenses/LICENSE-2.0
010 *
011 * Unless required by applicable law or agreed to in writing, software
012 * distributed under the License is distributed on an "AS IS" BASIS,
013 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
014 * See the License for the specific language governing permissions and
015 * limitations under the License.
016 *
017 * Author:
018 *    Franz Wilhelmstötter (franz.wilhelmstoetter@gmail.com)
019 */
020package io.jenetics;
021
022import static io.jenetics.internal.util.Hashes.hash;
023import static io.jenetics.internal.util.SerialIO.readLong;
024import static io.jenetics.internal.util.SerialIO.writeLong;
025import static io.jenetics.util.RandomRegistry.random;
026
027import java.io.DataInput;
028import java.io.DataOutput;
029import java.io.IOException;
030import java.io.InvalidObjectException;
031import java.io.ObjectInputStream;
032import java.io.Serial;
033import java.io.Serializable;
034
035import io.jenetics.util.ISeq;
036import io.jenetics.util.IntRange;
037import io.jenetics.util.LongRange;
038import io.jenetics.util.MSeq;
039import io.jenetics.util.Mean;
040
041/**
042 * NumericGene implementation which holds a 64-bit integer number.
043 *
044 * <p>This is a <a href="https://docs.oracle.com/javase/8/docs/api/java/lang/doc-files/ValueBased.html">
045 * value-based</a> class; use of identity-sensitive operations (including
046 * reference equality ({@code ==}), identity hash code, or synchronization) on
047 * instances of {@code LongGene} may have unpredictable results and should
048 * be avoided.
049 *
050 * @see LongChromosome
051 *
052 * @implNote
053 * This class is immutable and thread-safe.
054 *
055 * @author <a href="mailto:franz.wilhelmstoetter@gmail.com">Franz Wilhelmstötter</a>
056 * @since 1.6
057 * @version 6.1
058 */
059public final class LongGene
060        implements
061                NumericGene<Long, LongGene>,
062                Mean<LongGene>,
063                Comparable<LongGene>,
064                Serializable
065{
066
067        @Serial
068        private static final long serialVersionUID = 2L;
069
070        private final long _allele;
071        private final long _min;
072        private final long _max;
073
074        /**
075         * Create a new random {@code LongGene} with the given value and the
076         * given range. If the {@code value} isn't within the interval [min, max),
077         * no exception is thrown. In this case the method
078         * {@link LongGene#isValid()} returns {@code false}.
079         *
080         * @param allele the value of the gene.
081         * @param min the minimal valid value of this gene (inclusively).
082         * @param max the maximal valid value of this gene (exclusively).
083         */
084        private LongGene(final long allele, final long min, final long max) {
085                _allele = allele;
086                _min = min;
087                _max = max;
088        }
089
090        @Override
091        public Long allele() {
092                return _allele;
093        }
094
095        @Override
096        public Long min() {
097                return _min;
098        }
099
100        @Override
101        public Long max() {
102                return _max;
103        }
104
105        /**
106         * Return the range of {@code this} gene.
107         *
108         * @since 4.4
109         *
110         * @return the range of {@code this} gene
111         */
112        public LongRange range() {
113                return LongRange.of(_min, _max);
114        }
115
116        @Override
117        public byte byteValue() {
118                return (byte) _allele;
119        }
120
121        @Override
122        public short shortValue() {
123                return (short) _allele;
124        }
125
126        @Override
127        public int intValue() {
128                return (int) _allele;
129        }
130
131        @Override
132        public long longValue() {
133                return _allele;
134        }
135
136        @Override
137        public float floatValue() {
138                return (float) _allele;
139        }
140
141        @Override
142        public double doubleValue() {
143                return _allele;
144        }
145
146        @Override
147        public boolean isValid() {
148                return _allele >= _min && _allele < _max;
149        }
150
151        @Override
152        public int compareTo(final LongGene other) {
153                return Long.compare(_allele, other._allele);
154        }
155
156        @Override
157        public LongGene mean(final LongGene that) {
158                final long x = that._allele;
159                final long y = _allele;
160
161                // http://aggregate.org/MAGIC/#Average%20of%20Integers
162                return LongGene.of((x&y) + ((x^y) >> 1), _min, _max);
163        }
164
165        /**
166         * Create a new gene from the given {@code value} and the gene context.
167         *
168         * @since 5.0
169         * @param allele the value of the new gene.
170         * @return a new gene with the given value.
171         */
172        public LongGene newInstance(final long allele) {
173                return LongGene.of(allele, _min, _max);
174        }
175
176        @Override
177        public LongGene newInstance(final Long allele) {
178                return LongGene.of(allele, _min, _max);
179        }
180
181        @Override
182        public LongGene newInstance(final Number allele) {
183                final long value = allele instanceof Double || allele instanceof Float
184                        ? Math.round(allele.doubleValue())
185                        : allele.longValue();
186
187                return LongGene.of(value, _min, _max);
188        }
189
190        @Override
191        public LongGene newInstance() {
192                return LongGene.of(random().nextLong(_min, _max), _min, _max);
193        }
194
195        @Override
196        public int hashCode() {
197                return hash(_allele, hash(_min, hash(_max)));
198        }
199
200        @Override
201        public boolean equals(final Object obj) {
202                return obj == this ||
203                        obj instanceof LongGene other &&
204                        other._allele == _allele &&
205                        other._min == _min &&
206                        other._max == _max;
207        }
208
209        @Override
210        public String toString() {
211                return String.format("[%s]", _allele);
212        }
213
214        /* *************************************************************************
215         * Static factory methods.
216         * ************************************************************************/
217
218        /**
219         * Create a new random {@code LongGene} with the given value and the
220         * given range. If the {@code value} isn't within the interval [min, max],
221         * no exception is thrown. In this case the method
222         * {@link LongGene#isValid()} returns {@code false}.
223         *
224         * @param allele the value of the gene.
225         * @param min the minimal valid value of this gene (inclusively).
226         * @param max the maximal valid value of this gene (exclusively).
227         * @return a new {@code LongGene} with the given parameters.
228         */
229        public static LongGene of(final long allele, final long min, final long max) {
230                return new LongGene(allele, min, max);
231        }
232
233        /**
234         * Create a new random {@code LongGene} with the given value and the
235         * given range. If the {@code value} isn't within the interval [min, max],
236         * no exception is thrown. In this case the method
237         * {@link LongGene#isValid()} returns {@code false}.
238         *
239         * @since 3.2
240         *
241         * @param allele the value of the gene.
242         * @param range the long range to use
243         * @return a new random {@code LongGene}
244         * @throws NullPointerException if the given {@code range} is {@code null}.
245         */
246        public static LongGene of(final long allele, final LongRange range) {
247                return LongGene.of(allele, range.min(), range.max());
248        }
249
250        /**
251         * Create a new random {@code LongGene}. It is guaranteed that the value of
252         * the {@code LongGene} lies in the interval [min, max].
253         *
254         * @param min the minimal valid value of this gene (inclusively).
255         * @param max the maximal valid value of this gene (exclusively).
256         * @return a new {@code LongGene} with the given parameters.
257         * @throws IllegalArgumentException if {@code max} is greater than
258         *         or equal to {@code min}
259         */
260        public static LongGene of(final long min, final long max) {
261                return of(random().nextLong(min, max), min, max);
262        }
263
264        /**
265         * Create a new random {@code LongGene}. It is guaranteed that the value of
266         * the {@code LongGene} lies in the interval [min, max].
267         *
268         * @since 3.2
269         *
270         * @param range the long range to use
271         * @return a new random {@code LongGene}
272         * @throws NullPointerException if the given {@code range} is {@code null}
273         * @throws IllegalArgumentException if {@code max} is greater than
274         *         or equal to {@code min}
275         */
276        public static LongGene of(final LongRange range) {
277                return of(random().nextLong(range.min(), range.max()), range);
278        }
279
280        static ISeq<LongGene> seq(
281                final long min,
282                final long max,
283                final IntRange lengthRange
284        ) {
285                final var random = random();
286                final var length = random.nextInt(lengthRange.min(), lengthRange.max());
287
288                return MSeq.<LongGene>ofLength(length)
289                        .fill(() -> LongGene.of(random.nextLong(min, max), min, max))
290                        .toISeq();
291        }
292
293
294        /* *************************************************************************
295         *  Java object serialization
296         * ************************************************************************/
297
298        @Serial
299        private Object writeReplace() {
300                return new SerialProxy(SerialProxy.LONG_GENE, this);
301        }
302
303        @Serial
304        private void readObject(final ObjectInputStream stream)
305                throws InvalidObjectException
306        {
307                throw new InvalidObjectException("Serialization proxy required.");
308        }
309
310        void write(final DataOutput out) throws IOException {
311                writeLong(_allele, out);
312                writeLong(_min, out);
313                writeLong(_max, out);
314        }
315
316        static LongGene read(final DataInput in) throws IOException {
317                return of(readLong(in), readLong(in), readLong(in));
318        }
319
320}