How to resolve the algorithm Multiplicative order step by step in the Ruby programming language

Published on 12 May 2024 09:40 PM

How to resolve the algorithm Multiplicative order step by step in the Ruby programming language

Table of Contents

Problem Statement

The multiplicative order of a relative to m is the least positive integer n such that a^n is 1 (modulo m).

The multiplicative order of 37 relative to 1000 is 100 because 37^100 is 1 (modulo 1000), and no number smaller than 100 would do.

One possible algorithm that is efficient also for large numbers is the following: By the Chinese Remainder Theorem, it's enough to calculate the multiplicative order for each prime exponent p^k of m, and combine the results with the least common multiple operation. Now the order of a with regard to p^k must divide Φ(p^k). Call this number t, and determine it's factors q^e. Since each multiple of the order will also yield 1 when used as exponent for a, it's enough to find the least d such that (q^d)*(t/(q^e)) yields 1 when used as exponent.

Implement a routine to calculate the multiplicative order along these lines. You may assume that routines to determine the factorization into prime powers are available in some library. An algorithm for the multiplicative order can be found in Bach & Shallit, Algorithmic Number Theory, Volume I: Efficient Algorithms, The MIT Press, 1996: Exercise 5.8, page 115:Suppose you are given a prime p and a complete factorization of p-1.   Show how to compute the order of an element a in (Z/(p))* using O((lg p)4/(lg lg p)) bit operations.Solution, page 337:Let the prime factorization of p-1 be q1e1q2e2...qkek . We use the following observation: if x^((p-1)/qifi) = 1 (mod p) , and fi=ei or x^((p-1)/qifi+1) != 1 (mod p) , then qiei-fi||ordp x.   (This follows by combining Exercises 5.1 and 2.10.)

Hence it suffices to find, for each i , the exponent fi such that the condition above holds.This can be done as follows: first compute q1e1, q2e2, ... , qkek . This can be done using O((lg p)2) bit operations. Next, compute y1=(p-1)/q1e1, ... , yk=(p-1)/qkek . This can be done using O((lg p)2) bit operations. Now, using the binary method, compute x1=ay1(mod p), ... , xk=ayk(mod p) . This can be done using O(k(lg p)3) bit operations, and k=O((lg p)/(lg lg p)) by Theorem 8.8.10. Finally, for each i , repeatedly raise xi to the qi-th power (mod p) (as many as ei-1 times), checking to see when 1 is obtained.
This can be done using O((lg p)3) steps. The total cost is dominated by O(k(lg p)3) , which is O((lg p)4/(lg lg p)).

Instead of assuming a library call to factorize the modulus, we assume the caller of our Find_Order function has already factorized it. The Multiplicative_Order package is specified as follows ("multiplicative_order.ads"). Here is the implementation ("multiplicative_order.adb"): This is a sample program using the Multiplicative_Order package: The output from the sample program: Output: Uses prime/factor functions from Factors of an integer#Prime factoring. This implementation is not robust because of integer overflows. To properly deal with even moderately large numbers, an arbitrary precision integer package is a must. Translation of Julie, then revised to be more clojure idiomatic. It references some external modules for factoring and integer exponentiation. Assuming a function to efficiently calculate powers modulo some Integral, we get The dyadic verb mo converts its arguments to exact numbers a and m, executes mopk on the factorization of m, and combines the result with the least common multiple operation. The dyadic verb mopk expects a pair of prime and exponent in the second argument. It sets up a verb pm to calculate powers module p^k. Then calculates Φ(p^k) as t, factorizes it again into q and e, and calculates a^(t/(q^e)) as x. Now, it finds the least d such that subsequent application of pm yields 1. Finally, it combines the exponents q^d into a product. For example: (Uses the factors function from Factors of an integer#Julia.) Example output (using big to employ arbitrary-precision arithmetic where needed): For example, In Mathematica this is really easy, as this function is built-in: MultiplicativeOrder[k,n] gives the multiplicative order of k modulo n, defined as the smallest integer m such that k^m == 1 mod n. MultiplicativeOrder[k,n,{r1,r2,...}] gives the generalized multiplicative order of k modulo n, defined as the smallest integer m such that k^m==ri mod n for some i. Examples: gives back: Using modules: or The Racket function unit-group-order from racket/math computes the multiplicative order of an element a in Zn. An implementation of the algorithm in the tast description is shown below. Output: (formerly Perl 6) Built-in: Using Extensible prime generator#zkl and the GMP library for lcm (least common multiple), pow and powm ((n^e)%m) It would probably be nice to memoize the prime numbers but that isn't necessary for this task.

