In J it is implemented as (X|Y)=Y-X mul flr Y div X+0=X for all numerical X and Y, where mul is multiplication, div is division, and flr is flooring as defined here for complex numbers: http://www.jsoftware.com/help/dictionary/d011.htm
I believe it's implemented like in McDonnell's article which Jay already posted. Perhaps this can help? Louis > On 23 Jun 2017, at 09:34, Jay Foad <jay.f...@gmail.com> wrote: > > I urge you to read Eugene McDonnell's Complex Floor, which also discusses > Residue. I believe the design he comes up with in this paper was adopted more > or less verbatim in APL. Also bear in mind that Floor and Residue in APL have > to work well on all complex numbers, not just the Gaussian integers. > > Jay. > >> On 23 June 2017 at 01:42, Frederick Pitts <fred.pi...@comcast.net> wrote: >> Hello Jürgen, >> >> Some observations: >> >> 1) When performing a residue calculation on positive integers, a >> straight-forward integer division with remainder calculation >> suffices. For example, 5 ∣ 13 is computed with 13 / 5 = 2 r 3 and so 5 ∣ 13 >> = 3 where 3 is in the complete residue system >> { 0, 1, 2, 3, 4 }. When performing the calculation on negative integers, one >> has to take advantage of the fact that the >> integer division quotient and remainder are not unique in order to compute a >> residue that is in the complete residue system. >> For 5 ∣ ¯13, ¯13 / 5 = ¯2 r ¯3 where ¯3 is not in the CRS. However, ¯13 / 5 >> = ¯3 r 2 where 3 is in the CSR. The same concept applies Gaussian integers. >> >> 2) I suspect the decision to have the APL2 floor function round toward >> negative infinity, instead of toward zero, >> was made based on the desire to save cpu cycles and memory in the residue >> function code. >> >> 3) I read at least one math literature article discussing Gaussian integer >> Euclidean division algorithms, that recommended >> rounding down to the nearest real and imaginary part toward negative >> infinity. Unfortunately I cannot find >> the article right now. I will continue to look for it. None of the articles >> discussed using a complex integer floor function. >> >> 4) The reason MOD_TEST.apl shows total disagreement MODJ and the builtin >> residue function is that the complex floor function code change in SVN 965 >> relocated the CRS's on complex plane. Attached are CRS0-CRS1-6J-6-SVN964.out >> CRS0-CRS1-6J-6-SVN965.out. The first file contains a CRS map for modulus >> ¯6J¯6 produced with the residue function >> followed by a map for the same modulus produced with MODJ using SVN 964. The >> second file contains the same maps >> using SVN 965. Observe that for SVN 964 the residue function CRS is in the >> bottom half of the complex plane, but for SVN 965 it is in the top half. The >> CRS for the MODJ function is in the bottom half in both SVN cases. >> 5)The complex floor code change did not help with the issue that the builtin >> residue function is not idempotent for all possible arguments and >> consequently generates too many residues. See attached CRSOTST0-SVN965.out. >> For a grid >> of Gaussian integers with real and imaginary parts ranging from ¯15 to 15, >> using every value with every other value as modulus and second argument, >> there were 40 case where the order of CSR exceeded the modulus norm. I think >> that >> was the failure count with the previous SVN. >> >> Sincerely, I think the complex floor and ceiling functions should not be >> used by other functions even if IBM and ISO >> imply they are in their documentations. I'm not seeing them used in the >> Gaussian integer literature. Again, please correct me if I'm wrong. >> >> Regards, >> >> Fred >> >>> On Thu, 2017-06-22 at 18:08 +0200, Juergen Sauermann wrote: >>> Hi again, >>> >>> sorry a small typo below. Lines 19/20 should read: >>> >>> (¯6J¯5 - 0J¯11) ÷ ¯6J¯6 >>> 0J¯1 >>> >>> /// Jürgen >>> >>>> On 06/22/2017 05:44 PM, Juergen Sauermann wrote: >>>> Hi Fred at al., >>>> >>>> I have made another attempt to fix the residue function, SVN 965. >>>> >>>> For complex m∣b It now rounds down the real() and imag() parts of the >>>> quotient q←b÷m and returns b-q. >>>> Instead of always rounding towards 0 or -infinity, the rounding direction >>>> is now (compared to the previous >>>> attempt) determined by the quadrant in which the modulus m lies. >>>> >>>> There are still differences to the results displayed by MOD_test.apl, but >>>> I suppose they are >>>> caused by that program. For example, the first line of MOD_test.apl, says: >>>> >>>> LA RA MODJ | >>>> ¯6J¯6 ¯6J¯5 0J¯11 0J1 >>>> >>>> We have: >>>> >>>> (¯6J¯5 - 0J1) ÷ ¯6J¯6 >>>> 1 >>>> (0J¯11 - 0J1) ÷ ¯6J¯6 >>>> 1J1 >>>> >>>> so both 0J¯11 and 0J1 are valid remainders modulo ¯6J¯6. However, the >>>> magnitude of 0J¯11 (= 11) is larger than the magnitude of the divisor >>>> ¯6J¯6 (= around 8.4). >>>> I suppose most people expect the remainder of a division to be in some >>>> sense >>>> smaller than the divisor. >>>> >>>> Regarding Jay's idempotency requirement we now have: >>>> >>>> f←{6J6|⍵} >>>> f ¯3 ¯2 ¯3 ¯1 0 1 2 3 >>>> 3J6 4J6 3J6 5J6 0 ¯5J6 ¯4J6 ¯3J6 >>>> f f ¯3 ¯2 ¯3 ¯1 0 1 2 3 >>>> 3J6 4J6 3J6 5J6 0 ¯5J6 ¯4J6 ¯3J6 >>>> >>>> f←{5J3|⍵} >>>> f ¯3 ¯2 ¯3 ¯1 0 1 2 3 >>>> 2J3 3J3 2J3 4J3 0 ¯2J5 ¯1J5 0J5 >>>> f f ¯3 ¯2 ¯3 ¯1 0 1 2 3 >>>> 2J3 3J3 2J3 4J3 0 ¯2J5 ¯1J5 0J5 >>>> >>>> so at least the first modulus seems to work as well. The result is still >>>> different >>>> from APL2 as reported by Jay, but I can't tell why: >>>> >>>> IBM APL2: >>>> >>>> 5J3 ∣ 14J5 1J4 ¯4J1 >>>> ¯4J1 ¯4J1 ¯4J1 >>>> >>>> GNU APL: >>>> >>>> 5J3 ∣ 14J5 1J4 ¯4J1 >>>> 1J4 1J4 1J4 >>>> >>>> But both 1J4 and ¯4J1 are valid remainders. Interestingly Jay's >>>> idempotency requirement seems to >>>> be fulfilled by both the GNU APL and by IBM APL2, so that that requirement >>>> alone does not suffice >>>> to tell which result is correct. >>>> >>>> On the other hand this matter seems to be like discussing if the square >>>> root of 4 is 2 or -2 with >>>> both answers being correct. >>>> >>>> Best Regards, >>>> Jürgen Sauermann >>>> >>>> >>>> >>>>> On 06/21/2017 10:25 PM, Frederick Pitts wrote: >>>>> Jürgen, >>>>> >>>>> The proposed change to DIVJ does not work because 'q1' is a complex >>>>> number, so the '×' in '× q1' is the complex complement function, not the >>>>> sign function. I tried the proposed change and every test fails. >>>>> >>>>> I will try to hack DIVJ to use a floor towards zero instead of towards >>>>> minus infinity for the real and imaginary >>>>> parts of the quotient and see what happens. >>>>> >>>>> Correct me if I am wrong, but my mind set is that the APL residue >>>>> function has to satisfy the following invariants: >>>>> 1) The order of the complete residue system (residue count) for a given >>>>> modulo 'n' has to equal the norm of 'n'. >>>>> 2) And as Jay Foad so succinctly expressed it, the residue function has >>>>> to be idempotent with respect to its right argument, >>>>> e.g., ( n | m ) = n | n | m . >>>>> regardless of the implementation of the residue function. >>>>> >>>>> Later, >>>>> >>>>> Fred >>>>> >>>>> >>>>> >>>>> >>>>>> On Wed, 2017-06-21 at 19:46 +0200, Juergen Sauermann wrote: >>>>>> Hi Fred, >>>>>> >>>>>> I have a question about the MOD_test.apl that you kindly provided. >>>>>> >>>>>> In function DIVJ on line 57 ff it says: >>>>>> >>>>>> z ← q , a - b × q ← CMPLX ⌊ ( 9 11 ) ○ a ÷ b >>>>>> >>>>>> so the quotient is rounded down towards minus infinity. >>>>>> >>>>>> I wonder if that should be something like >>>>>> >>>>>> z ← q , (× q1) × a - b × q ← CMPLX ⌊ ∣ 9 11 ○ q1 ← a ÷ b >>>>>> >>>>>> so that the quotient is rounded towards 0? Interestingly IBM and ISO >>>>>> give different definitions for the residue in terms of APL: >>>>>> >>>>>> IBM (language reference, page 227): >>>>>> Z←L∣R >>>>>> Z is R-L×⌊ R÷L+L=0 >>>>>> >>>>>> ISO (chapter 7.2.9 Residue): >>>>>> R←Q∣P >>>>>> R←P-(×P)×|Q×⌊|P÷Q >>>>>> and return R if (×R)=×Q, or R+Q >>>>>> otherwise. >>>>>> >>>>>> That suggest that IBM rounds the quotient down towards minus infinity >>>>>> while ISO rounds >>>>>> towards 0. >>>>>> >>>>>> My naive view on remainder is that the nearest integer quotient shall be >>>>>> smaller in >>>>>> magnitude and not smaller in value. Regarding your proposal (which is >>>>>> different from >>>>>> both IBM and ISO) my concern is that may lead to different results for >>>>>> modulo N and >>>>>> modulo N×1J0 >>>>>> >>>>>> Best Regards, >>>>>> Jürgen Sauermann >>>>>> >>>>>> >>>>>>> On 06/21/2017 03:08 AM, Frederick Pitts wrote: >>>>>>> Jürgen, >>>>>>> >>>>>>> This message is being resent because last minute changes I made to >>>>>>> CRS0.apl and CRS1.apl do not output the >>>>>>> data I intended. This message has corrected versions of those files >>>>>>> attached. Please discard the old CRS0.apl and CRS1.apl files. The >>>>>>> first line of output is the modulo basis, the second line is the >>>>>>> calculated complete residue system values and the third line is the >>>>>>> number of residues in the CRS on the previous line. >>>>>>> >>>>>>> CRSOTST0.apl and CRSOTST1.apl are unchanged. >>>>>>> >>>>>>> Also please find attached MOD_TEST.apl which compares the residues >>>>>>> calculated by MODJ and the builtin residue function and reports >>>>>>> discrepancies. The first column of output is the modulo basis, the >>>>>>> second column the right argument to the functions, the third column the >>>>>>> MODJ result and the fourth column is the builtin residue function >>>>>>> result. >>>>>>> >>>>>>> Regards >>>>>>> >>>>>>> Fred >>>>>>> >>>>>>> Hello Jürgen, >>>>>>> SVN 964 moved us in the right direction but not completely out >>>>>>> of the >>>>>>> woods. SVN 964 still exhibits errors. For instance >>>>>>> 2J6 | 5J5 >>>>>>> ¯1J7 >>>>>>> 2J6 | ¯1J7 >>>>>>> ¯3J1 >>>>>>> 2J6 | ¯3J1 >>>>>>> ¯3J1 >>>>>>> I found this and previous residue function errors using the >>>>>>> attached APL >>>>>>> code files. The files with base name ending in '0' use the builtin >>>>>>> residue >>>>>>> function. Those with base name ending in '1' use a residue function >>>>>>> implemented >>>>>>> in APL. The files with base name beginning with 'CRSOTST' test if the >>>>>>> order of >>>>>>> the complete residue system (CRS) equals the norm of the modulo basis. >>>>>>> That >>>>>>> test fails for several modulo bases, 2J6 being one of them, using the >>>>>>> builtin >>>>>>> residue function. No errors are detected with the APL implementation. >>>>>>> The other files >>>>>>> can be used to plot the CRS for a given modulo basis where 'a' and 'b' >>>>>>> in >>>>>>> 'a + b * i' are limited to +15 to -15 range. A full screen terminal >>>>>>> window is >>>>>>> needed to see the plot. >>>>>>> My APL implementation of the residue function is very close to >>>>>>> what you >>>>>>> described in your previous email. Maybe comparing the two >>>>>>> implementations will >>>>>>> give insight into why the builtin residue function fails for some >>>>>>> modulo bases. >>>>>>> I make no assertion that my implementation is correct in all >>>>>>> aspects. >>>>>>> Regards, >>>>>>> Fred >>>>>>>> On Tue, 2017-06-20 at 14:14 +0200, Juergen Sauermann wrote: >>>>>>>> Hi Frederick, >>>>>>>> >>>>>>>> the algorithm for A ∣ B used in GNU APL is this: >>>>>>>> >>>>>>>> - compute the quotient Q←B÷A, >>>>>>>> - "round down" Q to the next (complex) integer Q1, >>>>>>>> - return B - Q1×A >>>>>>>> >>>>>>>> Now the problem seems to be what is meant by "round down". There are >>>>>>>> two candidates: >>>>>>>> >>>>>>>> Q1 ← ⌊ Q i.e. use APL floor >>>>>>>> to round down Q >>>>>>>> Q1 ← Complex( floor(Q.real(), floor(Q.imag()) ) i,e, use C/C++ >>>>>>>> floor() to round down Q. >>>>>>>> >>>>>>>> In your 5J3 ∣ 14J5 example, the quotient is 2.5J¯0.5, which gives >>>>>>>> different results for the APL floor ⌊ and the C/C++ floor(). >>>>>>>> >>>>>>>> The APL floor ⌊2.5J¯0.5 is 3J¯1 (a somewhat dubious invention in the >>>>>>>> ISO standard on page 19, which I used up to >>>>>>>> including SVN 963), while the C/C++ floor() is 2J¯1. The difference >>>>>>>> between the APL floor and the C/C++ floor is 1.0 which, >>>>>>>> multiplied by the divisor, explains the differences that we see. >>>>>>>> >>>>>>>> As of SVN 964 I have changed the residue function (∣) to use the C/C++ >>>>>>>> floor instead of the APL floor. The APL floor and >>>>>>>> Ceiling functions (⌊ and ⌈) are still using the apparently broken >>>>>>>> definition in the ISO standard. >>>>>>>> >>>>>>>> I hope this works better for you. At least I am getting this in SVN >>>>>>>> 964: >>>>>>>> >>>>>>>> 5J3 | 14J5 >>>>>>>> 1J4 >>>>>>>> 5J3 | 1J4 >>>>>>>> 1J4 >>>>>>>> >>>>>>>> whereas SVN 963 was giving: >>>>>>>> >>>>>>>> 5J3 | 14J5 >>>>>>>> ¯4J1 >>>>>>>> 5J3 | 1J4 >>>>>>>> ¯4J1 >>>>>>>> >>>>>>>> >>>>>>>> Best Regards, >>>>>>>> /// Jürgen >>>>>>>> >>>>>>>> >>>>>>>> >>>>>>>>> On 06/19/2017 07:03 PM, Frederick Pitts wrote: >>>>>>>>> Jürgen, >>>>>>>>> >>>>>>>>> With gnu apl (svn 961 on Fedora 25, Intel(R) Core(TM) i7-6700 >>>>>>>>> CPU), the residue function (∣) yields the following: >>>>>>>>> >>>>>>>>> 5J3 ∣ 14J5 >>>>>>>>> 1J4 >>>>>>>>> 5J3 | 1J4 >>>>>>>>> ¯4J1 >>>>>>>>> 5J3 | ¯4J1 >>>>>>>>> ¯4J1 >>>>>>>>> The above result means that two elements in the complete residue >>>>>>>>> system >>>>>>>>> (CSR) for mod 5J3 are equal, i.e. 1J4 = ¯4J1 mod 5J3, which is not >>>>>>>>> allowed. None of the elements of a CSR can be equal modulo the CSR's >>>>>>>>> basis. >>>>>>>>> >>>>>>>>> Regards, >>>>>>>>> >>>>>>>>> Fred >>>>>>>>> >>>>>>>> >>>>>> >>>> >>> >