Welcome to OStack Knowledge Sharing Community for programmer and developer-Open, Learning and Share
Welcome To Ask or Share your Answers For Others

Categories

0 votes
3.8k views
in Technique[技术] by (71.8m points)

python - Numpy matrix multiplication gives slightly different result in multiprocessing pool

I want to compute the numerical gradient of a function in parallel. However, I realized that evaluating my function in a multiprocessing pool gives me a different result than before. It seems like the matrix multiplication is different. I reconstructed the problem in the code below. For a smaller matrix size like (5,2) the results are equivalent. I am using Python 3.7.4 and Numpy 1.18.5. Is there a way to get equivalent results also for larger matrices?

import numpy as np
import multiprocessing as mp
import os

def setNumSubLibraryThreads(n):
    os.environ["OMP_NUM_THREADS"] = str(n)
    os.environ["OPENBLAS_NUM_THREADS"] = str(n)
    os.environ["MKL_NUM_THREADS"] = str(n)
    os.environ["VECLIB_MAXIMUM_THREADS"] = str(n)
    os.environ["NUMEXPR_NUM_THREADS"] = str(n)
    
def solveInParallel(f, args):
    cpu_count = mp.cpu_count()
    setNumSubLibraryThreads(1)
    p = mp.Pool(cpu_count)
    sol = np.array(p.map(f,args))
    p.close()
    setNumSubLibraryThreads(cpu_count)
    return sol

def g(args):
    mat1, mat2 = args
    return mat1.T.dot(mat2)

if __name__ == '__main__':
    np.random.seed(42)
    mat1 = np.random.random((1000,20))
    mat2 = np.random.random((1000,20))
    ref = mat1.T.dot(mat2)
    res_par_mat = solveInParallel(g, [(mat1, mat2)])
    print(res_par_mat[0] == ref)
np.__config__.show() gives                                                                                                                            
blas_mkl_info:
  NOT AVAILABLE
blis_info:
  NOT AVAILABLE
openblas_info:
    libraries = ['openblas', 'openblas']
    library_dirs = ['...']
    language = c
    define_macros = [('HAVE_CBLAS', None)]
blas_opt_info:
    libraries = ['openblas', 'openblas']
    library_dirs = [...]
    language = c
    define_macros = [('HAVE_CBLAS', None)]
lapack_mkl_info:
  NOT AVAILABLE
openblas_lapack_info:
    libraries = ['openblas', 'openblas']
    library_dirs = [...]
    language = c
    define_macros = [('HAVE_CBLAS', None)]
lapack_opt_info:
    libraries = ['openblas', 'openblas']
    library_dirs = [...]
    language = c
    define_macros = [('HAVE_CBLAS', None)]

与恶龙缠斗过久,自身亦成为恶龙;凝视深渊过久,深渊将回以凝视…
Welcome To Ask or Share your Answers For Others

1 Answer

0 votes
by (71.8m points)

Seems like I have to live with that round-off error.

import numpy as np
import multiprocessing as mp
from threadpoolctl import threadpool_limits

    
def solveInParallel(f, args):
    cpu_count = mp.cpu_count()
    with threadpool_limits(limits=1, user_api='blas'):
        p = mp.Pool(cpu_count)
        sol = np.array(p.map(f,args))
        p.close()
    return sol

def g2(args):
    mat1, mat2 = args
    return mat1.T.dot(mat2)

if __name__ == '__main__':
    np.random.seed(42)
    mat1 = np.random.random((1000,20))
    mat2 = np.random.random((1000,20))
    ref = mat1.T.dot(mat2)
    res = solveInParallel(g2, [(mat1, mat2)])
    print(res[0]== ref)

与恶龙缠斗过久,自身亦成为恶龙;凝视深渊过久,深渊将回以凝视…
Welcome to OStack Knowledge Sharing Community for programmer and developer-Open, Learning and Share
Click Here to Ask a Question

...