usertools.py 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293
  1. def monitor(f, input='print', output='print'):
  2. """
  3. Returns a wrapped copy of *f* that monitors evaluation by calling
  4. *input* with every input (*args*, *kwargs*) passed to *f* and
  5. *output* with every value returned from *f*. The default action
  6. (specify using the special string value ``'print'``) is to print
  7. inputs and outputs to stdout, along with the total evaluation
  8. count::
  9. >>> from mpmath import *
  10. >>> mp.dps = 5; mp.pretty = False
  11. >>> diff(monitor(exp), 1) # diff will eval f(x-h) and f(x+h)
  12. in 0 (mpf('0.99999999906867742538452148'),) {}
  13. out 0 mpf('2.7182818259274480055282064')
  14. in 1 (mpf('1.0000000009313225746154785'),) {}
  15. out 1 mpf('2.7182818309906424675501024')
  16. mpf('2.7182808')
  17. To disable either the input or the output handler, you may
  18. pass *None* as argument.
  19. Custom input and output handlers may be used e.g. to store
  20. results for later analysis::
  21. >>> mp.dps = 15
  22. >>> input = []
  23. >>> output = []
  24. >>> findroot(monitor(sin, input.append, output.append), 3.0)
  25. mpf('3.1415926535897932')
  26. >>> len(input) # Count number of evaluations
  27. 9
  28. >>> print(input[3]); print(output[3])
  29. ((mpf('3.1415076583334066'),), {})
  30. 8.49952562843408e-5
  31. >>> print(input[4]); print(output[4])
  32. ((mpf('3.1415928201669122'),), {})
  33. -1.66577118985331e-7
  34. """
  35. if not input:
  36. input = lambda v: None
  37. elif input == 'print':
  38. incount = [0]
  39. def input(value):
  40. args, kwargs = value
  41. print("in %s %r %r" % (incount[0], args, kwargs))
  42. incount[0] += 1
  43. if not output:
  44. output = lambda v: None
  45. elif output == 'print':
  46. outcount = [0]
  47. def output(value):
  48. print("out %s %r" % (outcount[0], value))
  49. outcount[0] += 1
  50. def f_monitored(*args, **kwargs):
  51. input((args, kwargs))
  52. v = f(*args, **kwargs)
  53. output(v)
  54. return v
  55. return f_monitored
  56. def timing(f, *args, **kwargs):
  57. """
  58. Returns time elapsed for evaluating ``f()``. Optionally arguments
  59. may be passed to time the execution of ``f(*args, **kwargs)``.
  60. If the first call is very quick, ``f`` is called
  61. repeatedly and the best time is returned.
  62. """
  63. once = kwargs.get('once')
  64. if 'once' in kwargs:
  65. del kwargs['once']
  66. if args or kwargs:
  67. if len(args) == 1 and not kwargs:
  68. arg = args[0]
  69. g = lambda: f(arg)
  70. else:
  71. g = lambda: f(*args, **kwargs)
  72. else:
  73. g = f
  74. from timeit import default_timer as clock
  75. t1=clock(); v=g(); t2=clock(); t=t2-t1
  76. if t > 0.05 or once:
  77. return t
  78. for i in range(3):
  79. t1=clock();
  80. # Evaluate multiple times because the timer function
  81. # has a significant overhead
  82. g();g();g();g();g();g();g();g();g();g()
  83. t2=clock()
  84. t=min(t,(t2-t1)/10)
  85. return t