Documentation

Mathlib.Algebra.Polynomial.Basic

Theory of univariate polynomials #

This file defines Polynomial R, the type of univariate polynomials over the semiring R, builds a semiring structure on it, and gives basic definitions that are expanded in other files in this directory.

Main definitions #

There are often two natural variants of lemmas involving sums, depending on whether one acts on the polynomials, or on the function. The naming convention is that one adds index when acting on the polynomials. For instance,

Implementation #

Polynomials are defined using R[ℕ], where R is a semiring. The variable X commutes with every polynomial p: lemma X_mul proves the identity X * p = p * X. The relationship to R[ℕ] is through a structure to make polynomials irreducible from the point of view of the kernel. Most operations are irreducible since Lean cannot compute anyway with AddMonoidAlgebra. There are two exceptions that we make semireducible:

The raw implementation of the equivalence between R[X] and R[ℕ] is done through ofFinsupp and toFinsupp (or, equivalently, rcases p when p is a polynomial gives an element q of R[ℕ], and conversely ⟨q⟩ gives back p). The equivalence is also registered as a ring equiv in Polynomial.toFinsuppIso. These should in general not be used once the basic API for polynomials is constructed.

structure Polynomial (R : Type u_1) [Semiring R] :
Type u_1

Polynomial R is the type of univariate polynomials over R, denoted as R[X] within the Polynomial namespace.

Polynomials should be seen as (semi-)rings with the additional constructor X. The embedding from R is called C.

  • ofFinsupp :: (
    • toFinsupp : AddMonoidAlgebra R

      The coefficients ℕ →₀ R of a polynomial in R[X].

  • )
