{
  "Selected_candidate": {
    "pr_number": 2707,
    "pr_title": "remove practice of evaluating factorial as 0 for negative numbers",
    "pr_body": "it is infinity for negative integers and remains unevaluated for non-integers. fix gosper test, which had a missing term.\nfix string representations\n",
    "issue_id": 2703,
    "issue_title": "uncommon convention for factorials of negative numbers",
    "issue_body": "from the docstring of the factorial function:\n\n\"For the sake of convenience and simplicity of procedures using\nthis function it is defined for negative integers and returns\nzero in this case.\"\n\nBy convention (and by some definitions), n! is usually infinity for n < 0. This is consistent with the Gamma function identity n! = Gamma(n+1) and with the convention that binomial coefficients are 0 outside of Pascal's triangle. In fact, in the same module, we have, in the doc string for binomial coefficients:\n\n\"For the sake of convenience for negative 'k' this function\nwill return zero no matter what valued is the other argument.\"\n\nThis is incompatible with sympy's convention on factorials, given the definition of the binomial coefficients: C(n,k) = n!/(k!(n-k)!)\n\nFor example take \"2 choose -1\", C(2,-1) = 0. However, going to sympy's definition of the factorial, we would have 2!/((-1)! \\* 3!) == 1/0 == oo, which is obviously wrong. Switching over to the Gamma function gives us the correct answer.\n\nDoes anyone know what \"procedures using\" the factorial function rely on the \"convenience and simplicity\" of the incorrect convention n! = 0  for n < 0? It would nice to bring the factorial function to consistency with other parts of sympy (like the binomial coefficients) and the more common convention in the math and cs communities.\n",
    "issue_closed_at": "2014-01-05T13:28:55Z",
    "base_commit": "5879fa2318e853807a6cf63981f2bca441cceb9f",
    "changes": [
      {
        "file": "sympy/functions/combinatorial/factorials.py",
        "type": "function",
        "name": "_eval_simplify",
        "class_name": "CombinatorialFunction",
        "code": "def _eval_simplify(self, ratio, measure):\n        from sympy.simplify.simplify import combsimp\n        expr = combsimp(self)\n        if measure(expr) <= ratio*measure(self):\n            return expr\n        return self"
      },
      {
        "file": "sympy/functions/combinatorial/factorials.py",
        "type": "class",
        "name": "factorial",
        "code": "class factorial(CombinatorialFunction):\n    \"\"\"Implementation of factorial function over nonnegative integers.\n       For the sake of convenience and simplicity of procedures using\n       this function it is defined for negative integers and returns\n       zero in this case.\n\n       The factorial is very important in combinatorics where it gives\n       the number of ways in which 'n' objects can be permuted. It also\n       arises in calculus, probability, number theory etc.\n\n       There is strict relation of factorial with gamma function. In\n       fact n! = gamma(n+1) for nonnegative integers. Rewrite of this\n       kind is very useful in case of combinatorial simplification.\n\n       Computation of the factorial is done using two algorithms. For\n       small arguments naive product is evaluated. However for bigger\n       input algorithm Prime-Swing is used. It is the fastest algorithm\n       known and computes n! via prime factorization of special class\n       of numbers, called here the 'Swing Numbers'.\n\n       Examples\n       ========\n\n       >>> from sympy import Symbol, factorial\n       >>> n = Symbol('n', integer=True)\n\n       >>> factorial(-2)\n       0\n\n       >>> factorial(0)\n       1\n\n       >>> factorial(7)\n       5040\n\n       >>> factorial(n)\n       factorial(n)\n\n       >>> factorial(2*n)\n       factorial(2*n)\n\n       See Also\n       ========\n\n       factorial2, RisingFactorial, FallingFactorial\n    \"\"\"\n\n    nargs = 1\n\n    def fdiff(self, argindex=1):\n        if argindex == 1:\n            return C.gamma(self.args[0] + 1)*C.polygamma(0, self.args[0] + 1)\n        else:\n            raise ArgumentIndexError(self, argindex)\n\n    _small_swing = [\n        1, 1, 1, 3, 3, 15, 5, 35, 35, 315, 63, 693, 231, 3003, 429, 6435, 6435, 109395,\n        12155, 230945, 46189, 969969, 88179, 2028117, 676039, 16900975, 1300075,\n        35102025, 5014575, 145422675, 9694845, 300540195, 300540195\n    ]\n\n    @classmethod\n    def _swing(cls, n):\n        if n < 33:\n            return cls._small_swing[n]\n        else:\n            N, primes = int(_sqrt(n)), []\n\n            for prime in sieve.primerange(3, N + 1):\n                p, q = 1, n\n\n                while True:\n                    q //= prime\n\n                    if q > 0:\n                        if q & 1 == 1:\n                            p *= prime\n                    else:\n                        break\n\n                if p > 1:\n                    primes.append(p)\n\n            for prime in sieve.primerange(N + 1, n//3 + 1):\n                if (n // prime) & 1 == 1:\n                    primes.append(prime)\n\n            L_product = R_product = 1\n\n            for prime in sieve.primerange(n//2 + 1, n + 1):\n                L_product *= prime\n\n            for prime in primes:\n                R_product *= prime\n\n            return L_product*R_product\n\n    @classmethod\n    def _recursive(cls, n):\n        if n < 2:\n            return 1\n        else:\n            return (cls._recursive(n//2)**2)*cls._swing(n)\n\n    @classmethod\n    def eval(cls, n):\n        n = sympify(n)\n\n        if n.is_Number:\n            if n is S.Zero:\n                return S.One\n            elif n is S.Infinity:\n                return S.Infinity\n            elif n.is_Integer:\n                if n.is_negative:\n                    return S.Zero\n                else:\n                    n, result = n.p, 1\n\n                    if n < 20:\n                        for i in range(2, n + 1):\n                            result *= i\n                    else:\n                        N, bits = n, 0\n\n                        while N != 0:\n                            if N & 1 == 1:\n                                bits += 1\n\n                            N = N >> 1\n\n                        result = cls._recursive(n)*2**(n - bits)\n\n                    return C.Integer(result)\n\n        if n.is_negative:\n            return S.Zero\n\n    def _eval_rewrite_as_gamma(self, n):\n        return C.gamma(n + 1)\n\n    def _eval_is_integer(self):\n        return self.args[0].is_integer\n\n    def _eval_is_positive(self):\n        if self.args[0].is_integer and self.args[0].is_positive:\n            return True"
      },
      {
        "file": "sympy/functions/combinatorial/factorials.py",
        "type": "function",
        "name": "eval",
        "class_name": "binomial",
        "code": "def eval(cls, n, k):\n        n, k = map(sympify, (n, k))\n\n        if k.is_Number:\n            if k.is_Integer:\n                if k < 0:\n                    return S.Zero\n                elif k == 0 or n == k:\n                    return S.One\n                elif n.is_Integer and n >= 0:\n                    n, k = int(n), int(k)\n\n                    if k > n:\n                        return S.Zero\n                    elif k > n // 2:\n                        k = n - k\n\n                    M, result = int(_sqrt(n)), 1\n\n                    for prime in sieve.primerange(2, n + 1):\n                        if prime > n - k:\n                            result *= prime\n                        elif prime > n // 2:\n                            continue\n                        elif prime > M:\n                            if n % prime < k % prime:\n                                result *= prime\n                        else:\n                            N, K = n, k\n                            exp = a = 0\n\n                            while N > 0:\n                                a = int((N % prime) < (K % prime + a))\n                                N, K = N // prime, K // prime\n                                exp = a + exp\n\n                            if exp > 0:\n                                result *= prime**exp\n\n                    return C.Integer(result)\n                elif n.is_Number:\n                    result = n - k + 1\n                    for i in xrange(2, k + 1):\n                        result *= n - k + i\n                        result /= i\n                    return result\n\n        elif k.is_negative:\n            return S.Zero\n        elif (n - k).simplify().is_negative:\n            return S.Zero\n        else:\n            d = n - k\n\n            if d.is_Integer:\n                return cls.eval(n, d)"
      },
      {
        "file": "sympy/functions/combinatorial/factorials.py",
        "type": "function",
        "name": "eval",
        "class_name": "binomial",
        "code": "def eval(cls, n, k):\n        n, k = map(sympify, (n, k))\n\n        if k.is_Number:\n            if k.is_Integer:\n                if k < 0:\n                    return S.Zero\n                elif k == 0 or n == k:\n                    return S.One\n                elif n.is_Integer and n >= 0:\n                    n, k = int(n), int(k)\n\n                    if k > n:\n                        return S.Zero\n                    elif k > n // 2:\n                        k = n - k\n\n                    M, result = int(_sqrt(n)), 1\n\n                    for prime in sieve.primerange(2, n + 1):\n                        if prime > n - k:\n                            result *= prime\n                        elif prime > n // 2:\n                            continue\n                        elif prime > M:\n                            if n % prime < k % prime:\n                                result *= prime\n                        else:\n                            N, K = n, k\n                            exp = a = 0\n\n                            while N > 0:\n                                a = int((N % prime) < (K % prime + a))\n                                N, K = N // prime, K // prime\n                                exp = a + exp\n\n                            if exp > 0:\n                                result *= prime**exp\n\n                    return C.Integer(result)\n                elif n.is_Number:\n                    result = n - k + 1\n                    for i in xrange(2, k + 1):\n                        result *= n - k + i\n                        result /= i\n                    return result\n\n        elif k.is_negative:\n            return S.Zero\n        elif (n - k).simplify().is_negative:\n            return S.Zero\n        else:\n            d = n - k\n\n            if d.is_Integer:\n                return cls.eval(n, d)"
      }
    ]
  },
  "Justification": "Candidate E addresses a fundamental mathematical inconsistency in the way the factorial function handles negative numbers, which can directly impact the overall behavior of polynomial factorizations when the Knuth-Cohen bound is used instead of the Mignotte bound. Given that the CURRENT bug report involves a significant improvement to the polynomial factorization method, the fixes applied in Candidate E could shed light on similar underlying issues in the Polynomial module's handling of edge cases related to polynomial properties. The patch similarities also indicate a broader effort to align SymPy’s behavior with common mathematical conventions, which is very relevant to the CURRENT bug report’s goal of improving existing functionality.",
  "instance_id": "sympy__sympy-19254",
  "repo": "sympy/sympy",
  "created_at": "2020-05-04T13:38:13Z",
  "problem_statement": "sympy.polys.factortools.dmp_zz_mignotte_bound improvement\nThe method `dup_zz_mignotte_bound(f, K)` can be significantly improved by using the **Knuth-Cohen bound** instead. After our research with Prof. Ag.Akritas we have implemented the Knuth-Cohen bound among others, and compare them among dozens of polynomials with different degree, density and coefficients range. Considering the results and the feedback from Mr.Kalevi Suominen, our proposal is that the mignotte_bound should be replaced by the knuth-cohen bound.\r\nAlso, `dmp_zz_mignotte_bound(f, u, K)` for mutli-variants polynomials should be replaced appropriately.\n",
  "patch": "diff --git a/sympy/polys/factortools.py b/sympy/polys/factortools.py\n--- a/sympy/polys/factortools.py\n+++ b/sympy/polys/factortools.py\n@@ -124,13 +124,64 @@ def dmp_trial_division(f, factors, u, K):\n \n \n def dup_zz_mignotte_bound(f, K):\n-    \"\"\"Mignotte bound for univariate polynomials in `K[x]`. \"\"\"\n-    a = dup_max_norm(f, K)\n-    b = abs(dup_LC(f, K))\n-    n = dup_degree(f)\n+    \"\"\"\n+    The Knuth-Cohen variant of Mignotte bound for\n+    univariate polynomials in `K[x]`.\n \n-    return K.sqrt(K(n + 1))*2**n*a*b\n+    Examples\n+    ========\n+\n+    >>> from sympy.polys import ring, ZZ\n+    >>> R, x = ring(\"x\", ZZ)\n+\n+    >>> f = x**3 + 14*x**2 + 56*x + 64\n+    >>> R.dup_zz_mignotte_bound(f)\n+    152\n+\n+    By checking `factor(f)` we can see that max coeff is 8\n+\n+    Also consider a case that `f` is irreducible for example `f = 2*x**2 + 3*x + 4`\n+    To avoid a bug for these cases, we return the bound plus the max coefficient of `f`\n+\n+    >>> f = 2*x**2 + 3*x + 4\n+    >>> R.dup_zz_mignotte_bound(f)\n+    6\n+\n+    Lastly,To see the difference between the new and the old Mignotte bound\n+    consider the irreducible polynomial::\n+\n+    >>> f = 87*x**7 + 4*x**6 + 80*x**5 + 17*x**4 + 9*x**3 + 12*x**2 + 49*x + 26\n+    >>> R.dup_zz_mignotte_bound(f)\n+    744\n+\n+    The new Mignotte bound is 744 whereas the old one (SymPy 1.5.1) is 1937664.\n+\n+\n+    References\n+    ==========\n+\n+    ..[1] [Abbott2013]_\n+\n+    \"\"\"\n+    from sympy import binomial\n+\n+    d = dup_degree(f)\n+    delta = _ceil(d / 2)\n+    delta2 = _ceil(delta / 2)\n+\n+    # euclidean-norm\n+    eucl_norm = K.sqrt( sum( [cf**2 for cf in f] ) )\n+\n+    # biggest values of binomial coefficients (p. 538 of reference)\n+    t1 = binomial(delta - 1, delta2)\n+    t2 = binomial(delta - 1, delta2 - 1)\n+\n+    lc = K.abs(dup_LC(f, K))   # leading coefficient\n+    bound = t1 * eucl_norm + t2 * lc   # (p. 538 of reference)\n+    bound += dup_max_norm(f, K) # add max coeff for irreducible polys\n+    bound = _ceil(bound / 2) * 2   # round up to even integer\n \n+    return bound\n \n def dmp_zz_mignotte_bound(f, u, K):\n     \"\"\"Mignotte bound for multivariate polynomials in `K[X]`. \"\"\"\n"
}