Let's start with the solution:

Step by Step solution about How to resolve the algorithm Multiplicative order step by step in the Ruby programming language

The Ruby code you provided is designed to determine the multiplicative order of numbers under a given modulus. Here's a detailed explanation:

  1. powerMod(b, p, m) Function:

    • This function calculates b to the power of p modulo m using a binary exponentiation algorithm. It considers each bit of the exponent p (in binary representation) and performs modular multiplication or squaring accordingly. This method efficiently calculates modular exponentiation, similar to the fast exponentiation method (sometimes called binary exponentiation or square-and-multiply).
  2. multOrder_(a, p, k) Function:

    • This function finds the multiplicative order of a modulo p^k.
    • It first computes pk and t. t is defined as the product of p-1 and p^(k-1).
    • It then iterates through the prime factors of t and calculates x for each prime factor q and its exponent e. x is computed as a^(t/q^e) modulo pk.
    • If x is not equal to 1, the function multiplies r by q and updates x to be x^q modulo pk.
    • Finally, it returns the value of r.
  3. multOrder(a, m) Function:

    • This function finds the multiplicative order of a modulo m.
    • It first obtains the prime factorization of m using the prime_division method.
    • It then applies the multOrder_ function to each prime factor and exponent pair in the factorization.
    • Finally, it returns the least common multiple (LCM) of the multiplicative orders obtained for each prime factor.
  4. Sample Usage:

    • The code provides several examples of using these functions to compute multiplicative orders and perform modular exponentiation.

    • The first example calculates the multiplicative order of 37 modulo 1000 and prints the result.

    • The next three examples calculate the multiplicative orders of 2, 17, and 54 modulo large prime numbers and print the results.

    • The last example checks if there exists a power r less than 9090 such that powerMod(54, r, b) == 1. If such a power exists, it prints the message 'Exists a power r < 9090 where powerMod(54,r,b)==1'; otherwise, it prints 'Everything checks.'

Source code in the ruby programming language

require 'prime'

def powerMod(b, p, m)
  p.to_s(2).each_char.inject(1) do |result, bit|
    result = (result * result) % m
    bit=='1' ? (result * b) % m : result
  end
end

def multOrder_(a, p, k)
  pk = p ** k
  t = (p - 1) * p ** (k - 1)
  r = 1
  for q, e in t.prime_division
    x = powerMod(a, t / q**e, pk)
    while x != 1
      r *= q
      x = powerMod(x, q, pk)
    end
  end      
  r
end

def multOrder(a, m)
  m.prime_division.inject(1) do |result, f|
    result.lcm(multOrder_(a, *f))
  end
end

puts multOrder(37, 1000)
b = 10**20-1
puts multOrder(2, b)
puts multOrder(17,b)
b = 100001
puts multOrder(54,b)
puts powerMod(54, multOrder(54,b), b)
if (1...multOrder(54,b)).any? {|r| powerMod(54, r, b) == 1}
  puts 'Exists a power r < 9090 where powerMod(54,r,b)==1'
else
  puts 'Everything checks.'
end


  

You may also check:How to resolve the algorithm Ascending primes step by step in the JavaScript programming language
You may also check:How to resolve the algorithm String length step by step in the HolyC programming language
You may also check:How to resolve the algorithm Loops/While step by step in the 68000 Assembly programming language
You may also check:How to resolve the algorithm 100 doors step by step in the SAS programming language
You may also check:How to resolve the algorithm O'Halloran numbers step by step in the Wren programming language