Basic Polynomial Algebra Subprograms (BPAS)  v. 1.791
FactorRefinement.hpp
1 #ifndef _FACTOR_REFINEMENT_H_
2 #define _FACTOR_REFINEMENT_H_
3 
4 #include <iostream>
5 #include "BPASFieldOfFractions.hpp"
6 #include "../../include/RationalFunction/rationalfunction_euclideanmethods.h"
7 #include "../../include/RationalNumberPolynomial/mrpolynomial.h"
8 #include <algorithm>
9 #include <vector>
10 #include <utility>
11 #include "../Utils/TemplateHelpers.hpp"
12 #include "../DataStructures/Factors.hpp"
13 
14 namespace FactorRefinement {
15  /*
16  *PolyRefine: it will do the factor refinement for two object a^e and b^f
17  *input a, e, b, f, which are the input for FactorRefinement algorithm
18  *input *ret_l,*ret_e,*G,*ret_r,*ret_f, should be empty pointers, the result of the algorithm will store in those pointers
19  *ret_l^ret_e is the refinement result for a^e
20  *ret_r^ret_f is the refinement result for b^f
21  *G is the gcd parts for FactorRefinement algorithm
22  */
23  template <class Domain>
24  static void PolyRefine(const Domain& a, int e, const Domain& b, int f, Domain* ret_l, int* ret_e, std::vector<Factor<Domain>>* ret_G, Domain* ret_r, int* ret_f) {
25  Domain g = a.gcd(b);
26  Domain a_ = a/g;
27  Domain b_ = b/g;
28  if(g.isOne() == true){
29  *ret_l = a;
30  //cout << "ret_l " << *ret_l << endl;
31  *ret_e = e;
32  //cout << "ret_e " << *ret_e << endl;
33  //Domain one;
34  //one.one();
35  //ret_G->push_back(std::make_pair(one,1));
36  ret_G->clear();
37  *ret_r = b;
38  *ret_f = f;
39  return;
40 
41  }
42  else if(a==b){
43  Domain one;
44  one.one();
45  ret_G->clear();
46  ret_G->push_back(std::make_pair(a,e+f));
47  *ret_l = one;
48  *ret_e = 1;
49  *ret_r = one;
50  *ret_f = 1;
51  return;
52  }
53  else{
54  Domain l1,l2;
55  int e1,e2;
56  Domain r1,r2;
57  int f1,f2;
58  std::vector<Factor<Domain>> G1,G2;
59  PolyRefine(a_,e,g,e+f,&l1,&e1,&G1,&r1,&f1);
60  PolyRefine(r1,f1,b_,f,&l2,&e2,&G2,&r2,&f2);
61  if(l2.isOne()==false){
62  G2.push_back(std::make_pair(l2,e2));
63  }
64  *ret_l = l1;
65  *ret_e = e1;
66  ret_G->clear();
67  ret_G->insert(ret_G->end(),G1.begin(),G1.end());
68  ret_G->insert(ret_G->end(),G2.begin(),G2.end());
69  *ret_r = r2;
70  *ret_f = f2;
71  return;
72 
73  }
74 
75  }
76 
77  /*
78  *PolyRefine: it will do the factor refinement for one object a^e and one list B
79  *input a, e, B which are the input for FactorRefinement algorithm
80  *input *ret_l,*ret_m,*Q,*S, should be empty pointers, the result of the algorithm will store in those pointers
81  *ret_l^ret_m is the refinement result for a^e
82  *S is the refinement result for B
83  *Q is the gcd parts for FactorRefinement algorithm
84  */
85  template <class Domain>
86  static void MergeRefinePolySeq(const Domain& a, int e,const std::vector<Factor<Domain>>& B, Domain* ret_l, int* ret_m, std::vector<Factor<Domain>>* ret_Q,std::vector<Factor<Domain>>* ret_S){
87  Domain l0 = a;
88  int m0 = e;
89 
90  //cout << "a in 2: " << a << endl;
91  //cout << "e in 2: " << e << endl;
92  std::vector<Factor<Domain>> Q;
93  std::vector<Factor<Domain>> S;
94 
95 
96  //cout << "B.size(): " << B.size() << endl;
97 
98  for (int i = 0; i < B.size(); ++i)
99  {
100 
101  Domain li;
102  int mi;
103  std::vector<Factor<Domain>> Gi;
104  Domain di;
105  int wi;
106 
107 
108  PolyRefine(l0,m0,B[i].first,B[i].second,&li,&mi,&Gi,&di,&wi);
109 
110  //cout << "li in 2: " << li << endl;
111  //cout << "di in 2: " << di << endl;
112 
113  Q.insert(Q.end(),Gi.begin(),Gi.end());
114  if(di.isOne()==false){
115  //cout << "di in 2: " << di << endl;
116  //cout << "wi in 2: " << wi << endl;
117  S.push_back(std::make_pair(di,wi));
118  }
119 
120  l0 = li;
121  m0 = mi;
122 
123  }
124 
125 
126 
127  *ret_l = l0;
128  *ret_m = m0;
129 
130 
131  //cout << "ret_l in 2: " << *ret_l << endl;
132  //cout << "ret_m in 2: " << *ret_m << endl;
133  ret_Q->insert(ret_Q->end(),Q.begin(),Q.end());
134  ret_S->insert(ret_S->end(),S.begin(),S.end());
135  return;
136  }
137  /*
138  *PolyRefine: it will do the factor refinement for two lists A and B
139  *input A and B which are the input for FactorRefinement algorithm
140  *input *ret_L,*ret_Q,*ret_S should be empty pointers, the result of the algorithm will store in those pointers
141  *ret_L is the refinement result for A
142  *ret_S is the refinement result for B
143  *ret_Q is the gcd parts for FactorRefinement algorithm
144  */
145  template <class Domain>
146  static void MergeRefineTwoSeq(const std::vector<Factor<Domain>>& A, const std::vector<Factor<Domain>>& B,std::vector<Factor<Domain>>* ret_L, std::vector<Factor<Domain>>* ret_Q,std::vector<Factor<Domain>>* ret_S){
147  std::vector<Factor<Domain>> L,Q,S;
148  S = B;
149  for (int i = 0; i < A.size(); ++i)
150  {
151  Domain l;
152  int m;
153  std::vector<Factor<Domain>> tempQ,tempS;
154  MergeRefinePolySeq(A[i].first,A[i].second,S,&l,&m,&tempQ,&tempS);
155  S = tempS;
156  Q.insert(Q.end(),tempQ.begin(),tempQ.end());
157  if(!l.isOne()){
158  L.push_back(std::make_pair(l,m));
159  //cout << "l: " << l << endl;
160  //cout << "m: " << m << endl;
161 
162  }
163  }
164 
165  ret_L->insert(ret_L->end(),L.begin(),L.end());
166  ret_Q->insert(ret_Q->end(),Q.begin(),Q.end());
167  ret_S->insert(ret_S->end(),S.begin(),S.end());
168 
169  //(3^1)*(2^2)
170  return;
171  }
172 
173  /*
174  *This function is similar with MergeRefineTwoSeq, it will call MergeRefineTwoSeq first
175  *then it check the result of MergeRefineTwoSeq, if the result is an empty vector, it will change the vector to one
176  *
177  *PolyRefine: it will do the factor refinement for two lists A and B
178  *input A and B which are the input for FactorRefinement algorithm
179  *input *ret_L,*ret_Q,*ret_S should be empty pointers, the result of the algorithm will store in those pointers
180  *ret_L is the refinement result for A
181  *ret_S is the refinement result for B
182  *ret_Q is the gcd parts for FactorRefinement algorithm
183  */
184  template <class Domain>
185  static void MergeRefineTwoSeqEmptyToIdentity(const std::vector<Factor<Domain>>& A, const std::vector<Factor<Domain>>& B,
186  std::vector<Factor<Domain>>* ret_L, std::vector<Factor<Domain>>* ret_Q,std::vector<Factor<Domain>>* ret_S){
187 
188  MergeRefineTwoSeq(A,B,&*ret_L,&*ret_Q,&*ret_S);
189 
190  Domain one;
191  one.one();
192  if(ret_L->size() == 0){
193  ret_L->push_back(std::make_pair(one,1));
194  }
195  if(ret_Q->size() == 0){
196  ret_Q->push_back(std::make_pair(one,1));
197  }
198  if(ret_S->size() == 0){
199  ret_S->push_back(std::make_pair(one,1));
200  }
201 
202  return;
203  }
204 };
205 
206 
207 
208 //template <class Domain>
209 //void PolyRefine(const Domain& a, int e, const Domain& b, int f, Domain* ret_l, int* ret_e, std::vector<Factor<Domain>>* ret_G, Domain* ret_r, int* ret_f);
210 
211 
212 
213 
214 
215 
216 
217 
218 
219 
220 #endif
Definition: FactorRefinement.hpp:14