001    /*
002     * $Id: ResidueRing.java 3211 2010-07-05 12:54:22Z kredel $
003     */
004    
005    package edu.jas.application;
006    
007    import java.util.Random;
008    import java.util.List;
009    import java.util.ArrayList;
010    import java.io.Reader;
011    
012    import org.apache.log4j.Logger;
013    
014    import edu.jas.structure.GcdRingElem;
015    import edu.jas.structure.RingFactory;
016    
017    import edu.jas.poly.GenPolynomial;
018    import edu.jas.poly.GenPolynomialRing;
019    
020    import edu.jas.ufd.GreatestCommonDivisor;
021    import edu.jas.ufd.GCDFactory;
022    
023    
024    /**
025     * Residue ring factory based on GenPolynomial with RingFactory interface.
026     * Objects of this class are immutable.
027     * @author Heinz Kredel
028     */
029    public class ResidueRing<C extends GcdRingElem<C> > 
030                 implements RingFactory< Residue<C> >  {
031    
032        private static final Logger logger = Logger.getLogger(ResidueRing.class);
033        //private boolean debug = logger.isDebugEnabled();
034    
035    
036        /**
037         * Greatest common divisor engine for coefficient content and primitive parts.
038         */
039        protected final GreatestCommonDivisor<C> engine;
040    
041    
042        /** Polynomial ideal for the reduction. 
043         */
044        public final Ideal<C> ideal;
045    
046    
047        /** Polynomial ring of the factory. 
048         * Shortcut to ideal.list.ring. 
049         */
050        public final GenPolynomialRing<C> ring;
051    
052    
053        /** Indicator if this ring is a field.
054         */
055        protected int isField = -1; // initially unknown
056    
057    
058        /** The constructor creates a ResidueRing object 
059         * from an Ideal. 
060         * @param i polynomial ideal.
061         */
062        public ResidueRing(Ideal<C> i) {
063            this(i,false);
064        }
065    
066    
067        /** The constructor creates a ResidueRing object 
068         * from an Ideal. 
069         * @param i polynomial ideal.
070         * @param isMaximal true, if ideal is maxmal.
071         */
072        public ResidueRing(Ideal<C> i, boolean isMaximal) {
073            ideal = i.GB(); // cheap if isGB
074            ring = ideal.list.ring;
075            if ( isMaximal ) {
076               isField = 1;
077            }
078            //engine = GCDFactory.<C>getImplementation( ring.coFac );
079            engine = GCDFactory.<C>getProxy( ring.coFac );
080            //System.out.println("rr engine = " + engine.getClass().getName());
081            //System.out.println("rr ring   = " + ring.getClass().getName());
082            //System.out.println("rr cofac  = " + ring.coFac.getClass().getName());
083        }
084    
085    
086        /**
087         * Is this structure finite or infinite.
088         * @return true if this structure is finite, else false.
089         * @see edu.jas.structure.ElemFactory#isFinite()
090         */
091        public boolean isFinite() {
092            return ideal.commonZeroTest() <= 0 && ring.coFac.isFinite();
093        }
094    
095    
096        /** Copy Residue element c.
097         * @param c
098         * @return a copy of c.
099         */
100        public Residue<C> copy(Residue<C> c) {
101            //System.out.println("rr copy in    = " + c.val);
102            if ( c == null ) { // where does this happen?
103               return getZERO(); // or null?
104            } 
105            Residue<C> r = new Residue<C>( this, c.val );
106            //System.out.println("rr copy out   = " + r.val);
107            //System.out.println("rr copy ideal = " + ideal.list.list);
108            return r; //new Residue<C>( c.ring, c.val );
109        }
110    
111    
112        /** Get the zero element.
113         * @return 0 as Residue.
114         */
115        public Residue<C> getZERO() {
116            return new Residue<C>( this, ring.getZERO() );
117        }
118    
119    
120        /**  Get the one element.
121         * @return 1 as Residue.
122         */
123        public Residue<C> getONE() {
124            Residue<C> one = new Residue<C>( this, ring.getONE() );
125            if ( one.isZERO() ) {
126               logger.warn("ideal is one, so all residues are 0");
127            }
128            return one;
129        }
130    
131    
132        /**  Get a list of the generating elements.
133         * @return list of generators for the algebraic structure.
134         * @see edu.jas.structure.ElemFactory#generators()
135         */
136        public List<Residue<C>> generators() {
137            List<GenPolynomial<C>> pgens = ring.generators();
138            List<Residue<C>> gens = new ArrayList<Residue<C>>( pgens.size() );
139            for ( GenPolynomial<C> p : pgens ) {
140                Residue<C> r = new Residue<C>( this, p );
141                gens.add(r);
142            }
143            return gens;
144        }
145    
146        
147        /**
148         * Query if this ring is commutative.
149         * @return true if this ring is commutative, else false.
150         */
151        public boolean isCommutative() {
152            return ring.isCommutative();
153        }
154    
155    
156        /**
157         * Query if this ring is associative.
158         * @return true if this ring is associative, else false.
159         */
160        public boolean isAssociative() {
161            return ring.isAssociative();
162        }
163    
164    
165        /**
166         * Query if this ring is a field.
167         * @return false.
168         */
169        public boolean isField() {
170            if ( isField > 0 ) { 
171               return true;
172            }
173            if ( isField == 0 ) { 
174               return false;
175            }
176            // ideal is prime or maximal ?
177            return false;
178        }
179    
180    
181        /**
182         * Characteristic of this ring.
183         * @return characteristic of this ring.
184         */
185        public java.math.BigInteger characteristic() {
186            return ring.characteristic();
187        }
188    
189    
190        /** Get a Residue element from a BigInteger value.
191         * @param a BigInteger.
192         * @return a Residue.
193         */
194        public Residue<C> fromInteger(java.math.BigInteger a) {
195            return new Residue<C>( this, ring.fromInteger(a) );
196        }
197    
198    
199        /** Get a Residue element from a long value.
200         * @param a long.
201         * @return a Residue.
202         */
203        public Residue<C> fromInteger(long a) {
204            return new Residue<C>( this, ring.fromInteger(a) );
205        }
206        
207    
208        /** Get the String representation as RingFactory.
209         * @see java.lang.Object#toString()
210         */
211        @Override
212         public String toString() {
213            return "ResidueRing[ " 
214                    + ideal.toString() + " ]";
215        }
216    
217    
218        /** Get a scripting compatible string representation.
219         * @return script compatible representation for this ElemFactory.
220         * @see edu.jas.structure.ElemFactory#toScript()
221         */
222        //JAVA6only: @Override
223        public String toScript() {
224            // Python case
225            return "RC(" + ideal.list.toScript() + ")";
226            //return "RC(" + ideal.toScript() + "," + ring.toScript()  + ")";
227        }
228    
229    
230        /** Comparison with any other object.
231         * @see java.lang.Object#equals(java.lang.Object)
232         */
233        @Override
234        @SuppressWarnings("unchecked") 
235        public boolean equals(Object b) {
236            if ( ! ( b instanceof ResidueRing ) ) {
237               return false;
238            }
239            ResidueRing<C> a = null;
240            try {
241                a = (ResidueRing<C>) b;
242            } catch (ClassCastException e) {
243            }
244            if ( a == null ) {
245                return false;
246            }
247            if ( ! ring.equals( a.ring ) ) {
248                return false;
249            }
250            return ideal.equals( a.ideal );
251        }
252    
253    
254        /** Hash code for this residue ring.
255         * @see java.lang.Object#hashCode()
256         */
257        @Override
258        public int hashCode() { 
259           int h;
260           h = ideal.hashCode();
261           return h;
262        }
263    
264    
265        /** Residue random.
266         * @param n such that 0 &le; v &le; (2<sup>n</sup>-1).
267         * @return a random residue element.
268         */
269        public Residue<C> random(int n) {
270          GenPolynomial<C> x = ring.random( n ).monic();
271          return new Residue<C>( this, x );
272        }
273    
274    
275        /**
276         * Generate a random residum polynomial.
277         * @param k bitsize of random coefficients.
278         * @param l number of terms.
279         * @param d maximal degree in each variable.
280         * @param q density of nozero exponents.
281         * @return a random residue polynomial.
282         */
283        public Residue<C> random(int k, int l, int d, float q) {
284          GenPolynomial<C> x = ring.random(k,l,d,q).monic();
285          return new Residue<C>( this, x );
286        }
287    
288    
289        /** Residue random.
290         * @param n such that 0 &le; v &le; (2<sup>n</sup>-1).
291         * @param rnd is a source for random bits.
292         * @return a random residue element.
293         */
294        public Residue<C> random(int n, Random rnd) {
295          GenPolynomial<C> x = ring.random( n, rnd ).monic();
296          return new Residue<C>( this, x);
297        }
298    
299    
300        /** Parse Residue from String.
301         * @param s String.
302         * @return Residue from s.
303         */
304        public Residue<C> parse(String s) {
305            GenPolynomial<C> x = ring.parse( s );
306            return new Residue<C>( this, x );
307        }
308    
309    
310        /** Parse Residue from Reader.
311         * @param r Reader.
312         * @return next Residue from r.
313         */
314        public Residue<C> parse(Reader r) {
315            GenPolynomial<C> x = ring.parse( r );
316            return new Residue<C>( this, x );
317        }
318    
319    }