Instances For
    def Polynomial.«term_[X]» :
    Lean.TrailingParserDescr

    Polynomial R is the type of univariate polynomials over R, denoted as R[X] within the Polynomial namespace.

    Polynomials should be seen as (semi-)rings with the additional constructor X. The embedding from R is called C.

    Instances For
      theorem Polynomial.forall_iff_forall_finsupp {R : Type u} [Semiring R] (P : Polynomial RProp) :
      (∀ (p : Polynomial R), P p) ∀ (q : AddMonoidAlgebra R ), P { toFinsupp := q }
      theorem Polynomial.exists_iff_exists_finsupp {R : Type u} [Semiring R] (P : Polynomial RProp) :
      (∃ (p : Polynomial R), P p) ∃ (q : AddMonoidAlgebra R ), P { toFinsupp := q }
      @[simp]
      theorem Polynomial.eta {R : Type u} [Semiring R] (f : Polynomial R) :
      { toFinsupp := f.toFinsupp } = f

      Conversions to and from AddMonoidAlgebra #

      Since R[X] is not defeq to R[ℕ], but instead is a structure wrapping it, we have to copy across all the arithmetic operators manually, along with the lemmas about how they unfold around Polynomial.ofFinsupp and Polynomial.toFinsupp.

      @[implicit_reducible]
      noncomputable instance Polynomial.instZero {R : Type u} [Semiring R] :
      Zero (Polynomial R)
      @[implicit_reducible]
      noncomputable instance Polynomial.instOne {R : Type u} [Semiring R] :
      One (Polynomial R)
      @[implicit_reducible]
      noncomputable instance Polynomial.instAdd {R : Type u} [Semiring R] :
      Add (Polynomial R)
      @[implicit_reducible]
      noncomputable instance Polynomial.instNeg {R : Type u} [Ring R] :
      Neg (Polynomial R)
      @[implicit_reducible]
      noncomputable instance Polynomial.instSub {R : Type u} [Ring R] :
      Sub (Polynomial R)
      @[implicit_reducible]
      noncomputable instance Polynomial.instMul {R : Type u} [Semiring R] :
      Mul (Polynomial R)
      @[implicit_reducible]
      noncomputable instance Polynomial.instNSMul {R : Type u} [Semiring R] :
      SMul (Polynomial R)
      @[implicit_reducible]
      noncomputable instance Polynomial.smulZeroClass {R : Type u} [Semiring R] {S : Type u_1} [SMulZeroClass S R] :
      @[implicit_reducible, instance 1]
      noncomputable instance Polynomial.pow {R : Type u} [Semiring R] :
      Pow (Polynomial R)
      @[simp]
      theorem Polynomial.ofFinsupp_zero {R : Type u} [Semiring R] :
      { toFinsupp := 0 } = 0
      @[simp]
      theorem Polynomial.ofFinsupp_one {R : Type u} [Semiring R] :
      { toFinsupp := 1 } = 1
      @[simp]
      theorem Polynomial.ofFinsupp_add {R : Type u} [Semiring R] {a b : AddMonoidAlgebra R } :
      { toFinsupp := a + b } = { toFinsupp := a } + { toFinsupp := b }
      @[simp]
      theorem Polynomial.ofFinsupp_neg {R : Type u} [Ring R] {a : AddMonoidAlgebra R } :
      { toFinsupp := -a } = -{ toFinsupp := a }
      @[simp]
      theorem Polynomial.ofFinsupp_sub {R : Type u} [Ring R] {a b : AddMonoidAlgebra R } :
      { toFinsupp := a - b } = { toFinsupp := a } - { toFinsupp := b }
      @[simp]
      theorem Polynomial.ofFinsupp_mul {R : Type u} [Semiring R] (a b : AddMonoidAlgebra R ) :
      { toFinsupp := a * b } = { toFinsupp := a } * { toFinsupp := b }
      @[simp]
      theorem Polynomial.ofFinsupp_nsmul {R : Type u} [Semiring R] (a : ) (b : AddMonoidAlgebra R ) :
      { toFinsupp := a b } = a { toFinsupp := b }
      @[simp]
      theorem Polynomial.ofFinsupp_smul {R : Type u} [Semiring R] {S : Type u_1} [SMulZeroClass S R] (a : S) (b : AddMonoidAlgebra R ) :
      { toFinsupp := a b } = a { toFinsupp := b }
      @[simp]
      theorem Polynomial.ofFinsupp_pow {R : Type u} [Semiring R] (a : AddMonoidAlgebra R ) (n : ) :
      { toFinsupp := a ^ n } = { toFinsupp := a } ^ n
      @[simp]
      theorem Polynomial.toFinsupp_add {R : Type u} [Semiring R] (a b : Polynomial R) :
      (a + b).toFinsupp = a.toFinsupp + b.toFinsupp
      @[simp]
      theorem Polynomial.toFinsupp_sub {R : Type u} [Ring R] (a b : Polynomial R) :
      (a - b).toFinsupp = a.toFinsupp - b.toFinsupp
      @[simp]
      theorem Polynomial.toFinsupp_mul {R : Type u} [Semiring R] (a b : Polynomial R) :
      (a * b).toFinsupp = a.toFinsupp * b.toFinsupp
      @[simp]
      theorem Polynomial.toFinsupp_nsmul {R : Type u} [Semiring R] (a : ) (b : Polynomial R) :
      (a b).toFinsupp = a b.toFinsupp
      @[simp]
      theorem Polynomial.toFinsupp_smul {R : Type u} [Semiring R] {S : Type u_1} [SMulZeroClass S R] (a : S) (b : Polynomial R) :
      (a b).toFinsupp = a b.toFinsupp
      @[simp]
      theorem Polynomial.toFinsupp_pow {R : Type u} [Semiring R] (a : Polynomial R) (n : ) :
      (a ^ n).toFinsupp = a.toFinsupp ^ n
      @[simp]
      theorem Polynomial.toFinsupp_inj {R : Type u} [Semiring R] {a b : Polynomial R} :
      a.toFinsupp = b.toFinsupp a = b
      @[simp]
      theorem Polynomial.toFinsupp_eq_zero {R : Type u} [Semiring R] {a : Polynomial R} :
      a.toFinsupp = 0 a = 0
      @[simp]
      theorem Polynomial.toFinsupp_eq_one {R : Type u} [Semiring R] {a : Polynomial R} :
      a.toFinsupp = 1 a = 1
      theorem Polynomial.ofFinsupp_inj {R : Type u} [Semiring R] {a b : AddMonoidAlgebra R } :
      { toFinsupp := a } = { toFinsupp := b } a = b

      A more convenient spelling of Polynomial.ofFinsupp.injEq in terms of Iff.

      @[simp]
      theorem Polynomial.ofFinsupp_eq_zero {R : Type u} [Semiring R] {a : AddMonoidAlgebra R } :
      { toFinsupp := a } = 0 a = 0
      @[simp]
      theorem Polynomial.ofFinsupp_eq_one {R : Type u} [Semiring R] {a : AddMonoidAlgebra R } :
      { toFinsupp := a } = 1 a = 1
      @[implicit_reducible]
      noncomputable instance Polynomial.inhabited {R : Type u} [Semiring R] :
      Inhabited (Polynomial R)
      @[implicit_reducible]
      noncomputable instance Polynomial.instNatCast {R : Type u} [Semiring R] :
      NatCast (Polynomial R)
      @[simp]
      theorem Polynomial.ofFinsupp_natCast {R : Type u} [Semiring R] (n : ) :
      { toFinsupp := n } = n
      @[simp]
      theorem Polynomial.toFinsupp_natCast {R : Type u} [Semiring R] (n : ) :
      (↑n).toFinsupp = n
      @[simp]
      theorem Polynomial.ofFinsupp_ofNat {R : Type u} [Semiring R] (n : ) [n.AtLeastTwo] :
      { toFinsupp := OfNat.ofNat n } = OfNat.ofNat n
      @[simp]
      theorem Polynomial.toFinsupp_ofNat {R : Type u} [Semiring R] (n : ) [n.AtLeastTwo] :
      (OfNat.ofNat n).toFinsupp = OfNat.ofNat n
      @[implicit_reducible]
      noncomputable instance Polynomial.semiring {R : Type u} [Semiring R] :
      @[implicit_reducible]
      noncomputable instance Polynomial.distribSMul {R : Type u} [Semiring R] {S : Type u_1} [DistribSMul S R] :
      @[implicit_reducible]
      noncomputable instance Polynomial.distribMulAction {R : Type u} [Semiring R] {S : Type u_1} [Monoid S] [DistribMulAction S R] :
      @[implicit_reducible]
      noncomputable instance Polynomial.module {R : Type u} [Semiring R] {S : Type u_1} [Semiring S] [Module S R] :
      instance Polynomial.smulCommClass {R : Type u} [Semiring R] {S₁ : Type u_1} {S₂ : Type u_2} [SMulZeroClass S₁ R] [SMulZeroClass S₂ R] [SMulCommClass S₁ S₂ R] :
      instance Polynomial.isScalarTower {R : Type u} [Semiring R] {S₁ : Type u_1} {S₂ : Type u_2} [SMul S₁ S₂] [SMulZeroClass S₁ R] [SMulZeroClass S₂ R] [IsScalarTower S₁ S₂ R] :
      @[implicit_reducible]
      noncomputable instance Polynomial.unique {R : Type u} [Semiring R] [Subsingleton R] :

      Ring isomorphism between R[X] and R[ℕ]. This is just an implementation detail, but it can be useful to transfer results from Finsupp to polynomials.

      Instances For
        @[simp]
        theorem Polynomial.toFinsuppIso_apply (R : Type u) [Semiring R] (self : Polynomial R) :
        (toFinsuppIso R) self = self.toFinsupp
        @[simp]
        theorem Polynomial.toFinsuppIso_symm_apply (R : Type u) [Semiring R] (toFinsupp : AddMonoidAlgebra R ) :
        (toFinsuppIso R).symm toFinsupp = { toFinsupp := toFinsupp }
        @[implicit_reducible]
        instance Polynomial.instDecidableEq (R : Type u) [Semiring R] [DecidableEq R] :
        DecidableEq (Polynomial R)

        Linear isomorphism between R[X] and R[ℕ]. This is just an implementation detail, but it can be useful to transfer results from Finsupp to polynomials.

        Instances For
          theorem Polynomial.ofFinsupp_sum {R : Type u} [Semiring R] {ι : Type u_1} (s : Finset ι) (f : ιAddMonoidAlgebra R ) :
          { toFinsupp := is, f i } = is, { toFinsupp := f i }
          theorem Polynomial.toFinsupp_sum {R : Type u} [Semiring R] {ι : Type u_1} (s : Finset ι) (f : ιPolynomial R) :
          (∑ is, f i).toFinsupp = is, (f i).toFinsupp
          def Polynomial.support {R : Type u} [Semiring R] :
          Polynomial RFinset

          The set of all n such that X^n has a non-zero coefficient.

          Instances For
            @[simp]
            theorem Polynomial.support_ofFinsupp {R : Type u} [Semiring R] (p : AddMonoidAlgebra R ) :
            { toFinsupp := p }.support = p.support
            @[simp]
            theorem Polynomial.support_zero {R : Type u} [Semiring R] :
            support 0 =
            @[simp]
            theorem Polynomial.support_eq_empty {R : Type u} [Semiring R] {p : Polynomial R} :
            p.support = p = 0
            @[simp]
            theorem Polynomial.support_nonempty {R : Type u} [Semiring R] {p : Polynomial R} :
            p.support.Nonempty p 0
            noncomputable def Polynomial.monomial {R : Type u} [Semiring R] (n : ) :

            monomial s a is the monomial a * X^s

            Instances For
              @[simp]
              theorem Polynomial.toFinsupp_monomial {R : Type u} [Semiring R] (n : ) (r : R) :
              @[simp]
              theorem Polynomial.ofFinsupp_single {R : Type u} [Semiring R] (n : ) (r : R) :
              { toFinsupp := AddMonoidAlgebra.single n r } = (monomial n) r
              @[simp]
              theorem Polynomial.monomial_zero_right {R : Type u} [Semiring R] (n : ) :
              (monomial n) 0 = 0
              @[deprecated map_add (since := "2025-11-15")]
              theorem Polynomial.monomial_add {R : Type u} [Semiring R] (n : ) (r s : R) :
              (monomial n) (r + s) = (monomial n) r + (monomial n) s
              theorem Polynomial.monomial_mul_monomial {R : Type u} [Semiring R] (n m : ) (r s : R) :
              (monomial n) r * (monomial m) s = (monomial (n + m)) (r * s)
              @[simp]
              theorem Polynomial.monomial_pow {R : Type u} [Semiring R] (n : ) (r : R) (k : ) :
              (monomial n) r ^ k = (monomial (n * k)) (r ^ k)
              theorem Polynomial.smul_monomial {R : Type u} [Semiring R] {S : Type u_1} [SMulZeroClass S R] (a : S) (n : ) (b : R) :
              a (monomial n) b = (monomial n) (a b)
              theorem Polynomial.monomial_injective {R : Type u} [Semiring R] (n : ) :
              Function.Injective (monomial n)
              @[simp]
              theorem Polynomial.monomial_eq_zero_iff {R : Type u} [Semiring R] (t : R) (n : ) :
              (monomial n) t = 0 t = 0
              theorem Polynomial.monomial_eq_monomial_iff {R : Type u} [Semiring R] {m n : } {a b : R} :
              (monomial m) a = (monomial n) b m = n a = b a = 0 b = 0
              theorem Polynomial.support_add {R : Type u} [Semiring R] {p q : Polynomial R} :
              (p + q).support p.support q.support
              noncomputable def Polynomial.C {R : Type u} [Semiring R] :

              C a is the constant polynomial a. C is provided as a ring homomorphism.

              Instances For
                @[simp]
                theorem Polynomial.monomial_zero_left {R : Type u} [Semiring R] (a : R) :
                (monomial 0) a = C a
                theorem Polynomial.C_0 {R : Type u} [Semiring R] :
                C 0 = 0
                theorem Polynomial.C_1 {R : Type u} [Semiring R] :
                C 1 = 1
                theorem Polynomial.C_ofNat {R : Type u} [Semiring R] (n : ) [n.AtLeastTwo] :
                C (OfNat.ofNat n) = OfNat.ofNat n
                theorem Polynomial.C_mul {R : Type u} {a b : R} [Semiring R] :
                C (a * b) = C a * C b
                theorem Polynomial.C_add {R : Type u} {a b : R} [Semiring R] :
                C (a + b) = C a + C b
                @[simp]
                theorem Polynomial.smul_C {R : Type u} [Semiring R] {S : Type u_1} [SMulZeroClass S R] (s : S) (r : R) :
                s C r = C (s r)
                theorem Polynomial.C_pow {R : Type u} {a : R} {n : } [Semiring R] :
                C (a ^ n) = C a ^ n
                theorem Polynomial.C_eq_natCast {R : Type u} [Semiring R] (n : ) :
                C n = n
                @[simp]
                theorem Polynomial.C_mul_monomial {R : Type u} {a b : R} {n : } [Semiring R] :
                C a * (monomial n) b = (monomial n) (a * b)
                @[simp]
                theorem Polynomial.monomial_mul_C {R : Type u} {a b : R} {n : } [Semiring R] :
                (monomial n) a * C b = (monomial n) (a * b)
                noncomputable def Polynomial.X {R : Type u} [Semiring R] :

                X is the polynomial variable (aka indeterminate).

                Instances For
                  theorem Polynomial.X_ne_C {R : Type u} [Semiring R] [Nontrivial R] (a : R) :
                  X C a
                  theorem Polynomial.X_mul {R : Type u} [Semiring R] {p : Polynomial R} :
                  X * p = p * X

                  X commutes with everything, even when the coefficients are noncommutative.

                  theorem Polynomial.X_pow_mul {R : Type u} [Semiring R] {p : Polynomial R} {n : } :
                  X ^ n * p = p * X ^ n
                  @[simp]
                  theorem Polynomial.X_mul_C {R : Type u} [Semiring R] (r : R) :
                  X * C r = C r * X

                  Prefer putting constants to the left of X.

                  This lemma is the loop-avoiding simp version of Polynomial.X_mul.

                  @[simp]
                  theorem Polynomial.X_pow_mul_C {R : Type u} [Semiring R] (r : R) (n : ) :
                  X ^ n * C r = C r * X ^ n

                  Prefer putting constants to the left of X ^ n.

                  This lemma is the loop-avoiding simp version of X_pow_mul.

                  theorem Polynomial.X_pow_mul_assoc {R : Type u} [Semiring R] {p q : Polynomial R} {n : } :
                  p * X ^ n * q = p * q * X ^ n
                  @[simp]
                  theorem Polynomial.X_pow_mul_assoc_C {R : Type u} [Semiring R] {p : Polynomial R} {n : } (r : R) :
                  p * X ^ n * C r = p * C r * X ^ n

                  Prefer putting constants to the left of X ^ n.

                  This lemma is the loop-avoiding simp version of X_pow_mul_assoc.

                  theorem Polynomial.commute_X_pow {R : Type u} [Semiring R] (p : Polynomial R) (n : ) :
                  Commute (X ^ n) p
                  @[simp]
                  theorem Polynomial.monomial_mul_X {R : Type u} [Semiring R] (n : ) (r : R) :
                  (monomial n) r * X = (monomial (n + 1)) r
                  @[simp]
                  theorem Polynomial.monomial_mul_X_pow {R : Type u} [Semiring R] (n : ) (r : R) (k : ) :
                  (monomial n) r * X ^ k = (monomial (n + k)) r
                  @[simp]
                  theorem Polynomial.X_mul_monomial {R : Type u} [Semiring R] (n : ) (r : R) :
                  X * (monomial n) r = (monomial (n + 1)) r
                  @[simp]
                  theorem Polynomial.X_pow_mul_monomial {R : Type u} [Semiring R] (k n : ) (r : R) :
                  X ^ k * (monomial n) r = (monomial (n + k)) r
                  def Polynomial.coeff {R : Type u} [Semiring R] :
                  Polynomial RR

                  coeff p n (often denoted p.coeff n) is the coefficient of X^n in p.

                  Instances For
                    @[simp]
                    theorem Polynomial.coeff_ofFinsupp {R : Type u} [Semiring R] (p : AddMonoidAlgebra R ) :
                    { toFinsupp := p }.coeff = p
                    theorem Polynomial.coeff_injective {R : Type u} [Semiring R] :
                    Function.Injective coeff
                    @[simp]
                    theorem Polynomial.coeff_inj {R : Type u} [Semiring R] {p q : Polynomial R} :
                    p.coeff = q.coeff p = q
                    theorem Polynomial.toFinsupp_apply {R : Type u} [Semiring R] (f : Polynomial R) (i : ) :
                    f.toFinsupp i = f.coeff i
                    theorem Polynomial.coeff_monomial {R : Type u} {a : R} {m n : } [Semiring R] :
                    ((monomial n) a).coeff m = if n = m then a else 0
                    @[simp]
                    theorem Polynomial.coeff_monomial_same {R : Type u} [Semiring R] (n : ) (c : R) :
                    ((monomial n) c).coeff n = c
                    theorem Polynomial.coeff_monomial_of_ne {R : Type u} [Semiring R] {m n : } (c : R) (h : m n) :
                    ((monomial n) c).coeff m = 0
                    @[simp]
                    theorem Polynomial.coeff_zero {R : Type u} [Semiring R] (n : ) :
                    coeff 0 n = 0
                    theorem Polynomial.coeff_one {R : Type u} [Semiring R] {n : } :
                    coeff 1 n = if n = 0 then 1 else 0
                    @[simp]
                    theorem Polynomial.coeff_one_zero {R : Type u} [Semiring R] :
                    coeff 1 0 = 1
                    @[simp]
                    theorem Polynomial.coeff_X_one {R : Type u} [Semiring R] :
                    X.coeff 1 = 1
                    @[simp]
                    theorem Polynomial.coeff_X_zero {R : Type u} [Semiring R] :
                    X.coeff 0 = 0
                    @[simp]
                    theorem Polynomial.coeff_monomial_succ {R : Type u} {a : R} {n : } [Semiring R] :
                    ((monomial (n + 1)) a).coeff 0 = 0
                    theorem Polynomial.coeff_X {R : Type u} {n : } [Semiring R] :
                    X.coeff n = if 1 = n then 1 else 0
                    theorem Polynomial.coeff_X_of_ne_one {R : Type u} [Semiring R] {n : } (hn : n 1) :
                    X.coeff n = 0
                    @[simp]
                    theorem Polynomial.mem_support_iff {R : Type u} {n : } [Semiring R] {p : Polynomial R} :
                    n p.support p.coeff n 0
                    theorem Polynomial.notMem_support_iff {R : Type u} {n : } [Semiring R] {p : Polynomial R} :
                    np.support p.coeff n = 0
                    theorem Polynomial.coeff_C {R : Type u} {a : R} {n : } [Semiring R] :
                    (C a).coeff n = if n = 0 then a else 0
                    @[simp]
                    theorem Polynomial.coeff_C_zero {R : Type u} {a : R} [Semiring R] :
                    (C a).coeff 0 = a
                    theorem Polynomial.coeff_C_ne_zero {R : Type u} {a : R} {n : } [Semiring R] (h : n 0) :
                    (C a).coeff n = 0
                    @[simp]
                    theorem Polynomial.coeff_C_succ {R : Type u} [Semiring R] {r : R} {n : } :
                    (C r).coeff (n + 1) = 0
                    @[simp]
                    theorem Polynomial.coeff_natCast_ite {R : Type u} {m n : } [Semiring R] :
                    (↑m).coeff n = (if n = 0 then m else 0)
                    @[simp]
                    theorem Polynomial.coeff_ofNat_zero {R : Type u} [Semiring R] (a : ) [a.AtLeastTwo] :
                    (OfNat.ofNat a).coeff 0 = OfNat.ofNat a
                    @[simp]
                    theorem Polynomial.coeff_ofNat_succ {R : Type u} [Semiring R] (a n : ) [h : a.AtLeastTwo] :
                    (OfNat.ofNat a).coeff (n + 1) = 0
                    theorem Polynomial.C_mul_X_pow_eq_monomial {R : Type u} {a : R} [Semiring R] {n : } :
                    C a * X ^ n = (monomial n) a
                    @[simp]
                    theorem Polynomial.toFinsupp_C_mul_X_pow {R : Type u} [Semiring R] (a : R) (n : ) :
                    theorem Polynomial.C_mul_X_eq_monomial {R : Type u} {a : R} [Semiring R] :
                    C a * X = (monomial 1) a
                    theorem Polynomial.C_injective {R : Type u} [Semiring R] :
                    Function.Injective C
                    @[simp]
                    theorem Polynomial.C_inj {R : Type u} {a b : R} [Semiring R] :
                    C a = C b a = b
                    @[simp]
                    theorem Polynomial.C_eq_zero {R : Type u} {a : R} [Semiring R] :
                    C a = 0 a = 0
                    theorem Polynomial.C_ne_zero {R : Type u} {a : R} [Semiring R] :
                    C a 0 a 0
                    theorem Polynomial.subsingleton_iff_subsingleton {R : Type u} [Semiring R] :
                    Subsingleton (Polynomial R) Subsingleton R
                    theorem Polynomial.forall_eq_iff_forall_eq {R : Type u} [Semiring R] :
                    (∀ (f g : Polynomial R), f = g) ∀ (a b : R), a = b
                    theorem Polynomial.ext_iff {R : Type u} [Semiring R] {p q : Polynomial R} :
                    p = q ∀ (n : ), p.coeff n = q.coeff n
                    theorem Polynomial.ext {R : Type u} [Semiring R] {p q : Polynomial R} :
                    (∀ (n : ), p.coeff n = q.coeff n)p = q

                    Monomials generate the additive monoid of polynomials.

                    theorem Polynomial.addHom_ext {R : Type u} [Semiring R] {M : Type u_1} [AddZeroClass M] {f g : Polynomial R →+ M} (h : ∀ (n : ) (a : R), f ((monomial n) a) = g ((monomial n) a)) :
                    f = g
                    theorem Polynomial.addHom_ext' {R : Type u} [Semiring R] {M : Type u_1} [AddZeroClass M] {f g : Polynomial R →+ M} (h : ∀ (n : ), f.comp (monomial n).toAddMonoidHom = g.comp (monomial n).toAddMonoidHom) :
                    f = g
                    theorem Polynomial.addHom_ext'_iff {R : Type u} [Semiring R] {M : Type u_1} [AddZeroClass M] {f g : Polynomial R →+ M} :
                    f = g ∀ (n : ), f.comp (monomial n).toAddMonoidHom = g.comp (monomial n).toAddMonoidHom
                    theorem Polynomial.lhom_ext' {R : Type u} [Semiring R] {M : Type u_1} [AddCommMonoid M] [Module R M] {f g : Polynomial R →ₗ[R] M} (h : ∀ (n : ), f ∘ₗ monomial n = g ∘ₗ monomial n) :
                    f = g
                    theorem Polynomial.lhom_ext'_iff {R : Type u} [Semiring R] {M : Type u_1} [AddCommMonoid M] [Module R M] {f g : Polynomial R →ₗ[R] M} :
                    f = g ∀ (n : ), f ∘ₗ monomial n = g ∘ₗ monomial n
                    theorem Polynomial.eq_zero_of_eq_zero {R : Type u} [Semiring R] (h : 0 = 1) (p : Polynomial R) :
                    p = 0
                    theorem Polynomial.support_monomial {R : Type u} [Semiring R] (n : ) {a : R} (H : a 0) :
                    ((monomial n) a).support = {n}
                    theorem Polynomial.support_monomial' {R : Type u} [Semiring R] (n : ) (a : R) :
                    ((monomial n) a).support {n}
                    theorem Polynomial.support_C {R : Type u} [Semiring R] {a : R} (h : a 0) :
                    (C a).support = {0}
                    theorem Polynomial.support_C_subset {R : Type u} [Semiring R] (a : R) :
                    (C a).support {0}
                    theorem Polynomial.support_C_mul_X {R : Type u} [Semiring R] {c : R} (h : c 0) :
                    (C c * X).support = {1}
                    theorem Polynomial.support_C_mul_X' {R : Type u} [Semiring R] (c : R) :
                    (C c * X).support {1}
                    theorem Polynomial.support_C_mul_X_pow {R : Type u} [Semiring R] (n : ) {c : R} (h : c 0) :
                    (C c * X ^ n).support = {n}
                    theorem Polynomial.support_C_mul_X_pow' {R : Type u} [Semiring R] (n : ) (c : R) :
                    (C c * X ^ n).support {n}
                    theorem Polynomial.support_binomial' {R : Type u} [Semiring R] (k m : ) (x y : R) :
                    (C x * X ^ k + C y * X ^ m).support {k, m}
                    theorem Polynomial.support_trinomial' {R : Type u} [Semiring R] (k m n : ) (x y z : R) :
                    (C x * X ^ k + C y * X ^ m + C z * X ^ n).support {k, m, n}
                    theorem Polynomial.X_pow_eq_monomial {R : Type u} [Semiring R] (n : ) :
                    X ^ n = (monomial n) 1
                    theorem Polynomial.smul_X_eq_monomial {R : Type u} {a : R} [Semiring R] {n : } :
                    a X ^ n = (monomial n) a
                    theorem Polynomial.support_X_pow {R : Type u} [Semiring R] (H : ¬1 = 0) (n : ) :
                    (X ^ n).support = {n}
                    theorem Polynomial.support_X_empty {R : Type u} [Semiring R] (H : 1 = 0) :
                    X.support =
                    theorem Polynomial.support_X {R : Type u} [Semiring R] (H : ¬1 = 0) :
                    X.support = {1}
                    theorem Polynomial.monomial_left_inj {R : Type u} [Semiring R] {a : R} (ha : a 0) {i j : } :
                    (monomial i) a = (monomial j) a i = j
                    theorem Polynomial.binomial_eq_binomial {R : Type u} [Semiring R] {k l m n : } {u v : R} (hu : u 0) (hv : v 0) :
                    C u * X ^ k + C v * X ^ l = C u * X ^ m + C v * X ^ n k = m l = n u = v k = n l = m u + v = 0 k = l m = n
                    theorem Polynomial.natCast_mul {R : Type u} [Semiring R] (n : ) (p : Polynomial R) :
                    n * p = n p
                    def Polynomial.sum {R : Type u} [Semiring R] {S : Type u_1} [AddCommMonoid S] (p : Polynomial R) (f : RS) :
                    S

                    Summing the values of a function applied to the coefficients of a polynomial

                    Instances For
                      theorem Polynomial.sum_def {R : Type u} [Semiring R] {S : Type u_1} [AddCommMonoid S] (p : Polynomial R) (f : RS) :
                      p.sum f = np.support, f n (p.coeff n)
                      theorem Polynomial.sum_eq_of_subset {R : Type u} [Semiring R] {S : Type u_1} [AddCommMonoid S] {p : Polynomial R} (f : RS) (hf : ∀ (i : ), f i 0 = 0) {s : Finset } (hs : p.support s) :
                      p.sum f = ns, f n (p.coeff n)
                      theorem Polynomial.mul_eq_sum_sum {R : Type u} [Semiring R] {p q : Polynomial R} :
                      p * q = ip.support, q.sum fun (j : ) (a : R) => (monomial (i + j)) (p.coeff i * a)

                      Expressing the product of two polynomials as a double sum.

                      @[simp]
                      theorem Polynomial.sum_zero_index {R : Type u} [Semiring R] {S : Type u_1} [AddCommMonoid S] (f : RS) :
                      sum 0 f = 0
                      @[simp]
                      theorem Polynomial.sum_monomial_index {R : Type u} [Semiring R] {S : Type u_1} [AddCommMonoid S] {n : } (a : R) (f : RS) (hf : f n 0 = 0) :
                      ((monomial n) a).sum f = f n a
                      @[simp]
                      theorem Polynomial.sum_C_index {R : Type u} [Semiring R] {a : R} {β : Type u_1} [AddCommMonoid β] {f : Rβ} (h : f 0 0 = 0) :
                      (C a).sum f = f 0 a
                      @[simp]
                      theorem Polynomial.sum_X_index {R : Type u} [Semiring R] {S : Type u_1} [AddCommMonoid S] {f : RS} (hf : f 1 0 = 0) :
                      X.sum f = f 1 1
                      theorem Polynomial.sum_add_index {R : Type u} [Semiring R] {S : Type u_1} [AddCommMonoid S] (p q : Polynomial R) (f : RS) (hf : ∀ (i : ), f i 0 = 0) (h_add : ∀ (a : ) (b₁ b₂ : R), f a (b₁ + b₂) = f a b₁ + f a b₂) :
                      (p + q).sum f = p.sum f + q.sum f
                      theorem Polynomial.sum_add' {R : Type u} [Semiring R] {S : Type u_1} [AddCommMonoid S] (p : Polynomial R) (f g : RS) :
                      p.sum (f + g) = p.sum f + p.sum g
                      theorem Polynomial.sum_add {R : Type u} [Semiring R] {S : Type u_1} [AddCommMonoid S] (p : Polynomial R) (f g : RS) :
                      (p.sum fun (n : ) (x : R) => f n x + g n x) = p.sum f + p.sum g
                      theorem Polynomial.sum_smul_index {R : Type u} [Semiring R] {S : Type u_1} [AddCommMonoid S] (p : Polynomial R) (b : R) (f : RS) (hf : ∀ (i : ), f i 0 = 0) :
                      (b p).sum f = p.sum fun (n : ) (a : R) => f n (b * a)
                      theorem Polynomial.sum_smul_index' {R : Type u} [Semiring R] {S : Type u_1} {T : Type u_2} [DistribSMul T R] [AddCommMonoid S] (p : Polynomial R) (b : T) (f : RS) (hf : ∀ (i : ), f i 0 = 0) :
                      (b p).sum f = p.sum fun (n : ) (a : R) => f n (b a)
                      theorem Polynomial.smul_sum {R : Type u} [Semiring R] {S : Type u_1} {T : Type u_2} [AddCommMonoid S] [DistribSMul T S] (p : Polynomial R) (b : T) (f : RS) :
                      b p.sum f = p.sum fun (n : ) (a : R) => b f n a
                      @[simp]
                      theorem Polynomial.sum_monomial_eq {R : Type u} [Semiring R] (p : Polynomial R) :
                      (p.sum fun (n : ) (a : R) => (monomial n) a) = p
                      theorem Polynomial.sum_C_mul_X_pow_eq {R : Type u} [Semiring R] (p : Polynomial R) :
                      (p.sum fun (n : ) (a : R) => C a * X ^ n) = p
                      theorem Polynomial.induction_on {R : Type u} [Semiring R] {motive : Polynomial RProp} (p : Polynomial R) (C : ∀ (a : R), motive (C a)) (add : ∀ (p q : Polynomial R), motive pmotive qmotive (p + q)) (monomial : ∀ (n : ) (a : R), motive (Polynomial.C a * X ^ n)motive (Polynomial.C a * X ^ (n + 1))) :
                      motive p
                      theorem Polynomial.induction_on' {R : Type u} [Semiring R] {motive : Polynomial RProp} (p : Polynomial R) (add : ∀ (p q : Polynomial R), motive pmotive qmotive (p + q)) (monomial : ∀ (n : ) (a : R), motive ((monomial n) a)) :
                      motive p

                      To prove something about polynomials, it suffices to show the condition is closed under taking sums, and it holds for monomials.

                      @[irreducible]
                      noncomputable def Polynomial.erase {R : Type u_1} [Semiring R] (n : ) :

                      erase p n is the polynomial p in which the X^n term has been erased.

                      Instances For
                        theorem Polynomial.erase_def {R : Type u_1} [Semiring R] (n : ) (x✝ : Polynomial R) :
                        erase n x✝ = match x✝ with | { toFinsupp := p } => { toFinsupp := AddMonoidAlgebra.erase n p }
                        @[simp]
                        theorem Polynomial.ofFinsupp_erase {R : Type u} [Semiring R] (p : AddMonoidAlgebra R ) (n : ) :
                        { toFinsupp := AddMonoidAlgebra.erase n p } = erase n { toFinsupp := p }
                        @[simp]
                        theorem Polynomial.support_erase {R : Type u} [Semiring R] (p : Polynomial R) (n : ) :
                        theorem Polynomial.monomial_add_erase {R : Type u} [Semiring R] (p : Polynomial R) (n : ) :
                        (monomial n) (p.coeff n) + erase n p = p
                        theorem Polynomial.coeff_erase {R : Type u} [Semiring R] (p : Polynomial R) (n i : ) :
                        (erase n p).coeff i = if i = n then 0 else p.coeff i
                        @[simp]
                        theorem Polynomial.erase_zero {R : Type u} [Semiring R] (n : ) :
                        erase n 0 = 0
                        @[simp]
                        theorem Polynomial.erase_monomial {R : Type u} [Semiring R] {n : } {a : R} :
                        erase n ((monomial n) a) = 0
                        @[simp]
                        theorem Polynomial.erase_same {R : Type u} [Semiring R] (p : Polynomial R) (n : ) :
                        (erase n p).coeff n = 0
                        @[simp]
                        theorem Polynomial.erase_ne {R : Type u} [Semiring R] (p : Polynomial R) (n i : ) (h : i n) :
                        (erase n p).coeff i = p.coeff i
                        noncomputable def Polynomial.update {R : Type u} [Semiring R] (p : Polynomial R) (n : ) (a : R) :

                        Replace the coefficient of a p : R[X] at a given degree n : ℕ by a given value a : R. If a = 0, this is equal to p.erase n If p.natDegree < n and a ≠ 0, this increases the degree to n.

                        Instances For
                          theorem Polynomial.coeff_update {R : Type u} [Semiring R] (p : Polynomial R) (n : ) (a : R) :
                          theorem Polynomial.coeff_update_apply {R : Type u} [Semiring R] (p : Polynomial R) (n : ) (a : R) (i : ) :
                          (p.update n a).coeff i = if i = n then a else p.coeff i
                          @[simp]
                          theorem Polynomial.coeff_update_same {R : Type u} [Semiring R] (p : Polynomial R) (n : ) (a : R) :
                          (p.update n a).coeff n = a
                          theorem Polynomial.coeff_update_ne {R : Type u} [Semiring R] (p : Polynomial R) {n : } (a : R) {i : } (h : i n) :
                          (p.update n a).coeff i = p.coeff i
                          @[simp]
                          theorem Polynomial.update_zero_eq_erase {R : Type u} [Semiring R] (p : Polynomial R) (n : ) :
                          p.update n 0 = erase n p
                          theorem Polynomial.support_update {R : Type u} [Semiring R] (p : Polynomial R) (n : ) (a : R) [Decidable (a = 0)] :
                          (p.update n a).support = if a = 0 then p.support.erase n else insert n p.support
                          theorem Polynomial.support_update_ne_zero {R : Type u} [Semiring R] (p : Polynomial R) (n : ) {a : R} (ha : a 0) :
                          (p.update n a).support = insert n p.support
                          noncomputable def Polynomial.coeffs {R : Type u} [Semiring R] (p : Polynomial R) :

                          The finset of nonzero coefficients of a polynomial.

                          Instances For
                            @[simp]
                            theorem Polynomial.coeffs_zero {R : Type u} [Semiring R] :
                            coeffs 0 =
                            theorem Polynomial.mem_coeffs_iff {R : Type u} [Semiring R] {p : Polynomial R} {c : R} :
                            c p.coeffs np.support, c = p.coeff n
                            theorem Polynomial.coeff_mem_coeffs {R : Type u} [Semiring R] {p : Polynomial R} {n : } (h : p.coeff n 0) :
                            p.coeff n p.coeffs
                            @[simp]
                            theorem Polynomial.coeffs_empty_iff {R : Type u} [Semiring R] {p : Polynomial R} :
                            p.coeffs = p = 0
                            @[simp]
                            theorem Polynomial.coeffs_nonempty_iff {R : Type u} [Semiring R] {p : Polynomial R} :
                            p.coeffs.Nonempty p 0
                            theorem Polynomial.coeffs_monomial {R : Type u} [Semiring R] (n : ) {c : R} (hc : c 0) :
                            ((monomial n) c).coeffs = {c}
                            @[implicit_reducible]
                            noncomputable instance Polynomial.commSemiring {R : Type u} [CommSemiring R] :
                            @[implicit_reducible]
                            noncomputable instance Polynomial.instZSMul {R : Type u} [Ring R] :
                            SMul (Polynomial R)
                            @[simp]
                            theorem Polynomial.ofFinsupp_zsmul {R : Type u} [Ring R] (a : ) (b : AddMonoidAlgebra R ) :
                            { toFinsupp := a b } = a { toFinsupp := b }
                            @[simp]
                            theorem Polynomial.toFinsupp_zsmul {R : Type u} [Ring R] (a : ) (b : Polynomial R) :
                            (a b).toFinsupp = a b.toFinsupp
                            @[implicit_reducible]
                            noncomputable instance Polynomial.instIntCast {R : Type u} [Ring R] :
                            IntCast (Polynomial R)
                            @[simp]
                            theorem Polynomial.ofFinsupp_intCast {R : Type u} [Ring R] (z : ) :
                            { toFinsupp := z } = z
                            @[simp]
                            theorem Polynomial.toFinsupp_intCast {R : Type u} [Ring R] (z : ) :
                            (↑z).toFinsupp = z
                            @[implicit_reducible]
                            noncomputable instance Polynomial.ring {R : Type u} [Ring R] :
                            @[simp]
                            theorem Polynomial.coeff_neg {R : Type u} [Ring R] (p : Polynomial R) (n : ) :
                            (-p).coeff n = -p.coeff n
                            @[simp]
                            theorem Polynomial.coeff_sub {R : Type u} [Ring R] (p q : Polynomial R) (n : ) :
                            (p - q).coeff n = p.coeff n - q.coeff n
                            @[simp]
                            theorem Polynomial.monomial_neg {R : Type u} [Ring R] (n : ) (a : R) :
                            (monomial n) (-a) = -(monomial n) a
                            theorem Polynomial.monomial_sub {R : Type u} {a b : R} [Ring R] (n : ) :
                            (monomial n) (a - b) = (monomial n) a - (monomial n) b
                            @[simp]
                            theorem Polynomial.support_neg {R : Type u} [Ring R] {p : Polynomial R} :
                            theorem Polynomial.C_eq_intCast {R : Type u} [Ring R] (n : ) :
                            C n = n
                            theorem Polynomial.C_neg {R : Type u} {a : R} [Ring R] :
                            C (-a) = -C a
                            theorem Polynomial.C_sub {R : Type u} {a b : R} [Ring R] :
                            C (a - b) = C a - C b
                            @[implicit_reducible]
                            noncomputable instance Polynomial.commRing {R : Type u} [CommRing R] :
                            @[simp]
                            theorem Polynomial.X_ne_zero {R : Type u} [Semiring R] [Nontrivial R] :
                            X 0

                            See also Polynomial.isCancelMulZero_iff: in order for R[X] to have cancellative multiplication (stronger than NoZeroDivisors in general, but equivalent if R is a ring), R must have both cancellative multiplication and cancellative addition.

                            theorem Polynomial.nnqsmul_eq_C_mul {R : Type u} [DivisionSemiring R] (q : ℚ≥0) (f : Polynomial R) :
                            q f = C q * f
                            theorem Polynomial.qsmul_eq_C_mul {R : Type u} [DivisionRing R] (a : ) (f : Polynomial R) :
                            a f = C a * f
                            noncomputable def Polynomial.ofMultiset {R : Type u} [CommRing R] :

                            The map sending a collection of roots into a polynomial, as a morphism.

                            Instances For
                              @[simp]
                              theorem Polynomial.ofMultiset_apply {R : Type u} [CommRing R] (s : Multiset R) :
                              ofMultiset s = (Multiset.map (fun (a : R) => X - C a) s).prod
                              @[implicit_reducible]
                              instance Polynomial.repr {R : Type u} [Semiring R] [Repr R] [DecidableEq R] :
                              Repr (Polynomial R)