From 30be930f0f11312c2dfd46eb94f6502df22cbbd2 Mon Sep 17 00:00:00 2001 From: jverzani Date: Wed, 16 Apr 2025 14:31:16 -0400 Subject: [PATCH 01/22] WIP --- quarto/ODEs/differential_equations.qmd | 20 +- quarto/ODEs/solve.qmd | 14 +- quarto/README-quarto.md | 12 +- quarto/_quarto.yml | 2 +- .../vector_fields.qmd | 305 ++++ quarto/references.bib | 22 + quarto/staging/Project.toml | 2 + quarto/staging/matrix-calculus-notes.html | 1278 +++++++++++++++++ quarto/staging/matrix-calculus-notes.qmd | 937 ++++++++++++ 9 files changed, 2576 insertions(+), 16 deletions(-) create mode 100644 quarto/staging/Project.toml create mode 100644 quarto/staging/matrix-calculus-notes.html create mode 100644 quarto/staging/matrix-calculus-notes.qmd diff --git a/quarto/ODEs/differential_equations.qmd b/quarto/ODEs/differential_equations.qmd index 082e324..9efc5c1 100644 --- a/quarto/ODEs/differential_equations.qmd +++ b/quarto/ODEs/differential_equations.qmd @@ -79,7 +79,7 @@ R(0) &= 0\\ \end{align*} $$ -In `Julia` we define these, `N` to model the total population, and `u0` to be the proportions. +In `Julia` we define these parameter values and `N` to model the total population and `u0` to be represent the proportions. ```{julia} @@ -94,7 +94,7 @@ An *estimated* set of values for $k$ and $b$ are $k=1/3$, coming from the averag Okay, the mathematical modeling is done; now we try to solve for the unknown functions using `DifferentialEquations`. -To warm up, if $b=0$ then $i'(t) = -k \cdot i(t)$ describes the infected. (There is no circulation of people in this case.) The solution would be achieved through: +To warm up, if $b=0$ then $i'(t) = -k \cdot i(t)$ describes the infected. (There is no circulation of people in this case.) This is a single ODE. The solution would be achieved through: ```{julia} @@ -102,10 +102,12 @@ To warm up, if $b=0$ then $i'(t) = -k \cdot i(t)$ describes the infected. (The k = 1/3 f(u,p,t) = -k * u # solving u′(t) = - k u(t) + +uᵢ0= I0/N time_span = (0.0, 20.0) -prob = ODEProblem(f, I0/N, time_span) -sol = solve(prob, Tsit5(), reltol=1e-8, abstol=1e-8) +prob = ODEProblem(f, uᵢ0, time_span) +sol = solve(prob, Tsit5(); reltol=1e-8, abstol=1e-8) plot(sol) ``` @@ -120,7 +122,7 @@ $$ \frac{di}{dt} = -k \cdot i(t) = F(i(t), k, t) $$ -where $F$ depends on the current value ($i$), a parameter ($k$), and the time ($t$). We did not utilize $p$ above for the parameter, as it was easy not to, but could have, and will in the following. The time variable $t$ does not appear by itself in our equation, so only `f(u, p, t) = -k * u` was used, `u` the generic name for a solution which in this case is $i$. +where $F$ depends on the current value ($i$), a parameter ($k$), and the time ($t$). We did not utilize $p$ above for the parameter, as it was easy not to, but could have, and will in the following. The time variable $t$ does not appear by itself in our equation, so only `f(u, p, t) = -k * u` was used, `u` the generic name for a solution which in this case was labeled with an $i$. The problem we set up needs an initial value (the $u0$) and a time span to solve over. Here we want time to model real time, so use floating point values. @@ -167,7 +169,7 @@ The `sir!` function has the trailing `!` indicating – by convention – it *mu ::: -With the update function defined, the problem is setup and a solution found with in the same manner: +With the update function defined, the problem is setup and a solution is found using the same manner as before: ```{julia} @@ -193,7 +195,7 @@ p = (k=1/2, b=2) # change b from 1/2 to 2 -- more daily contact prob = ODEProblem(sir!, u0, time_span, p) sol = solve(prob, Tsit5()) -plot(sol) +plot(sol; legend=:right) ``` The graphs are somewhat similar, but the steady state is reached much more quickly and nearly everyone became infected. @@ -252,7 +254,7 @@ end p ``` -The 3-dimensional graph with `plotly` can have its viewing angle adjusted with the mouse. When looking down on the $x-y$ plane, which code `b` and `k`, we can see the rapid growth along a line related to $b/k$. +(A 3-dimensional graph with `plotly` or `Makie` can have its viewing angle adjusted with the mouse. When looking down on the $x-y$ plane, which code `b` and `k`, we can see the rapid growth along a line related to $b/k$.) Smith and Moore point out that $k$ is roughly the reciprocal of the number of days an individual is sick enough to infect others. This can be estimated during a breakout. However, they go on to note that there is no direct way to observe $b$, but there is an indirect way. @@ -382,7 +384,7 @@ SOL = solve(trajectory_problem, Tsit5(); p = ps, callback=cb) plot(t -> SOL(t)[1], t -> SOL(t)[2], TSPAN...; legend=false) ``` -Finally, we note that the `ModelingToolkit` package provides symbolic-numeric computing. This allows the equations to be set up symbolically, as in `SymPy` before being passed off to `DifferentialEquations` to solve numerically. The above example with no wind resistance could be translated into the following: +Finally, we note that the `ModelingToolkit` package provides symbolic-numeric computing. This allows the equations to be set up symbolically, as has been illustrated with `SymPy`, before being passed off to `DifferentialEquations` to solve numerically. The above example with no wind resistance could be translated into the following: ```{julia} diff --git a/quarto/ODEs/solve.qmd b/quarto/ODEs/solve.qmd index ec4e3c6..d9fc00d 100644 --- a/quarto/ODEs/solve.qmd +++ b/quarto/ODEs/solve.qmd @@ -118,10 +118,10 @@ function solve(prob::Problem, alg::EulerMethod) end ``` -The post has a more elegant means to unpack the parameters from the structures, but for each of the above, the parameters are unpacked, and then the corresponding algorithm employed. As of version `v1.7` of `Julia`, the syntax `(;g,y0,v0,tspan) = prob` could also be employed. +The post has a more elegant means to unpack the parameters from the structures, but for each of the above, the parameters are unpacked using the dot notation for `getproperty`, and then the corresponding algorithm employed. As of version `v1.7` of `Julia`, the syntax `(;g,y0,v0,tspan) = prob` could also have been employed. -The exact formulas, `y(t) = y0 + v0*(t - t0) - g*(t - t0)^2/2` and `v(t) = v0 - g*(t - t0)`, follow from well-known physics formulas. Each answer is wrapped in a `Solution` type so that the answers found can be easily extracted in a uniform manner. +The exact answers, `y(t) = y0 + v0*(t - t0) - g*(t - t0)^2/2` and `v(t) = v0 - g*(t - t0)`, follow from well-known physics formulas for constant-acceleration motion. Each answer is wrapped in a `Solution` type so that the answers found can be easily extracted in a uniform manner. For example, plots of each can be obtained through: @@ -138,7 +138,9 @@ plot!(sol_exact.t, sol_exact.y; label="exact solution", ls=:auto) title!("On the Earth"; xlabel="t", legend=:bottomleft) ``` -Following the post, since the time step `dt = 0.1` is not small enough, the error of the Euler method is rather large. Next we change the algorithm parameter, `dt`, to be smaller: +Following the post, since the time step `dt = 0.1` is not small enough, the error of the Euler method is readily identified. + +Next we change the algorithm parameter, `dt`, to be smaller: ```{julia} @@ -155,7 +157,7 @@ title!("On the Earth"; xlabel="t", legend=:bottomleft) It is worth noting that only the first line is modified, and only the method requires modification. -Were the moon to be considered, the gravitational constant would need adjustment. This parameter is part of the problem, not the solution algorithm. +Were the moon to be considered, the gravitational constant would need adjustment. This parameter is a property of the problem, not the solution algorithm, as `dt` is. Such adjustments are made by passing different values to the `Problem` constructor: @@ -175,7 +177,9 @@ title!("On the Moon"; xlabel="t", legend=:bottomleft) The code above also adjusts the time span in addition to the graviational constant. The algorithm for exact formula is set to use the `dt` value used in the `euler` formula, for easier comparison. Otherwise, outside of the labels, the patterns are the same. Only those things that need changing are changed, the rest comes from defaults. -The above shows the benefits of using a common interface. Next, the post illustrates how *other* authors could extend this code, simply by adding a *new* `solve` method. For example, +The above shows the benefits of using a common interface. + +Next, the post illustrates how *other* authors could extend this code, simply by adding a *new* `solve` method. For example, a sympletic method conserves a quantity, so can track long-term evolution without drift. ```{julia} diff --git a/quarto/README-quarto.md b/quarto/README-quarto.md index e133c7e..6c4a801 100644 --- a/quarto/README-quarto.md +++ b/quarto/README-quarto.md @@ -4,12 +4,22 @@ Short cut. Run first command until happy, then run second to publish ``` quarto render -#julia adjust_plotly.jl # <-- no longer needed # maybe git config --global http.postBuffer 157286400 quarto publish gh-pages --no-render ``` +But better to + +``` +quarto render +# commit changes and push +# fix typos +quarto render +quarto publish gh-pages --no-render +``` + + To compile the pages through quarto diff --git a/quarto/_quarto.yml b/quarto/_quarto.yml index b41a532..766b3a0 100644 --- a/quarto/_quarto.yml +++ b/quarto/_quarto.yml @@ -1,4 +1,4 @@ -version: "0.23" +version: "0.24" engine: julia project: diff --git a/quarto/differentiable_vector_calculus/vector_fields.qmd b/quarto/differentiable_vector_calculus/vector_fields.qmd index bc270f3..b8172b6 100644 --- a/quarto/differentiable_vector_calculus/vector_fields.qmd +++ b/quarto/differentiable_vector_calculus/vector_fields.qmd @@ -839,6 +839,311 @@ Taking $\partial/\partial{a_i}$ gives equations $2a_i\sigma_i^2 + \lambda = 0$, For the special case of a common variance, $\sigma_i=\sigma$, the above simplifies to $a_i = 1/n$ and the estimator is $\sum X_i/n$, the familiar sample mean, $\bar{X}$. +##### Example: The mean value theorem + +[Perturbing the Mean Value Theorem: Implicit Functions, the Morse Lemma, and Beyond](https://www.jstor.org/stable/48661587) by Lowry-Duda, and Wheeler presents an interesting take on the mean-value theorem by asking if the endpoint $b$ moves continuously, does the value $c$ move continuously? + +Fix the left-hand endpoint, $a_0$, and consider: + +$$ +F(b,c) = \frac{f(b) - f(a_0)}{b-a_0} - f'(c). +$$ + +Solutions to $F(b,c)=0$ satisfy the mean value theorem for $f$. +Suppose $(b_0,c_0)$ is one such solution. +By using the implicit function theorem, the question of finding a $C(b)$ such that $C$ is continuous near $b_0$ and satisfied $F(b, C(b)) =0$ for $b$ near $b_0$ can be characterized. + +To analyze this question, Lowry-Duda and Wheeler fix a set of points $a_0 = 0$, $b_0=3$ and consider functions $f$ with $f(a_0) = f(b_0) = 0$. Similar to how Rolle's theorem easily proves the mean value theorem, this choice imposes no loss of generality. + +Suppose further that $c_0 = 1$, where $c_0$ solves the mean value theorem: + +$$ +f'(c_0) = \frac{f(b_0) - f(a_0)}{b_0 - a_0}. +$$ + + +Again, this is no loss of generality. By construction $(b_0, c_0)$ is a zero of the just defined $F$. + +We are interested in the shape of the level set $F(b,c) = 0$ which reveals other solutions $(b,c)$. For a given $f$, a contour plot, with $b>c$, can reveal this shape. + +To find a source of examples for such functions, polynomials are considered, beginning with these constraints: + +$$ +f(a_0) = 0, f(b_0) = 0, f(c_0) = 1, f'(c_0) = 0 +$$ + +With four conditions, we might guess a cubic parabola with four unknowns should fit. We use `SymPy` to identify the coefficients. + +```{julia} +a₀, b₀, c₀ = 0, 3, 1 +@syms x +@syms a[0:3] +p = sum(aᵢ*x^(i-1) for (i,aᵢ) ∈ enumerate(a)) +dp = diff(p,x) +p, dp +``` + +The constraints are specified as follows; `solve` has no issue with this system of equations. + +```{julia} +eqs = (p(x=>a₀) ~ 0, + p(x=>b₀) ~ 0, + p(x=>c₀) ~ 1, + dp(x=>c₀) ~ 0) +d = solve(eqs, a) +q = p(d...) +``` + +We can plot $q$ and emphasize the three points with: + +```{julia} +xlims = (-0.5, 3.5) +plot(q; xlims, legend=false) +scatter!([a₀, b₀, c₀], [0,0,1]; marker=(5, 0.25)) +``` + + +We now make a plot of the level curve $F(x,y)=0$ using `contour` and the constraint that $b>c$ to graphically identify $C(b)$: + +```{julia} +dq = diff(q, x) +λ(b,c) = b > c ? (q(b) - q(a₀)) / (b - a₀) - dq(c) : -Inf +bs = cs = range(0.5,3.5, 100) +plot(; legend=false) +contour!(bs, cs, λ; levels=[0]) +plot!(identity; line=(1, 0.25)) +scatter!([b₀], [c₀]; marker=(5, 0.25)) +``` + +The curve that passes through the point $(3,1)$ is clearly continuous, and following it, we see continuous changes in $b$ result in continuous changes in $c$. + + + +Following a behind-the-scenes blog post by [Lowry-Duda](https://davidlowryduda.com/choosing-functions-for-mvt-abscissa/) we wrap some of the above into a function to find a polynomial given a set of conditions on values for its self or its derivatives at a point. + +```{julia} +function _interpolate(conds; x=x) + np1 = length(conds) + n = np1 - 1 + as = [Sym("a$i") for i in 0:n] + p = sum(as[i+1] * x^i for i in 0:n) + # set p⁽ᵏ⁾(xᵢ) = v + eqs = Tuple(diff(p, x, k)(x => xᵢ) ~ v for (xᵢ, k, v) ∈ conds) + soln = solve(eqs, as) + p(soln...) +end + +# sets p⁽⁰⁾(a₀) = 0, p⁽⁰⁾(b₀) = 0, p⁽⁰⁾(c₀) = 1, p⁽¹⁾(c₀) = 0 +basic_conditions = [(a₀,0,0), (b₀,0,0), (c₀,0,1), (c₀,1,0)] +_interpolate(basic_conditions; x) +``` + +Before moving on, polynomial interpolation can suffer from the Runge phenomenon, where there can be severe oscillations between the points. To tamp these down, an additional *control* point is added which is adjusted to minimize the size of the derivative through the value $\int \| f'(x) \|^2 dx$ (the $L_2$ norm of the derivative): + +```{julia} +function interpolate(conds) + @syms x, D + # set f'(2) = D, then adjust D to minimize L₂ below + new_conds = vcat(conds, [(2, 1, D)]) + p = _interpolate(new_conds; x) + + # measure size of p with ∫₀⁴f'(x)^2 dx + dp = diff(p, x) + L₂ = integrate(dp^2, (x, 0, 4)) + dL₂ = diff(L₂, D) + soln = first(solve(dL₂ ~ 0, D)) # critical point to minimum L₂ + + p(D => soln) +end +q = interpolate(basic_conditions) +``` + +We also make a plotting function to show both `q` and the level curve of `F`: + +```{julia} +function plot_q_level_curve(q; title="", layout=[1;1]) + x = only(free_symbols(q)) # fish out x + dq = diff(q, x) + + xlims = ylims = (-0.5, 4.5) + + p₁ = plot(; xlims, ylims, title, + legend=false, aspect_ratio=:equal) + plot!(p₁, q; xlims, ylims) + scatter!(p₁, [a₀, b₀, c₀], [0,0,1]; marker=(5, 0.25)) + + λ(b,c) = b > c ? (q(b) - q(a₀)) / (b - a₀) - dq(c) : -Inf + bs = cs = range(xlims..., 100) + + p₂ = plot(; xlims, ylims, legend=false, aspect_ratio=:equal) + contour!(p₂, bs, cs, λ; levels=[0]) + plot!(p₂, identity; line=(1, 0.25)) + scatter!(p₂, [b₀], [c₀]; marker=(5, 0.25)) + + plot(p₁, p₂; layout) +end +``` + + +```{julia} +plot_q_level_curve(q; layout=(1,2)) +``` + +Like previously, this highlights the presence of a continuous function in $b$ yielding $c$. + +This is not the only possibility. Another such from their paper (Figure 3) looks like the following where some additional constraints are added ($f''(c_0) = 0, f'''(c_0)=3, f'(b_0)=-3$): + +```{julia} +new_conds = [(c₀, 2, 0), (c₀, 3, 3), (b₀, 1, -3)] +q = interpolate(vcat(basic_conditions, new_conds)) + +plot_q_level_curve(q;layout=(1,2)) +``` + +For this shape, if $b$ increases away from $b_0$, the secant line connecting $(a_0,0)$ and $(b, f(b)$ will have a negative slope, but there are no points nearby $x=c_0$ where the derivative has a tangent line with negative slope, so the continuous function is only on the left side of $b_0$. Mathematically, as $f$ is increasing $c_0$ -- as $f'''(c_0) = 3 > 0$ -- and $f$ is decreasing at $f(b_0)$ -- as $f'(b_0) = -1 < 0$, the signs alone suggest the scenario. The contour plot reveals, not one, but two one-sided functions of $b$ giving $c$. + +---- + +Now to characterize all possibilities. + +Suppose $F(x,y)$ is differentiable. Then $F(x,y)$ has this approximation (where $F_x$ and $F_y$ are the partial derivatives): + +$$ +F(x,y) \approx F(x_0,y_0) + F_x(x_0,y_0) (x - x_0) + F_y(x_0,y_0) (y-y_0) +$$ + +If $(x_0,y_0)$ is a zero of $F$, then the above can be solved for $y$ assuming $F_y$ does not vanish: + +$$ +y \approx y_0 - \frac{F_x(x_0, y_0)}{F_y(x_0, y_0)} \cdot (x - x_0) +$$ + +The main tool used in the authors' investigation is the implicit function theorem. The implicit function theorem states there is some function continuously describing $y$, not just approximately, under the above assumption of $F_y$ not vanishing. + + +Again, with $F(b,c) = (f(b) - f(a_0)) / (b -a_0) - f'(c)$ and assuming $f$ has at least two continuous derivatives, then: + +$$ +\begin{align*} +F(b_0,c_0) &= 0,\\ +F_c(b_0, c_0) &= -f''(c_0). +\end{align*} +$$ + +Assuming $f''(c_0)$ is *non*-zero, then this proves that if $b$ moves continuously, a corresponding solution to the mean value theorem will as well, or there is a continuous function $C(b)$ with $F(b,C(b)) = 0$. + +Further, they establish if $f'(b_0) \neq f'(c_0)$ then there is a continuous $B(c)$ near $c_0$ such that $F(B(c),c) = 0$; and that there are no other nearby solutions to $F(b,c)=0$ near $(b_0, c_0)$. + + +This leaves for consideration the possibilities when $f''(c_0) = 0$ and $f'(b_0) = f'(c_0)$. + +One such possibility looks like: + +```{julia} +new_conds = [(c₀, 2, 0), (c₀, 3, 3), (b₀, 1, 0), (b₀, 2, 3)] +q = interpolate(vcat(basic_conditions, new_conds)) +plot_q_level_curve(q;layout=(1,2)) +``` + +This picture shows more than one possible choice for a continuous function, as the contour plot has this looping intersection point at $(b_0,c_0)$. + + +To characterize possible behaviors, the authors recall the [Morse lemma](https://en.wikipedia.org/wiki/Morse_theory) applied to functions $f:R^2 \rightarrow R$ with vanishing gradient, but non-vanishing Hession. This states that after some continuous change of coordinates, $f$ looks like $\pm u^2 \pm v^2$. Only this one-dimensional Morse lemma (and a generalization) is required for this analysis: + +> if $g(x)$ is three-times continuously differentiable with $g(x_0) = g'(x_0) = 0$ but $g''(x_0) \neq 0$ then *near* $x_0$ $g(x)$ can be transformed through a continuous change of coordinates to look like $\pm u^2$, where the sign is the sign of the second derivative of $g$. + +That is, locally the function can be continuously transformed into a parabola opening up or down depending on the sign of the second derivative. Their proof starts with Taylor's remainder theorem to find a candidate for the change of coordinates and shows with the implicit function theorem this is a viable change. + + +Setting: +$$ +\begin{align*} +g_1(b) &= (f(b) - f(a_0))/(b - a_0) - f'(c_0)\\ +g_2(c) &= f'(c) - f'(c_0). +\end{align*} +$$ + +Then $F(c, b) = g_1(b) - g_2(c)$. + +By construction, $g_2(c_0) = 0$ and $g_2^{(k)}(c_0) = f^{(k+1)}(c_0)$, +Adjusting $f$ to have a vanishing second -- but not third -- derivative at $c_0$ means $g_2$ will satisfy the assumptions of the lemma assuming $f$ has at least four continuous derivatives (as all our example polynomials do). + +As for $g_1$, we have by construction $g_1(b_0) = 0$. By differentiation we get a pattern for some constants $c_j = (j+1)\cdot(j+2)\cdots \cdot k$ with $c_k = 1$. + +$$ +g^{(k)}(b) = k! \cdot \frac{f(a_0) - f(b)}{(a_0-b)^{k+1}} - \sum_{j=1}^k c_j \frac{f^{(j)}(b)}{(a_0 - b)^{k-j+1}}. +$$ + +Of note that when $f(a_0) = f(b_0) = 0$ that if $f^{(k)}(b_0)$ is the first non-vanishing derivative of $f$ at $b_0$ that $g^{(k)}(b_0) = f^{(k)}(b_0)/(b_0 - a_0)$ (they have the same sign). + + +In particular, if $f(a_0) = f(b_0) = 0$ and $f'(b_0)=0$ and $f''(b_0)$ is non-zero, the lemma applies to $g_1$, again assuming $f$ has at least four continuous derivatives. + +Let $\sigma_1 = \text{sign}(f''(b_0))$ and $\sigma_2 = \text{sign}(f'''(c_0))$, then we have $F(b,c) = \sigma_1 u^2 - \sigma_2 v^2$ after some change of variables. The authors conclude: + +* If $\sigma_1$ and $\sigma_2$ have different signs, then $F(b,c) = 0$ is like $u^2 = - v^2$ which has only one isolated solution, as the left hand side and right hand sign will have different signs except when $0$. +* If $\sigma_1$ and $\sigma_2$ have the same sign, then $F(b,c) = 0$ is like $u^2 = v^2$ which has two solutions $u = \pm v$. + +Applied to the problem at hand: + +* if $f''(b_0)$ and $f'''(c_0)$ have different signs, the $c_0$ can not be extended to a continuous function near $b_0$. +* if the two have the same sign, then there are two such functions possible. + +```{julia} +conds₁ = [(b₀,1,0), (b₀,2,3), (c₀,2,0), (c₀,3,-3)] +conds₂ = [(b₀,1,0), (b₀,2,3), (c₀,2,0), (c₀,3, 3)] + +q₁ = interpolate(vcat(basic_conditions, conds₁)) +q₂ = interpolate(vcat(basic_conditions, conds₂)) + +p₁ = plot_q_level_curve(q₁) +p₂ = plot_q_level_curve(q₂) + +plot(p₁, p₂; layout=(1,2)) +``` + +There are more possibilities, as pointed out in the article. + +Say a function, $h$, has *a zero of order $k$ at $x_0$* if the first $k-1$ derivatives of $h$ are zero at $x_0$, but that $h^{(k)}(x_0) \neq 0$. Now suppose $f$ has order $k$ at $b_0$ and order $l$ at $c_0$. Then $g_1$ will be order $k$ at $b_0$ and $g_2$ will have order $l-1$ at $c_0$. In the above, we had orders $2$ and $3$ respectively. + +A generalization of the Morse lemma to the function, $h$ having a zero of order $k$ at $x_0$ is $h(x) = \pm u^k$ where if $k$ is odd either sign is possible and if $k$ is even, then the sign is that of $h^{(k)}(x_0)$. + +With this, we get the following possibilities for $f$ with a zero of order $k$ at $b_0$ and $l$ at $c_0$: + +* If $l$ is even, then there is one continuous solution near $(b_0,c_0)$ + +* If $l$ is odd and $k$ is even and $f^{(k)}(b_0)$ and $f^{(l)}(c_0)$ have the *same* sign, then there are two continuous solutions + +* If $l$ is odd and $k$ is even and $f^{(k)}(b_0)$ and $f^{(l)}(c_0)$ have *opposite* signs, the $(b_0, c_0)$ is an isolated solution. + +* If $l$ is add and $k$ is odd, then there are two continuous solutions, but only defined in a a one-sided neighborhood of $b_0$ where $f^{(k)}(b_0) f^{(l)}(c_0) (b - b_0) > 0$. + + +To visualize these four cases, we take $(l=2,k=1)$, $(l=3, k=2)$ (twice) and $(l=3, k=3)$. + +```{julia} +condsₑ = [(c₀,2,3), (b₀,1,-3)] +condsₒₑ₊₊ = [(c₀,2,0), (c₀,3, 10), (b₀,1,0), (b₀,2,10)] +condsₒₑ₊₋ = [(c₀,2,0), (c₀,3,-20), (b₀,1,0), (b₀,2,20)] +condsₒₒ = [(c₀,2,0), (c₀,3,-20), (b₀,1,0), (b₀,2, 0), (b₀,3, 20)] + +qₑ = interpolate(vcat(basic_conditions, condsₑ)) +qₒₑ₊₊ = interpolate(vcat(basic_conditions, condsₒₑ₊₊)) +qₒₑ₊₋ = interpolate(vcat(basic_conditions, condsₒₑ₊₋)) +qₒₒ = interpolate(vcat(basic_conditions, condsₒₒ)) + +p₁ = plot_q_level_curve(qₑ; title = "(e,.)") +p₂ = plot_q_level_curve(qₒₑ₊₊; title = "(o,e,same)") +p₃ = plot_q_level_curve(qₒₑ₊₋; title = "(o,e,different)") +p₄ = plot_q_level_curve(qₒₒ; title = "(o,o)") + +plot(p₁, p₂, p₃, p₄; layout=(1,4)) +``` + + +This handles most cases, but leaves the possibility that a function with infinite vanishing derivatives to consider. We steer the interested reader to the article for thoughts on that. + + ## Questions diff --git a/quarto/references.bib b/quarto/references.bib index 48a8b35..b096c05 100644 --- a/quarto/references.bib +++ b/quarto/references.bib @@ -87,3 +87,25 @@ edition = {Julia adaptation}, URL = {https://tobydriscoll.net/fnc-julia/frontmatter.html}, eprint = {https://epubs.siam.org/doi/pdf/10.1137/1.9781611975086} } + +## matrix calculus + +@misc{BrightEdelmanJohnson, + title={Matrix Calculus (for Machine Learning and Beyond)}, + author={Paige Bright and Alan Edelman and Steven G. Johnson}, + year={2025}, + eprint={2501.14787}, + archivePrefix={arXiv}, + primaryClass={math.HO}, + url={https://arxiv.org/abs/2501.14787}, +} + +@misc{CarlssonNikitinTroedssonWendt, + title={The bilinear Hessian for large scale optimization}, + author={Marcus Carlsson and Viktor Nikitin and Erik Troedsson and Herwig Wendt}, + year={2025}, + eprint={2502.03070}, + archivePrefix={arXiv}, + primaryClass={math.OC}, + url={https://arxiv.org/abs/2502.03070}, +} \ No newline at end of file diff --git a/quarto/staging/Project.toml b/quarto/staging/Project.toml new file mode 100644 index 0000000..c319267 --- /dev/null +++ b/quarto/staging/Project.toml @@ -0,0 +1,2 @@ +[deps] +quarto_jll = "b7163347-bfae-5fd9-aba4-19f139889d78" diff --git a/quarto/staging/matrix-calculus-notes.html b/quarto/staging/matrix-calculus-notes.html new file mode 100644 index 0000000..dd6fd95 --- /dev/null +++ b/quarto/staging/matrix-calculus-notes.html @@ -0,0 +1,1278 @@ + + + + + + + + + +matrix-calculus-notes + + + + + + + + + + + + + + + + + + + + + + + +
+ +
+ +
+
+

Matrix Calculus

+
+ + + +
+ + + + +
+ + + +
+ + +

XXX Add in examples from paper XXX optimization? large number of parameters? ,…

+
+
+
+ +
+
+Based on Bright, Edelman, and Johnson’s notes +
+
+
+

This section samples material from the notes Matrix Calculus (for Machine Learning and Beyond) by Paige Bright, Alan Edelman, and Steven G. Johnson. These notes cover material taught in a course at MIT. Support materials for their course in Julia are available at https://github.com/mitmath/matrixcalc/tree/main. For more details and examples, please refer to the source.

+
+
+

We have seen several “derivatives” of a function, based on the number of inputs and outputs. The first one was for functions \(f: R \rightarrow R\).

+

Then \(f\) has a derivative at \(x\) if this limit exists

+

\[ +\lim_{h \rightarrow 0}\frac{f(x + h) - f(x)}{h}. +\]

+

The derivative of the function \(x\) is this limit for a given \(x\). Common notation is:

+

\[ +f'(x) = \frac{dy}{dx} = \lim_{h \rightarrow 0}\frac{f(x + h) - f(x)}{h} +\]

+

(when the limit exists).

+

This limit gets expressed in different ways:

+
    +
  • linearization write \(f(x+\Delta x) - f(x) \approx f'(x)\Delta x\), where \(\delta x\) is a small displacement from \(x\). The reason there isn’t equality is the unwritten higher order terms that vanish in a limit.

  • +
  • Alternate limits. Another way of writing this is in terms of explicit smaller order terms:

  • +
+

\[ +(f(x+h) - f(x)) - f'(x)h = \mathscr{o}(h), +\]

+

which means if we divide both sides by \(h\) and take the limit, we will get \(0\) on the right and the relationship on the left.

+
    +
  • Differential notation simply writes this as \(dy = f(x)dx\). More verbosely, we might write
  • +
+

\[ +df = f(x+dx) - f(x) = f'(x) dx. +\]

+

Here \(dx\) is a differential, made rigorous by a limit, which hides the higher order terms.

+

In these notes the limit has been defined, with suitable modification, for functions of vectors (multiple values) with scalar or vector outputs.

+

For example, when \(f: R \rightarrow R^m\) was a vector-valued function the derivative was defined similarly through a limit of \((f(t + \Delta t) - f(t))/{\Delta t}\), where each component needed to have a limit. This can be rewritten through \(f(t + dt) - f(t) = f'(t) dt\), again using differentials to avoid the higher order terms.

+

When \(f: R^n \rightarrow R\) is a scalar-valued function of a derivative, differentiability was defined by a gradient existing with \(f(c+h) - f(c) - \nabla{f}(c) \cdot h\) being \(\mathscr{o}(\|h\|)\). In other words \(df = f(c + dh) - f(c) = \nabla{f}(c) \cdot dh\). The gradient has the same shape as \(c\), a column vector. If we take the row vector (e.g. \(f'(c) = \nabla{f}(c)^T\)) then again we see \(df = f(c+dh) - f(c) = f'(c) dh\), where the last term uses matrix multiplication of a row vector times a column vector.

+

Finally, when \(f:R^n \rightarrow R^m\), the Jacobian was defined and characterized by \(\| f(x + dx) - f(x) - J_f(x)dx \|\) being \(\mathscr{o}(\|dx\|)\). Again, we can express this through \(df = f(x + dx) - f(x) = f'(x)dx\) where \(f'(x) = J_f(x)\).

+

In writing \(df = f(x + dx) - f(x) = f'(x) dx\) generically, some underlying facts are left implicit: \(dx\) has the same shape as \(x\) (so can be added); \(f'(x) dx\) may mean usual multiplication or matrix multiplication; and there is an underlying concept of distance and size that allows the above to be rigorous. This may be an abolute value or a norm.

+

Further, various differentiation rules apply such as the sum, product, and chain rule.

+

The @BrightEdelmanJohnson notes cover differentiation of functions in this uniform manner and then extend the form by treating derivatives as linear operators.

+

A linear operator is a mathematical object which satisfies

+

\[ +f[\alpha v + \beta w] = \alpha f[v] + \beta f[w]. +\]

+

where the \(\alpha\) and \(\beta\) are scalars, and \(v\) and \(w\) possibly not and come from a vector space. Regular multiplication and matrix multiplication are familiar linear operations, but there are many others.

+

The referenced notes identify \(f'(x) dx\) with \(f'(x)[dx]\), the latter emphasizing \(f'(x)\) acts on \(dx\) and the notation is not commutative (e.g., it is not \(dx f'(x)\)).

+

Linear operators are related to vector spaces.

+

A vector space is a set of mathematical objects which can be added together and also multiplied by a scalar. Vectors of similar size, as previously discussed, are the typical example, with vector addition and scalar multiplication previously discussed topics. Matrices of similar size (and some subclasses) also form a vector space. Additionally, many other set of objects form vector spaces. An example might be polynomial functions of degree \(n\) or less; continuous functions, or functions with a certain number of derivatives.

+

Take differentiable functions as an example, then the simplest derivative rules \([af(x) + bg(x)]' = a[f(x)]' + b[g(x)]'\) show the linearity of the derivative in this setting. This linearity is different from how the derivative is a linear operator on \(dx\).

+

A vector space is described by a basis – a minimal set of vectors needed to describe the space, after consideration of linear combinations. For many vectors, this the set of special vectors with \(1\) as one of the entries, and \(0\) otherwise.

+

A key fact about a basis is every vector in the vector space can be expressed uniquely as a linear combination of the basis vectors.

+

Vectors and matrices have properties that are generalizations of the real numbers. As vectors and matrices form vector spaces, the concept of addition of vectors and matrices is defined, as is scalar multiplication. Additionally, we have seen:

+
    +
  • The dot product between two vectors of the same length is defined easily (\(v\cdot w = \Sigma_i v_i w_i\)). It is coupled with the length as \(\|v\|^2 = v\cdot v\).

  • +
  • Matrix multiplication is defined for two properly sized matrices. If \(A\) is \(m \times k\) and \(B\) is \(k \times n\) then \(AB\) is a \(m\times n\) matrix with \((i,j)\) term given by the dot product of the \(i\)th row of \(A\) (viewed as a vector) and the \(j\)th column of \(B\) (viewed as a vector). Matrix multiplication is associative but not commutative. (E.g. \((AB)C = A(BC)\) but \(AB\) and \(BA\) need not be equal (or even defined, as the shapes may not match up).

  • +
  • A square matrix \(A\) has an inverse \(A^{-1}\) if \(AA^{-1} = A^{-1}A = I\), where \(I\) is the identity matrix (a matrix which is zero except on its diagonal entries which are all \(1\)). Square matrices may or may not have an inverse. When they don’t the matrix is called singular.

  • +
  • Viewing a vector as a matrix is possible. The association is typically through a column vector.

  • +
  • The transpose of a matrix comes by permuting the rows and columns. The transpose of a column vector is a row vector, so \(v\cdot w = v^T w\), where we use a superscript \(T\) for the transpose. The transpose of a product, is the product of the transposes – reversed: \((AB)^T = B^T A^T\); the tranpose of a transpose is an identity operation: \((A^T)^T = A\); the inverse of a transpose is the tranpose of the inverse: \((A^{-1})^T = (A^T){-1}\).

  • +
  • Matrices for which \(A = A^T\) are called symmetric.

  • +
  • A few of the operations on matrices are the transpose and the inverse. These return a matrix, when defined. There is also the determinant and the trace, which return a scalar from a matrix. The trace is just the sum of the diagonal; the determinant is more involved to compute, but was previously seen to have a relationship to the volume of a certain parallellpiped. There are a few other operations described in the following.

  • +
+

.

+
+

Scalar-valued functions of a vector

+

Suppose \(f: R^n \rightarrow R\), a scalar-valued function of a vector. Then the directional derivative at \(x\) in the direction \(v\) was defined for a scalar \(\alpha\) by:

+

\[ +\frac{\partial}{\partial \alpha}f(x + \alpha v) \mid_{\alpha = 0} = +\lim_{\Delta\alpha \rightarrow 0} \frac{f(x + \Delta\alpha v) - f(x)}{\Delta\alpha}. +\]

+

This rate of change in the direction of \(v\) can be expressed through the linear operator \(f'(x)\) via

+

\[ +f(x + d\alpha v) - f(x) = f'(x) [d\alpha v] = d\alpha f'(x)[v], +\]

+

using linearity to move the scalar part outside the \([]\). This connects the partial derivative at \(x\) in the direction of \(v\) with \(f'(x)\):

+

\[ +\frac{\partial}{\partial \alpha}f(x + \alpha v) \mid_{\alpha = 0} = +f'(x)[v]. +\]

+

Not only does this give a connection in notation with the derivative, it naturally illustrates how the derivative as a linear operator can act on non-infinitesimal values.

+

Previously, we wrote \(\nabla f \cdot v\) for the directional derivative, where the gradient is a column vector. The above uses the identification \(f' = (\nabla f)^T\).

+

For \(f: R^n \rightarrow R\) we have

+

\[ +df = f(x + dx) - f(x) = f'(x) [dx] +\]

+

is a scalar, so if \(dx\) is a column vector, \(f'(x)\) is a row vector with the same number of components (just as \(\nabla f\) is a column vector with the same number of components).

+
+
Examples
+

@BrightEdelmanJohnson include this example to show that the computation of derivatives using components can be avoided. Consider \(f(x) = x^T A x\) where \(x\) is a vector in \(R^n\) and \(A\) is an \(n\times n\) matrix. Then \(f: R^n \rightarrow R\) and its derivative can be computed:

+

\[ +\begin{align*} +df &= f(x + dx) - f(x)\\ +&= (x + dx)^T A (x + dx) - x^TAx \\ +&= x^TAx + dx^TA x + x^TAx + dx^T A dx - x^TAx\\ +&= dx^TA x + x^TAdx \\ +&= (dx^TAx)^T + x^TAdx \\ +&= x^T A^T dx + x^T A dx\\ +&= x^T(A^T + A) dx +\end{align*} +\]

+

The term \(dx^t A dx\) is dropped, as it is higher order (goes to zero faster), it containing two \(dx\) terms. In the second to last step, an identity operation (taking the transpose of the scalar quantity) is taken to simplify the algebra. Finally, as \(df = f'(x)[dx]\) the identity of \(f'(x) = x^T(A^T+A)\) is made, or taking transposes \(\nabla f = (A + A^T)x\).

+

Compare the elegance above, with the component version, even though simplified, it still requires a specification of the size to carry the following out:

+
+
using SymPy
+@syms x[1:3]::real A[1:3, 1:3]::real
+u = x' * A * x
+grad_u = [diff(u, xi) for xi in x]
+
+

\(\left[\begin{smallmatrix}2 A₁_{₁} x₁ + A₁_{₂} x₂ + A₁_{₃} x₃ + A₂_{₁} x₂ + A₃_{₁} x₃\\A₁_{₂} x₁ + A₂_{₁} x₁ + 2 A₂_{₂} x₂ + A₂_{₃} x₃ + A₃_{₂} x₃\\A₁_{₃} x₁ + A₂_{₃} x₂ + A₃_{₁} x₁ + A₃_{₂} x₂ + 2 A₃_{₃} x₃\end{smallmatrix}\right]\)

+
+
+

Compare to the formula for the gradient just derived:

+
+
grad_u_1 = (A + A')*x
+
+

\(\left[\begin{smallmatrix}2 A₁_{₁} x₁ + x₂ \left(A₁_{₂} + A₂_{₁}\right) + x₃ \left(A₁_{₃} + A₃_{₁}\right)\\2 A₂_{₂} x₂ + x₁ \left(A₁_{₂} + A₂_{₁}\right) + x₃ \left(A₂_{₃} + A₃_{₂}\right)\\2 A₃_{₃} x₃ + x₁ \left(A₁_{₃} + A₃_{₁}\right) + x₂ \left(A₂_{₃} + A₃_{₂}\right)\end{smallmatrix}\right]\)

+
+
+

The two are, of course, equal

+
+
all(a == b for (a,b)  zip(grad_u, grad_u_1))
+
+
true
+
+
+
+

For \(f: R^n \rightarrow R^m\), @BrightEdelmanJohnson give an example of computing the Jacobian without resorting to component wise computations. Let \(f(x) = Ax\) with \(A\) being a \(m \times n\) matrix, it follows that

+

\[ +\begin{align*} +df &= f(x + dx) - f(x)\\ +&= A(x + dx) - Ax\\ +&= Adx\\ +&= f'(x) dx. +\end{align*} +\]

+

The Jacobian is the linear operator \(A\) acting on \(dx\).

+
+
+
+

Sum and product rules for the derivative

+

Using the differential notation – which implicitly ignores higher order terms as they vanish in a limit – the sum and product rules can be derived.

+

For the sum rule, let \(f(x) = g(x) + h(x)\). Then

+

\[ +\begin{align*} +df &= f(x + dx) - f(x) \\ +&= f'(x) dx\\ +&= \left(g(x+dx) + h(x+dx)\right) - \left(g(x) + h(x)\right)\\ +&= \left(g(x + dx) - g(x)\right) + \left(h(x + dx) - h(x)\right)\\ +&= g'(x)dx + h'(x) dx\\ +&= \left(g'(x) + h'(x)\right) dx +\end{align*} +\]

+

Comparing we get \(f'(x) = g'(x) + h'(x)\).

+

The sum rule has the same derivation as was done with univariate, scalar functions. Similarly for the product rule.

+

The product rule has with \(f(x) = g(x)h(x)\)

+

\[ +\begin{align*} +df &= f(x + dx) - f(x) \\ +&= g(x+dx)h(x + dx) - g(x) h(x)\\ +&= \left(g(x) + g'(x)dx\right)\left(h(x) + h'(x) dx\right) - \left(g(x) h(x)\right) \\ +&= g(x)h(x) + g'(x) dx h(x) + g(x) h'(x) dx + g'(x)dx h'(x) dx - g(x) h(x)\\ +&= gh + dg h + gdh + dg dh - gh\\ + &= dg h + gdh, +\end{align*} +\]

+

after dropping the higher order term and cancelling \(gh\) terms of opposite signs in the fourth row.

+
+
Examples
+

These two rules can be used to show the last two examples:

+

First, to differentiate \(f(x) = x^TAx\):

+

\[ +\begin{align*} +df &= dx^T (Ax) + x^T d(Ax) \\ +&= x^T A^T dx + x^T A dx \\ +&= x^T(A^T + A) dx +\end{align*} +\]

+

Again, taking the transpose of the scalar quantity \(x^TAdx\) to simplify the expression.

+

When \(A^T = A\) (\(A\) is symmetric) this simplifies to a more familiar looking \(2x^TA\), but we see that this requires assumptions not needed in the scalar case.

+

Next, if \(f(x) = Ax\) then

+

\[ +df = (dA)x + A(dx) = 0x + A dx = A dx, +\]

+

\(A\) being a constant here.

+
+
+
Example
+

@BrightEdelmanJohnson consider what in Julia is .*. That is the operation:

+

\[ +v .* w = +\begin{bmatrix} +v_1w_1 \\ +v_2w_2 \\ +\vdots\\ +v_nw_n +\end{bmatrix} += +\begin{bmatrix} +v_1 & 0 & \cdots & 0 \\ +0 & v_2 & \cdots & 0 \\ + & & \vdots & \\ +0 & 0 & \cdots & v_n +\end{bmatrix} +\begin{bmatrix} +w_1 \\ +w_2 \\ +\vdots\\ +w_n +\end{bmatrix} += \text{diag}(v) w. +\]

+

They compute the derivative of \(f(x) = A(x .* x)\) for some fixed matrix \(A\) of the proper size.

+

We can see that \(d (\text{diag}(v)w) = d(\text{diag}(v)) w + \text{diag}(v) dw = (dx) .* w + x .* dw\). So

+

\(df = A(dx .* x + x .* dx) = 2A(x .* dx)\), as \(.*\) is commutative by its definition. Writing this as \(df = 2A(x .* dx) = 2A(\text{diag}(x) dx) = (2A\text{diag}(x)) dx\), we identify \(f'(x) = 2A\text{diag}(x)\).

+

This operation is called the Hadamard product and it extends to matrices and arrays.

+
+
+
+

The chain rule

+

Like the product rule, the chain rule is shown by @BrightEdelmanJohnson in this notation with \(f(x) = g(h(x))\):

+

\[ +\begin{align*} +df &= f(x + dx) - f(x)\\ +&= g(h(x + dx)) - g(h(x))\\ +&= g(h(x) + h'(x)[dx]) - g(h(x))\\ +&= g'(h(x)) [h'(x) [dx]]\\ +&= (g'(h(x)) h'(x)) [dx] +\end{align*} +\]

+

(The limit requires a bit more detail.)

+

The operator \(f'(x)= g'(h(x)) h'(x)\) is a product of matrices.

+
+

Computational differences with expressions from the chain rule

+

Of note here is the application of the chain rule to three (or more compositions):

+

The derivative of \(f(x) = a(x) b(x) c(x)\) can be expressed as

+

\[ +f' = (a'b')c' \text{ or } f' = a'(b'c') +\]

+

Multiplying left to right (the first) is called reverse mode; multiplying right to left (the second) is called forward mode. The distinction becomes important when considering the computational cost of the multiplications.

+
    +
  • If \(f: R^n \rightarrow R^m\) has \(n\) much bigger than \(1\) and \(m=1\), then it is much faster to do left to right multiplication
  • +
  • if \(f:R^n \rightarrow R^m\) has \(n=1\) and \(m\) much bigger than one, the it is faster to do right to left multiplication.
  • +
+

The basic idea comes down to the shape of the matrices. When \(m=1\), the derviative is a product of matrices of size \(n\times j\) \(j\times k\) and \(k \times 1\) yielding a matrix of size \(n \times 1\) matching the function dimension. Matrix multiplication of an \(m \times q\) times \(q \times n\) takes an order of \(mqn\) operations. The multiplication of left to right is then

+

The first operation takes \(njk\) operation leaving an \(n\times k\) matrix, the next multiplication then takes another \(nk1\) operations or \(njk + nk\) together. Whereas computing from the right to left is first \(jk1\) operations leaving a \(j \times 1\) matrix. The next operation would take another \(nk1\) operations. In totalL

+
    +
  • left to right is \(njk + nk\) = \(nk \cdot (1 + j)\).
  • +
  • right to left is \(jk + j = j\cdot (k+1)\).
  • +
+

When \(j=k\), say, we can compare and see the second is a factor less in terms of operations. This can be quite significant in higher dimensions, whereas the dimensions of calculus (where \(n\) and \(m\) are \(3\) or less) it is not an issue.

+
+
Example
+

Using the BenchmarkTools package, we can check the time to compute various products:

+
+
using BenchmarkTools
+n,j,k,m = 20,15,10,1
+@btime A*(B*C) setup=(A=rand(n,j);B=rand(j,k); C=rand(k,m));
+@btime (A*B)*C setup=(A=rand(n,j);B=rand(j,k); C=rand(k,m));
+
+
  420.814 ns (4 allocations: 432 bytes)
+  702.678 ns (4 allocations: 1.88 KiB)
+
+
+

The latter computation is about 1.5 times slower.

+

Whereas the relationship is changed when the first matrix is skinny and the last is not:

+
+
@btime A*(B*C) setup=(A=rand(m,k);B=rand(k,j); C=rand(j,n));
+@btime (A*B)*C setup=(A=rand(m,k);B=rand(k,j); C=rand(j,n));
+
+
  901.488 ns (4 allocations: 1.88 KiB)
+  623.468 ns (4 allocations: 432 bytes)
+
+
+
+
+
Example
+

In calculus, we have \(n\) and \(m\) are \(1\),\(2\),or \(3\). But that need not be the case, especially if differentiation is over a parameter space.

+

XXXX (Maybe the ariplain wing, but please, something origi

+
+
+
+
+

Derivatives of matrix functions

+

What is the the derivative of \(f(A) = A^2\)?

+

The function \(f\) takes a \(n\times n\) matrix and returns a matrix of the same size. This innocuous question isn’t directly handled by the Jacobian, which is defined for vector valued function \(f:R^n \rightarrow R^m\).

+

This derivative can be derived directly from the product rule:

+

\[ +\begin{align*} +f(A) &= [AA]'\\ +&= A dA + dA A +\end{align*} +\]

+

That is \(f'(A)\) is the operator \(f'(A)[\delta A] = A \delta A + \delta A A\) and not \(2A\delta A\), as \(A\) may not commute with \(\delta A\).

+
+

Vectorization of a matrix

+

Alternatively, we can identify \(A\) through its components, as a vector in \(R^{n^2}\) and then leverage the Jacobian.

+

One such identification is vectorization – consecutively stacking the column vectors into a vector. In Julia the vec function does this operation:

+
+
@syms A[1:2, 1:2]
+vec(A)
+
+

\(\left[\begin{smallmatrix}A₁_{₁}\\A₂_{₁}\\A₁_{₂}\\A₂_{₂}\end{smallmatrix}\right]\)

+
+
+

The stacking by column follows how Julia stores matrices and how Julia references a matrices entries by linear index:

+
+
vec(A) == [A[i] for i in eachindex(A)]
+
+
true
+
+
+

With this vectorization operation, \(f\) may be viewed as \(\tilde{f}:R^{n^2} \rightarrow R^{n^2}\) through:

+

\[ +\tilde{f}(\text{vec}(A)) = \text{vec}(f(A)) +\]

+

We use SymPy to compute the Jacobian of this vector valued function.

+
+
@syms A[1:3, 1:3]::real
+f(x) = x^2
+J = vec(f(A)).jacobian(vec(A)) # jacobian of f̃
+
+

\(\left[\begin{smallmatrix}2 A₁_{₁} & A₁_{₂} & A₁_{₃} & A₂_{₁} & 0 & 0 & A₃_{₁} & 0 & 0\\A₂_{₁} & A₁_{₁} + A₂_{₂} & A₂_{₃} & 0 & A₂_{₁} & 0 & 0 & A₃_{₁} & 0\\A₃_{₁} & A₃_{₂} & A₁_{₁} + A₃_{₃} & 0 & 0 & A₂_{₁} & 0 & 0 & A₃_{₁}\\A₁_{₂} & 0 & 0 & A₁_{₁} + A₂_{₂} & A₁_{₂} & A₁_{₃} & A₃_{₂} & 0 & 0\\0 & A₁_{₂} & 0 & A₂_{₁} & 2 A₂_{₂} & A₂_{₃} & 0 & A₃_{₂} & 0\\0 & 0 & A₁_{₂} & A₃_{₁} & A₃_{₂} & A₂_{₂} + A₃_{₃} & 0 & 0 & A₃_{₂}\\A₁_{₃} & 0 & 0 & A₂_{₃} & 0 & 0 & A₁_{₁} + A₃_{₃} & A₁_{₂} & A₁_{₃}\\0 & A₁_{₃} & 0 & 0 & A₂_{₃} & 0 & A₂_{₁} & A₂_{₂} + A₃_{₃} & A₂_{₃}\\0 & 0 & A₁_{₃} & 0 & 0 & A₂_{₃} & A₃_{₁} & A₃_{₂} & 2 A₃_{₃}\end{smallmatrix}\right]\)

+
+
+

We do this via linear algebra first, then see a more elegant manner following the notes.

+

A basic course in linear algebra shows that any linear operator on a finite vector space can be represented as a matrix. The basic idea is to represent what the operator does to each basis element and put these values as columns of the matrix.

+

In this \(3 \times 3\) case, the linear operator works on an object with \(9\) slots and returns an object with \(9\) slots, so the matrix will be \(9 \times 9\).

+

The basis elements are simply the matrices with a \(1\) in spot \((i,j)\) and zero elsewhere. Here we generate them through a function:

+
+
basis(i,j,A) = (b=zeros(Int, size(A)...); b[i,j] = 1; b)
+JJ = [vec(basis(i,j,A)*A + A*basis(i,j,A)) for  j in 1:3 for i in 1:3]
+
+
9-element Vector{Vector{Sym{PyCall.PyObject}}}:
+ [2*A₁_₁, A₂_₁, A₃_₁, A₁_₂, 0, 0, A₁_₃, 0, 0]
+ [A₁_₂, A₁_₁ + A₂_₂, A₃_₂, 0, A₁_₂, 0, 0, A₁_₃, 0]
+ [A₁_₃, A₂_₃, A₁_₁ + A₃_₃, 0, 0, A₁_₂, 0, 0, A₁_₃]
+ [A₂_₁, 0, 0, A₁_₁ + A₂_₂, A₂_₁, A₃_₁, A₂_₃, 0, 0]
+ [0, A₂_₁, 0, A₁_₂, 2*A₂_₂, A₃_₂, 0, A₂_₃, 0]
+ [0, 0, A₂_₁, A₁_₃, A₂_₃, A₂_₂ + A₃_₃, 0, 0, A₂_₃]
+ [A₃_₁, 0, 0, A₃_₂, 0, 0, A₁_₁ + A₃_₃, A₂_₁, A₃_₁]
+ [0, A₃_₁, 0, 0, A₃_₂, 0, A₁_₂, A₂_₂ + A₃_₃, A₃_₂]
+ [0, 0, A₃_₁, 0, 0, A₃_₂, A₁_₃, A₂_₃, 2*A₃_₃]
+
+
+

The elements of JJ show the representation of each of the \(9\) basis elements under the linear transformation.

+

To construct the matrix representing the linear operator, we need to concatenate these horizontally as column vectors

+
+
JJ = hcat(JJ...)
+
+

\(\left[\begin{smallmatrix}2 A₁_{₁} & A₁_{₂} & A₁_{₃} & A₂_{₁} & 0 & 0 & A₃_{₁} & 0 & 0\\A₂_{₁} & A₁_{₁} + A₂_{₂} & A₂_{₃} & 0 & A₂_{₁} & 0 & 0 & A₃_{₁} & 0\\A₃_{₁} & A₃_{₂} & A₁_{₁} + A₃_{₃} & 0 & 0 & A₂_{₁} & 0 & 0 & A₃_{₁}\\A₁_{₂} & 0 & 0 & A₁_{₁} + A₂_{₂} & A₁_{₂} & A₁_{₃} & A₃_{₂} & 0 & 0\\0 & A₁_{₂} & 0 & A₂_{₁} & 2 A₂_{₂} & A₂_{₃} & 0 & A₃_{₂} & 0\\0 & 0 & A₁_{₂} & A₃_{₁} & A₃_{₂} & A₂_{₂} + A₃_{₃} & 0 & 0 & A₃_{₂}\\A₁_{₃} & 0 & 0 & A₂_{₃} & 0 & 0 & A₁_{₁} + A₃_{₃} & A₁_{₂} & A₁_{₃}\\0 & A₁_{₃} & 0 & 0 & A₂_{₃} & 0 & A₂_{₁} & A₂_{₂} + A₃_{₃} & A₂_{₃}\\0 & 0 & A₁_{₃} & 0 & 0 & A₂_{₃} & A₃_{₁} & A₃_{₂} & 2 A₃_{₃}\end{smallmatrix}\right]\)

+
+
+

The matrix \(JJ\) is identical to \(J\), above:

+
+
all(j == jj for (j, jj) in zip(J, JJ))
+
+
true
+
+
+
+
+

Kronecker products

+

But how can we see the Jacobian, \(J\), from the linear operator \(f'(A)[\delta A] = \delta A A + A \delta A\)?

+

To make this less magical, a related operation to vec is defined.

+

The \(\text{vec}\) function takes a matrix and stacks its columns.

+

The \(\text{vec}\) function can turn a matrix into a vector, so it can be used for finding the Jacobian, as above. However the shape of the matrix is lost, as are the fundamental matrix operations, like multiplication.

+

The Kronecker product replicates values making a bigger matrix. That is, if \(A\) and \(B\) are matrices, the Kronecker product replaces each value in \(A\) with that value times \(B\), making a bigger matrix, as each entry in \(A\) is replaced by an entry with size \(B\).

+

Formally,

+

\[ +A \otimes B = +\begin{bmatrix} +a_{11}B & a_{12}B & \cdots & a_{1n}B \\ +a_{21}B & a_{22}B & \cdots & a_{2n}B \\ + &\vdots & & \\ +a_{m1}B & a_{m2}B & \cdots & a_{mn}B +\end{bmatrix} +\]

+

The function kron forms this product:

+
+
@syms A[1:2, 1:3] B[1:3, 1:4]
+kron(A, B) # same as hcat((vcat((A[i,j]*B for i in 1:2)...) for j in 1:3)...)
+
+

\(\left[\begin{smallmatrix}A₁_{₁} B₁_{₁} & A₁_{₁} B₁_{₂} & A₁_{₁} B₁_{₃} & A₁_{₁} B₁_{₄} & A₁_{₂} B₁_{₁} & A₁_{₂} B₁_{₂} & A₁_{₂} B₁_{₃} & A₁_{₂} B₁_{₄} & A₁_{₃} B₁_{₁} & A₁_{₃} B₁_{₂} & A₁_{₃} B₁_{₃} & A₁_{₃} B₁_{₄}\\A₁_{₁} B₂_{₁} & A₁_{₁} B₂_{₂} & A₁_{₁} B₂_{₃} & A₁_{₁} B₂_{₄} & A₁_{₂} B₂_{₁} & A₁_{₂} B₂_{₂} & A₁_{₂} B₂_{₃} & A₁_{₂} B₂_{₄} & A₁_{₃} B₂_{₁} & A₁_{₃} B₂_{₂} & A₁_{₃} B₂_{₃} & A₁_{₃} B₂_{₄}\\A₁_{₁} B₃_{₁} & A₁_{₁} B₃_{₂} & A₁_{₁} B₃_{₃} & A₁_{₁} B₃_{₄} & A₁_{₂} B₃_{₁} & A₁_{₂} B₃_{₂} & A₁_{₂} B₃_{₃} & A₁_{₂} B₃_{₄} & A₁_{₃} B₃_{₁} & A₁_{₃} B₃_{₂} & A₁_{₃} B₃_{₃} & A₁_{₃} B₃_{₄}\\A₂_{₁} B₁_{₁} & A₂_{₁} B₁_{₂} & A₂_{₁} B₁_{₃} & A₂_{₁} B₁_{₄} & A₂_{₂} B₁_{₁} & A₂_{₂} B₁_{₂} & A₂_{₂} B₁_{₃} & A₂_{₂} B₁_{₄} & A₂_{₃} B₁_{₁} & A₂_{₃} B₁_{₂} & A₂_{₃} B₁_{₃} & A₂_{₃} B₁_{₄}\\A₂_{₁} B₂_{₁} & A₂_{₁} B₂_{₂} & A₂_{₁} B₂_{₃} & A₂_{₁} B₂_{₄} & A₂_{₂} B₂_{₁} & A₂_{₂} B₂_{₂} & A₂_{₂} B₂_{₃} & A₂_{₂} B₂_{₄} & A₂_{₃} B₂_{₁} & A₂_{₃} B₂_{₂} & A₂_{₃} B₂_{₃} & A₂_{₃} B₂_{₄}\\A₂_{₁} B₃_{₁} & A₂_{₁} B₃_{₂} & A₂_{₁} B₃_{₃} & A₂_{₁} B₃_{₄} & A₂_{₂} B₃_{₁} & A₂_{₂} B₃_{₂} & A₂_{₂} B₃_{₃} & A₂_{₂} B₃_{₄} & A₂_{₃} B₃_{₁} & A₂_{₃} B₃_{₂} & A₂_{₃} B₃_{₃} & A₂_{₃} B₃_{₄}\end{smallmatrix}\right]\)

+
+
+

The \(m\times n\) matrix \(A\) and \(j \times k\) matrix \(B\) has a Kronecker product with size \(mj \times nk\).

+

The Kronecker product has a certain algebra, including:

+
    +
  • transposes: \((A \otimes B)^T) = A^T \otimes B^T\)
  • +
  • multiplication: \((A\otimes B)(C \otimes D) = (AC) \otimes (BD)\)
  • +
  • inverses: \((A \otimes B)^{-1} = (A^{-1}) \otimes (B^{-1})\)
  • +
  • orthogonal: \((A\otimes B)^T = (A\otimes B)\) if both \(A\) and \(B\) has the same property
  • +
  • determinants: \(\det(A\otimes B) = \det(A)^m \det(B)^n\), where \(A\) is \(n\times n\), \(B\) is \(m \times m\).
  • +
  • trace (sum of diagonal): \(\text{tr}(A \otimes B) = \text{tr}(A)\text{tr}(B)\).
  • +
+

The main equation coupling vec and kron is the fact that if \(A\), \(B\), and \(C\) have appropriate sizes, then:

+

\[ +(A \otimes B) \text{vec}(C) = \text{vec}(B C A^T). +\]

+

Appropriate sizes for \(A\), \(B\), and \(C\) are determined by the various products in \(BCA^T\).

+

If \(A\) is \(m \times n\) and \(B\) is \(r \times s\), then since \(BC\) is defined, \(C\) has \(s\) rows, and since \(CA^T\) is defined, \(C\) must have \(n\) columns, as \(A^T\) is \(n \times m\), so \(C\) must be \(s\times n\). Checking this is correct on the other side, \(A \times B\) would be size \(mr \times ns\) and \(\vec{C}\) would be size \(sn\), so that product works, size wise.

+

The referred to notes have an explanation for this formula, but we confirm with an example with \(m=n-2\), \(r=s=3\):

+
+
@syms A[1:2, 1:2]::real B[1:3, 1:3]::real C[1:3, 1:2]::real
+L, R = kron(A,B)*vec(C),  vec(B*C*A')
+all(l == r for (l, r)  zip(L, R))
+
+
true
+
+
+
+

Now to use this relationship to recognize \(df = A dA + dA A\) with the Jacobian computed from \(\text{vec}{f(a)}\).

+

We have \(\text{vec}(A dA + dA A) = \text{vec}(A dA) + \text{vec}(dA A)\), by obvious linearity of \(\text{vec}\). Now inserting an identity matrix, \(I\), which is symmteric, we have:

+

\[ +\text{vec}(A dA) = \text{vec}(A dA I^T) = (I \otimes A) \text{vec}(dA), +\]

+

and

+

\[ +\text{vec}(dA A) = \text{vec}(I dA (A^T)^T) = (A^T \otimes I) \text{vec}(dA) +\]

+

This leaves

+

\[ +\text{vec}(A dA + dA A) = +\left((I \otimes A) + (A^T \otimes I)\right) \text{vec}(dA) +\]

+

We should then get the Jacobian we computed from the following:

+
+
@syms A[1:3, 1:3]::real
+using LinearAlgebra: I
+J = vec(A^2).jacobian(vec(A))
+JJ = kron(I(3), A) + kron(A', I(3))
+all(j == jj for (j,jj) in zip(J,JJ))
+
+
true
+
+
+

This technique can also be used with other powers, say \(f(A) = A^3\), where the resulting \(df = A^2 dA + A dA A + dA A^2\) is one answer that can be compared to a Jacobian through

+

\[ +\begin{align*} +df &= \text{vec}(A^2 dA I^T) + \text{vec}(A dA A) + \text{vec}(I dA A^2)\\ +&= (I \otimes A^2)\text{vec}(dA) + (A^T \otimes A) \text{vec}(dA) + ((A^T)^2 \otimes I) \text{vec}(dA) +\end{align*} +\]

+

The above shows how to relate the derivative of a matrix function to the Jacobian of a vectorized function, but only for illustration. It is decidely not necessary to express the derivative of \(f\) in terms of the derivative of its vectorized counterpart.

+
+
Example: derivative of the inverse
+

What is the derivative of \(f(A) = A^{-1}\). When \(A\) is a scalar, we related it to the reciprocal of the derivative of \(f\) at some other point. The same technique is available. Starting with \(I = AA^{-1}\) and noting \(dI\) is \(0\) we have

+

\[ +\begin{align*} +0 &= d(AA^{-1})\\ +&= dAA^{-1} + A d(A^{-1}) +\end{align*} +\]

+

So, \(d(A^{-1}) = -A^{-1} dA A^{-1}\).

+

This could be re-expressed as a linear operator through

+

\[ +\text{vec}(dA^{-1}) = +\left((A^{-1})^T \otimes A^{-1}\right) \text{vec}(dA) += \left((A^T)^{-1} \otimes A^{-1}\right) \text{vec}(dA). +\]

+
+
+
Example: derivative of the determinant
+

Let \(f(A) = \text{det}(A)\). What is the derivative?

+

First, the determinant of a square, \(n\times n\), matrix \(A\) is a scalar summary of \(A\) with different means to compute it, but one recursive one in particular is helpful here:

+

\[ +\text{det}(A) = a_{1j}C_{1j} + a_{2j}C_{2j} + \cdots a_{nj}C_{nj} +\]

+

for any \(j\). The cofactor \(C_{ij}\) is the determinant of the \((n-1)\times(n-1)\) matrix with the \(i\)th row and \(j\)th column deleted times \((-1)^{i+j}\).

+

To find the gradient of \(f\), we differentiate by each of the \(A_{ij}\) variables, and so

+

\[ +\frac{\partial\text{det}(A)}{\partial A_{ij}} = +\frac{\partial (a_{1j}C_{1j} + a_{2j}C_{2j} + \cdots a_{nj}C_{nj})}{\partial A_{ij}} = +C_{ij}, +\]

+

as each cofactor in the expansion has no dependence on \(A_{ij}\) as the cofactor removes the \(i\)th row and \(j\)th column.

+

So the gradient is the matrix of cofactors.

+

@BrightEdelmanJohnson also give a different proof, starting with this observation

+

\[ +\text{det}(I + dA) - \text{det}(I) = \text{tr}(dA) +\]

+

Assuming that, then by the fact \(\text{det}(AB) = \text{det}(A)\text{det}(B)\):

+

\[ +\begin{align*} +\text{det}(A + A(A^{-1}dA)) - \text{det}(A) &= \text{det}(A)\cdot(\text{det}(I+ A^{-1}dA) - \text{det}(I)) \\ +&= \text{det}(A) \text{tr}(A^{-1}dA)\\ +&= \text{tr}(\text{det}(A)A^{-1}dA)\\ +\end{align*} +\]

+

This agrees through a formula to compute the inverse of a matrix through its cofactor matrix divided by its determinant.

+

That the trace gets involved, can be seen from this computation, which shows the only first-order terms are from the diagonal sum:

+
+
using LinearAlgebra
+@syms dA[1:2, 1:2]
+det(I + dA) - det(I)
+
+

\(dA₁_{₁} dA₂_{₂} + dA₁_{₁} - dA₁_{₂} dA₂_{₁} + dA₂_{₂}\)

+
+
+
+
+
+
+

The adjoint method

+

The chain rule brings about a series of products. The adjoint method illustrated below, shows how to approach the computation of the series in a direction that minimizes the computational cost, illustrating why reverse mode is preferred to forward mode when a scalar function of several variables is considered.

+

@BrightEdelmanJohnson consider the derivative of

+

\[ +g(p) = f(A(p)^{-1} b) +\]

+

This might arise from applying a scalar-valued \(f\) to the solution of \(Ax = b\), where \(A\) is parameterized by \(p\).

+

The chain rule gives the following computation to find the derivative (or gradient):

+

\[ +\begin{align*} +dg +&= f'(x)[dx]\\ +&= f'(x) [d(A(p)^{1} b)]\\ +&= f'(x)[-A(p)^{-1} dA A(p)^{-1} b + 0]\\ +&= -f'(x) A(p)^{-1} dA A(p)^{-1} b. +\end{align*} +\]

+

By writing \(dA = A'(p)[dp]\) and setting \(v^T = f'(x)A(p)^{-1}\) this becomes

+

\[ +dg = -v^T dA A(p)^{-1} b = -v^T dA x +\]

+

This product of three terms can be computed in two directions:

+

From left to right:

+

First \(v\) is found by solving \(v^T = f'(x) A^{-1}\) through the solving of \(v = (A^{-1})^T (f'(x))^T = (A^T)^{-1} \nabla(f)\) or by solving \(A^T v = \nabla f\). This is called the adjoint equation.

+

The partial derivatives in \(g\) is related to each partial derivative of \(dA\) through:

+

\[ +\frac{\partial g}{\partial p_k} = -v^T\frac{\partial A}{\partial p_k} x, +\]

+

as the scalar factor commutes through. With \(v\) and \(x\) solved for (via the adjoint equation and from solving \(Ax=b\)) the partials in \(p_k\) are computed with dot products. There are just two costly operations.

+

From right to left:

+

The value of \(x\) can be solved for, as above, but computing the value of

+

\[ +\frac{\partial g}{\partial p_k} = +-f'(x) \left(A^{-1} \frac{\partial A}{\partial p_k} x \right) +\]

+

requires a costly solve for each \(p_k\), and \(p\) may have many components. As mentioned above, the reverse mode offers advantages when there are many input parameters (\(p\)) and a single output parameter.

+
+
Example
+

Suppose \(x(p)\) solves some system of equations \(h(x(p),p) = 0\) in \(R^n\) (\(n\) possibly just \(1\)) and \(g(p) = f(x(p))\) is some non-linear transformation of \(x\). What is the derivative of \(g\) in \(p\)?

+

Suppose the implicit function theorem applies to \(h(x,p) = 0\), that is – locally – there is an implicitly defined function \(x(p)\) with a derivative. Moreover by differentiating both sides it can be identified:

+

\[ +0 = \frac{\partial h}{\partial p} dp + \frac{\partial h}{\partial x} dx +\]

+

which can be solved for \(dx\) to give

+

\[ +dx = -\left(\frac{\partial h}{\partial x}\right)^{-1} \frac{\partial h}{\partial p} dp. +\]

+

The chain rule then gives

+

\[ +dg = f'(x) dx = -f'(x) \left(\frac{\partial h}{\partial x}\right)^{-1} \frac{\partial h}{\partial p} dp. +\]

+

This can be computed in two directions:

+

From left to right:

+

Call \(A =\left(\frac{\partial h}{\partial x}\right)^{-1}\). Then define \(v\) indirectly through \(v^T = f'(x) A^{-1}\). With this: \(v = (A^{-1})^T (f'(x))^T = (A^T)^{-1} \nabla{f}\) which is found by solving \(A^Tv = \nabla{f}\). Again, this is the adjoint equation.

+

The value of \(dA\) is related to each partial derivative for which

+

\[ +\frac{\partial g}{\partial p_k} = -v^T\frac{\partial A}{\partial p_k} x, +\]

+

as the scalar factor commutes through. With \(v\) and \(x\) solved for (via the adjoint equation and from solving \(Ax=b\)) the partials in \(p_k\) are computed with dot products.

+

However, from right to left, the value of \(x\) can be solved for, but computing the value of

+

\[ +\frac{\partial g}{\partial p_k} = +-f'(x) +\left(A^{-1} \frac{\partial A}{\partial p_k} x \right) +\]

+

requires a costly solve for each \(p_k\), and \(p\) may have many components. The reverse mode offers advantages when there are many input parameters (\(p\)) and a single output parameter.

+
+
+
Example
+

Suppose \(x(p)\) solves some system of equations \(h(x(p),p) = 0\) in \(R^n\) (\(n\) possibly just 1$) and \(g(p) = f(x(p))\) is some non-linear transformation of \(x\). What is the derivative of \(g\) in \(p\)?

+

Suppose the implicit function theorem applies to \(h(x,p) = 0\), that is locally the response \(x(p)\) has a derivative, and moreover by the chain rule

+

\[ +0 = \frac{\partial h}{\partial p} dp + \frac{\partial h}{\partial x} dx. +\]

+

Solving the above for \(dx\) gives:

+

\[ +dx = -\left(\frac{\partial h}{\partial x}\right)^{-1} \frac{\partial h}{\partial p} dp. +\]

+

The chain rule applied to \(g(p) = f(x(p))\) then yields

+

\[ +dg = f'(x) dx = - f'(x) \left(\frac{\partial h}{\partial x}\right)^{-1} \frac{\partial h}{\partial p} dp. +\]

+

Setting

+

\[ +v^T = -f'(x) \left(\frac{\partial h}{\partial x}\right)^{-1} +\]

+

then \(v\) can be solved from taking adjoints (as before). Let \(A = \partial h/\partial x\), the \(v^T = -f'(x) A^{-1}\) or \(v = -(A^{-1})^T (f'(x))^t= -(A^T)^{-1} \nabla f\). As before it would take two solves to get both \(g\) and its gradient.

+
+
+
+

Second derivatives, Hessian

+

@CarlssonNikitinTroedssonWendt

+

We reference a theorem presented by Carlsson, Nikitin, Troedsson, and Wendt for exposition with some modification

+
+
+
+ +
+
+

Theorem 1. Let \(f:X \rightarrow Y\), where \(X,Y\) are finite dimensional inner product spaces with elements in \(R\). Suppose \(f\) is smooth (a certain number of derivatives). Then for each \(x\) in \(X\) there exists a unique linear operator, \(f'(x)\), and a unique bilinear symmetric operator \(f'': X \oplus X \rightarrow Y\) such that

+

\[ +f(x + \delta x) = f(x) + f'(x)[\delta x] + \frac{1}{2}f''(x)[\delta x, \delta x] + \mathscr(||\delta x ||^2). +\]

+
+
+
+

New terms include bilinear, symmetric, and inner product. An operator (\(X\oplus X \rightarrow Y\)) is bilinear if it is a linear operator in each of its two arguments. Such an operator is symmetric if interchanging its two arguments makes no difference in its output. Finally, an inner product space is one with a generalization of the dot product. An inner product takes two vectors \(x\) and \(y\) and returns a scalar; it is denoted \(\langle x,y\rangle\); and has properties of symmetry, linearity, and non-negativity (\(\langle x,x\rangle \geq 0\), and equal \(0\) only if \(x\) is the zero vector.) Inner products can be used to form a norm (or length) for a vector through \(||x||^2 = \langle x,x\rangle\).

+

We reference this, as the values denoted \(f'\) and \(f''\) are unique. So if we identify them one way, we have identified them.

+

Specializing to \(X=R^n\) and \(Y=R^1\), we have, \(f'=\nabla f^T\) and \(f''\) is the Hessian.

+

Take \(n=2\). Previously we wrote a formula for Taylor’s theorem for \(f:R^n \rightarrow R\) that with \(n=2\) has with \(x=\langle x_1,x_2\rangle\):

+

\[ +\begin{align*} +f(x + dx) &= f(x) + +\frac{\partial f}{\partial x_1} dx_1 + \frac{\partial f}{\partial x_2} dx_2\\ +&+ \frac{1}{2}\left( +\frac{\partial^2 f}{\partial x_1^2}dx_1^2 + +\frac{\partial^2 f}{\partial x_1 \partial x_2}dx_1dx_2 + +\frac{\partial^2 f}{\partial x_2^2}dx_2^2 +\right) + \mathscr{o}(dx). +\end{align*} +\]

+

We can see \(\nabla{f} \cdot dx = f'(x) dx\) to tidy up part of the first line, and more over the second line can be seen to be a matrix product:

+

\[ +[dx_1 dx_2] +\begin{bmatrix} +\frac{\partial^2 f}{\partial x_1^2} & +\frac{\partial^2 f}{\partial x_1 \partial x_2}\\ +\frac{\partial^2 f}{\partial x_2 \partial x_1} & +\frac{\partial^2 f}{\partial x_2^2} +\end{bmatrix} +\begin{bmatrix} +dx_1\\ +dx_2 +\end{bmatrix} += dx^T H dx, +\]

+

\(H\) being the Hessian with entries \(H_{ij} = \frac{\partial f}{\partial x_i \partial x_j}\).

+

This formula – \(f(x+dx)-f(x) \approx f'(x)dx + dx^T H dx\) – is valid for any \(n\), showing \(n=2\) was just for ease of notation when expressing in the coordinates and not as matrices.

+

By uniqueness, we have under these assumptions that the Hessian is symmetric and the expression \(dx^T H dx\) is a bilinear form, which we can identify as \(f''(x)[dx,dx]\).

+

That the Hessian is symmetric could also be derived under these assumptions by directly computing that the mixed partials can have their order exchanged. But in this framework, as explained by @BrightEdelmanJohnson it is a result of the underlying vector space having an addition that is commutative (e.g. \(u+v = v+u\)).

+

The mapping \((u,v) \rightarrow u^T A v\) for a matrix \(A\) is bilinear. For a fixed \(u\), it is linear as it can be viewed as \((u^TA)[v]\) and matrix multiplication is linear. Similarly for a fixed \(v\).

+

@BrightEdelmanJohnson extend this characterization to a broader setting. The second derivative can be viewed as expressing first-order change in \(f'(x)\), a linear operator. The value \(df'\) has the same shape as \(f'\), which is a linear operator, so \(df'\) acts on vectors, say \(dx\), then:

+

\[ +df'[dx] = f''(x)[dx'][dx] = f''(x)[dx', dx] +\]

+

The prime in \(dx'\) is just notation, not a derivative operation for \(dx\).

+

With this view, we can see that \(f''(x)\) has two vectors it acts on. By definition it is linear in \(dx\). However, as \(f'(x)\) is a linear operator and the sum and product rules apply to derivatives, this operator is linear in \(dx'\) as well. So \(f''(x)\) is bilinear and as mentioned earlier symmetric.

+
+

Polarization

+

@BrightEdelmanJohnson interpret \(f''\) by looking at the image under \(f\) of \(x + dx + dx'\). If \(x\) is a vector, then this has a geometrical picture, from vector addtion, relating \(x + dx\), \(x+dx'\), and \(x + dx + dx'\).

+

The image for \(x +dx\) is to second order \(f(x) + f'(x)[dx] + (1/2)f''(x)[dx, dx]\), similarly \(x + dx'\) is to second order \(f(x) + f'(x)[dx'] + (1/2)f''(x)[dx', dx']\). The key formula for \(f''(x)\) is

+

\[ +\begin{align*} +f(x + dx + dx') &= f(x) + f'(x)[dx + dx'] + \frac{1}{2}f''(x)[dx, dx']\\ +&= f(x) + f'(x)[dx] + (1/2)f''(x)[dx, dx] +&+ f(x) + f'(x)[dx] + (1/2)f''(x)[dx', dx'] +&+ f''(x)[dx, dx'] +\end{align} +\]

+

This gives a means to compute \(f''\) in terms of \(f''\) acting on diagonal terms, where the two vectors are equal:

+

\[ +f''(x)[dx, dx'] = \frac{1}{2} f''(x)[dx+dx',dx+dx'] - f''(x)[dx,dx] - f''(x)[dx',dx'] +\]

+
+
+

XXX does this fit in?

+

However, as a description of second-order change in \(f\), we recover the initial terms in the Taylor series

+

\[ +f(x + \delta x) = f(x) + f'(x)\delta x + (1/2) f''(x)[\delta x, \delta x] + \mathscr{o}(||\delta x||^2). +\]

+
+
+

Examples

+
+
Example: second derivative of \(x^TAx\)
+

Consider an expression from earlier \(f(x) = x^T A x\) for some constant \(A\). Then \(f''\) is found by noting that \(f' = (\nabla f)^T = x^T(A + A^T)\), or \(\nabla f = (A^T + A)x\) and \(f'' = H = A^T + A\) is the Jacobian of the gradient.

+

By rearranging terms, it can be shown that \(f(x) = 1/2 x^THx = 1/2 f''[x,x]\).

+
+
+
Example: second derivative of \(\text{det}(A)\)
+

Consider \(f(A) = \text{det}(A)\). We saw previously that:

+

\[ +\begin{align*} +\text{tr}(A + B) &= \text{tr}(A) + \text{tr}(B)\\ +\text{det}(A + dA') &= \text{det}(A) + \text{det}(A)\text{tr}(A^{-1}dA')\\ +(A + dA') &= A^{-1} - A^{-1} dA' A^{-1} +\end{align*} +\]

+

These are all used to simplify:

+

\[ +\begin{align*} +\text{det}(A+dA')&\text{tr}((A + dA')^{-1} dA) - \text{det}(A) \text{tr}(A^{-1}dA) \\ +&= \left( +\text{det}(A) + \text{det}(A)\text{tr}(A^{-1}dA') +\right) +\text{tr}((A^{-1} - A^{-1}dA' A^{-1})dA) - \text{det}(A) \text{tr}(A^{-1}dA) \\ +&= +\text{det}(A) \text{tr}(A^{-1}dA)\\ +&+ \text{det}(A)\text{tr}(A^{-1}dA')\text{tr}(A^{-1}dA) \\ +&- \text{det}(A)\text{tr}(A^{-1}dA' A^{-1}dA)\\ +&- \text{det}(A)\text{tr}(A^{-1}dA')\text{tr}(A^{-1}dA' A^{-1}dA)\\ +&- \text{det}(A) \text{tr}(A^{-1}dA) \\ +&= \text{det}(A)\text{tr}(A^{-1}dA')\text{tr}(A^{-1}dA) - \text{det}(A)\text{tr}(A^{-1}dA' A^{-1}dA)\\ +&+ \text{third order term} +\end{align*} +\]

+

So, after dropping the third-order term, we see: \[ +\begin{align*} +f''(A)&[dA,dA'] \\ +&= \text{det}(A)\text{tr}(A^{-1}dA')\text{tr}(A^{-1}dA) +- \text{det}(A)\text{tr}(A^{-1}dA' A^{-1}dA). +\end{align*} +\]

+
+
+
+ +
+ + +
+ + + + + \ No newline at end of file diff --git a/quarto/staging/matrix-calculus-notes.qmd b/quarto/staging/matrix-calculus-notes.qmd new file mode 100644 index 0000000..a9e457e --- /dev/null +++ b/quarto/staging/matrix-calculus-notes.qmd @@ -0,0 +1,937 @@ +# Matrix Calculus + +XXX Add in examples from paper XXX +optimization? large number of parameters? ,... + +::: {.callout-note} +## Based on Bright, Edelman, and Johnson's notes + +This section samples material from the notes [Matrix Calculus (for Machine Learning and Beyond)](https://arxiv.org/abs/2501.14787) by Paige Bright, Alan Edelman, and Steven G. Johnson. These notes cover material taught in a course at MIT. Support materials for their course in `Julia` are available at [https://github.com/mitmath/matrixcalc/tree/main](https://github.com/mitmath/matrixcalc/tree/main). For more details and examples, please refer to the source. +::: + + +We have seen several "derivatives" of a function, based on the number of inputs and outputs. The first one was for functions $f: R \rightarrow R$. + +Then $f$ has a derivative at $x$ if this limit exists + +$$ +\lim_{h \rightarrow 0}\frac{f(x + h) - f(x)}{h}. +$$ + +The derivative of the function $x$ is this limit for a given $x$. Common notation is: + +$$ +f'(x) = \frac{dy}{dx} = \lim_{h \rightarrow 0}\frac{f(x + h) - f(x)}{h} +$$ + +(when the limit exists). + + +This limit gets expressed in different ways: + +* linearization write $f(x+\Delta x) - f(x) \approx f'(x)\Delta x$, where $\delta x$ is a small displacement from $x$. The reason there isn't equality is the unwritten higher order terms that vanish in a limit. + +* Alternate limits. Another way of writing this is in terms of explicit smaller order terms: + +$$ +(f(x+h) - f(x)) - f'(x)h = \mathscr{o}(h), +$$ + +which means if we divide both sides by $h$ and take the limit, we will get $0$ on the right and the relationship on the left. + +* Differential notation simply writes this as $dy = f(x)dx$. More verbosely, we might write + +$$ +df = f(x+dx) - f(x) = f'(x) dx. +$$ + +Here $dx$ is a differential, made rigorous by a limit, which hides the higher order terms. + + +In these notes the limit has been defined, with suitable modification, for functions of vectors (multiple values) with scalar or vector outputs. + + +For example, when $f: R \rightarrow R^m$ was a vector-valued function the derivative was defined similarly through a limit of $(f(t + \Delta t) - f(t))/{\Delta t}$, where each component needed to have a limit. This can be rewritten through $f(t + dt) - f(t) = f'(t) dt$, again using differentials to avoid the higher order terms. + +When $f: R^n \rightarrow R$ is a scalar-valued function of a derivative, differentiability was defined by a gradient existing with $f(c+h) - f(c) - \nabla{f}(c) \cdot h$ being $\mathscr{o}(\|h\|)$. In other words $df = f(c + dh) - f(c) = \nabla{f}(c) \cdot dh$. The gradient has the same shape as $c$, a column vector. If we take the row vector (e.g. $f'(c) = \nabla{f}(c)^T$) then again we see $df = f(c+dh) - f(c) = f'(c) dh$, where the last term uses matrix multiplication of a row vector times a column vector. + +Finally, when $f:R^n \rightarrow R^m$, the Jacobian was defined and characterized by +$\| f(x + dx) - f(x) - J_f(x)dx \|$ being $\mathscr{o}(\|dx\|)$. Again, we can express this through $df = f(x + dx) - f(x) = f'(x)dx$ where $f'(x) = J_f(x)$. + + +In writing $df = f(x + dx) - f(x) = f'(x) dx$ generically, some underlying facts are left implicit: $dx$ has the same shape as $x$ (so can be added); $f'(x) dx$ may mean usual multiplication or matrix multiplication; and there is an underlying concept of distance and size that allows the above to be rigorous. This may be an abolute value or a norm. + +Further, various differentiation rules apply such as the sum, product, and chain rule. + + +The @BrightEdelmanJohnson notes cover differentiation of functions in this uniform manner and then extend the form by treating derivatives as *linear operators*. + + +A [linear operator](https://en.wikipedia.org/wiki/Operator_(mathematics)) is a mathematical object which satisfies + +$$ +f[\alpha v + \beta w] = \alpha f[v] + \beta f[w]. +$$ + +where the $\alpha$ and $\beta$ are scalars, and $v$ and $w$ possibly not and come from a *vector space*. Regular multiplication and matrix multiplication are familiar linear operations, but there are many others. + +The referenced notes identify $f'(x) dx$ with $f'(x)[dx]$, the latter emphasizing $f'(x)$ acts on $dx$ and the notation is not commutative (e.g., it is not $dx f'(x)$). + +Linear operators are related to vector spaces. + +A [vector space](https://en.wikipedia.org/wiki/Vector_space) is a set of mathematical objects which can be added together and also multiplied by a scalar. Vectors of similar size, as previously discussed, are the typical example, with vector addition and scalar multiplication previously discussed topics. Matrices of similar size (and some subclasses) also form a vector space. Additionally, many other set of objects form vector spaces. An example might be polynomial functions of degree $n$ or less; continuous functions, or functions with a certain number of derivatives. + +Take differentiable functions as an example, then the simplest derivative rules $[af(x) + bg(x)]' = a[f(x)]' + b[g(x)]'$ show the linearity of the derivative in this setting. This linearity is different from how the derivative is a linear operator on $dx$. + +A vector space is described by a *basis* -- a minimal set of vectors needed to describe the space, after consideration of linear combinations. For many vectors, this the set of special vectors with $1$ as one of the entries, and $0$ otherwise. + +A key fact about a basis is every vector in the vector space can be expressed *uniquely* as a linear combination of the basis vectors. + +Vectors and matrices have properties that are generalizations of the real numbers. As vectors and matrices form vector spaces, the concept of addition of vectors and matrices is defined, as is scalar multiplication. Additionally, we have seen: + +* The dot product between two vectors of the same length is defined easily ($v\cdot w = \Sigma_i v_i w_i$). It is coupled with the length as $\|v\|^2 = v\cdot v$. + +* Matrix multiplication is defined for two properly sized matrices. If $A$ is $m \times k$ and $B$ is $k \times n$ then $AB$ is a $m\times n$ matrix with $(i,j)$ term given by the dot product of the $i$th row of $A$ (viewed as a vector) and the $j$th column of $B$ (viewed as a vector). Matrix multiplication is associative but *not* commutative. (E.g. $(AB)C = A(BC)$ but $AB$ and $BA$ need not be equal (or even defined, as the shapes may not match up). + +* A square matrix $A$ has an *inverse* $A^{-1}$ if $AA^{-1} = A^{-1}A = I$, where $I$ is the identity matrix (a matrix which is zero except on its diagonal entries which are all $1$). Square matrices may or may not have an inverse. When they don't the matrix is called singular. + +* Viewing a vector as a matrix is possible. The association is typically through a *column* vector. + +* The transpose of a matrix comes by permuting the rows and columns. The transpose of a column vector is a row vector, so $v\cdot w = v^T w$, where we use a superscript $T$ for the transpose. The transpose of a product, is the product of the transposes -- reversed: $(AB)^T = B^T A^T$; the tranpose of a transpose is an identity operation: $(A^T)^T = A$; the inverse of a transpose is the tranpose of the inverse: $(A^{-1})^T = (A^T){-1}$. + +* Matrices for which $A = A^T$ are called symmetric. + +* A few of the operations on matrices are the transpose and the inverse. These return a matrix, when defined. There is also the determinant and the trace, which return a scalar from a matrix. The trace is just the sum of the diagonal; the determinant is more involved to compute, but was previously seen to have a relationship to the volume of a certain parallellpiped. There are a few other operations described in the following. + +. + +## Scalar-valued functions of a vector + +Suppose $f: R^n \rightarrow R$, a scalar-valued function of a vector. Then the directional derivative at $x$ in the direction $v$ was defined for a scalar $\alpha$ by: + +$$ +\frac{\partial}{\partial \alpha}f(x + \alpha v) \mid_{\alpha = 0} = +\lim_{\Delta\alpha \rightarrow 0} \frac{f(x + \Delta\alpha v) - f(x)}{\Delta\alpha}. +$$ + +This rate of change in the direction of $v$ can be expressed through the linear operator $f'(x)$ via + +$$ +f(x + d\alpha v) - f(x) = f'(x) [d\alpha v] = d\alpha f'(x)[v], +$$ + +using linearity to move the scalar part outside the $[]$. This connects the partial derivative at $x$ in the direction of $v$ with $f'(x)$: + +$$ +\frac{\partial}{\partial \alpha}f(x + \alpha v) \mid_{\alpha = 0} = +f'(x)[v]. +$$ + + +Not only does this give a connection in notation with the derivative, it naturally illustrates how the derivative as a linear operator can act on non-infinitesimal values. + +Previously, we wrote $\nabla f \cdot v$ for the directional derivative, where the gradient is a column vector. The above uses the identification +$f' = (\nabla f)^T$. + +For $f: R^n \rightarrow R$ we have + +$$ +df = f(x + dx) - f(x) = f'(x) [dx] +$$ + +is a scalar, so if $dx$ is a column vector, $f'(x)$ is a row vector with the same number of components (just as $\nabla f$ is a column vector with the same number of components). + +##### Examples + +@BrightEdelmanJohnson include this example to show that the computation of derivatives using components can be avoided. Consider $f(x) = x^T A x$ where $x$ is a vector in $R^n$ and $A$ is an $n\times n$ matrix. Then $f: R^n \rightarrow R$ and its derivative can be computed: + +$$ +\begin{align*} +df &= f(x + dx) - f(x)\\ +&= (x + dx)^T A (x + dx) - x^TAx \\ +&= x^TAx + dx^TA x + x^TAx + dx^T A dx - x^TAx\\ +&= dx^TA x + x^TAdx \\ +&= (dx^TAx)^T + x^TAdx \\ +&= x^T A^T dx + x^T A dx\\ +&= x^T(A^T + A) dx +\end{align*} +$$ + +The term $dx^t A dx$ is dropped, as it is higher order (goes to zero faster), it containing two $dx$ terms. +In the second to last step, an identity operation (taking the transpose of the scalar quantity) is taken to simplify the algebra. Finally, as $df = f'(x)[dx]$ the identity of $f'(x) = x^T(A^T+A)$ is made, or taking transposes $\nabla f = (A + A^T)x$. + +Compare the elegance above, with the component version, even though simplified, it still requires a specification of the size to carry the following out: + +```{julia} +using SymPy +@syms x[1:3]::real A[1:3, 1:3]::real +u = x' * A * x +grad_u = [diff(u, xi) for xi in x] +``` + +Compare to the formula for the gradient just derived: + +```{julia} +grad_u_1 = (A + A')*x +``` + +The two are, of course, equal + +```{julia} +all(a == b for (a,b) ∈ zip(grad_u, grad_u_1)) +``` + + +---- + +For $f: R^n \rightarrow R^m$, @BrightEdelmanJohnson give an example of computing the Jacobian without resorting to component wise computations. Let $f(x) = Ax$ with $A$ being a $m \times n$ matrix, it follows that + +$$ +\begin{align*} +df &= f(x + dx) - f(x)\\ +&= A(x + dx) - Ax\\ +&= Adx\\ +&= f'(x) dx. +\end{align*} +$$ + +The Jacobian is the linear operator $A$ acting on $dx$. + + +## Sum and product rules for the derivative + +Using the differential notation -- which implicitly ignores higher order terms as they vanish in a limit -- the sum and product rules can be derived. + +For the sum rule, let $f(x) = g(x) + h(x)$. Then + +$$ +\begin{align*} +df &= f(x + dx) - f(x) \\ +&= f'(x) dx\\ +&= \left(g(x+dx) + h(x+dx)\right) - \left(g(x) + h(x)\right)\\ +&= \left(g(x + dx) - g(x)\right) + \left(h(x + dx) - h(x)\right)\\ +&= g'(x)dx + h'(x) dx\\ +&= \left(g'(x) + h'(x)\right) dx +\end{align*} +$$ + +Comparing we get $f'(x) = g'(x) + h'(x)$. + +The sum rule has the same derivation as was done with univariate, scalar functions. Similarly for the product rule. + +The product rule has with $f(x) = g(x)h(x)$ + +$$ +\begin{align*} +df &= f(x + dx) - f(x) \\ +&= g(x+dx)h(x + dx) - g(x) h(x)\\ +&= \left(g(x) + g'(x)dx\right)\left(h(x) + h'(x) dx\right) - \left(g(x) h(x)\right) \\ +&= g(x)h(x) + g'(x) dx h(x) + g(x) h'(x) dx + g'(x)dx h'(x) dx - g(x) h(x)\\ +&= gh + dg h + gdh + dg dh - gh\\ + &= dg h + gdh, +\end{align*} +$$ + +**after** dropping the higher order term and cancelling $gh$ terms of opposite signs in the fourth row. + + +##### Examples + +These two rules can be used to show the last two examples: + +First, to differentiate $f(x) = x^TAx$: + +$$ +\begin{align*} +df &= dx^T (Ax) + x^T d(Ax) \\ +&= x^T A^T dx + x^T A dx \\ +&= x^T(A^T + A) dx +\end{align*} +$$ + +Again, taking the transpose of the scalar quantity $x^TAdx$ to simplify the expression. + +When $A^T = A$ ($A$ is symmetric) this simplifies to a more familiar looking $2x^TA$, but we see that this requires assumptions not needed in the scalar case. + + +Next, if $f(x) = Ax$ then + +$$ +df = (dA)x + A(dx) = 0x + A dx = A dx, +$$ + +$A$ being a constant here. + +##### Example + +@BrightEdelmanJohnson consider what in `Julia` is `.*`. That is the operation: + +$$ +v .* w = +\begin{bmatrix} +v_1w_1 \\ +v_2w_2 \\ +\vdots\\ +v_nw_n +\end{bmatrix} += +\begin{bmatrix} +v_1 & 0 & \cdots & 0 \\ +0 & v_2 & \cdots & 0 \\ + & & \vdots & \\ +0 & 0 & \cdots & v_n +\end{bmatrix} +\begin{bmatrix} +w_1 \\ +w_2 \\ +\vdots\\ +w_n +\end{bmatrix} += \text{diag}(v) w. +$$ + +They compute the derivative of $f(x) = A(x .* x)$ for some fixed matrix $A$ of the proper size. + +We can see that $d (\text{diag}(v)w) = d(\text{diag}(v)) w + \text{diag}(v) dw = (dx) .* w + x .* dw$. So + +$df = A(dx .* x + x .* dx) = 2A(x .* dx)$, as $.*$ is commutative by its definition. Writing this as $df = 2A(x .* dx) = 2A(\text{diag}(x) dx) = (2A\text{diag}(x)) dx$, we identify $f'(x) = 2A\text{diag}(x)$. + + +This operation is called the [Hadamard product](https://en.wikipedia.org/wiki/Hadamard_product_(matrices)) and it extends to matrices and arrays. + + +## The chain rule + +Like the product rule, the chain rule is shown by @BrightEdelmanJohnson in this notation with $f(x) = g(h(x))$: + +$$ +\begin{align*} +df &= f(x + dx) - f(x)\\ +&= g(h(x + dx)) - g(h(x))\\ +&= g(h(x) + h'(x)[dx]) - g(h(x))\\ +&= g'(h(x)) [h'(x) [dx]]\\ +&= (g'(h(x)) h'(x)) [dx] +\end{align*} +$$ + +(The limit requires a bit more detail.) + +The operator $f'(x)= g'(h(x)) h'(x)$ is a product of matrices. + +### Computational differences with expressions from the chain rule + +Of note here is the application of the chain rule to three (or more compositions): + +The derivative of $f(x) = a(x) b(x) c(x)$ can be expressed as + +$$ +f' = (a'b')c' \text{ or } f' = a'(b'c') +$$ + +Multiplying left to right (the first) is called reverse mode; multiplying right to left (the second) is called forward mode. The distinction becomes important when considering the computational cost of the multiplications. + +* If $f: R^n \rightarrow R^m$ has $n$ much bigger than $1$ and $m=1$, then it is much faster to do left to right multiplication +* if $f:R^n \rightarrow R^m$ has $n=1$ and $m$ much bigger than one, the it is faster to do right to left multiplication. + + +The basic idea comes down to the shape of the matrices. When $m=1$, the derviative is a product of matrices of size $n\times j$ $j\times k$ and $k \times 1$ yielding a matrix of size $n \times 1$ matching the function dimension. Matrix multiplication of an $m \times q$ times $q \times n$ takes an order of $mqn$ operations. The multiplication of left to right is then + +The first operation takes $njk$ operation leaving an $n\times k$ matrix, the next multiplication then takes another $nk1$ operations or $njk + nk$ together. Whereas computing from the right to left is first $jk1$ operations leaving a $j \times 1$ matrix. The next operation would take another $nk1$ operations. In totalL + +* left to right is $njk + nk$ = $nk \cdot (1 + j)$. +* right to left is $jk + j = j\cdot (k+1)$. + +When $j=k$, say, we can compare and see the second is a factor less in terms of operations. This can be quite significant in higher dimensions, whereas the dimensions of calculus (where $n$ and $m$ are $3$ or less) it is not an issue. + + +##### Example + +Using the `BenchmarkTools` package, we can check the time to compute various products: + +```{julia} +using BenchmarkTools +n,j,k,m = 20,15,10,1 +@btime A*(B*C) setup=(A=rand(n,j);B=rand(j,k); C=rand(k,m)); +@btime (A*B)*C setup=(A=rand(n,j);B=rand(j,k); C=rand(k,m)); +``` + +The latter computation is about 1.5 times slower. + +Whereas the relationship is changed when the first matrix is skinny and the last is not: + +```{julia} +@btime A*(B*C) setup=(A=rand(m,k);B=rand(k,j); C=rand(j,n)); +@btime (A*B)*C setup=(A=rand(m,k);B=rand(k,j); C=rand(j,n)); +``` + +##### Example + +In calculus, we have $n$ and $m$ are $1$,$2$,or $3$. But that need not be the case, especially if differentiation is over a parameter space. + +XXXX +(Maybe the ariplain wing, but please, something origi + + + +## Derivatives of matrix functions + +What is the the derivative of $f(A) = A^2$? + +The function $f$ takes a $n\times n$ matrix and returns a matrix of the same size. This innocuous question isn't directly +handled by the Jacobian, which is defined for vector valued function $f:R^n \rightarrow R^m$. + +This derivative can be derived directly from the *product rule*: + +$$ +\begin{align*} +f(A) &= [AA]'\\ +&= A dA + dA A +\end{align*} +$$ + +That is $f'(A)$ is the operator $f'(A)[\delta A] = A \delta A + \delta A A$ and not $2A\delta A$, as $A$ may not commute with $\delta A$. + + +### Vectorization of a matrix + +Alternatively, we can identify $A$ through its +components, as a vector in $R^{n^2}$ and then leverage the Jacobian. + +One such identification is vectorization -- consecutively stacking the +column vectors into a vector. In `Julia` the `vec` function does this +operation: + +```{julia} +@syms A[1:2, 1:2] +vec(A) +``` + +The stacking by column follows how `Julia` stores matrices and how `Julia` references a matrices entries by linear index: + +```{julia} +vec(A) == [A[i] for i in eachindex(A)] +``` + +With this vectorization operation, $f$ may be viewed as +$\tilde{f}:R^{n^2} \rightarrow R^{n^2}$ through: + + +$$ +\tilde{f}(\text{vec}(A)) = \text{vec}(f(A)) +$$ + +We use `SymPy` to compute the Jacobian of this vector valued function. + +```{julia} +@syms A[1:3, 1:3]::real +f(x) = x^2 +J = vec(f(A)).jacobian(vec(A)) # jacobian of f̃ +``` + + +We do this via linear algebra first, then see a more elegant manner following the notes. + +A basic course in linear algebra shows that any linear operator on a finite vector space can be represented as a matrix. The basic idea is to represent what the operator does to each *basis* element and put these values as columns of the matrix. + + +In this $3 \times 3$ case, the linear operator works on an object with $9$ slots and returns an object with $9$ slots, so the matrix will be $9 \times 9$. + +The basis elements are simply the matrices with a $1$ in spot $(i,j)$ and zero elsewhere. Here we generate them through a function: + +```{julia} +basis(i,j,A) = (b=zeros(Int, size(A)...); b[i,j] = 1; b) +JJ = [vec(basis(i,j,A)*A + A*basis(i,j,A)) for j in 1:3 for i in 1:3] +``` + +The elements of `JJ` show the representation of each of the $9$ basis elements under the linear transformation. + +To construct the matrix representing the linear operator, we need to concatenate these horizontally as column vectors + +```{julia} +JJ = hcat(JJ...) +``` + +The matrix $JJ$ is identical to $J$, above: + +```{julia} +all(j == jj for (j, jj) in zip(J, JJ)) +``` + +### Kronecker products + +But how can we see the Jacobian, $J$, from the linear operator $f'(A)[\delta A] = \delta A A + A \delta A$? + +To make this less magical, a related operation to `vec` is defined. + +The $\text{vec}$ function takes a matrix and stacks its columns. + +The $\text{vec}$ function can turn a matrix into a vector, so it can be used for finding the Jacobian, as above. However the shape of the matrix is lost, as are the fundamental matrix operations, like multiplication. + + +The [Kronecker product](https://en.wikipedia.org/wiki/Kronecker_product) replicates values making a bigger matrix. That is, if $A$ and $B$ are matrices, the Kronecker product replaces each value in $A$ with that value times $B$, making a bigger matrix, as each entry in $A$ is replaced by an entry with size $B$. + +Formally, + + +$$ +A \otimes B = +\begin{bmatrix} +a_{11}B & a_{12}B & \cdots & a_{1n}B \\ +a_{21}B & a_{22}B & \cdots & a_{2n}B \\ + &\vdots & & \\ +a_{m1}B & a_{m2}B & \cdots & a_{mn}B +\end{bmatrix} +$$ + +The function `kron` forms this product: + +```{julia} +@syms A[1:2, 1:3] B[1:3, 1:4] +kron(A, B) # same as hcat((vcat((A[i,j]*B for i in 1:2)...) for j in 1:3)...) +``` + + + +The $m\times n$ matrix $A$ and $j \times k$ matrix $B$ has a Kronecker product with size $mj \times nk$. + +The Kronecker product has a certain algebra, including: + +* transposes: $(A \otimes B)^T) = A^T \otimes B^T$ +* multiplication: $(A\otimes B)(C \otimes D) = (AC) \otimes (BD)$ +* inverses: $(A \otimes B)^{-1} = (A^{-1}) \otimes (B^{-1})$ +* orthogonal: $(A\otimes B)^T = (A\otimes B)$ if both $A$ and $B$ has the same property +* determinants: $\det(A\otimes B) = \det(A)^m \det(B)^n$, where $A$ is $n\times n$, $B$ is $m \times m$. +* trace (sum of diagonal): $\text{tr}(A \otimes B) = \text{tr}(A)\text{tr}(B)$. + +The main equation coupling `vec` and `kron` is the fact that if $A$, $B$, and $C$ have appropriate sizes, then: + +$$ +(A \otimes B) \text{vec}(C) = \text{vec}(B C A^T). +$$ + +Appropriate sizes for $A$, $B$, and $C$ are determined by the various products in $BCA^T$. + +If $A$ is $m \times n$ and $B$ is $r \times s$, then since $BC$ is defined, $C$ has $s$ rows, and since $CA^T$ is defined, $C$ must have $n$ columns, as $A^T$ is $n \times m$, so $C$ must be $s\times n$. Checking this is correct on the other side, $A \times B$ would be size $mr \times ns$ and $\vec{C}$ would be size $sn$, so that product works, size wise. + +The referred to notes have an explanation for this formula, but we confirm with an example with $m=n-2$, $r=s=3$: + +```{julia} +@syms A[1:2, 1:2]::real B[1:3, 1:3]::real C[1:3, 1:2]::real +L, R = kron(A,B)*vec(C), vec(B*C*A') +all(l == r for (l, r) ∈ zip(L, R)) +``` + +---- + +Now to use this relationship to recognize $df = A dA + dA A$ with the Jacobian computed from $\text{vec}{f(a)}$. + +We have $\text{vec}(A dA + dA A) = \text{vec}(A dA) + \text{vec}(dA A)$, by obvious linearity of $\text{vec}$. Now inserting an identity matrix, $I$, which is symmteric, we have: + +$$ +\text{vec}(A dA) = \text{vec}(A dA I^T) = (I \otimes A) \text{vec}(dA), +$$ + +and + +$$ +\text{vec}(dA A) = \text{vec}(I dA (A^T)^T) = (A^T \otimes I) \text{vec}(dA) +$$ + +This leaves + +$$ +\text{vec}(A dA + dA A) = +\left((I \otimes A) + (A^T \otimes I)\right) \text{vec}(dA) +$$ + +We should then get the Jacobian we computed from the following: + +```{julia} +@syms A[1:3, 1:3]::real +using LinearAlgebra: I +J = vec(A^2).jacobian(vec(A)) +JJ = kron(I(3), A) + kron(A', I(3)) +all(j == jj for (j,jj) in zip(J,JJ)) +``` + +This technique can also be used with other powers, say $f(A) = A^3$, where the resulting $df = A^2 dA + A dA A + dA A^2$ is one answer that can be compared to a Jacobian through + +$$ +\begin{align*} +df &= \text{vec}(A^2 dA I^T) + \text{vec}(A dA A) + \text{vec}(I dA A^2)\\ +&= (I \otimes A^2)\text{vec}(dA) + (A^T \otimes A) \text{vec}(dA) + ((A^T)^2 \otimes I) \text{vec}(dA) +\end{align*} +$$ + +The above shows how to relate the derivative of a matrix function to +the Jacobian of a vectorized function, but only for illustration. It +is decidely not necessary to express the derivative of $f$ in terms of +the derivative of its vectorized counterpart. + + +##### Example: derivative of the inverse + +What is the derivative of $f(A) = A^{-1}$. When $A$ is a scalar, we related it to the reciprocal of the derivative of $f$ at some other point. The same technique is available. Starting with $I = AA^{-1}$ and noting $dI$ is $0$ we have + +$$ +\begin{align*} +0 &= d(AA^{-1})\\ +&= dAA^{-1} + A d(A^{-1}) +\end{align*} +$$ + +So, $d(A^{-1}) = -A^{-1} dA A^{-1}$. + + +This could be re-expressed as a linear operator through + +$$ +\text{vec}(dA^{-1}) = +\left((A^{-1})^T \otimes A^{-1}\right) \text{vec}(dA) += \left((A^T)^{-1} \otimes A^{-1}\right) \text{vec}(dA). +$$ + +##### Example: derivative of the determinant + +Let $f(A) = \text{det}(A)$. What is the derivative? + +First, the determinant of a square, $n\times n$, matrix $A$ is a scalar summary of $A$ with different means to compute it, but one recursive one in particular is helpful here: + +$$ +\text{det}(A) = a_{1j}C_{1j} + a_{2j}C_{2j} + \cdots a_{nj}C_{nj} +$$ + +for any $j$. The *cofactor* $C_{ij}$ is the determinant of the $(n-1)\times(n-1)$ matrix with the $i$th row and $j$th column deleted times $(-1)^{i+j}$. + +To find the *gradient* of $f$, we differentiate by each of the $A_{ij}$ variables, and so + +$$ +\frac{\partial\text{det}(A)}{\partial A_{ij}} = +\frac{\partial (a_{1j}C_{1j} + a_{2j}C_{2j} + \cdots a_{nj}C_{nj})}{\partial A_{ij}} = +C_{ij}, +$$ + +as each cofactor in the expansion has no dependence on $A_{ij}$ as the cofactor removes the $i$th row and $j$th column. + +So the gradient is the matrix of cofactors. + +@BrightEdelmanJohnson also give a different proof, starting with this observation + +$$ +\text{det}(I + dA) - \text{det}(I) = \text{tr}(dA) +$$ + +Assuming that, then by the fact $\text{det}(AB) = \text{det}(A)\text{det}(B)$: + +$$ +\begin{align*} +\text{det}(A + A(A^{-1}dA)) - \text{det}(A) &= \text{det}(A)\cdot(\text{det}(I+ A^{-1}dA) - \text{det}(I)) \\ +&= \text{det}(A) \text{tr}(A^{-1}dA)\\ +&= \text{tr}(\text{det}(A)A^{-1}dA)\\ +\end{align*} +$$ + +This agrees through a formula to compute the inverse of a matrix through its cofactor matrix divided by its determinant. + +That the trace gets involved, can be seen from this computation, which shows the only first-order terms are from the diagonal sum: + +```{julia} +using LinearAlgebra +@syms dA[1:2, 1:2] +det(I + dA) - det(I) +``` + +## The adjoint method + +The chain rule brings about a series of products. The adjoint method illustrated below, shows how to approach the computation of the series in a direction that minimizes the computational cost, illustrating why reverse mode is preferred to forward mode when a scalar function of several variables is considered. + + +@BrightEdelmanJohnson consider the derivative of + +$$ +g(p) = f(A(p)^{-1} b) +$$ + +This might arise from applying a scalar-valued $f$ to the solution of $Ax = b$, where $A$ is parameterized by $p$. + +The chain rule gives the following computation to find the derivative (or gradient): + +$$ +\begin{align*} +dg +&= f'(x)[dx]\\ +&= f'(x) [d(A(p)^{1} b)]\\ +&= f'(x)[-A(p)^{-1} dA A(p)^{-1} b + 0]\\ +&= -f'(x) A(p)^{-1} dA A(p)^{-1} b. +\end{align*} +$$ + +By writing $dA = A'(p)[dp]$ and setting $v^T = f'(x)A(p)^{-1}$ this becomes + +$$ +dg = -v^T dA A(p)^{-1} b = -v^T dA x +$$ + +This product of three terms can be computed in two directions: + + From left to right: + + First $v$ is found by solving $v^T = f'(x) A^{-1}$ through +the solving of +$v = (A^{-1})^T (f'(x))^T = (A^T)^{-1} \nabla(f)$ +or by solving $A^T v = \nabla f$. This is called the *adjoint* equation. + +The partial derivatives in $g$ is related to each partial derivative of $dA$ through: + +$$ +\frac{\partial g}{\partial p_k} = -v^T\frac{\partial A}{\partial p_k} x, +$$ + +as the scalar factor commutes through. With $v$ and $x$ solved for (via the adjoint equation and from solving $Ax=b$) the partials in $p_k$ are computed with dot products. There are just two costly operations. + +From right to left: + +The value of $x$ can be solved for, as above, but computing the value of + +$$ +\frac{\partial g}{\partial p_k} = +-f'(x) \left(A^{-1} \frac{\partial A}{\partial p_k} x \right) +$$ + +requires a costly solve for each $p_k$, and $p$ may have many components. As mentioned above, the reverse mode offers advantages when there are many input parameters ($p$) and a single output parameter. + +##### Example + + +Suppose $x(p)$ solves some system of equations $h(x(p),p) = 0$ in $R^n$ ($n$ possibly just $1$) and $g(p) = f(x(p))$ is some non-linear transformation of $x$. What is the derivative of $g$ in $p$? + +Suppose the *implicit function theorem* applies to $h(x,p) = 0$, that is -- *locally* -- there is an implicitly defined function $x(p)$ with a derivative. Moreover by differentiating both sides it can be identified: + +$$ +0 = \frac{\partial h}{\partial p} dp + \frac{\partial h}{\partial x} dx +$$ + +which can be solved for $dx$ to give + +$$ +dx = -\left(\frac{\partial h}{\partial x}\right)^{-1} \frac{\partial h}{\partial p} dp. +$$ + +The chain rule then gives + +$$ +dg = f'(x) dx = -f'(x) \left(\frac{\partial h}{\partial x}\right)^{-1} \frac{\partial h}{\partial p} dp. +$$ + +This can be computed in two directions: + +From left to right: + +Call $A =\left(\frac{\partial h}{\partial x}\right)^{-1}$. Then define $v$ indirectly through $v^T = f'(x) A^{-1}$. With this: +$v = (A^{-1})^T (f'(x))^T = (A^T)^{-1} \nabla{f}$ +which is found by solving +$A^Tv = \nabla{f}$. +Again, this is the *adjoint* equation. + +The value of $dA$ is related to each partial derivative for which + +$$ +\frac{\partial g}{\partial p_k} = -v^T\frac{\partial A}{\partial p_k} x, +$$ + +as the scalar factor commutes through. With $v$ and $x$ solved for (via the adjoint equation and from solving $Ax=b$) the partials in $p_k$ are computed with dot products. + +However, from right to left, the value of $x$ can be solved for, but computing the value of + +$$ +\frac{\partial g}{\partial p_k} = +-f'(x) +\left(A^{-1} \frac{\partial A}{\partial p_k} x \right) +$$ + +requires a costly solve for each $p_k$, and $p$ may have many components. The reverse mode offers advantages when there are many input parameters ($p$) and a single output parameter. + +##### Example + + +Suppose $x(p)$ solves some system of equations $h(x(p),p) = 0$ in $R^n$ ($n$ possibly just 1$) and $g(p) = f(x(p))$ is some non-linear transformation of $x$. What is the derivative of $g$ in $p$? + +Suppose the *implicit function theorem* applies to $h(x,p) = 0$, that is *locally* the response $x(p)$ has a derivative, and moreover by the chain rule + +$$ +0 = \frac{\partial h}{\partial p} dp + \frac{\partial h}{\partial x} dx. +$$ + +Solving the above for $dx$ gives: + +$$ +dx = -\left(\frac{\partial h}{\partial x}\right)^{-1} \frac{\partial h}{\partial p} dp. +$$ + +The chain rule applied to $g(p) = f(x(p))$ then yields + +$$ +dg = f'(x) dx = - f'(x) \left(\frac{\partial h}{\partial x}\right)^{-1} \frac{\partial h}{\partial p} dp. +$$ + +Setting + +$$ +v^T = -f'(x) \left(\frac{\partial h}{\partial x}\right)^{-1} +$$ + +then $v$ can be solved from taking adjoints (as before). Let $A = \partial h/\partial x$, the $v^T = -f'(x) A^{-1}$ or $v = -(A^{-1})^T (f'(x))^t= -(A^T)^{-1} \nabla f$. As before it would take two solves to get both $g$ and its gradient. + + +## Second derivatives, Hessian + +@CarlssonNikitinTroedssonWendt + +We reference a theorem presented by [Carlsson, Nikitin, Troedsson, and Wendt](https://arxiv.org/pdf/2502.03070v1) for exposition with some modification + +::: {.callout-note appearance="minimal"} +Theorem 1. Let $f:X \rightarrow Y$, where $X,Y$ are finite dimensional *inner product* spaces with elements in $R$. Suppose $f$ is smooth (a certain number of derivatives). Then for each $x$ in $X$ there exists a unique linear operator, $f'(x)$, and a unique *bilinear* *symmetric* operator $f'': X \oplus X \rightarrow Y$ such that + +$$ +f(x + \delta x) = f(x) + f'(x)[\delta x] + \frac{1}{2}f''(x)[\delta x, \delta x] + \mathscr(||\delta x ||^2). +$$ + +::: + +New terms include *bilinear*, *symmetric*, and *inner product*. An operator ($X\oplus X \rightarrow Y$) is bilinear if it is a linear operator in each of its two arguments. Such an operator is *symmetric* if interchanging its two arguments makes no difference in its output. Finally, an *inner product* space is one with a generalization of the dot product. An inner product takes two vectors $x$ and $y$ and returns a scalar; it is denoted $\langle x,y\rangle$; and has properties of symmetry, linearity, and non-negativity ($\langle x,x\rangle \geq 0$, and equal $0$ only if $x$ is the zero vector.) Inner products can be used to form a norm (or length) for a vector through $||x||^2 = \langle x,x\rangle$. + + +We reference this, as the values denoted $f'$ and $f''$ are *unique*. So if we identify them one way, we have identified them. + +Specializing to $X=R^n$ and $Y=R^1$, we have, $f'=\nabla f^T$ and $f''$ is the Hessian. + +Take $n=2$. Previously we wrote a formula for Taylor's theorem for $f:R^n \rightarrow R$ that with $n=2$ has with $x=\langle x_1,x_2\rangle$: + +$$ +\begin{align*} +f(x + dx) &= f(x) + +\frac{\partial f}{\partial x_1} dx_1 + \frac{\partial f}{\partial x_2} dx_2\\ +&+ \frac{1}{2}\left( +\frac{\partial^2 f}{\partial x_1^2}dx_1^2 + +\frac{\partial^2 f}{\partial x_1 \partial x_2}dx_1dx_2 + +\frac{\partial^2 f}{\partial x_2^2}dx_2^2 +\right) + \mathscr{o}(dx). +\end{align*} +$$ + + +We can see $\nabla{f} \cdot dx = f'(x) dx$ to tidy up part of the first line, and more over the second line can be seen to be a matrix product: + +$$ +[dx_1 dx_2] +\begin{bmatrix} +\frac{\partial^2 f}{\partial x_1^2} & +\frac{\partial^2 f}{\partial x_1 \partial x_2}\\ +\frac{\partial^2 f}{\partial x_2 \partial x_1} & +\frac{\partial^2 f}{\partial x_2^2} +\end{bmatrix} +\begin{bmatrix} +dx_1\\ +dx_2 +\end{bmatrix} += dx^T H dx, +$$ + +$H$ being the *Hessian* with entries $H_{ij} = \frac{\partial f}{\partial x_i \partial x_j}$. + +This formula -- $f(x+dx)-f(x) \approx f'(x)dx + dx^T H dx$ -- is valid for any $n$, showing $n=2$ was just for ease of notation when expressing in the coordinates and not as matrices. + +By uniqueness, we have under these assumptions that the Hessian is *symmetric* and the expression $dx^T H dx$ is a *bilinear* form, which we can identify as $f''(x)[dx,dx]$. + +That the Hessian is symmetric could also be derived under these assumptions by directly computing that the mixed partials can have their order exchanged. But in this framework, as explained by @BrightEdelmanJohnson it is a result of the underlying vector space having an addition that is commutative (e.g. $u+v = v+u$). + +The mapping $(u,v) \rightarrow u^T A v$ for a matrix $A$ is bilinear. For a fixed $u$, it is linear as it can be viewed as $(u^TA)[v]$ and matrix multiplication is linear. Similarly for a fixed $v$. + +@BrightEdelmanJohnson extend this characterization to a broader setting. +The second derivative can be viewed as expressing first-order change in $f'(x)$, a linear operator. The value $df'$ has the same shape as $f'$, which is a linear operator, so $df'$ acts on vectors, say $dx$, then: + +$$ +df'[dx] = f''(x)[dx'][dx] = f''(x)[dx', dx] +$$ + +The prime in $dx'$ is just notation, not a derivative operation for $dx$. + +With this view, we can see that $f''(x)$ has two vectors it acts on. By definition it is linear in $dx$. However, as $f'(x)$ is a linear operator and the sum and product rules apply to derivatives, this operator is linear in $dx'$ as well. So $f''(x)$ is bilinear and as mentioned earlier symmetric. + +### Polarization + +@BrightEdelmanJohnson interpret $f''$ by looking at the image under $f$ of $x + dx + dx'$. If $x$ is a vector, then this has a geometrical picture, from vector addtion, relating $x + dx$, $x+dx'$, and $x + dx + dx'$. + +The image for $x +dx$ is to second order $f(x) + f'(x)[dx] + (1/2)f''(x)[dx, dx]$, similarly $x + dx'$ is to second order $f(x) + f'(x)[dx'] + (1/2)f''(x)[dx', dx']$. The key formula for $f''(x)$ is + +$$ +\begin{align*} +f(x + dx + dx') &= f(x) + f'(x)[dx + dx'] + \frac{1}{2}f''(x)[dx, dx']\\ +&= f(x) + f'(x)[dx] + (1/2)f''(x)[dx, dx] +&+ f(x) + f'(x)[dx] + (1/2)f''(x)[dx', dx'] +&+ f''(x)[dx, dx'] +\end{align} +$$ + +This gives a means to compute $f''$ in terms of $f''$ acting on diagonal terms, where the two vectors are equal: + +$$ +f''(x)[dx, dx'] = \frac{1}{2} f''(x)[dx+dx',dx+dx'] - f''(x)[dx,dx] - f''(x)[dx',dx'] +$$ + +### XXX does this fit in? + +However, as a description of second-order change in $f$, we recover the initial terms in the Taylor series + +$$ +f(x + \delta x) = f(x) + f'(x)\delta x + (1/2) f''(x)[\delta x, \delta x] + \mathscr{o}(||\delta x||^2). +$$ + +### Examples + +##### Example: second derivative of $x^TAx$ + +Consider an expression from earlier $f(x) = x^T A x$ for some constant $A$. Then $f''$ is found by noting that $f' = (\nabla f)^T = x^T(A + A^T)$, or $\nabla f = (A^T + A)x$ and $f'' = H = A^T + A$ is the Jacobian of the gradient. + +By rearranging terms, it can be shown that $f(x) = 1/2 x^THx = 1/2 f''[x,x]$. + +##### Example: second derivative of $\text{det}(A)$ + +Consider $f(A) = \text{det}(A)$. We saw previously that: + +$$ +\begin{align*} +\text{tr}(A + B) &= \text{tr}(A) + \text{tr}(B)\\ +\text{det}(A + dA') &= \text{det}(A) + \text{det}(A)\text{tr}(A^{-1}dA')\\ +(A + dA') &= A^{-1} - A^{-1} dA' A^{-1} +\end{align*} +$$ + +These are all used to simplify: + +$$ +\begin{align*} +\text{det}(A+dA')&\text{tr}((A + dA')^{-1} dA) - \text{det}(A) \text{tr}(A^{-1}dA) \\ +&= \left( +\text{det}(A) + \text{det}(A)\text{tr}(A^{-1}dA') +\right) +\text{tr}((A^{-1} - A^{-1}dA' A^{-1})dA) - \text{det}(A) \text{tr}(A^{-1}dA) \\ +&= +\text{det}(A) \text{tr}(A^{-1}dA)\\ +&+ \text{det}(A)\text{tr}(A^{-1}dA')\text{tr}(A^{-1}dA) \\ +&- \text{det}(A)\text{tr}(A^{-1}dA' A^{-1}dA)\\ +&- \text{det}(A)\text{tr}(A^{-1}dA')\text{tr}(A^{-1}dA' A^{-1}dA)\\ +&- \text{det}(A) \text{tr}(A^{-1}dA) \\ +&= \text{det}(A)\text{tr}(A^{-1}dA')\text{tr}(A^{-1}dA) - \text{det}(A)\text{tr}(A^{-1}dA' A^{-1}dA)\\ +&+ \text{third order term} +\end{align*} +$$ + +So, after dropping the third-order term, we see: +$$ +\begin{align*} +f''(A)&[dA,dA'] \\ +&= \text{det}(A)\text{tr}(A^{-1}dA')\text{tr}(A^{-1}dA) +- \text{det}(A)\text{tr}(A^{-1}dA' A^{-1}dA). +\end{align*} +$$ From b5f03009210a075b86980f118d6d00ded7e6d648 Mon Sep 17 00:00:00 2001 From: jverzani Date: Wed, 23 Apr 2025 11:35:45 -0400 Subject: [PATCH 02/22] WIP --- .../vector_fields.qmd | 32 +++++- quarto/staging/matrix-calculus-notes.html | 104 +++++++++--------- quarto/staging/matrix-calculus-notes.qmd | 49 +++++---- 3 files changed, 113 insertions(+), 72 deletions(-) diff --git a/quarto/differentiable_vector_calculus/vector_fields.qmd b/quarto/differentiable_vector_calculus/vector_fields.qmd index b8172b6..3e3363f 100644 --- a/quarto/differentiable_vector_calculus/vector_fields.qmd +++ b/quarto/differentiable_vector_calculus/vector_fields.qmd @@ -24,7 +24,7 @@ For a scalar function $f: R^n \rightarrow R$, the gradient of $f$, $\nabla{f}$, | $f: R\rightarrow R$ | univariate | familiar graph of function | $f$ | | $f: R\rightarrow R^m$ | vector-valued | space curve when n=2 or 3 | $\vec{r}$, $\vec{N}$ | | $f: R^n\rightarrow R$ | scalar | a surface when n=2 | $f$ | -| $F: R^n\rightarrow R^n$ | vector field | a vector field when n=2 | $F$ | +| $F: R^n\rightarrow R^n$ | vector field | a vector field when n=2, 3| $F$ | | $F: R^n\rightarrow R^m$ | multivariable | n=2,m=3 describes a surface | $F$, $\Phi$ | @@ -34,7 +34,9 @@ After an example where the use of a multivariable function is of necessity, we d ## Vector fields -We have seen that the gradient of a scalar function, $f:R^2 \rightarrow R$, takes a point in $R^2$ and associates a vector in $R^2$. As such $\nabla{f}:R^2 \rightarrow R^2$ is a vector field. A vector field can be visualized by sampling a region and representing the field at those points. The details, as previously mentioned, are in the `vectorfieldplot` function of `CalculusWithJulia`. +We have seen that the gradient of a scalar function, $f:R^2 \rightarrow R$, takes a point in $R^2$ and associates a vector in $R^2$. As such $\nabla{f}:R^2 \rightarrow R^2$ is a vector field. A vector field is a vector-valued function from $R^n \rightarrow R^n$ for $n \geq 2$. + +An input/output pair can be visualized by identifying the input values as a point, and the output as a vector visualized by anchoring the vector at the point. A vector field is a sampling of such pairs, usually taken over some ordered grid. The details, as previously mentioned, are in the `vectorfieldplot` function of `CalculusWithJulia`. ```{julia} @@ -1071,7 +1073,7 @@ Adjusting $f$ to have a vanishing second -- but not third -- derivative at $c_0$ As for $g_1$, we have by construction $g_1(b_0) = 0$. By differentiation we get a pattern for some constants $c_j = (j+1)\cdot(j+2)\cdots \cdot k$ with $c_k = 1$. $$ -g^{(k)}(b) = k! \cdot \frac{f(a_0) - f(b)}{(a_0-b)^{k+1}} - \sum_{j=1}^k c_j \frac{f^{(j)}(b)}{(a_0 - b)^{k-j+1}}. +g^{(k)}(b) = k! \cdot \frac{f(a_0) - f(b)}{(a_0-b)^{k+1}} - \sum_{j=1}^k c_j \frac{f^{(j)}(b)}{(a_0 - b)^{k-j+1}}. $$ Of note that when $f(a_0) = f(b_0) = 0$ that if $f^{(k)}(b_0)$ is the first non-vanishing derivative of $f$ at $b_0$ that $g^{(k)}(b_0) = f^{(k)}(b_0)/(b_0 - a_0)$ (they have the same sign). @@ -1146,6 +1148,30 @@ This handles most cases, but leaves the possibility that a function with infinit ## Questions +##### Question + +```{julia} +#| echo: false +gr() +p1 = vectorfieldplot((x,y) -> [x,y], xlim=(-4,4), ylim=(-4,4), nx=9, ny=9, title="A"); +p2 = vectorfieldplot((x,y) -> [x-y,x], xlim=(-4,4), ylim=(-4,4), nx=9, ny=9,title="B"); +p3 = vectorfieldplot((x,y) -> [y,0], xlim=(-4,4), ylim=(-4,4), nx=9, ny=9, title="C"); +p4 = vectorfieldplot((x,y) -> [-y,x], xlim=(-4,4), ylim=(-4,4), nx=9, ny=9, title="D"); +plot(p1, p2, p3, p4; layout=[2,2]) +``` + +In the above figure, match the function with the vector field plot. + +```{julia} +#| echo: false +plotly() +matchq(("`F(x,y)=[-y ,x]`", "`F(x,y)=[y,0]`", + "`F(x,y)=[x-y,x]`", "`F(x,y)=[x,y]`"), + ("A", "B", "C", "D"), + (4,3,2,1); + label="For each function mark the correct vector field plot" + ) +``` ###### Question diff --git a/quarto/staging/matrix-calculus-notes.html b/quarto/staging/matrix-calculus-notes.html index dd6fd95..9f7b7e8 100644 --- a/quarto/staging/matrix-calculus-notes.html +++ b/quarto/staging/matrix-calculus-notes.html @@ -126,6 +126,7 @@ window.Quarto = {

XXX Add in examples from paper XXX optimization? large number of parameters? ,…

+

Mention numerator layout from https://en.wikipedia.org/wiki/Matrix_calculus#Layout_conventions

@@ -140,18 +141,18 @@ Based on Bright, Edelman, and Johnson’s notes

We have seen several “derivatives” of a function, based on the number of inputs and outputs. The first one was for functions \(f: R \rightarrow R\).

-

Then \(f\) has a derivative at \(x\) if this limit exists

+

In this case, we saw that \(f\) has a derivative at \(c\) if this limit exists:

\[ -\lim_{h \rightarrow 0}\frac{f(x + h) - f(x)}{h}. +\lim_{h \rightarrow 0}\frac{f(c + h) - f(c)}{h}. \]

-

The derivative of the function \(x\) is this limit for a given \(x\). Common notation is:

+

The derivative as a function of \(x\) using this rule for any \(x\) in the domain. Common notation is:

\[ f'(x) = \frac{dy}{dx} = \lim_{h \rightarrow 0}\frac{f(x + h) - f(x)}{h} \]

(when the limit exists).

-

This limit gets expressed in different ways:

+

This limit gets re-expressed in different ways:

    -
  • linearization write \(f(x+\Delta x) - f(x) \approx f'(x)\Delta x\), where \(\delta x\) is a small displacement from \(x\). The reason there isn’t equality is the unwritten higher order terms that vanish in a limit.

  • +
  • linearization writes \(f(x+\Delta x) - f(x) \approx f'(x)\Delta x\), where \(\Delta x\) is a small displacement from \(x\). The reason there isn’t equality is the unwritten higher order terms that vanish in a limit.

  • Alternate limits. Another way of writing this is in terms of explicit smaller order terms:

\[ @@ -159,37 +160,39 @@ f'(x) = \frac{dy}{dx} = \lim_{h \rightarrow 0}\frac{f(x + h) - f(x)}{h} \]

which means if we divide both sides by \(h\) and take the limit, we will get \(0\) on the right and the relationship on the left.

    -
  • Differential notation simply writes this as \(dy = f(x)dx\). More verbosely, we might write
  • +
  • Differential notation simply writes this as \(dy = f'(x)dx\). Focusing on \(f\) and not \(y=f(x)\), we might write

\[ df = f(x+dx) - f(x) = f'(x) dx. \]

-

Here \(dx\) is a differential, made rigorous by a limit, which hides the higher order terms.

+

We will see all the derivatives encountered so far can be similarly expressed.

+

In the above, \(df\) and \(dx\) are differentials, made rigorous by a limit, which hides the higher order terms.

In these notes the limit has been defined, with suitable modification, for functions of vectors (multiple values) with scalar or vector outputs.

For example, when \(f: R \rightarrow R^m\) was a vector-valued function the derivative was defined similarly through a limit of \((f(t + \Delta t) - f(t))/{\Delta t}\), where each component needed to have a limit. This can be rewritten through \(f(t + dt) - f(t) = f'(t) dt\), again using differentials to avoid the higher order terms.

-

When \(f: R^n \rightarrow R\) is a scalar-valued function of a derivative, differentiability was defined by a gradient existing with \(f(c+h) - f(c) - \nabla{f}(c) \cdot h\) being \(\mathscr{o}(\|h\|)\). In other words \(df = f(c + dh) - f(c) = \nabla{f}(c) \cdot dh\). The gradient has the same shape as \(c\), a column vector. If we take the row vector (e.g. \(f'(c) = \nabla{f}(c)^T\)) then again we see \(df = f(c+dh) - f(c) = f'(c) dh\), where the last term uses matrix multiplication of a row vector times a column vector.

+

When \(f: R^n \rightarrow R\) is a scalar-valued function with vector inputs, differentiability was defined by a gradient existing with \(f(c+h) - f(c) - \nabla{f}(c) \cdot h\) being \(\mathscr{o}(\|h\|)\). In other words \(df = f(c + dh) - f(c) = \nabla{f}(c) \cdot dh\). The gradient has the same shape as \(c\), a column vector. If we take the row vector (e.g. \(f'(c) = \nabla{f}(c)^T\)) then again we see \(df = f(c+dh) - f(c) = f'(c) dh\), where the last term uses matrix multiplication of a row vector times a column vector.

Finally, when \(f:R^n \rightarrow R^m\), the Jacobian was defined and characterized by \(\| f(x + dx) - f(x) - J_f(x)dx \|\) being \(\mathscr{o}(\|dx\|)\). Again, we can express this through \(df = f(x + dx) - f(x) = f'(x)dx\) where \(f'(x) = J_f(x)\).

In writing \(df = f(x + dx) - f(x) = f'(x) dx\) generically, some underlying facts are left implicit: \(dx\) has the same shape as \(x\) (so can be added); \(f'(x) dx\) may mean usual multiplication or matrix multiplication; and there is an underlying concept of distance and size that allows the above to be rigorous. This may be an abolute value or a norm.

-

Further, various differentiation rules apply such as the sum, product, and chain rule.

+

Further, various differentiation rules apply such as the sum, product, and chain rules.

The @BrightEdelmanJohnson notes cover differentiation of functions in this uniform manner and then extend the form by treating derivatives as linear operators.

A linear operator is a mathematical object which satisfies

\[ f[\alpha v + \beta w] = \alpha f[v] + \beta f[w]. \]

where the \(\alpha\) and \(\beta\) are scalars, and \(v\) and \(w\) possibly not and come from a vector space. Regular multiplication and matrix multiplication are familiar linear operations, but there are many others.

-

The referenced notes identify \(f'(x) dx\) with \(f'(x)[dx]\), the latter emphasizing \(f'(x)\) acts on \(dx\) and the notation is not commutative (e.g., it is not \(dx f'(x)\)).

+

The referenced notes identify \(f'(x) dx\) as \(f'(x)[dx]\), the latter emphasizing \(f'(x)\) acts on \(dx\) and the notation is not commutative (e.g., it is not \(dx f'(x)\)). The use of \([]\) is to indicate that \(f'(x)\) “acts” on \(dx\) in a linear manner. It may be multiplication, matrix multiplication, or something else. Parentheses are not used which might imply function application or multiplication.

Linear operators are related to vector spaces.

A vector space is a set of mathematical objects which can be added together and also multiplied by a scalar. Vectors of similar size, as previously discussed, are the typical example, with vector addition and scalar multiplication previously discussed topics. Matrices of similar size (and some subclasses) also form a vector space. Additionally, many other set of objects form vector spaces. An example might be polynomial functions of degree \(n\) or less; continuous functions, or functions with a certain number of derivatives.

-

Take differentiable functions as an example, then the simplest derivative rules \([af(x) + bg(x)]' = a[f(x)]' + b[g(x)]'\) show the linearity of the derivative in this setting. This linearity is different from how the derivative is a linear operator on \(dx\).

-

A vector space is described by a basis – a minimal set of vectors needed to describe the space, after consideration of linear combinations. For many vectors, this the set of special vectors with \(1\) as one of the entries, and \(0\) otherwise.

-

A key fact about a basis is every vector in the vector space can be expressed uniquely as a linear combination of the basis vectors.

+

Take differentiable functions as an example, then the simplest derivative rules \([af(x) + bg(x)]' = a[f(x)]' + b[g(x)]'\) show the linearity of the derivative in this setting.

+

A finite vector space is described by a basis – a minimal set of vectors needed to describe the space, after consideration of linear combinations. For some typical vector spaces, this the set of special vectors with \(1\) as one of the entries, and \(0\) otherwise.

+

A key fact about a basis for a finite vector space is every vector in the vector space can be expressed uniquely as a linear combination of the basis vectors.

Vectors and matrices have properties that are generalizations of the real numbers. As vectors and matrices form vector spaces, the concept of addition of vectors and matrices is defined, as is scalar multiplication. Additionally, we have seen:

  • The dot product between two vectors of the same length is defined easily (\(v\cdot w = \Sigma_i v_i w_i\)). It is coupled with the length as \(\|v\|^2 = v\cdot v\).

  • -
  • Matrix multiplication is defined for two properly sized matrices. If \(A\) is \(m \times k\) and \(B\) is \(k \times n\) then \(AB\) is a \(m\times n\) matrix with \((i,j)\) term given by the dot product of the \(i\)th row of \(A\) (viewed as a vector) and the \(j\)th column of \(B\) (viewed as a vector). Matrix multiplication is associative but not commutative. (E.g. \((AB)C = A(BC)\) but \(AB\) and \(BA\) need not be equal (or even defined, as the shapes may not match up).

  • -
  • A square matrix \(A\) has an inverse \(A^{-1}\) if \(AA^{-1} = A^{-1}A = I\), where \(I\) is the identity matrix (a matrix which is zero except on its diagonal entries which are all \(1\)). Square matrices may or may not have an inverse. When they don’t the matrix is called singular.

  • +
  • Matrix multiplication is defined for two properly sized matrices. If \(A\) is \(m \times k\) and \(B\) is \(k \times n\) then \(AB\) is a \(m\times n\) matrix with \((i,j)\) term given by the dot product of the \(i\)th row of \(A\) (viewed as a vector) and the \(j\)th column of \(B\) (viewed as a vector). Matrix multiplication is associative but not commutative. (E.g. \((AB)C = A(BC)\) but \(AB\) and \(BA\) need not be equal, or even defined, as the shapes may not match up).

  • +
  • A square matrix \(A\) has an inverse \(A^{-1}\) if \(AA^{-1} = A^{-1}A = I\), where \(I\) is the identity matrix (a matrix which is zero except on its diagonal entries, which are all \(1\)). Square matrices may or may not have an inverse. A matrix without an inverse is called singular.

  • Viewing a vector as a matrix is possible. The association is typically through a column vector.

  • -
  • The transpose of a matrix comes by permuting the rows and columns. The transpose of a column vector is a row vector, so \(v\cdot w = v^T w\), where we use a superscript \(T\) for the transpose. The transpose of a product, is the product of the transposes – reversed: \((AB)^T = B^T A^T\); the tranpose of a transpose is an identity operation: \((A^T)^T = A\); the inverse of a transpose is the tranpose of the inverse: \((A^{-1})^T = (A^T){-1}\).

  • +
  • The transpose of a matrix comes by permuting the rows and columns. The transpose of a column vector is a row vector, so \(v\cdot w = v^T w\), where we use a superscript \(T\) for the transpose. The transpose of a product, is the product of the transposes – reversed: \((AB)^T = B^T A^T\); the tranpose of a transpose is an identity operation: \((A^T)^T = A\); the inverse of a transpose is the tranpose of the inverse: \((A^{-1})^T = (A^T)^{-1}\).

  • +
  • The adjoint of a matrix is related to the transpose, only complex conjugates are also taken.

  • Matrices for which \(A = A^T\) are called symmetric.

  • A few of the operations on matrices are the transpose and the inverse. These return a matrix, when defined. There is also the determinant and the trace, which return a scalar from a matrix. The trace is just the sum of the diagonal; the determinant is more involved to compute, but was previously seen to have a relationship to the volume of a certain parallellpiped. There are a few other operations described in the following.

@@ -233,26 +236,26 @@ df &= f(x + dx) - f(x)\\ \]

The term \(dx^t A dx\) is dropped, as it is higher order (goes to zero faster), it containing two \(dx\) terms. In the second to last step, an identity operation (taking the transpose of the scalar quantity) is taken to simplify the algebra. Finally, as \(df = f'(x)[dx]\) the identity of \(f'(x) = x^T(A^T+A)\) is made, or taking transposes \(\nabla f = (A + A^T)x\).

Compare the elegance above, with the component version, even though simplified, it still requires a specification of the size to carry the following out:

-
+
using SymPy
 @syms x[1:3]::real A[1:3, 1:3]::real
 u = x' * A * x
 grad_u = [diff(u, xi) for xi in x]
-
+

\(\left[\begin{smallmatrix}2 A₁_{₁} x₁ + A₁_{₂} x₂ + A₁_{₃} x₃ + A₂_{₁} x₂ + A₃_{₁} x₃\\A₁_{₂} x₁ + A₂_{₁} x₁ + 2 A₂_{₂} x₂ + A₂_{₃} x₃ + A₃_{₂} x₃\\A₁_{₃} x₁ + A₂_{₃} x₂ + A₃_{₁} x₁ + A₃_{₂} x₂ + 2 A₃_{₃} x₃\end{smallmatrix}\right]\)

Compare to the formula for the gradient just derived:

-
+
grad_u_1 = (A + A')*x
-
+

\(\left[\begin{smallmatrix}2 A₁_{₁} x₁ + x₂ \left(A₁_{₂} + A₂_{₁}\right) + x₃ \left(A₁_{₃} + A₃_{₁}\right)\\2 A₂_{₂} x₂ + x₁ \left(A₁_{₂} + A₂_{₁}\right) + x₃ \left(A₂_{₃} + A₃_{₂}\right)\\2 A₃_{₃} x₃ + x₁ \left(A₁_{₃} + A₃_{₁}\right) + x₂ \left(A₂_{₃} + A₃_{₂}\right)\end{smallmatrix}\right]\)

The two are, of course, equal

-
+
all(a == b for (a,b)  zip(grad_u, grad_u_1))
-
+
true
@@ -384,24 +387,24 @@ f' = (a'b')c' \text{ or } f' = a'(b'c')
Example

Using the BenchmarkTools package, we can check the time to compute various products:

-
+
using BenchmarkTools
 n,j,k,m = 20,15,10,1
 @btime A*(B*C) setup=(A=rand(n,j);B=rand(j,k); C=rand(k,m));
 @btime (A*B)*C setup=(A=rand(n,j);B=rand(j,k); C=rand(k,m));
-
  420.814 ns (4 allocations: 432 bytes)
-  702.678 ns (4 allocations: 1.88 KiB)
+
  452.474 ns (4 allocations: 432 bytes)
+  837.192 ns (4 allocations: 1.88 KiB)

The latter computation is about 1.5 times slower.

Whereas the relationship is changed when the first matrix is skinny and the last is not:

-
+
@btime A*(B*C) setup=(A=rand(m,k);B=rand(k,j); C=rand(j,n));
 @btime (A*B)*C setup=(A=rand(m,k);B=rand(k,j); C=rand(j,n));
-
  901.488 ns (4 allocations: 1.88 KiB)
-  623.468 ns (4 allocations: 432 bytes)
+
  1.098 μs (4 allocations: 1.88 KiB)
+  779.020 ns (4 allocations: 432 bytes)
@@ -415,30 +418,31 @@ f' = (a'b')c' \text{ or } f' = a'(b'c')

Derivatives of matrix functions

What is the the derivative of \(f(A) = A^2\)?

-

The function \(f\) takes a \(n\times n\) matrix and returns a matrix of the same size. This innocuous question isn’t directly handled by the Jacobian, which is defined for vector valued function \(f:R^n \rightarrow R^m\).

+

The function \(f\) takes a \(n\times n\) matrix and returns a matrix of the same size. This innocuous question isn’t directly handled, here, by the Jacobian, which is defined for vector-valued functions \(f:R^n \rightarrow R^m\).

This derivative can be derived directly from the product rule:

\[ \begin{align*} -f(A) &= [AA]'\\ +f'(A) &= [AA]'\\ &= A dA + dA A \end{align*} \]

That is \(f'(A)\) is the operator \(f'(A)[\delta A] = A \delta A + \delta A A\) and not \(2A\delta A\), as \(A\) may not commute with \(\delta A\).

+

XXX THIS ISN”T EVEN RIGHT

Vectorization of a matrix

Alternatively, we can identify \(A\) through its components, as a vector in \(R^{n^2}\) and then leverage the Jacobian.

One such identification is vectorization – consecutively stacking the column vectors into a vector. In Julia the vec function does this operation:

-
+
@syms A[1:2, 1:2]
 vec(A)
-
+

\(\left[\begin{smallmatrix}A₁_{₁}\\A₂_{₁}\\A₁_{₂}\\A₂_{₂}\end{smallmatrix}\right]\)

The stacking by column follows how Julia stores matrices and how Julia references a matrices entries by linear index:

-
+
vec(A) == [A[i] for i in eachindex(A)]
-
+
true
@@ -447,11 +451,11 @@ f(A) &= [AA]'\\ \tilde{f}(\text{vec}(A)) = \text{vec}(f(A)) \]

We use SymPy to compute the Jacobian of this vector valued function.

-
+
@syms A[1:3, 1:3]::real
 f(x) = x^2
 J = vec(f(A)).jacobian(vec(A)) # jacobian of f̃
-
+

\(\left[\begin{smallmatrix}2 A₁_{₁} & A₁_{₂} & A₁_{₃} & A₂_{₁} & 0 & 0 & A₃_{₁} & 0 & 0\\A₂_{₁} & A₁_{₁} + A₂_{₂} & A₂_{₃} & 0 & A₂_{₁} & 0 & 0 & A₃_{₁} & 0\\A₃_{₁} & A₃_{₂} & A₁_{₁} + A₃_{₃} & 0 & 0 & A₂_{₁} & 0 & 0 & A₃_{₁}\\A₁_{₂} & 0 & 0 & A₁_{₁} + A₂_{₂} & A₁_{₂} & A₁_{₃} & A₃_{₂} & 0 & 0\\0 & A₁_{₂} & 0 & A₂_{₁} & 2 A₂_{₂} & A₂_{₃} & 0 & A₃_{₂} & 0\\0 & 0 & A₁_{₂} & A₃_{₁} & A₃_{₂} & A₂_{₂} + A₃_{₃} & 0 & 0 & A₃_{₂}\\A₁_{₃} & 0 & 0 & A₂_{₃} & 0 & 0 & A₁_{₁} + A₃_{₃} & A₁_{₂} & A₁_{₃}\\0 & A₁_{₃} & 0 & 0 & A₂_{₃} & 0 & A₂_{₁} & A₂_{₂} + A₃_{₃} & A₂_{₃}\\0 & 0 & A₁_{₃} & 0 & 0 & A₂_{₃} & A₃_{₁} & A₃_{₂} & 2 A₃_{₃}\end{smallmatrix}\right]\)

@@ -459,10 +463,10 @@ f(A) &= [AA]'\\

A basic course in linear algebra shows that any linear operator on a finite vector space can be represented as a matrix. The basic idea is to represent what the operator does to each basis element and put these values as columns of the matrix.

In this \(3 \times 3\) case, the linear operator works on an object with \(9\) slots and returns an object with \(9\) slots, so the matrix will be \(9 \times 9\).

The basis elements are simply the matrices with a \(1\) in spot \((i,j)\) and zero elsewhere. Here we generate them through a function:

-
+
basis(i,j,A) = (b=zeros(Int, size(A)...); b[i,j] = 1; b)
 JJ = [vec(basis(i,j,A)*A + A*basis(i,j,A)) for  j in 1:3 for i in 1:3]
-
+
9-element Vector{Vector{Sym{PyCall.PyObject}}}:
  [2*A₁_₁, A₂_₁, A₃_₁, A₁_₂, 0, 0, A₁_₃, 0, 0]
  [A₁_₂, A₁_₁ + A₂_₂, A₃_₂, 0, A₁_₂, 0, 0, A₁_₃, 0]
@@ -477,16 +481,16 @@ f(A) &= [AA]'\\
 

The elements of JJ show the representation of each of the \(9\) basis elements under the linear transformation.

To construct the matrix representing the linear operator, we need to concatenate these horizontally as column vectors

-
+
JJ = hcat(JJ...)
-
+

\(\left[\begin{smallmatrix}2 A₁_{₁} & A₁_{₂} & A₁_{₃} & A₂_{₁} & 0 & 0 & A₃_{₁} & 0 & 0\\A₂_{₁} & A₁_{₁} + A₂_{₂} & A₂_{₃} & 0 & A₂_{₁} & 0 & 0 & A₃_{₁} & 0\\A₃_{₁} & A₃_{₂} & A₁_{₁} + A₃_{₃} & 0 & 0 & A₂_{₁} & 0 & 0 & A₃_{₁}\\A₁_{₂} & 0 & 0 & A₁_{₁} + A₂_{₂} & A₁_{₂} & A₁_{₃} & A₃_{₂} & 0 & 0\\0 & A₁_{₂} & 0 & A₂_{₁} & 2 A₂_{₂} & A₂_{₃} & 0 & A₃_{₂} & 0\\0 & 0 & A₁_{₂} & A₃_{₁} & A₃_{₂} & A₂_{₂} + A₃_{₃} & 0 & 0 & A₃_{₂}\\A₁_{₃} & 0 & 0 & A₂_{₃} & 0 & 0 & A₁_{₁} + A₃_{₃} & A₁_{₂} & A₁_{₃}\\0 & A₁_{₃} & 0 & 0 & A₂_{₃} & 0 & A₂_{₁} & A₂_{₂} + A₃_{₃} & A₂_{₃}\\0 & 0 & A₁_{₃} & 0 & 0 & A₂_{₃} & A₃_{₁} & A₃_{₂} & 2 A₃_{₃}\end{smallmatrix}\right]\)

The matrix \(JJ\) is identical to \(J\), above:

-
+
all(j == jj for (j, jj) in zip(J, JJ))
-
+
true
@@ -509,10 +513,10 @@ a_{m1}B & a_{m2}B & \cdots & a_{mn}B \end{bmatrix} \]

The function kron forms this product:

-
+
@syms A[1:2, 1:3] B[1:3, 1:4]
 kron(A, B) # same as hcat((vcat((A[i,j]*B for i in 1:2)...) for j in 1:3)...)
-
+

\(\left[\begin{smallmatrix}A₁_{₁} B₁_{₁} & A₁_{₁} B₁_{₂} & A₁_{₁} B₁_{₃} & A₁_{₁} B₁_{₄} & A₁_{₂} B₁_{₁} & A₁_{₂} B₁_{₂} & A₁_{₂} B₁_{₃} & A₁_{₂} B₁_{₄} & A₁_{₃} B₁_{₁} & A₁_{₃} B₁_{₂} & A₁_{₃} B₁_{₃} & A₁_{₃} B₁_{₄}\\A₁_{₁} B₂_{₁} & A₁_{₁} B₂_{₂} & A₁_{₁} B₂_{₃} & A₁_{₁} B₂_{₄} & A₁_{₂} B₂_{₁} & A₁_{₂} B₂_{₂} & A₁_{₂} B₂_{₃} & A₁_{₂} B₂_{₄} & A₁_{₃} B₂_{₁} & A₁_{₃} B₂_{₂} & A₁_{₃} B₂_{₃} & A₁_{₃} B₂_{₄}\\A₁_{₁} B₃_{₁} & A₁_{₁} B₃_{₂} & A₁_{₁} B₃_{₃} & A₁_{₁} B₃_{₄} & A₁_{₂} B₃_{₁} & A₁_{₂} B₃_{₂} & A₁_{₂} B₃_{₃} & A₁_{₂} B₃_{₄} & A₁_{₃} B₃_{₁} & A₁_{₃} B₃_{₂} & A₁_{₃} B₃_{₃} & A₁_{₃} B₃_{₄}\\A₂_{₁} B₁_{₁} & A₂_{₁} B₁_{₂} & A₂_{₁} B₁_{₃} & A₂_{₁} B₁_{₄} & A₂_{₂} B₁_{₁} & A₂_{₂} B₁_{₂} & A₂_{₂} B₁_{₃} & A₂_{₂} B₁_{₄} & A₂_{₃} B₁_{₁} & A₂_{₃} B₁_{₂} & A₂_{₃} B₁_{₃} & A₂_{₃} B₁_{₄}\\A₂_{₁} B₂_{₁} & A₂_{₁} B₂_{₂} & A₂_{₁} B₂_{₃} & A₂_{₁} B₂_{₄} & A₂_{₂} B₂_{₁} & A₂_{₂} B₂_{₂} & A₂_{₂} B₂_{₃} & A₂_{₂} B₂_{₄} & A₂_{₃} B₂_{₁} & A₂_{₃} B₂_{₂} & A₂_{₃} B₂_{₃} & A₂_{₃} B₂_{₄}\\A₂_{₁} B₃_{₁} & A₂_{₁} B₃_{₂} & A₂_{₁} B₃_{₃} & A₂_{₁} B₃_{₄} & A₂_{₂} B₃_{₁} & A₂_{₂} B₃_{₂} & A₂_{₂} B₃_{₃} & A₂_{₂} B₃_{₄} & A₂_{₃} B₃_{₁} & A₂_{₃} B₃_{₂} & A₂_{₃} B₃_{₃} & A₂_{₃} B₃_{₄}\end{smallmatrix}\right]\)

@@ -533,11 +537,11 @@ a_{m1}B & a_{m2}B & \cdots & a_{mn}B

Appropriate sizes for \(A\), \(B\), and \(C\) are determined by the various products in \(BCA^T\).

If \(A\) is \(m \times n\) and \(B\) is \(r \times s\), then since \(BC\) is defined, \(C\) has \(s\) rows, and since \(CA^T\) is defined, \(C\) must have \(n\) columns, as \(A^T\) is \(n \times m\), so \(C\) must be \(s\times n\). Checking this is correct on the other side, \(A \times B\) would be size \(mr \times ns\) and \(\vec{C}\) would be size \(sn\), so that product works, size wise.

The referred to notes have an explanation for this formula, but we confirm with an example with \(m=n-2\), \(r=s=3\):

-
+
@syms A[1:2, 1:2]::real B[1:3, 1:3]::real C[1:3, 1:2]::real
 L, R = kron(A,B)*vec(C),  vec(B*C*A')
 all(l == r for (l, r)  zip(L, R))
-
+
true
@@ -557,13 +561,13 @@ a_{m1}B & a_{m2}B & \cdots & a_{mn}B \left((I \otimes A) + (A^T \otimes I)\right) \text{vec}(dA) \]

We should then get the Jacobian we computed from the following:

-
+
@syms A[1:3, 1:3]::real
 using LinearAlgebra: I
 J = vec(A^2).jacobian(vec(A))
 JJ = kron(I(3), A) + kron(A', I(3))
 all(j == jj for (j,jj) in zip(J,JJ))
-
+
true
@@ -622,11 +626,11 @@ C_{ij}, \]

This agrees through a formula to compute the inverse of a matrix through its cofactor matrix divided by its determinant.

That the trace gets involved, can be seen from this computation, which shows the only first-order terms are from the diagonal sum:

-
+
using LinearAlgebra
 @syms dA[1:2, 1:2]
 det(I + dA) - det(I)
-
+

\(dA₁_{₁} dA₂_{₂} + dA₁_{₁} - dA₁_{₂} dA₂_{₁} + dA₂_{₂}\)

diff --git a/quarto/staging/matrix-calculus-notes.qmd b/quarto/staging/matrix-calculus-notes.qmd index a9e457e..032c48d 100644 --- a/quarto/staging/matrix-calculus-notes.qmd +++ b/quarto/staging/matrix-calculus-notes.qmd @@ -3,6 +3,9 @@ XXX Add in examples from paper XXX optimization? large number of parameters? ,... + +Mention numerator layout from https://en.wikipedia.org/wiki/Matrix_calculus#Layout_conventions + ::: {.callout-note} ## Based on Bright, Edelman, and Johnson's notes @@ -12,13 +15,13 @@ This section samples material from the notes [Matrix Calculus (for Machine Learn We have seen several "derivatives" of a function, based on the number of inputs and outputs. The first one was for functions $f: R \rightarrow R$. -Then $f$ has a derivative at $x$ if this limit exists +In this case, we saw that $f$ has a derivative at $c$ if this limit exists: $$ -\lim_{h \rightarrow 0}\frac{f(x + h) - f(x)}{h}. +\lim_{h \rightarrow 0}\frac{f(c + h) - f(c)}{h}. $$ -The derivative of the function $x$ is this limit for a given $x$. Common notation is: +The derivative as a function of $x$ using this rule for any $x$ in the domain. Common notation is: $$ f'(x) = \frac{dy}{dx} = \lim_{h \rightarrow 0}\frac{f(x + h) - f(x)}{h} @@ -27,9 +30,9 @@ $$ (when the limit exists). -This limit gets expressed in different ways: +This limit gets re-expressed in different ways: -* linearization write $f(x+\Delta x) - f(x) \approx f'(x)\Delta x$, where $\delta x$ is a small displacement from $x$. The reason there isn't equality is the unwritten higher order terms that vanish in a limit. +* linearization writes $f(x+\Delta x) - f(x) \approx f'(x)\Delta x$, where $\Delta x$ is a small displacement from $x$. The reason there isn't equality is the unwritten higher order terms that vanish in a limit. * Alternate limits. Another way of writing this is in terms of explicit smaller order terms: @@ -39,13 +42,15 @@ $$ which means if we divide both sides by $h$ and take the limit, we will get $0$ on the right and the relationship on the left. -* Differential notation simply writes this as $dy = f(x)dx$. More verbosely, we might write +* Differential notation simply writes this as $dy = f'(x)dx$. Focusing on $f$ and not $y=f(x)$, we might write $$ df = f(x+dx) - f(x) = f'(x) dx. $$ -Here $dx$ is a differential, made rigorous by a limit, which hides the higher order terms. +We will see all the derivatives encountered so far can be similarly expressed. + +In the above, $df$ and $dx$ are differentials, made rigorous by a limit, which hides the higher order terms. In these notes the limit has been defined, with suitable modification, for functions of vectors (multiple values) with scalar or vector outputs. @@ -53,7 +58,7 @@ In these notes the limit has been defined, with suitable modification, for funct For example, when $f: R \rightarrow R^m$ was a vector-valued function the derivative was defined similarly through a limit of $(f(t + \Delta t) - f(t))/{\Delta t}$, where each component needed to have a limit. This can be rewritten through $f(t + dt) - f(t) = f'(t) dt$, again using differentials to avoid the higher order terms. -When $f: R^n \rightarrow R$ is a scalar-valued function of a derivative, differentiability was defined by a gradient existing with $f(c+h) - f(c) - \nabla{f}(c) \cdot h$ being $\mathscr{o}(\|h\|)$. In other words $df = f(c + dh) - f(c) = \nabla{f}(c) \cdot dh$. The gradient has the same shape as $c$, a column vector. If we take the row vector (e.g. $f'(c) = \nabla{f}(c)^T$) then again we see $df = f(c+dh) - f(c) = f'(c) dh$, where the last term uses matrix multiplication of a row vector times a column vector. +When $f: R^n \rightarrow R$ is a scalar-valued function with vector inputs, differentiability was defined by a gradient existing with $f(c+h) - f(c) - \nabla{f}(c) \cdot h$ being $\mathscr{o}(\|h\|)$. In other words $df = f(c + dh) - f(c) = \nabla{f}(c) \cdot dh$. The gradient has the same shape as $c$, a column vector. If we take the row vector (e.g. $f'(c) = \nabla{f}(c)^T$) then again we see $df = f(c+dh) - f(c) = f'(c) dh$, where the last term uses matrix multiplication of a row vector times a column vector. Finally, when $f:R^n \rightarrow R^m$, the Jacobian was defined and characterized by $\| f(x + dx) - f(x) - J_f(x)dx \|$ being $\mathscr{o}(\|dx\|)$. Again, we can express this through $df = f(x + dx) - f(x) = f'(x)dx$ where $f'(x) = J_f(x)$. @@ -61,7 +66,7 @@ $\| f(x + dx) - f(x) - J_f(x)dx \|$ being $\mathscr{o}(\|dx\|)$. Again, we can e In writing $df = f(x + dx) - f(x) = f'(x) dx$ generically, some underlying facts are left implicit: $dx$ has the same shape as $x$ (so can be added); $f'(x) dx$ may mean usual multiplication or matrix multiplication; and there is an underlying concept of distance and size that allows the above to be rigorous. This may be an abolute value or a norm. -Further, various differentiation rules apply such as the sum, product, and chain rule. +Further, various differentiation rules apply such as the sum, product, and chain rules. The @BrightEdelmanJohnson notes cover differentiation of functions in this uniform manner and then extend the form by treating derivatives as *linear operators*. @@ -75,29 +80,35 @@ $$ where the $\alpha$ and $\beta$ are scalars, and $v$ and $w$ possibly not and come from a *vector space*. Regular multiplication and matrix multiplication are familiar linear operations, but there are many others. -The referenced notes identify $f'(x) dx$ with $f'(x)[dx]$, the latter emphasizing $f'(x)$ acts on $dx$ and the notation is not commutative (e.g., it is not $dx f'(x)$). +The referenced notes identify $f'(x) dx$ as $f'(x)[dx]$, the latter emphasizing $f'(x)$ acts on $dx$ and the notation is not commutative (e.g., it is not $dx f'(x)$). The use of $[]$ is to indicate that $f'(x)$ "acts" on $dx$ in a linear manner. It may be multiplication, matrix multiplication, or something else. Parentheses are not used which might imply function application or multiplication. + + + + Linear operators are related to vector spaces. A [vector space](https://en.wikipedia.org/wiki/Vector_space) is a set of mathematical objects which can be added together and also multiplied by a scalar. Vectors of similar size, as previously discussed, are the typical example, with vector addition and scalar multiplication previously discussed topics. Matrices of similar size (and some subclasses) also form a vector space. Additionally, many other set of objects form vector spaces. An example might be polynomial functions of degree $n$ or less; continuous functions, or functions with a certain number of derivatives. -Take differentiable functions as an example, then the simplest derivative rules $[af(x) + bg(x)]' = a[f(x)]' + b[g(x)]'$ show the linearity of the derivative in this setting. This linearity is different from how the derivative is a linear operator on $dx$. +Take differentiable functions as an example, then the simplest derivative rules $[af(x) + bg(x)]' = a[f(x)]' + b[g(x)]'$ show the linearity of the derivative in this setting. -A vector space is described by a *basis* -- a minimal set of vectors needed to describe the space, after consideration of linear combinations. For many vectors, this the set of special vectors with $1$ as one of the entries, and $0$ otherwise. +A finite vector space is described by a *basis* -- a minimal set of vectors needed to describe the space, after consideration of linear combinations. For some typical vector spaces, this the set of special vectors with $1$ as one of the entries, and $0$ otherwise. -A key fact about a basis is every vector in the vector space can be expressed *uniquely* as a linear combination of the basis vectors. +A key fact about a basis for a finite vector space is every vector in the vector space can be expressed *uniquely* as a linear combination of the basis vectors. Vectors and matrices have properties that are generalizations of the real numbers. As vectors and matrices form vector spaces, the concept of addition of vectors and matrices is defined, as is scalar multiplication. Additionally, we have seen: * The dot product between two vectors of the same length is defined easily ($v\cdot w = \Sigma_i v_i w_i$). It is coupled with the length as $\|v\|^2 = v\cdot v$. -* Matrix multiplication is defined for two properly sized matrices. If $A$ is $m \times k$ and $B$ is $k \times n$ then $AB$ is a $m\times n$ matrix with $(i,j)$ term given by the dot product of the $i$th row of $A$ (viewed as a vector) and the $j$th column of $B$ (viewed as a vector). Matrix multiplication is associative but *not* commutative. (E.g. $(AB)C = A(BC)$ but $AB$ and $BA$ need not be equal (or even defined, as the shapes may not match up). +* Matrix multiplication is defined for two properly sized matrices. If $A$ is $m \times k$ and $B$ is $k \times n$ then $AB$ is a $m\times n$ matrix with $(i,j)$ term given by the dot product of the $i$th row of $A$ (viewed as a vector) and the $j$th column of $B$ (viewed as a vector). Matrix multiplication is associative but *not* commutative. (E.g. $(AB)C = A(BC)$ but $AB$ and $BA$ need not be equal, or even defined, as the shapes may not match up). -* A square matrix $A$ has an *inverse* $A^{-1}$ if $AA^{-1} = A^{-1}A = I$, where $I$ is the identity matrix (a matrix which is zero except on its diagonal entries which are all $1$). Square matrices may or may not have an inverse. When they don't the matrix is called singular. +* A square matrix $A$ has an *inverse* $A^{-1}$ if $AA^{-1} = A^{-1}A = I$, where $I$ is the identity matrix (a matrix which is zero except on its diagonal entries, which are all $1$). Square matrices may or may not have an inverse. A matrix without an inverse is called *singular*. * Viewing a vector as a matrix is possible. The association is typically through a *column* vector. -* The transpose of a matrix comes by permuting the rows and columns. The transpose of a column vector is a row vector, so $v\cdot w = v^T w$, where we use a superscript $T$ for the transpose. The transpose of a product, is the product of the transposes -- reversed: $(AB)^T = B^T A^T$; the tranpose of a transpose is an identity operation: $(A^T)^T = A$; the inverse of a transpose is the tranpose of the inverse: $(A^{-1})^T = (A^T){-1}$. +* The *transpose* of a matrix comes by permuting the rows and columns. The transpose of a column vector is a row vector, so $v\cdot w = v^T w$, where we use a superscript $T$ for the transpose. The transpose of a product, is the product of the transposes -- reversed: $(AB)^T = B^T A^T$; the tranpose of a transpose is an identity operation: $(A^T)^T = A$; the inverse of a transpose is the tranpose of the inverse: $(A^{-1})^T = (A^T)^{-1}$. + +* The *adjoint* of a matrix is related to the transpose, only complex conjugates are also taken. * Matrices for which $A = A^T$ are called symmetric. @@ -377,20 +388,20 @@ XXXX What is the the derivative of $f(A) = A^2$? -The function $f$ takes a $n\times n$ matrix and returns a matrix of the same size. This innocuous question isn't directly -handled by the Jacobian, which is defined for vector valued function $f:R^n \rightarrow R^m$. +The function $f$ takes a $n\times n$ matrix and returns a matrix of the same size. This innocuous question isn't directly handled, here, by the Jacobian, which is defined for vector-valued functions $f:R^n \rightarrow R^m$. This derivative can be derived directly from the *product rule*: $$ \begin{align*} -f(A) &= [AA]'\\ +f'(A) &= [AA]'\\ &= A dA + dA A \end{align*} $$ That is $f'(A)$ is the operator $f'(A)[\delta A] = A \delta A + \delta A A$ and not $2A\delta A$, as $A$ may not commute with $\delta A$. +XXX THIS ISN"T EVEN RIGHT ### Vectorization of a matrix From fa5f9f449d3e234610fcaf4ee8ee971cf07aa5a9 Mon Sep 17 00:00:00 2001 From: jverzani Date: Wed, 30 Apr 2025 17:57:44 -0400 Subject: [PATCH 03/22] add matrix calculus notes --- quarto/_quarto.yml | 1 + .../Project.toml | 1 + .../matrix_calculus_notes.qmd} | 428 +++--- quarto/staging/Project.toml | 2 - quarto/staging/matrix-calculus-notes.html | 1282 ----------------- 5 files changed, 205 insertions(+), 1509 deletions(-) rename quarto/{staging/matrix-calculus-notes.qmd => differentiable_vector_calculus/matrix_calculus_notes.qmd} (67%) delete mode 100644 quarto/staging/Project.toml delete mode 100644 quarto/staging/matrix-calculus-notes.html diff --git a/quarto/_quarto.yml b/quarto/_quarto.yml index 766b3a0..eab49c1 100644 --- a/quarto/_quarto.yml +++ b/quarto/_quarto.yml @@ -100,6 +100,7 @@ book: - differentiable_vector_calculus/scalar_functions.qmd - differentiable_vector_calculus/scalar_functions_applications.qmd - differentiable_vector_calculus/vector_fields.qmd + - differentiable_vector_calculus/matrix_calculus_notes.qmd - differentiable_vector_calculus/plots_plotting.qmd - part: integral_vector_calculus.qmd diff --git a/quarto/differentiable_vector_calculus/Project.toml b/quarto/differentiable_vector_calculus/Project.toml index ebc683f..6c11110 100644 --- a/quarto/differentiable_vector_calculus/Project.toml +++ b/quarto/differentiable_vector_calculus/Project.toml @@ -1,4 +1,5 @@ [deps] +BenchmarkTools = "6e4b80f9-dd63-53aa-95a3-0cdb28fa8baf" CSV = "336ed68f-0bac-5ca0-87d4-7b16caf5d00b" CalculusWithJulia = "a2e0e22d-7d4c-5312-9169-8b992201a882" Contour = "d38c429a-6771-53c6-b99e-75d170b6e991" diff --git a/quarto/staging/matrix-calculus-notes.qmd b/quarto/differentiable_vector_calculus/matrix_calculus_notes.qmd similarity index 67% rename from quarto/staging/matrix-calculus-notes.qmd rename to quarto/differentiable_vector_calculus/matrix_calculus_notes.qmd index 032c48d..8c5dab8 100644 --- a/quarto/staging/matrix-calculus-notes.qmd +++ b/quarto/differentiable_vector_calculus/matrix_calculus_notes.qmd @@ -1,17 +1,14 @@ # Matrix Calculus -XXX Add in examples from paper XXX -optimization? large number of parameters? ,... +This section illustrates a more general setting for taking derivatives, that unifies the different expositions taken prior. - -Mention numerator layout from https://en.wikipedia.org/wiki/Matrix_calculus#Layout_conventions - -::: {.callout-note} +::: {.callout-note appearance="minimal"} ## Based on Bright, Edelman, and Johnson's notes -This section samples material from the notes [Matrix Calculus (for Machine Learning and Beyond)](https://arxiv.org/abs/2501.14787) by Paige Bright, Alan Edelman, and Steven G. Johnson. These notes cover material taught in a course at MIT. Support materials for their course in `Julia` are available at [https://github.com/mitmath/matrixcalc/tree/main](https://github.com/mitmath/matrixcalc/tree/main). For more details and examples, please refer to the source. +This section has essentially no original contribution, as it basically samples material from the notes [Matrix Calculus (for Machine Learning and Beyond)](https://arxiv.org/abs/2501.14787) by Paige Bright, Alan Edelman, and Steven G. Johnson. Their notes cover material taught in a course at MIT. Support materials for their course in `Julia` are available at [https://github.com/mitmath/matrixcalc/tree/main](https://github.com/mitmath/matrixcalc/tree/main). For more details and examples, please refer to the source. ::: +## Review We have seen several "derivatives" of a function, based on the number of inputs and outputs. The first one was for functions $f: R \rightarrow R$. @@ -21,7 +18,9 @@ $$ \lim_{h \rightarrow 0}\frac{f(c + h) - f(c)}{h}. $$ -The derivative as a function of $x$ using this rule for any $x$ in the domain. Common notation is: +The derivative as a function of $x$ uses this rule for any $x$ in the domain. + +Common notation is: $$ f'(x) = \frac{dy}{dx} = \lim_{h \rightarrow 0}\frac{f(x + h) - f(x)}{h} @@ -48,28 +47,64 @@ $$ df = f(x+dx) - f(x) = f'(x) dx. $$ -We will see all the derivatives encountered so far can be similarly expressed. - In the above, $df$ and $dx$ are differentials, made rigorous by a limit, which hides the higher order terms. +We will see all the derivatives encountered so far can be similarly expressed as this last characterization. -In these notes the limit has been defined, with suitable modification, for functions of vectors (multiple values) with scalar or vector outputs. - +### Univariate, vector-valued For example, when $f: R \rightarrow R^m$ was a vector-valued function the derivative was defined similarly through a limit of $(f(t + \Delta t) - f(t))/{\Delta t}$, where each component needed to have a limit. This can be rewritten through $f(t + dt) - f(t) = f'(t) dt$, again using differentials to avoid the higher order terms. +### Multivariate, scalar-valued When $f: R^n \rightarrow R$ is a scalar-valued function with vector inputs, differentiability was defined by a gradient existing with $f(c+h) - f(c) - \nabla{f}(c) \cdot h$ being $\mathscr{o}(\|h\|)$. In other words $df = f(c + dh) - f(c) = \nabla{f}(c) \cdot dh$. The gradient has the same shape as $c$, a column vector. If we take the row vector (e.g. $f'(c) = \nabla{f}(c)^T$) then again we see $df = f(c+dh) - f(c) = f'(c) dh$, where the last term uses matrix multiplication of a row vector times a column vector. + +### Multivariate, vector-valued Finally, when $f:R^n \rightarrow R^m$, the Jacobian was defined and characterized by $\| f(x + dx) - f(x) - J_f(x)dx \|$ being $\mathscr{o}(\|dx\|)$. Again, we can express this through $df = f(x + dx) - f(x) = f'(x)dx$ where $f'(x) = J_f(x)$. -In writing $df = f(x + dx) - f(x) = f'(x) dx$ generically, some underlying facts are left implicit: $dx$ has the same shape as $x$ (so can be added); $f'(x) dx$ may mean usual multiplication or matrix multiplication; and there is an underlying concept of distance and size that allows the above to be rigorous. This may be an abolute value or a norm. +### Vector spaces -Further, various differentiation rules apply such as the sum, product, and chain rules. +The generalization of the derivative involves linear operators which are defined for vector spaces. +A [vector space](https://en.wikipedia.org/wiki/Vector_space) is a set of mathematical objects which can be added together and also multiplied by a scalar. Vectors of similar size, as previously discussed, are the typical example, with vector addition and scalar multiplication already defined. Matrices of similar size (and some subclasses) also form a vector space. -The @BrightEdelmanJohnson notes cover differentiation of functions in this uniform manner and then extend the form by treating derivatives as *linear operators*. +Additionally, many other set of objects form vector spaces. Certain families of functions form examples such as: polynomial functions of degree $n$ or less; continuous functions, or functions with a certain number of derivatives. The last two are infinite dimensional; our focus here is on finite dimensional vector spaces. + +Let's take differentiable functions as an example. These form a vector space as the derivative of a linear combination of differentiable functions is defined through the simplest derivative rule: $[af(x) + bg(x)]' = a[f(x)]' + b[g(x)]'$. If $f$ and $g$ are differentiable, then so is $af(x)+bg(x)$. + +A finite vector space is described by a *basis* -- a minimal set of vectors needed to describe the space, after consideration of linear combinations. For some typical vector spaces, this is the set of special vectors with $1$ as one of the entries, and $0$ otherwise. + +A key fact about a basis for a finite vector space is every vector in the vector space can be expressed *uniquely* as a linear combination of the basis vectors. The set of numbers used in the linear combination, along with an order to the basis, means an element in a finite vector space can be associated with a unique coordinate vector. + +Vectors and matrices have properties that are generalizations of the real numbers. As vectors and matrices form vector spaces, the concept of addition of vectors and matrices is defined, as is scalar multiplication. Additionally, we have seen: + +* The dot product between two vectors of the same length is defined easily ($v\cdot w = \Sigma_i v_i w_i$). It is coupled with the length as $\|v\|^2 = v\cdot v$. + +* Matrix multiplication is defined for two properly sized matrices. If $A$ is $m \times k$ and $B$ is $k \times n$ then $AB$ is a $m\times n$ matrix with $(i,j)$ term given by the dot product of the $i$th row of $A$ (viewed as a vector) and the $j$th column of $B$ (viewed as a vector). Matrix multiplication is associative but *not* commutative. (E.g. $(AB)C = A(BC)$ but $AB$ and $BA$ need not be equal, or even defined, as the shapes may not match up.) + +* A square matrix $A$ has an *inverse* $A^{-1}$ if $AA^{-1} = A^{-1}A = I$, where $I$ is the identity matrix (a matrix which is zero except on its diagonal entries, which are all $1$). Square matrices may or may not have an inverse. A matrix without an inverse is called *singular*. + +* Viewing a vector as a matrix is possible. The association chosen here is common and is through a *column* vector. + +* The *transpose* of a matrix comes by permuting the rows and columns. The transpose of a column vector is a row vector, so $v\cdot w = v^T w$, where we use a superscript $T$ for the transpose. The transpose of a product, is the product of the transposes -- reversed: $(AB)^T = B^T A^T$; the tranpose of a transpose is an identity operation: $(A^T)^T = A$; the inverse of a transpose is the tranpose of the inverse: $(A^{-1})^T = (A^T)^{-1}$. + +* Matrices for which $A = A^T$ are called symmetric. + +* The *adjoint* of a matrix is related to the transpose, only complex conjugates are also taken. When a matrix has real components, the adjoint and transpose are identical operations. + +* The trace of a square matrix is just the sum of its diagonal terms + +* The determinant of a square matrix is involved to compute, but was previously seen to have a relationship to the volume of a certain parallellpiped. + +These operations have different inputs and outputs: the determinant and trace take a (square) matrix and return a scalar; the inverse takes a square matrix and returns a square matrix (when defined); the transpose and adjoint take a rectangular matrix and return a rectangular matrix. + +In addition to these, there are a few other key operations on matrices described in the following. + +### Linear operators + +The @BrightEdelmanJohnson notes cover differentiation of functions in this uniform manner extending the form by treating derivatives more generally as *linear operators*. A [linear operator](https://en.wikipedia.org/wiki/Operator_(mathematics)) is a mathematical object which satisfies @@ -78,45 +113,29 @@ $$ f[\alpha v + \beta w] = \alpha f[v] + \beta f[w]. $$ -where the $\alpha$ and $\beta$ are scalars, and $v$ and $w$ possibly not and come from a *vector space*. Regular multiplication and matrix multiplication are familiar linear operations, but there are many others. +where the $\alpha$ and $\beta$ are scalars, and $v$ and $w$ come from a *vector space*. + +Taking the real numbers as a vector space, then regular multiplication is a linear operation, as $c \cdot (ax + by) = a\cdot(cx) + b\cdot(cy)$ using the distributive and commutative properties. + +Taking $n$-dimensional vectors as vector space, matrix multiplication by an $n \times n$ matrix on the left will be a linear operator as $M(av + bw) = a(Mv) + b(Mw)$, using distribution and the commutative properties of scalar multiplication. + +We saw differential functions form a vector space, the derivative is a linear operator, as $[af(x) + bg(x)]' = af'(x) + bg'(x)$. + + +::: {.callout-note appearance="minimal"} +## The use of `[]` The referenced notes identify $f'(x) dx$ as $f'(x)[dx]$, the latter emphasizing $f'(x)$ acts on $dx$ and the notation is not commutative (e.g., it is not $dx f'(x)$). The use of $[]$ is to indicate that $f'(x)$ "acts" on $dx$ in a linear manner. It may be multiplication, matrix multiplication, or something else. Parentheses are not used which might imply function application or multiplication. +::: +## The derivative as a linear operator +We take the view that a derivative is a linear operator where $df = f(x+dx) + f(x) = f'(x)[dx]$. +In writing $df = f(x + dx) - f(x) = f'(x)[dx]$ generically, some underlying facts are left implicit: $dx$ has the same shape as $x$ (so can be added) and there is an underlying concept of distance and size that allows the above to be made rigorous. This may be an abolute value or a norm. -Linear operators are related to vector spaces. - -A [vector space](https://en.wikipedia.org/wiki/Vector_space) is a set of mathematical objects which can be added together and also multiplied by a scalar. Vectors of similar size, as previously discussed, are the typical example, with vector addition and scalar multiplication previously discussed topics. Matrices of similar size (and some subclasses) also form a vector space. Additionally, many other set of objects form vector spaces. An example might be polynomial functions of degree $n$ or less; continuous functions, or functions with a certain number of derivatives. - -Take differentiable functions as an example, then the simplest derivative rules $[af(x) + bg(x)]' = a[f(x)]' + b[g(x)]'$ show the linearity of the derivative in this setting. - -A finite vector space is described by a *basis* -- a minimal set of vectors needed to describe the space, after consideration of linear combinations. For some typical vector spaces, this the set of special vectors with $1$ as one of the entries, and $0$ otherwise. - -A key fact about a basis for a finite vector space is every vector in the vector space can be expressed *uniquely* as a linear combination of the basis vectors. - -Vectors and matrices have properties that are generalizations of the real numbers. As vectors and matrices form vector spaces, the concept of addition of vectors and matrices is defined, as is scalar multiplication. Additionally, we have seen: - -* The dot product between two vectors of the same length is defined easily ($v\cdot w = \Sigma_i v_i w_i$). It is coupled with the length as $\|v\|^2 = v\cdot v$. - -* Matrix multiplication is defined for two properly sized matrices. If $A$ is $m \times k$ and $B$ is $k \times n$ then $AB$ is a $m\times n$ matrix with $(i,j)$ term given by the dot product of the $i$th row of $A$ (viewed as a vector) and the $j$th column of $B$ (viewed as a vector). Matrix multiplication is associative but *not* commutative. (E.g. $(AB)C = A(BC)$ but $AB$ and $BA$ need not be equal, or even defined, as the shapes may not match up). - -* A square matrix $A$ has an *inverse* $A^{-1}$ if $AA^{-1} = A^{-1}A = I$, where $I$ is the identity matrix (a matrix which is zero except on its diagonal entries, which are all $1$). Square matrices may or may not have an inverse. A matrix without an inverse is called *singular*. - -* Viewing a vector as a matrix is possible. The association is typically through a *column* vector. - -* The *transpose* of a matrix comes by permuting the rows and columns. The transpose of a column vector is a row vector, so $v\cdot w = v^T w$, where we use a superscript $T$ for the transpose. The transpose of a product, is the product of the transposes -- reversed: $(AB)^T = B^T A^T$; the tranpose of a transpose is an identity operation: $(A^T)^T = A$; the inverse of a transpose is the tranpose of the inverse: $(A^{-1})^T = (A^T)^{-1}$. - -* The *adjoint* of a matrix is related to the transpose, only complex conjugates are also taken. - -* Matrices for which $A = A^T$ are called symmetric. - -* A few of the operations on matrices are the transpose and the inverse. These return a matrix, when defined. There is also the determinant and the trace, which return a scalar from a matrix. The trace is just the sum of the diagonal; the determinant is more involved to compute, but was previously seen to have a relationship to the volume of a certain parallellpiped. There are a few other operations described in the following. - -. - -## Scalar-valued functions of a vector +##### Example: directional derivatives Suppose $f: R^n \rightarrow R$, a scalar-valued function of a vector. Then the directional derivative at $x$ in the direction $v$ was defined for a scalar $\alpha$ by: @@ -128,10 +147,10 @@ $$ This rate of change in the direction of $v$ can be expressed through the linear operator $f'(x)$ via $$ -f(x + d\alpha v) - f(x) = f'(x) [d\alpha v] = d\alpha f'(x)[v], +df = f(x + d\alpha v) - f(x) = f'(x) [d\alpha v] = d\alpha f'(x)[v], $$ -using linearity to move the scalar part outside the $[]$. This connects the partial derivative at $x$ in the direction of $v$ with $f'(x)$: +using linearity to move the scalar multiplication by $d\alpha$ outside the action of the linear operator. This connects the partial derivative at $x$ in the direction of $v$ with $f'(x)$: $$ \frac{\partial}{\partial \alpha}f(x + \alpha v) \mid_{\alpha = 0} = @@ -139,28 +158,25 @@ f'(x)[v]. $$ -Not only does this give a connection in notation with the derivative, it naturally illustrates how the derivative as a linear operator can act on non-infinitesimal values. +Not only does this give a connection in notation with the derivative, it naturally illustrates how the derivative as a linear operator can act on non-infinitesimal values, in this case on $v$. -Previously, we wrote $\nabla f \cdot v$ for the directional derivative, where the gradient is a column vector. The above uses the identification -$f' = (\nabla f)^T$. +Previously, we wrote $\nabla f \cdot v$ for the directional derivative, where the gradient is a column vector. -For $f: R^n \rightarrow R$ we have +The above uses the identification $f' = (\nabla f)^T$. +For $f: R^n \rightarrow R$ we have $df = f(x + dx) - f(x) = f'(x) [dx]$ is a scalar, so if $dx$ is a column vector, $f'(x)$ is a row vector with the same number of components (just as $\nabla f$ is a column vector with the same number of components). The operation $f'(x)[dx]$ is just matrix multiplication, which is a linear operation. -$$ -df = f(x + dx) - f(x) = f'(x) [dx] -$$ +##### Example: derivative of a matrix expression -is a scalar, so if $dx$ is a column vector, $f'(x)$ is a row vector with the same number of components (just as $\nabla f$ is a column vector with the same number of components). +@BrightEdelmanJohnson include this example to show that the computation of derivatives using components can be avoided. Consider $f(x) = x^T A x$ where $x$ is a vector in $R^n$ and $A$ is an $n\times n$ matrix. This type of expression is common. -##### Examples -@BrightEdelmanJohnson include this example to show that the computation of derivatives using components can be avoided. Consider $f(x) = x^T A x$ where $x$ is a vector in $R^n$ and $A$ is an $n\times n$ matrix. Then $f: R^n \rightarrow R$ and its derivative can be computed: +Then $f: R^n \rightarrow R$ and its derivative can be computed: $$ \begin{align*} df &= f(x + dx) - f(x)\\ &= (x + dx)^T A (x + dx) - x^TAx \\ -&= x^TAx + dx^TA x + x^TAx + dx^T A dx - x^TAx\\ +&= \textcolor{blue}{x^TAx} + dx^TA x + \textcolor{blue}{x^TAx} + \textcolor{red}{dx^T A dx} - \textcolor{blue}{x^TAx}\\ &= dx^TA x + x^TAdx \\ &= (dx^TAx)^T + x^TAdx \\ &= x^T A^T dx + x^T A dx\\ @@ -169,9 +185,9 @@ df &= f(x + dx) - f(x)\\ $$ The term $dx^t A dx$ is dropped, as it is higher order (goes to zero faster), it containing two $dx$ terms. -In the second to last step, an identity operation (taking the transpose of the scalar quantity) is taken to simplify the algebra. Finally, as $df = f'(x)[dx]$ the identity of $f'(x) = x^T(A^T+A)$ is made, or taking transposes $\nabla f = (A + A^T)x$. +In the second to last step, an identity operation (taking the transpose of the scalar quantity) is taken to simplify the algebra. Finally, as $df = f'(x)[dx]$ the identity of $f'(x) = x^T(A^T+A)$ is made, or taking transposes $\nabla f(x) = (A + A^T)x$. -Compare the elegance above, with the component version, even though simplified, it still requires a specification of the size to carry the following out: +Compare the elegance above with the component version, even though simplified, which still requires a specification of the size to carry the following out: ```{julia} using SymPy @@ -193,7 +209,7 @@ all(a == b for (a,b) ∈ zip(grad_u, grad_u_1)) ``` ----- +##### Example: derivative of matrix application For $f: R^n \rightarrow R^m$, @BrightEdelmanJohnson give an example of computing the Jacobian without resorting to component wise computations. Let $f(x) = Ax$ with $A$ being a $m \times n$ matrix, it follows that @@ -202,14 +218,18 @@ $$ df &= f(x + dx) - f(x)\\ &= A(x + dx) - Ax\\ &= Adx\\ -&= f'(x) dx. +&= f'(x)[dx]. \end{align*} $$ -The Jacobian is the linear operator $A$ acting on $dx$. +The Jacobian is the linear operator $A$ acting on $dx$. (Seeing that $Adx = f'(x)[dx]$ implies $f'(x)=A$ comes as this action is true for any $dx$, hence the actions must be the same.) + +## Differentation rules + +Various differentiation rules are still available such as the sum, product, and chain rules. -## Sum and product rules for the derivative +### Sum and product rules for the derivative Using the differential notation -- which implicitly ignores higher order terms as they vanish in a limit -- the sum and product rules can be derived. @@ -218,60 +238,60 @@ For the sum rule, let $f(x) = g(x) + h(x)$. Then $$ \begin{align*} df &= f(x + dx) - f(x) \\ -&= f'(x) dx\\ +&= f'(x)[dx]\\ &= \left(g(x+dx) + h(x+dx)\right) - \left(g(x) + h(x)\right)\\ &= \left(g(x + dx) - g(x)\right) + \left(h(x + dx) - h(x)\right)\\ -&= g'(x)dx + h'(x) dx\\ -&= \left(g'(x) + h'(x)\right) dx +&= g'(x)[dx] + h'(x)[dx]\\ +&= \left(g'(x) + h'(x)\right)[dx] \end{align*} $$ -Comparing we get $f'(x) = g'(x) + h'(x)$. +Comparing we get $f'(x)dx = (g'(x) + h'(x))[dx]$ or $f'(x) = g'(x) + h'(x)$. The sum rule has the same derivation as was done with univariate, scalar functions. Similarly for the product rule. -The product rule has with $f(x) = g(x)h(x)$ +The product rule for $f(x) = g(x)h(x)$ comes as: $$ \begin{align*} df &= f(x + dx) - f(x) \\ &= g(x+dx)h(x + dx) - g(x) h(x)\\ &= \left(g(x) + g'(x)dx\right)\left(h(x) + h'(x) dx\right) - \left(g(x) h(x)\right) \\ -&= g(x)h(x) + g'(x) dx h(x) + g(x) h'(x) dx + g'(x)dx h'(x) dx - g(x) h(x)\\ -&= gh + dg h + gdh + dg dh - gh\\ - &= dg h + gdh, +&= \textcolor{blue}{g(x)h(x)} + g'(x) dx h(x) + g(x) h'(x) dx + \textcolor{red}{g'(x)dx h'(x) dx} - \textcolor{blue}{g(x) h(x)}\\ +&= g'(x)dxh(x) + g(x)h'(x) dx\\ +&= (g'(x)h(x) + g(x)h'(x)) dx \end{align*} $$ **after** dropping the higher order term and cancelling $gh$ terms of opposite signs in the fourth row. -##### Examples +##### Example -These two rules can be used to show the last two examples: +These two rules can be used to directly show the last two examples. -First, to differentiate $f(x) = x^TAx$: - -$$ -\begin{align*} -df &= dx^T (Ax) + x^T d(Ax) \\ -&= x^T A^T dx + x^T A dx \\ -&= x^T(A^T + A) dx -\end{align*} -$$ - -Again, taking the transpose of the scalar quantity $x^TAdx$ to simplify the expression. - -When $A^T = A$ ($A$ is symmetric) this simplifies to a more familiar looking $2x^TA$, but we see that this requires assumptions not needed in the scalar case. - - -Next, if $f(x) = Ax$ then +First, if $f(x) = Ax$ and $A$ is a constant, then: $$ df = (dA)x + A(dx) = 0x + A dx = A dx, $$ -$A$ being a constant here. +Next, to differentiate $f(x) = x^TAx$: + +$$ +\begin{align*} +df &= dx^T (Ax) + x^T d(Ax) \\ +&= (dx^T (Ax))^T + x^T A dx \\ +&= x^T A^T dx + x^T A dx \\ +&= x^T(A^T + A) dx +\end{align*} +$$ + +In the second line the transpose of the scalar quantity $x^TAdx$ it taken to simplify the expression and the first calculation is used. + +When $A^T = A$ ($A$ is symmetric) this simplifies to a more familiar looking $2x^TA$, but we see that this requires assumptions not needed in the scalar case. + + ##### Example @@ -303,7 +323,7 @@ $$ They compute the derivative of $f(x) = A(x .* x)$ for some fixed matrix $A$ of the proper size. -We can see that $d (\text{diag}(v)w) = d(\text{diag}(v)) w + \text{diag}(v) dw = (dx) .* w + x .* dw$. So +We can see by the product rule that $d (\text{diag}(v)w) = d(\text{diag}(v)) w + \text{diag}(v) dw = (dx) .* w + x .* dw$. So $df = A(dx .* x + x .* dx) = 2A(x .* dx)$, as $.*$ is commutative by its definition. Writing this as $df = 2A(x .* dx) = 2A(\text{diag}(x) dx) = (2A\text{diag}(x)) dx$, we identify $f'(x) = 2A\text{diag}(x)$. @@ -311,7 +331,12 @@ $df = A(dx .* x + x .* dx) = 2A(x .* dx)$, as $.*$ is commutative by its definit This operation is called the [Hadamard product](https://en.wikipedia.org/wiki/Hadamard_product_(matrices)) and it extends to matrices and arrays. -## The chain rule +::: {.callout-note appearance="minimal"} +## Numerator layout +The Wikipedia page on [matrix calculus](https://en.wikipedia.org/wiki/Matrix_calculus#Layout_conventions) has numerous such "identities" for derivatives of different common matrix/vector expressions. As vectors are viewed as column vectors; the "numerator layout" identities apply. +::: + +### The chain rule Like the product rule, the chain rule is shown by @BrightEdelmanJohnson in this notation with $f(x) = g(h(x))$: @@ -320,13 +345,12 @@ $$ df &= f(x + dx) - f(x)\\ &= g(h(x + dx)) - g(h(x))\\ &= g(h(x) + h'(x)[dx]) - g(h(x))\\ +&= g(h(x)) + g'(h(x))[h'(x)[dx]] - g(h(x))\\ &= g'(h(x)) [h'(x) [dx]]\\ &= (g'(h(x)) h'(x)) [dx] \end{align*} $$ -(The limit requires a bit more detail.) - The operator $f'(x)= g'(h(x)) h'(x)$ is a product of matrices. ### Computational differences with expressions from the chain rule @@ -341,16 +365,18 @@ $$ Multiplying left to right (the first) is called reverse mode; multiplying right to left (the second) is called forward mode. The distinction becomes important when considering the computational cost of the multiplications. -* If $f: R^n \rightarrow R^m$ has $n$ much bigger than $1$ and $m=1$, then it is much faster to do left to right multiplication -* if $f:R^n \rightarrow R^m$ has $n=1$ and $m$ much bigger than one, the it is faster to do right to left multiplication. +* If $f: R^n \rightarrow R^m$ has $n$ much bigger than $1$ and $m=1$, then it is much faster to do left to right multiplication (many more inputs than outputs) +* if $f:R^n \rightarrow R^m$ has $n=1$ and $m$ much bigger than one, the it is faster to do right to left multiplication (many outputs than inputs) -The basic idea comes down to the shape of the matrices. When $m=1$, the derviative is a product of matrices of size $n\times j$ $j\times k$ and $k \times 1$ yielding a matrix of size $n \times 1$ matching the function dimension. Matrix multiplication of an $m \times q$ times $q \times n$ takes an order of $mqn$ operations. The multiplication of left to right is then +The basic idea comes down to the shape of the matrices. When $m=1$, the derviative is a product of matrices of size $n\times j$, $j\times k$, and $k \times 1$ yielding a matrix of size $n \times 1$ matching the function dimension. Matrix multiplication of an $m \times q$ matrix times a $q \times n$ matrix takes an order of $mqn$ operations. -The first operation takes $njk$ operation leaving an $n\times k$ matrix, the next multiplication then takes another $nk1$ operations or $njk + nk$ together. Whereas computing from the right to left is first $jk1$ operations leaving a $j \times 1$ matrix. The next operation would take another $nk1$ operations. In totalL +The operations involved in multiplication of left to right can be quantified. The first operation takes $njk$ operation leaving an $n\times k$ matrix, the next multiplication then takes another $nk1$ operations or $njk + nk$ together. + +Whereas computing from the right to left is first $jk1$ operations leaving a $j \times 1$ matrix. The next operation would take another $nk1$ operations. In total: * left to right is $njk + nk$ = $nk \cdot (1 + j)$. -* right to left is $jk + j = j\cdot (k+1)$. +* right to left is $jk + jn = j\cdot (k+n)$. When $j=k$, say, we can compare and see the second is a factor less in terms of operations. This can be quite significant in higher dimensions, whereas the dimensions of calculus (where $n$ and $m$ are $3$ or less) it is not an issue. @@ -362,8 +388,8 @@ Using the `BenchmarkTools` package, we can check the time to compute various pro ```{julia} using BenchmarkTools n,j,k,m = 20,15,10,1 -@btime A*(B*C) setup=(A=rand(n,j);B=rand(j,k); C=rand(k,m)); -@btime (A*B)*C setup=(A=rand(n,j);B=rand(j,k); C=rand(k,m)); +@btime A*(B*C) setup=(A=rand(n,j); B=rand(j,k); C=rand(k,m)); +@btime (A*B)*C setup=(A=rand(n,j); B=rand(j,k); C=rand(k,m)); ``` The latter computation is about 1.5 times slower. @@ -371,18 +397,15 @@ The latter computation is about 1.5 times slower. Whereas the relationship is changed when the first matrix is skinny and the last is not: ```{julia} -@btime A*(B*C) setup=(A=rand(m,k);B=rand(k,j); C=rand(j,n)); -@btime (A*B)*C setup=(A=rand(m,k);B=rand(k,j); C=rand(j,n)); +@btime A*(B*C) setup=(A=rand(m,k); B=rand(k,j); C=rand(j,n)); +@btime (A*B)*C setup=(A=rand(m,k); B=rand(k,j); C=rand(j,n)); ``` ##### Example In calculus, we have $n$ and $m$ are $1$,$2$,or $3$. But that need not be the case, especially if differentiation is over a parameter space. -XXXX -(Maybe the ariplain wing, but please, something origi - - +XXX insert example XXX ## Derivatives of matrix functions @@ -394,22 +417,20 @@ This derivative can be derived directly from the *product rule*: $$ \begin{align*} -f'(A) &= [AA]'\\ -&= A dA + dA A +df &= d(A^2) = d(AA)\\ +&= dA A + A dA \end{align*} $$ That is $f'(A)$ is the operator $f'(A)[\delta A] = A \delta A + \delta A A$ and not $2A\delta A$, as $A$ may not commute with $\delta A$. -XXX THIS ISN"T EVEN RIGHT - ### Vectorization of a matrix Alternatively, we can identify $A$ through its components, as a vector in $R^{n^2}$ and then leverage the Jacobian. One such identification is vectorization -- consecutively stacking the -column vectors into a vector. In `Julia` the `vec` function does this +column vectors into a single vector. In `Julia` the `vec` function does this operation: ```{julia} @@ -507,12 +528,13 @@ The $m\times n$ matrix $A$ and $j \times k$ matrix $B$ has a Kronecker product w The Kronecker product has a certain algebra, including: -* transposes: $(A \otimes B)^T) = A^T \otimes B^T$ -* multiplication: $(A\otimes B)(C \otimes D) = (AC) \otimes (BD)$ -* inverses: $(A \otimes B)^{-1} = (A^{-1}) \otimes (B^{-1})$ +* transposes: $(A \otimes B)^T = A^T \otimes B^T$ * orthogonal: $(A\otimes B)^T = (A\otimes B)$ if both $A$ and $B$ has the same property -* determinants: $\det(A\otimes B) = \det(A)^m \det(B)^n$, where $A$ is $n\times n$, $B$ is $m \times m$. -* trace (sum of diagonal): $\text{tr}(A \otimes B) = \text{tr}(A)\text{tr}(B)$. +* trace (sum of diagonal): $\text{tr}(A \otimes B) = \text{tr}(A)\text{tr}(B)$.* determinants: $\det(A\otimes B) = \det(A)^m \det(B)^n$, where $A$ is $n\times n$, $B$ is $m \times m$. +* inverses: $(A \otimes B)^{-1} = (A^{-1}) \otimes (B^{-1})$ +* multiplication: $(A\otimes B)(C \otimes D) = (AC) \otimes (BD)$ + + The main equation coupling `vec` and `kron` is the fact that if $A$, $B$, and $C$ have appropriate sizes, then: @@ -524,7 +546,7 @@ Appropriate sizes for $A$, $B$, and $C$ are determined by the various products i If $A$ is $m \times n$ and $B$ is $r \times s$, then since $BC$ is defined, $C$ has $s$ rows, and since $CA^T$ is defined, $C$ must have $n$ columns, as $A^T$ is $n \times m$, so $C$ must be $s\times n$. Checking this is correct on the other side, $A \times B$ would be size $mr \times ns$ and $\vec{C}$ would be size $sn$, so that product works, size wise. -The referred to notes have an explanation for this formula, but we confirm with an example with $m=n-2$, $r=s=3$: +The referred to notes have an explanation for this formula, but we confirm with an example with $m=n=2$ and $r=s=3$: ```{julia} @syms A[1:2, 1:2]::real B[1:3, 1:3]::real C[1:3, 1:2]::real @@ -536,7 +558,7 @@ all(l == r for (l, r) ∈ zip(L, R)) Now to use this relationship to recognize $df = A dA + dA A$ with the Jacobian computed from $\text{vec}{f(a)}$. -We have $\text{vec}(A dA + dA A) = \text{vec}(A dA) + \text{vec}(dA A)$, by obvious linearity of $\text{vec}$. Now inserting an identity matrix, $I$, which is symmteric, we have: +We have $\text{vec}(A dA + dA A) = \text{vec}(A dA) + \text{vec}(dA A)$, by obvious linearity of $\text{vec}$. Now inserting an identity matrix, $I$, which is symmteric, in a useful spot we have: $$ \text{vec}(A dA) = \text{vec}(A dA I^T) = (I \otimes A) \text{vec}(dA), @@ -545,7 +567,7 @@ $$ and $$ -\text{vec}(dA A) = \text{vec}(I dA (A^T)^T) = (A^T \otimes I) \text{vec}(dA) +\text{vec}(dA A) = \text{vec}(I dA (A^T)^T) = (A^T \otimes I) \text{vec}(dA). $$ This leaves @@ -576,13 +598,14 @@ $$ The above shows how to relate the derivative of a matrix function to the Jacobian of a vectorized function, but only for illustration. It -is decidely not necessary to express the derivative of $f$ in terms of +is certainly not necessary to express the derivative of $f$ in terms of the derivative of its vectorized counterpart. -##### Example: derivative of the inverse +##### Example: derivative of the matrix inverse -What is the derivative of $f(A) = A^{-1}$. When $A$ is a scalar, we related it to the reciprocal of the derivative of $f$ at some other point. The same technique is available. Starting with $I = AA^{-1}$ and noting $dI$ is $0$ we have +What is the derivative of $f(A) = A^{-1}$? The same technique used to find the derivative of the inverse of a univariate, scalar-valued function is useful. +Starting with $I = AA^{-1}$ and noting $dI$ is $0$ we have $$ \begin{align*} @@ -602,11 +625,11 @@ $$ = \left((A^T)^{-1} \otimes A^{-1}\right) \text{vec}(dA). $$ -##### Example: derivative of the determinant +##### Example: derivative of the matrix determinant Let $f(A) = \text{det}(A)$. What is the derivative? -First, the determinant of a square, $n\times n$, matrix $A$ is a scalar summary of $A$ with different means to compute it, but one recursive one in particular is helpful here: +First, the determinant of a square, $n\times n$, matrix $A$ is a scalar summary of $A$. There are different means to compute the determinant, but this recursive one in particular is helpful here: $$ \text{det}(A) = a_{1j}C_{1j} + a_{2j}C_{2j} + \cdots a_{nj}C_{nj} @@ -626,10 +649,10 @@ as each cofactor in the expansion has no dependence on $A_{ij}$ as the cofactor So the gradient is the matrix of cofactors. -@BrightEdelmanJohnson also give a different proof, starting with this observation +@BrightEdelmanJohnson also give a different proof, starting with this observation: $$ -\text{det}(I + dA) - \text{det}(I) = \text{tr}(dA) +\text{det}(I + dA) - \text{det}(I) = \text{tr}(dA). $$ Assuming that, then by the fact $\text{det}(AB) = \text{det}(A)\text{det}(B)$: @@ -638,7 +661,7 @@ $$ \begin{align*} \text{det}(A + A(A^{-1}dA)) - \text{det}(A) &= \text{det}(A)\cdot(\text{det}(I+ A^{-1}dA) - \text{det}(I)) \\ &= \text{det}(A) \text{tr}(A^{-1}dA)\\ -&= \text{tr}(\text{det}(A)A^{-1}dA)\\ +&= \text{tr}(\text{det}(A)A^{-1}dA). \end{align*} $$ @@ -663,7 +686,7 @@ $$ g(p) = f(A(p)^{-1} b) $$ -This might arise from applying a scalar-valued $f$ to the solution of $Ax = b$, where $A$ is parameterized by $p$. +This might arise from applying a scalar-valued $f$ to the solution of $Ax = b$, where $A$ is parameterized by $p$. The number of parameters might be quite large, so how the resulting computation is organized might effect the computational costs. The chain rule gives the following computation to find the derivative (or gradient): @@ -671,28 +694,28 @@ $$ \begin{align*} dg &= f'(x)[dx]\\ -&= f'(x) [d(A(p)^{1} b)]\\ +&= f'(x) [d(A(p)^{-1} b)]\\ &= f'(x)[-A(p)^{-1} dA A(p)^{-1} b + 0]\\ -&= -f'(x) A(p)^{-1} dA A(p)^{-1} b. +&= -\textcolor{red}{f'(x) A(p)^{-1}} dA\textcolor{blue}{A(p)^{-1}[b]}. \end{align*} $$ -By writing $dA = A'(p)[dp]$ and setting $v^T = f'(x)A(p)^{-1}$ this becomes +By setting $v^T = f'(x)A(p)^{-1}$ and writing $x = A(p)^{-1}[b]$ this becomes $$ -dg = -v^T dA A(p)^{-1} b = -v^T dA x +dg = -v^T dA x. $$ This product of three terms can be computed in two directions: - From left to right: +*From left to right:* First $v$ is found by solving $v^T = f'(x) A^{-1}$ through the solving of $v = (A^{-1})^T (f'(x))^T = (A^T)^{-1} \nabla(f)$ or by solving $A^T v = \nabla f$. This is called the *adjoint* equation. -The partial derivatives in $g$ is related to each partial derivative of $dA$ through: +The partial derivatives in $p$ of $g$ are related to each partial derivative of $dA$ through: $$ \frac{\partial g}{\partial p_k} = -v^T\frac{\partial A}{\partial p_k} x, @@ -700,7 +723,7 @@ $$ as the scalar factor commutes through. With $v$ and $x$ solved for (via the adjoint equation and from solving $Ax=b$) the partials in $p_k$ are computed with dot products. There are just two costly operations. -From right to left: +*From right to left:* The value of $x$ can be solved for, as above, but computing the value of @@ -709,64 +732,17 @@ $$ -f'(x) \left(A^{-1} \frac{\partial A}{\partial p_k} x \right) $$ -requires a costly solve for each $p_k$, and $p$ may have many components. As mentioned above, the reverse mode offers advantages when there are many input parameters ($p$) and a single output parameter. +requires a costly solve of $A^{-1}\frac{\partial A}{\partial p_k} x$ for each $p_k$, and $p$ may have many components. This is the difference: left to right only has the solve of the one adjoint equation. + +As mentioned above, the reverse mode offers advantages when there are many input parameters ($p$) and a single output parameter. + + ##### Example Suppose $x(p)$ solves some system of equations $h(x(p),p) = 0$ in $R^n$ ($n$ possibly just $1$) and $g(p) = f(x(p))$ is some non-linear transformation of $x$. What is the derivative of $g$ in $p$? -Suppose the *implicit function theorem* applies to $h(x,p) = 0$, that is -- *locally* -- there is an implicitly defined function $x(p)$ with a derivative. Moreover by differentiating both sides it can be identified: - -$$ -0 = \frac{\partial h}{\partial p} dp + \frac{\partial h}{\partial x} dx -$$ - -which can be solved for $dx$ to give - -$$ -dx = -\left(\frac{\partial h}{\partial x}\right)^{-1} \frac{\partial h}{\partial p} dp. -$$ - -The chain rule then gives - -$$ -dg = f'(x) dx = -f'(x) \left(\frac{\partial h}{\partial x}\right)^{-1} \frac{\partial h}{\partial p} dp. -$$ - -This can be computed in two directions: - -From left to right: - -Call $A =\left(\frac{\partial h}{\partial x}\right)^{-1}$. Then define $v$ indirectly through $v^T = f'(x) A^{-1}$. With this: -$v = (A^{-1})^T (f'(x))^T = (A^T)^{-1} \nabla{f}$ -which is found by solving -$A^Tv = \nabla{f}$. -Again, this is the *adjoint* equation. - -The value of $dA$ is related to each partial derivative for which - -$$ -\frac{\partial g}{\partial p_k} = -v^T\frac{\partial A}{\partial p_k} x, -$$ - -as the scalar factor commutes through. With $v$ and $x$ solved for (via the adjoint equation and from solving $Ax=b$) the partials in $p_k$ are computed with dot products. - -However, from right to left, the value of $x$ can be solved for, but computing the value of - -$$ -\frac{\partial g}{\partial p_k} = --f'(x) -\left(A^{-1} \frac{\partial A}{\partial p_k} x \right) -$$ - -requires a costly solve for each $p_k$, and $p$ may have many components. The reverse mode offers advantages when there are many input parameters ($p$) and a single output parameter. - -##### Example - - -Suppose $x(p)$ solves some system of equations $h(x(p),p) = 0$ in $R^n$ ($n$ possibly just 1$) and $g(p) = f(x(p))$ is some non-linear transformation of $x$. What is the derivative of $g$ in $p$? - Suppose the *implicit function theorem* applies to $h(x,p) = 0$, that is *locally* the response $x(p)$ has a derivative, and moreover by the chain rule $$ @@ -782,16 +758,16 @@ $$ The chain rule applied to $g(p) = f(x(p))$ then yields $$ -dg = f'(x) dx = - f'(x) \left(\frac{\partial h}{\partial x}\right)^{-1} \frac{\partial h}{\partial p} dp. +dg = f'(x) dx = - f'(x) \left(\frac{\partial h}{\partial x}\right)^{-1} \frac{\partial h}{\partial p} dp = -v^T\frac{\partial h}{\partial p} dp, $$ -Setting +by setting $$ -v^T = -f'(x) \left(\frac{\partial h}{\partial x}\right)^{-1} +v^T = f'(x) \left(\frac{\partial h}{\partial x}\right)^{-1}. $$ -then $v$ can be solved from taking adjoints (as before). Let $A = \partial h/\partial x$, the $v^T = -f'(x) A^{-1}$ or $v = -(A^{-1})^T (f'(x))^t= -(A^T)^{-1} \nabla f$. As before it would take two solves to get both $g$ and its gradient. +Here $v$ can be solved for by taking adjoints (as before). Let $A = \partial h/\partial x$, then $v^T = f'(x) A^{-1}$ or $v = (A^{-1})^T (f'(x))^t= (A^T)^{-1} \nabla f$. That is $v$ solves $A^Tv=\nabla f$. As before it would take two solves to get both $g$ and its gradient. ## Second derivatives, Hessian @@ -854,57 +830,58 @@ This formula -- $f(x+dx)-f(x) \approx f'(x)dx + dx^T H dx$ -- is valid for any $ By uniqueness, we have under these assumptions that the Hessian is *symmetric* and the expression $dx^T H dx$ is a *bilinear* form, which we can identify as $f''(x)[dx,dx]$. -That the Hessian is symmetric could also be derived under these assumptions by directly computing that the mixed partials can have their order exchanged. But in this framework, as explained by @BrightEdelmanJohnson it is a result of the underlying vector space having an addition that is commutative (e.g. $u+v = v+u$). +That the Hessian is symmetric could also be derived under these assumptions by directly computing that the mixed partials can have their order exchanged. But in this framework, as explained by @BrightEdelmanJohnson (and shown later) it is a result of the underlying vector space having an addition that is commutative (e.g. $u+v = v+u$). The mapping $(u,v) \rightarrow u^T A v$ for a matrix $A$ is bilinear. For a fixed $u$, it is linear as it can be viewed as $(u^TA)[v]$ and matrix multiplication is linear. Similarly for a fixed $v$. @BrightEdelmanJohnson extend this characterization to a broader setting. -The second derivative can be viewed as expressing first-order change in $f'(x)$, a linear operator. The value $df'$ has the same shape as $f'$, which is a linear operator, so $df'$ acts on vectors, say $dx$, then: + +We have for some function $f$ $$ -df'[dx] = f''(x)[dx'][dx] = f''(x)[dx', dx] +df = f(x + dx) - f(x) = f'(x)[dx] $$ -The prime in $dx'$ is just notation, not a derivative operation for $dx$. +Then if $d\tilde{x}$ is another differential change with the same shape as $x$ we can look at the differential of $f'(x)$: -With this view, we can see that $f''(x)$ has two vectors it acts on. By definition it is linear in $dx$. However, as $f'(x)$ is a linear operator and the sum and product rules apply to derivatives, this operator is linear in $dx'$ as well. So $f''(x)$ is bilinear and as mentioned earlier symmetric. +$$ +d(f') = f'(x + d\tilde{x}) - f'(x) = f''(x)[d\tilde{x}] +$$ -### Polarization -@BrightEdelmanJohnson interpret $f''$ by looking at the image under $f$ of $x + dx + dx'$. If $x$ is a vector, then this has a geometrical picture, from vector addtion, relating $x + dx$, $x+dx'$, and $x + dx + dx'$. +Now, $d(f')$ has the same shape as $f'$, a linear operator, hence $d(f')$ is also a linear operator. Acting on $dx$, we have -The image for $x +dx$ is to second order $f(x) + f'(x)[dx] + (1/2)f''(x)[dx, dx]$, similarly $x + dx'$ is to second order $f(x) + f'(x)[dx'] + (1/2)f''(x)[dx', dx']$. The key formula for $f''(x)$ is +$$ +d(f')[dx] = f''(x)[d\tilde{x}][dx] = f''(x)[d\tilde{x}, dx]. +$$ + +The last equality a definition. As $f''$ is linear in the the application to $d\tilde{x}$ and also linear in application to $dx$, $f''(x)$ is a bilinear operator. + +Moreover, the following shows it is *symmetric*: $$ \begin{align*} -f(x + dx + dx') &= f(x) + f'(x)[dx + dx'] + \frac{1}{2}f''(x)[dx, dx']\\ -&= f(x) + f'(x)[dx] + (1/2)f''(x)[dx, dx] -&+ f(x) + f'(x)[dx] + (1/2)f''(x)[dx', dx'] -&+ f''(x)[dx, dx'] -\end{align} +f''(x)[d\tilde{x}][dx] &= (f'(x + d\tilde{x}) - f'(x))[dx]\\ +&= f'(x + d\tilde{x})[dx] - f'(x)[dx]\\ +&= (f(x + d\tilde{x} + dx) - f(x + d\tilde{x})) - (f(x+dx) - f(x))\\ +&= (f(x + dx + d\tilde{x}) - f(x + dx)) - (f(x + d\tilde{x}) - f(x))\\ +&= f'(x + dx)[d\tilde{x}] - f'(x)[d\tilde{x}]\\ +&= f''(x)[dx][d\tilde{x}] +\end{align*} $$ -This gives a means to compute $f''$ in terms of $f''$ acting on diagonal terms, where the two vectors are equal: +So $f''(x)[d\tilde{x},dx] = f''(x)[dx, d\tilde{x}]$. The key is the commutivity of vector addition to say $dx + d\tilde{x} = d\tilde{x} + dx$ in the third line. -$$ -f''(x)[dx, dx'] = \frac{1}{2} f''(x)[dx+dx',dx+dx'] - f''(x)[dx,dx] - f''(x)[dx',dx'] -$$ +##### Example: Hessian is symmetric -### XXX does this fit in? +As mentioned earlier, the Hessian is the matrix arising from finding the second derivative of a multivariate, scalar-valued function $f:R^n \rightarrow R$. As a bilinear form on a finite vector space, it can be written as $\tilde{x}^T A x$. As this second derivative is symmetric, and this value above a scalar, it follows that $\tilde{x}^T A x = \tilde{x}^T A^T x$. That is $H = A$ must also be symmetric from general principles. -However, as a description of second-order change in $f$, we recover the initial terms in the Taylor series - -$$ -f(x + \delta x) = f(x) + f'(x)\delta x + (1/2) f''(x)[\delta x, \delta x] + \mathscr{o}(||\delta x||^2). -$$ - -### Examples ##### Example: second derivative of $x^TAx$ -Consider an expression from earlier $f(x) = x^T A x$ for some constant $A$. Then $f''$ is found by noting that $f' = (\nabla f)^T = x^T(A + A^T)$, or $\nabla f = (A^T + A)x$ and $f'' = H = A^T + A$ is the Jacobian of the gradient. +Consider an expression from earlier $f(x) = x^T A x$ for some constant $A$. -By rearranging terms, it can be shown that $f(x) = 1/2 x^THx = 1/2 f''[x,x]$. +We have seen that $f' = (\nabla f)^T = x^T(A+A^T)$. That is $\nabla f = (A^T+A)x$ is linear in $x$. The Jacobian of $\nabla f$ is the Hessian, $H = f'' = A + A^T$. ##### Example: second derivative of $\text{det}(A)$ @@ -939,10 +916,11 @@ $$ $$ So, after dropping the third-order term, we see: + $$ \begin{align*} -f''(A)&[dA,dA'] \\ -&= \text{det}(A)\text{tr}(A^{-1}dA')\text{tr}(A^{-1}dA) -- \text{det}(A)\text{tr}(A^{-1}dA' A^{-1}dA). +f''(A)[dA,dA'] +&= \text{det}(A)\text{tr}(A^{-1}dA')\text{tr}(A^{-1}dA)\\ +&\quad - \text{det}(A)\text{tr}(A^{-1}dA' A^{-1}dA). \end{align*} $$ diff --git a/quarto/staging/Project.toml b/quarto/staging/Project.toml deleted file mode 100644 index c319267..0000000 --- a/quarto/staging/Project.toml +++ /dev/null @@ -1,2 +0,0 @@ -[deps] -quarto_jll = "b7163347-bfae-5fd9-aba4-19f139889d78" diff --git a/quarto/staging/matrix-calculus-notes.html b/quarto/staging/matrix-calculus-notes.html deleted file mode 100644 index 9f7b7e8..0000000 --- a/quarto/staging/matrix-calculus-notes.html +++ /dev/null @@ -1,1282 +0,0 @@ - - - - - - - - - -matrix-calculus-notes - - - - - - - - - - - - - - - - - - - - - - - -
- -
- -
-
-

Matrix Calculus

-
- - - -
- - - - -
- - - -
- - -

XXX Add in examples from paper XXX optimization? large number of parameters? ,…

-

Mention numerator layout from https://en.wikipedia.org/wiki/Matrix_calculus#Layout_conventions

-
-
-
- -
-
-Based on Bright, Edelman, and Johnson’s notes -
-
-
-

This section samples material from the notes Matrix Calculus (for Machine Learning and Beyond) by Paige Bright, Alan Edelman, and Steven G. Johnson. These notes cover material taught in a course at MIT. Support materials for their course in Julia are available at https://github.com/mitmath/matrixcalc/tree/main. For more details and examples, please refer to the source.

-
-
-

We have seen several “derivatives” of a function, based on the number of inputs and outputs. The first one was for functions \(f: R \rightarrow R\).

-

In this case, we saw that \(f\) has a derivative at \(c\) if this limit exists:

-

\[ -\lim_{h \rightarrow 0}\frac{f(c + h) - f(c)}{h}. -\]

-

The derivative as a function of \(x\) using this rule for any \(x\) in the domain. Common notation is:

-

\[ -f'(x) = \frac{dy}{dx} = \lim_{h \rightarrow 0}\frac{f(x + h) - f(x)}{h} -\]

-

(when the limit exists).

-

This limit gets re-expressed in different ways:

-
    -
  • linearization writes \(f(x+\Delta x) - f(x) \approx f'(x)\Delta x\), where \(\Delta x\) is a small displacement from \(x\). The reason there isn’t equality is the unwritten higher order terms that vanish in a limit.

  • -
  • Alternate limits. Another way of writing this is in terms of explicit smaller order terms:

  • -
-

\[ -(f(x+h) - f(x)) - f'(x)h = \mathscr{o}(h), -\]

-

which means if we divide both sides by \(h\) and take the limit, we will get \(0\) on the right and the relationship on the left.

-
    -
  • Differential notation simply writes this as \(dy = f'(x)dx\). Focusing on \(f\) and not \(y=f(x)\), we might write
  • -
-

\[ -df = f(x+dx) - f(x) = f'(x) dx. -\]

-

We will see all the derivatives encountered so far can be similarly expressed.

-

In the above, \(df\) and \(dx\) are differentials, made rigorous by a limit, which hides the higher order terms.

-

In these notes the limit has been defined, with suitable modification, for functions of vectors (multiple values) with scalar or vector outputs.

-

For example, when \(f: R \rightarrow R^m\) was a vector-valued function the derivative was defined similarly through a limit of \((f(t + \Delta t) - f(t))/{\Delta t}\), where each component needed to have a limit. This can be rewritten through \(f(t + dt) - f(t) = f'(t) dt\), again using differentials to avoid the higher order terms.

-

When \(f: R^n \rightarrow R\) is a scalar-valued function with vector inputs, differentiability was defined by a gradient existing with \(f(c+h) - f(c) - \nabla{f}(c) \cdot h\) being \(\mathscr{o}(\|h\|)\). In other words \(df = f(c + dh) - f(c) = \nabla{f}(c) \cdot dh\). The gradient has the same shape as \(c\), a column vector. If we take the row vector (e.g. \(f'(c) = \nabla{f}(c)^T\)) then again we see \(df = f(c+dh) - f(c) = f'(c) dh\), where the last term uses matrix multiplication of a row vector times a column vector.

-

Finally, when \(f:R^n \rightarrow R^m\), the Jacobian was defined and characterized by \(\| f(x + dx) - f(x) - J_f(x)dx \|\) being \(\mathscr{o}(\|dx\|)\). Again, we can express this through \(df = f(x + dx) - f(x) = f'(x)dx\) where \(f'(x) = J_f(x)\).

-

In writing \(df = f(x + dx) - f(x) = f'(x) dx\) generically, some underlying facts are left implicit: \(dx\) has the same shape as \(x\) (so can be added); \(f'(x) dx\) may mean usual multiplication or matrix multiplication; and there is an underlying concept of distance and size that allows the above to be rigorous. This may be an abolute value or a norm.

-

Further, various differentiation rules apply such as the sum, product, and chain rules.

-

The @BrightEdelmanJohnson notes cover differentiation of functions in this uniform manner and then extend the form by treating derivatives as linear operators.

-

A linear operator is a mathematical object which satisfies

-

\[ -f[\alpha v + \beta w] = \alpha f[v] + \beta f[w]. -\]

-

where the \(\alpha\) and \(\beta\) are scalars, and \(v\) and \(w\) possibly not and come from a vector space. Regular multiplication and matrix multiplication are familiar linear operations, but there are many others.

-

The referenced notes identify \(f'(x) dx\) as \(f'(x)[dx]\), the latter emphasizing \(f'(x)\) acts on \(dx\) and the notation is not commutative (e.g., it is not \(dx f'(x)\)). The use of \([]\) is to indicate that \(f'(x)\) “acts” on \(dx\) in a linear manner. It may be multiplication, matrix multiplication, or something else. Parentheses are not used which might imply function application or multiplication.

-

Linear operators are related to vector spaces.

-

A vector space is a set of mathematical objects which can be added together and also multiplied by a scalar. Vectors of similar size, as previously discussed, are the typical example, with vector addition and scalar multiplication previously discussed topics. Matrices of similar size (and some subclasses) also form a vector space. Additionally, many other set of objects form vector spaces. An example might be polynomial functions of degree \(n\) or less; continuous functions, or functions with a certain number of derivatives.

-

Take differentiable functions as an example, then the simplest derivative rules \([af(x) + bg(x)]' = a[f(x)]' + b[g(x)]'\) show the linearity of the derivative in this setting.

-

A finite vector space is described by a basis – a minimal set of vectors needed to describe the space, after consideration of linear combinations. For some typical vector spaces, this the set of special vectors with \(1\) as one of the entries, and \(0\) otherwise.

-

A key fact about a basis for a finite vector space is every vector in the vector space can be expressed uniquely as a linear combination of the basis vectors.

-

Vectors and matrices have properties that are generalizations of the real numbers. As vectors and matrices form vector spaces, the concept of addition of vectors and matrices is defined, as is scalar multiplication. Additionally, we have seen:

-
    -
  • The dot product between two vectors of the same length is defined easily (\(v\cdot w = \Sigma_i v_i w_i\)). It is coupled with the length as \(\|v\|^2 = v\cdot v\).

  • -
  • Matrix multiplication is defined for two properly sized matrices. If \(A\) is \(m \times k\) and \(B\) is \(k \times n\) then \(AB\) is a \(m\times n\) matrix with \((i,j)\) term given by the dot product of the \(i\)th row of \(A\) (viewed as a vector) and the \(j\)th column of \(B\) (viewed as a vector). Matrix multiplication is associative but not commutative. (E.g. \((AB)C = A(BC)\) but \(AB\) and \(BA\) need not be equal, or even defined, as the shapes may not match up).

  • -
  • A square matrix \(A\) has an inverse \(A^{-1}\) if \(AA^{-1} = A^{-1}A = I\), where \(I\) is the identity matrix (a matrix which is zero except on its diagonal entries, which are all \(1\)). Square matrices may or may not have an inverse. A matrix without an inverse is called singular.

  • -
  • Viewing a vector as a matrix is possible. The association is typically through a column vector.

  • -
  • The transpose of a matrix comes by permuting the rows and columns. The transpose of a column vector is a row vector, so \(v\cdot w = v^T w\), where we use a superscript \(T\) for the transpose. The transpose of a product, is the product of the transposes – reversed: \((AB)^T = B^T A^T\); the tranpose of a transpose is an identity operation: \((A^T)^T = A\); the inverse of a transpose is the tranpose of the inverse: \((A^{-1})^T = (A^T)^{-1}\).

  • -
  • The adjoint of a matrix is related to the transpose, only complex conjugates are also taken.

  • -
  • Matrices for which \(A = A^T\) are called symmetric.

  • -
  • A few of the operations on matrices are the transpose and the inverse. These return a matrix, when defined. There is also the determinant and the trace, which return a scalar from a matrix. The trace is just the sum of the diagonal; the determinant is more involved to compute, but was previously seen to have a relationship to the volume of a certain parallellpiped. There are a few other operations described in the following.

  • -
-

.

-
-

Scalar-valued functions of a vector

-

Suppose \(f: R^n \rightarrow R\), a scalar-valued function of a vector. Then the directional derivative at \(x\) in the direction \(v\) was defined for a scalar \(\alpha\) by:

-

\[ -\frac{\partial}{\partial \alpha}f(x + \alpha v) \mid_{\alpha = 0} = -\lim_{\Delta\alpha \rightarrow 0} \frac{f(x + \Delta\alpha v) - f(x)}{\Delta\alpha}. -\]

-

This rate of change in the direction of \(v\) can be expressed through the linear operator \(f'(x)\) via

-

\[ -f(x + d\alpha v) - f(x) = f'(x) [d\alpha v] = d\alpha f'(x)[v], -\]

-

using linearity to move the scalar part outside the \([]\). This connects the partial derivative at \(x\) in the direction of \(v\) with \(f'(x)\):

-

\[ -\frac{\partial}{\partial \alpha}f(x + \alpha v) \mid_{\alpha = 0} = -f'(x)[v]. -\]

-

Not only does this give a connection in notation with the derivative, it naturally illustrates how the derivative as a linear operator can act on non-infinitesimal values.

-

Previously, we wrote \(\nabla f \cdot v\) for the directional derivative, where the gradient is a column vector. The above uses the identification \(f' = (\nabla f)^T\).

-

For \(f: R^n \rightarrow R\) we have

-

\[ -df = f(x + dx) - f(x) = f'(x) [dx] -\]

-

is a scalar, so if \(dx\) is a column vector, \(f'(x)\) is a row vector with the same number of components (just as \(\nabla f\) is a column vector with the same number of components).

-
-
Examples
-

@BrightEdelmanJohnson include this example to show that the computation of derivatives using components can be avoided. Consider \(f(x) = x^T A x\) where \(x\) is a vector in \(R^n\) and \(A\) is an \(n\times n\) matrix. Then \(f: R^n \rightarrow R\) and its derivative can be computed:

-

\[ -\begin{align*} -df &= f(x + dx) - f(x)\\ -&= (x + dx)^T A (x + dx) - x^TAx \\ -&= x^TAx + dx^TA x + x^TAx + dx^T A dx - x^TAx\\ -&= dx^TA x + x^TAdx \\ -&= (dx^TAx)^T + x^TAdx \\ -&= x^T A^T dx + x^T A dx\\ -&= x^T(A^T + A) dx -\end{align*} -\]

-

The term \(dx^t A dx\) is dropped, as it is higher order (goes to zero faster), it containing two \(dx\) terms. In the second to last step, an identity operation (taking the transpose of the scalar quantity) is taken to simplify the algebra. Finally, as \(df = f'(x)[dx]\) the identity of \(f'(x) = x^T(A^T+A)\) is made, or taking transposes \(\nabla f = (A + A^T)x\).

-

Compare the elegance above, with the component version, even though simplified, it still requires a specification of the size to carry the following out:

-
-
using SymPy
-@syms x[1:3]::real A[1:3, 1:3]::real
-u = x' * A * x
-grad_u = [diff(u, xi) for xi in x]
-
-

\(\left[\begin{smallmatrix}2 A₁_{₁} x₁ + A₁_{₂} x₂ + A₁_{₃} x₃ + A₂_{₁} x₂ + A₃_{₁} x₃\\A₁_{₂} x₁ + A₂_{₁} x₁ + 2 A₂_{₂} x₂ + A₂_{₃} x₃ + A₃_{₂} x₃\\A₁_{₃} x₁ + A₂_{₃} x₂ + A₃_{₁} x₁ + A₃_{₂} x₂ + 2 A₃_{₃} x₃\end{smallmatrix}\right]\)

-
-
-

Compare to the formula for the gradient just derived:

-
-
grad_u_1 = (A + A')*x
-
-

\(\left[\begin{smallmatrix}2 A₁_{₁} x₁ + x₂ \left(A₁_{₂} + A₂_{₁}\right) + x₃ \left(A₁_{₃} + A₃_{₁}\right)\\2 A₂_{₂} x₂ + x₁ \left(A₁_{₂} + A₂_{₁}\right) + x₃ \left(A₂_{₃} + A₃_{₂}\right)\\2 A₃_{₃} x₃ + x₁ \left(A₁_{₃} + A₃_{₁}\right) + x₂ \left(A₂_{₃} + A₃_{₂}\right)\end{smallmatrix}\right]\)

-
-
-

The two are, of course, equal

-
-
all(a == b for (a,b)  zip(grad_u, grad_u_1))
-
-
true
-
-
-
-

For \(f: R^n \rightarrow R^m\), @BrightEdelmanJohnson give an example of computing the Jacobian without resorting to component wise computations. Let \(f(x) = Ax\) with \(A\) being a \(m \times n\) matrix, it follows that

-

\[ -\begin{align*} -df &= f(x + dx) - f(x)\\ -&= A(x + dx) - Ax\\ -&= Adx\\ -&= f'(x) dx. -\end{align*} -\]

-

The Jacobian is the linear operator \(A\) acting on \(dx\).

-
-
-
-

Sum and product rules for the derivative

-

Using the differential notation – which implicitly ignores higher order terms as they vanish in a limit – the sum and product rules can be derived.

-

For the sum rule, let \(f(x) = g(x) + h(x)\). Then

-

\[ -\begin{align*} -df &= f(x + dx) - f(x) \\ -&= f'(x) dx\\ -&= \left(g(x+dx) + h(x+dx)\right) - \left(g(x) + h(x)\right)\\ -&= \left(g(x + dx) - g(x)\right) + \left(h(x + dx) - h(x)\right)\\ -&= g'(x)dx + h'(x) dx\\ -&= \left(g'(x) + h'(x)\right) dx -\end{align*} -\]

-

Comparing we get \(f'(x) = g'(x) + h'(x)\).

-

The sum rule has the same derivation as was done with univariate, scalar functions. Similarly for the product rule.

-

The product rule has with \(f(x) = g(x)h(x)\)

-

\[ -\begin{align*} -df &= f(x + dx) - f(x) \\ -&= g(x+dx)h(x + dx) - g(x) h(x)\\ -&= \left(g(x) + g'(x)dx\right)\left(h(x) + h'(x) dx\right) - \left(g(x) h(x)\right) \\ -&= g(x)h(x) + g'(x) dx h(x) + g(x) h'(x) dx + g'(x)dx h'(x) dx - g(x) h(x)\\ -&= gh + dg h + gdh + dg dh - gh\\ - &= dg h + gdh, -\end{align*} -\]

-

after dropping the higher order term and cancelling \(gh\) terms of opposite signs in the fourth row.

-
-
Examples
-

These two rules can be used to show the last two examples:

-

First, to differentiate \(f(x) = x^TAx\):

-

\[ -\begin{align*} -df &= dx^T (Ax) + x^T d(Ax) \\ -&= x^T A^T dx + x^T A dx \\ -&= x^T(A^T + A) dx -\end{align*} -\]

-

Again, taking the transpose of the scalar quantity \(x^TAdx\) to simplify the expression.

-

When \(A^T = A\) (\(A\) is symmetric) this simplifies to a more familiar looking \(2x^TA\), but we see that this requires assumptions not needed in the scalar case.

-

Next, if \(f(x) = Ax\) then

-

\[ -df = (dA)x + A(dx) = 0x + A dx = A dx, -\]

-

\(A\) being a constant here.

-
-
-
Example
-

@BrightEdelmanJohnson consider what in Julia is .*. That is the operation:

-

\[ -v .* w = -\begin{bmatrix} -v_1w_1 \\ -v_2w_2 \\ -\vdots\\ -v_nw_n -\end{bmatrix} -= -\begin{bmatrix} -v_1 & 0 & \cdots & 0 \\ -0 & v_2 & \cdots & 0 \\ - & & \vdots & \\ -0 & 0 & \cdots & v_n -\end{bmatrix} -\begin{bmatrix} -w_1 \\ -w_2 \\ -\vdots\\ -w_n -\end{bmatrix} -= \text{diag}(v) w. -\]

-

They compute the derivative of \(f(x) = A(x .* x)\) for some fixed matrix \(A\) of the proper size.

-

We can see that \(d (\text{diag}(v)w) = d(\text{diag}(v)) w + \text{diag}(v) dw = (dx) .* w + x .* dw\). So

-

\(df = A(dx .* x + x .* dx) = 2A(x .* dx)\), as \(.*\) is commutative by its definition. Writing this as \(df = 2A(x .* dx) = 2A(\text{diag}(x) dx) = (2A\text{diag}(x)) dx\), we identify \(f'(x) = 2A\text{diag}(x)\).

-

This operation is called the Hadamard product and it extends to matrices and arrays.

-
-
-
-

The chain rule

-

Like the product rule, the chain rule is shown by @BrightEdelmanJohnson in this notation with \(f(x) = g(h(x))\):

-

\[ -\begin{align*} -df &= f(x + dx) - f(x)\\ -&= g(h(x + dx)) - g(h(x))\\ -&= g(h(x) + h'(x)[dx]) - g(h(x))\\ -&= g'(h(x)) [h'(x) [dx]]\\ -&= (g'(h(x)) h'(x)) [dx] -\end{align*} -\]

-

(The limit requires a bit more detail.)

-

The operator \(f'(x)= g'(h(x)) h'(x)\) is a product of matrices.

-
-

Computational differences with expressions from the chain rule

-

Of note here is the application of the chain rule to three (or more compositions):

-

The derivative of \(f(x) = a(x) b(x) c(x)\) can be expressed as

-

\[ -f' = (a'b')c' \text{ or } f' = a'(b'c') -\]

-

Multiplying left to right (the first) is called reverse mode; multiplying right to left (the second) is called forward mode. The distinction becomes important when considering the computational cost of the multiplications.

-
    -
  • If \(f: R^n \rightarrow R^m\) has \(n\) much bigger than \(1\) and \(m=1\), then it is much faster to do left to right multiplication
  • -
  • if \(f:R^n \rightarrow R^m\) has \(n=1\) and \(m\) much bigger than one, the it is faster to do right to left multiplication.
  • -
-

The basic idea comes down to the shape of the matrices. When \(m=1\), the derviative is a product of matrices of size \(n\times j\) \(j\times k\) and \(k \times 1\) yielding a matrix of size \(n \times 1\) matching the function dimension. Matrix multiplication of an \(m \times q\) times \(q \times n\) takes an order of \(mqn\) operations. The multiplication of left to right is then

-

The first operation takes \(njk\) operation leaving an \(n\times k\) matrix, the next multiplication then takes another \(nk1\) operations or \(njk + nk\) together. Whereas computing from the right to left is first \(jk1\) operations leaving a \(j \times 1\) matrix. The next operation would take another \(nk1\) operations. In totalL

-
    -
  • left to right is \(njk + nk\) = \(nk \cdot (1 + j)\).
  • -
  • right to left is \(jk + j = j\cdot (k+1)\).
  • -
-

When \(j=k\), say, we can compare and see the second is a factor less in terms of operations. This can be quite significant in higher dimensions, whereas the dimensions of calculus (where \(n\) and \(m\) are \(3\) or less) it is not an issue.

-
-
Example
-

Using the BenchmarkTools package, we can check the time to compute various products:

-
-
using BenchmarkTools
-n,j,k,m = 20,15,10,1
-@btime A*(B*C) setup=(A=rand(n,j);B=rand(j,k); C=rand(k,m));
-@btime (A*B)*C setup=(A=rand(n,j);B=rand(j,k); C=rand(k,m));
-
-
  452.474 ns (4 allocations: 432 bytes)
-  837.192 ns (4 allocations: 1.88 KiB)
-
-
-

The latter computation is about 1.5 times slower.

-

Whereas the relationship is changed when the first matrix is skinny and the last is not:

-
-
@btime A*(B*C) setup=(A=rand(m,k);B=rand(k,j); C=rand(j,n));
-@btime (A*B)*C setup=(A=rand(m,k);B=rand(k,j); C=rand(j,n));
-
-
  1.098 μs (4 allocations: 1.88 KiB)
-  779.020 ns (4 allocations: 432 bytes)
-
-
-
-
-
Example
-

In calculus, we have \(n\) and \(m\) are \(1\),\(2\),or \(3\). But that need not be the case, especially if differentiation is over a parameter space.

-

XXXX (Maybe the ariplain wing, but please, something origi

-
-
-
-
-

Derivatives of matrix functions

-

What is the the derivative of \(f(A) = A^2\)?

-

The function \(f\) takes a \(n\times n\) matrix and returns a matrix of the same size. This innocuous question isn’t directly handled, here, by the Jacobian, which is defined for vector-valued functions \(f:R^n \rightarrow R^m\).

-

This derivative can be derived directly from the product rule:

-

\[ -\begin{align*} -f'(A) &= [AA]'\\ -&= A dA + dA A -\end{align*} -\]

-

That is \(f'(A)\) is the operator \(f'(A)[\delta A] = A \delta A + \delta A A\) and not \(2A\delta A\), as \(A\) may not commute with \(\delta A\).

-

XXX THIS ISN”T EVEN RIGHT

-
-

Vectorization of a matrix

-

Alternatively, we can identify \(A\) through its components, as a vector in \(R^{n^2}\) and then leverage the Jacobian.

-

One such identification is vectorization – consecutively stacking the column vectors into a vector. In Julia the vec function does this operation:

-
-
@syms A[1:2, 1:2]
-vec(A)
-
-

\(\left[\begin{smallmatrix}A₁_{₁}\\A₂_{₁}\\A₁_{₂}\\A₂_{₂}\end{smallmatrix}\right]\)

-
-
-

The stacking by column follows how Julia stores matrices and how Julia references a matrices entries by linear index:

-
-
vec(A) == [A[i] for i in eachindex(A)]
-
-
true
-
-
-

With this vectorization operation, \(f\) may be viewed as \(\tilde{f}:R^{n^2} \rightarrow R^{n^2}\) through:

-

\[ -\tilde{f}(\text{vec}(A)) = \text{vec}(f(A)) -\]

-

We use SymPy to compute the Jacobian of this vector valued function.

-
-
@syms A[1:3, 1:3]::real
-f(x) = x^2
-J = vec(f(A)).jacobian(vec(A)) # jacobian of f̃
-
-

\(\left[\begin{smallmatrix}2 A₁_{₁} & A₁_{₂} & A₁_{₃} & A₂_{₁} & 0 & 0 & A₃_{₁} & 0 & 0\\A₂_{₁} & A₁_{₁} + A₂_{₂} & A₂_{₃} & 0 & A₂_{₁} & 0 & 0 & A₃_{₁} & 0\\A₃_{₁} & A₃_{₂} & A₁_{₁} + A₃_{₃} & 0 & 0 & A₂_{₁} & 0 & 0 & A₃_{₁}\\A₁_{₂} & 0 & 0 & A₁_{₁} + A₂_{₂} & A₁_{₂} & A₁_{₃} & A₃_{₂} & 0 & 0\\0 & A₁_{₂} & 0 & A₂_{₁} & 2 A₂_{₂} & A₂_{₃} & 0 & A₃_{₂} & 0\\0 & 0 & A₁_{₂} & A₃_{₁} & A₃_{₂} & A₂_{₂} + A₃_{₃} & 0 & 0 & A₃_{₂}\\A₁_{₃} & 0 & 0 & A₂_{₃} & 0 & 0 & A₁_{₁} + A₃_{₃} & A₁_{₂} & A₁_{₃}\\0 & A₁_{₃} & 0 & 0 & A₂_{₃} & 0 & A₂_{₁} & A₂_{₂} + A₃_{₃} & A₂_{₃}\\0 & 0 & A₁_{₃} & 0 & 0 & A₂_{₃} & A₃_{₁} & A₃_{₂} & 2 A₃_{₃}\end{smallmatrix}\right]\)

-
-
-

We do this via linear algebra first, then see a more elegant manner following the notes.

-

A basic course in linear algebra shows that any linear operator on a finite vector space can be represented as a matrix. The basic idea is to represent what the operator does to each basis element and put these values as columns of the matrix.

-

In this \(3 \times 3\) case, the linear operator works on an object with \(9\) slots and returns an object with \(9\) slots, so the matrix will be \(9 \times 9\).

-

The basis elements are simply the matrices with a \(1\) in spot \((i,j)\) and zero elsewhere. Here we generate them through a function:

-
-
basis(i,j,A) = (b=zeros(Int, size(A)...); b[i,j] = 1; b)
-JJ = [vec(basis(i,j,A)*A + A*basis(i,j,A)) for  j in 1:3 for i in 1:3]
-
-
9-element Vector{Vector{Sym{PyCall.PyObject}}}:
- [2*A₁_₁, A₂_₁, A₃_₁, A₁_₂, 0, 0, A₁_₃, 0, 0]
- [A₁_₂, A₁_₁ + A₂_₂, A₃_₂, 0, A₁_₂, 0, 0, A₁_₃, 0]
- [A₁_₃, A₂_₃, A₁_₁ + A₃_₃, 0, 0, A₁_₂, 0, 0, A₁_₃]
- [A₂_₁, 0, 0, A₁_₁ + A₂_₂, A₂_₁, A₃_₁, A₂_₃, 0, 0]
- [0, A₂_₁, 0, A₁_₂, 2*A₂_₂, A₃_₂, 0, A₂_₃, 0]
- [0, 0, A₂_₁, A₁_₃, A₂_₃, A₂_₂ + A₃_₃, 0, 0, A₂_₃]
- [A₃_₁, 0, 0, A₃_₂, 0, 0, A₁_₁ + A₃_₃, A₂_₁, A₃_₁]
- [0, A₃_₁, 0, 0, A₃_₂, 0, A₁_₂, A₂_₂ + A₃_₃, A₃_₂]
- [0, 0, A₃_₁, 0, 0, A₃_₂, A₁_₃, A₂_₃, 2*A₃_₃]
-
-
-

The elements of JJ show the representation of each of the \(9\) basis elements under the linear transformation.

-

To construct the matrix representing the linear operator, we need to concatenate these horizontally as column vectors

-
-
JJ = hcat(JJ...)
-
-

\(\left[\begin{smallmatrix}2 A₁_{₁} & A₁_{₂} & A₁_{₃} & A₂_{₁} & 0 & 0 & A₃_{₁} & 0 & 0\\A₂_{₁} & A₁_{₁} + A₂_{₂} & A₂_{₃} & 0 & A₂_{₁} & 0 & 0 & A₃_{₁} & 0\\A₃_{₁} & A₃_{₂} & A₁_{₁} + A₃_{₃} & 0 & 0 & A₂_{₁} & 0 & 0 & A₃_{₁}\\A₁_{₂} & 0 & 0 & A₁_{₁} + A₂_{₂} & A₁_{₂} & A₁_{₃} & A₃_{₂} & 0 & 0\\0 & A₁_{₂} & 0 & A₂_{₁} & 2 A₂_{₂} & A₂_{₃} & 0 & A₃_{₂} & 0\\0 & 0 & A₁_{₂} & A₃_{₁} & A₃_{₂} & A₂_{₂} + A₃_{₃} & 0 & 0 & A₃_{₂}\\A₁_{₃} & 0 & 0 & A₂_{₃} & 0 & 0 & A₁_{₁} + A₃_{₃} & A₁_{₂} & A₁_{₃}\\0 & A₁_{₃} & 0 & 0 & A₂_{₃} & 0 & A₂_{₁} & A₂_{₂} + A₃_{₃} & A₂_{₃}\\0 & 0 & A₁_{₃} & 0 & 0 & A₂_{₃} & A₃_{₁} & A₃_{₂} & 2 A₃_{₃}\end{smallmatrix}\right]\)

-
-
-

The matrix \(JJ\) is identical to \(J\), above:

-
-
all(j == jj for (j, jj) in zip(J, JJ))
-
-
true
-
-
-
-
-

Kronecker products

-

But how can we see the Jacobian, \(J\), from the linear operator \(f'(A)[\delta A] = \delta A A + A \delta A\)?

-

To make this less magical, a related operation to vec is defined.

-

The \(\text{vec}\) function takes a matrix and stacks its columns.

-

The \(\text{vec}\) function can turn a matrix into a vector, so it can be used for finding the Jacobian, as above. However the shape of the matrix is lost, as are the fundamental matrix operations, like multiplication.

-

The Kronecker product replicates values making a bigger matrix. That is, if \(A\) and \(B\) are matrices, the Kronecker product replaces each value in \(A\) with that value times \(B\), making a bigger matrix, as each entry in \(A\) is replaced by an entry with size \(B\).

-

Formally,

-

\[ -A \otimes B = -\begin{bmatrix} -a_{11}B & a_{12}B & \cdots & a_{1n}B \\ -a_{21}B & a_{22}B & \cdots & a_{2n}B \\ - &\vdots & & \\ -a_{m1}B & a_{m2}B & \cdots & a_{mn}B -\end{bmatrix} -\]

-

The function kron forms this product:

-
-
@syms A[1:2, 1:3] B[1:3, 1:4]
-kron(A, B) # same as hcat((vcat((A[i,j]*B for i in 1:2)...) for j in 1:3)...)
-
-

\(\left[\begin{smallmatrix}A₁_{₁} B₁_{₁} & A₁_{₁} B₁_{₂} & A₁_{₁} B₁_{₃} & A₁_{₁} B₁_{₄} & A₁_{₂} B₁_{₁} & A₁_{₂} B₁_{₂} & A₁_{₂} B₁_{₃} & A₁_{₂} B₁_{₄} & A₁_{₃} B₁_{₁} & A₁_{₃} B₁_{₂} & A₁_{₃} B₁_{₃} & A₁_{₃} B₁_{₄}\\A₁_{₁} B₂_{₁} & A₁_{₁} B₂_{₂} & A₁_{₁} B₂_{₃} & A₁_{₁} B₂_{₄} & A₁_{₂} B₂_{₁} & A₁_{₂} B₂_{₂} & A₁_{₂} B₂_{₃} & A₁_{₂} B₂_{₄} & A₁_{₃} B₂_{₁} & A₁_{₃} B₂_{₂} & A₁_{₃} B₂_{₃} & A₁_{₃} B₂_{₄}\\A₁_{₁} B₃_{₁} & A₁_{₁} B₃_{₂} & A₁_{₁} B₃_{₃} & A₁_{₁} B₃_{₄} & A₁_{₂} B₃_{₁} & A₁_{₂} B₃_{₂} & A₁_{₂} B₃_{₃} & A₁_{₂} B₃_{₄} & A₁_{₃} B₃_{₁} & A₁_{₃} B₃_{₂} & A₁_{₃} B₃_{₃} & A₁_{₃} B₃_{₄}\\A₂_{₁} B₁_{₁} & A₂_{₁} B₁_{₂} & A₂_{₁} B₁_{₃} & A₂_{₁} B₁_{₄} & A₂_{₂} B₁_{₁} & A₂_{₂} B₁_{₂} & A₂_{₂} B₁_{₃} & A₂_{₂} B₁_{₄} & A₂_{₃} B₁_{₁} & A₂_{₃} B₁_{₂} & A₂_{₃} B₁_{₃} & A₂_{₃} B₁_{₄}\\A₂_{₁} B₂_{₁} & A₂_{₁} B₂_{₂} & A₂_{₁} B₂_{₃} & A₂_{₁} B₂_{₄} & A₂_{₂} B₂_{₁} & A₂_{₂} B₂_{₂} & A₂_{₂} B₂_{₃} & A₂_{₂} B₂_{₄} & A₂_{₃} B₂_{₁} & A₂_{₃} B₂_{₂} & A₂_{₃} B₂_{₃} & A₂_{₃} B₂_{₄}\\A₂_{₁} B₃_{₁} & A₂_{₁} B₃_{₂} & A₂_{₁} B₃_{₃} & A₂_{₁} B₃_{₄} & A₂_{₂} B₃_{₁} & A₂_{₂} B₃_{₂} & A₂_{₂} B₃_{₃} & A₂_{₂} B₃_{₄} & A₂_{₃} B₃_{₁} & A₂_{₃} B₃_{₂} & A₂_{₃} B₃_{₃} & A₂_{₃} B₃_{₄}\end{smallmatrix}\right]\)

-
-
-

The \(m\times n\) matrix \(A\) and \(j \times k\) matrix \(B\) has a Kronecker product with size \(mj \times nk\).

-

The Kronecker product has a certain algebra, including:

-
    -
  • transposes: \((A \otimes B)^T) = A^T \otimes B^T\)
  • -
  • multiplication: \((A\otimes B)(C \otimes D) = (AC) \otimes (BD)\)
  • -
  • inverses: \((A \otimes B)^{-1} = (A^{-1}) \otimes (B^{-1})\)
  • -
  • orthogonal: \((A\otimes B)^T = (A\otimes B)\) if both \(A\) and \(B\) has the same property
  • -
  • determinants: \(\det(A\otimes B) = \det(A)^m \det(B)^n\), where \(A\) is \(n\times n\), \(B\) is \(m \times m\).
  • -
  • trace (sum of diagonal): \(\text{tr}(A \otimes B) = \text{tr}(A)\text{tr}(B)\).
  • -
-

The main equation coupling vec and kron is the fact that if \(A\), \(B\), and \(C\) have appropriate sizes, then:

-

\[ -(A \otimes B) \text{vec}(C) = \text{vec}(B C A^T). -\]

-

Appropriate sizes for \(A\), \(B\), and \(C\) are determined by the various products in \(BCA^T\).

-

If \(A\) is \(m \times n\) and \(B\) is \(r \times s\), then since \(BC\) is defined, \(C\) has \(s\) rows, and since \(CA^T\) is defined, \(C\) must have \(n\) columns, as \(A^T\) is \(n \times m\), so \(C\) must be \(s\times n\). Checking this is correct on the other side, \(A \times B\) would be size \(mr \times ns\) and \(\vec{C}\) would be size \(sn\), so that product works, size wise.

-

The referred to notes have an explanation for this formula, but we confirm with an example with \(m=n-2\), \(r=s=3\):

-
-
@syms A[1:2, 1:2]::real B[1:3, 1:3]::real C[1:3, 1:2]::real
-L, R = kron(A,B)*vec(C),  vec(B*C*A')
-all(l == r for (l, r)  zip(L, R))
-
-
true
-
-
-
-

Now to use this relationship to recognize \(df = A dA + dA A\) with the Jacobian computed from \(\text{vec}{f(a)}\).

-

We have \(\text{vec}(A dA + dA A) = \text{vec}(A dA) + \text{vec}(dA A)\), by obvious linearity of \(\text{vec}\). Now inserting an identity matrix, \(I\), which is symmteric, we have:

-

\[ -\text{vec}(A dA) = \text{vec}(A dA I^T) = (I \otimes A) \text{vec}(dA), -\]

-

and

-

\[ -\text{vec}(dA A) = \text{vec}(I dA (A^T)^T) = (A^T \otimes I) \text{vec}(dA) -\]

-

This leaves

-

\[ -\text{vec}(A dA + dA A) = -\left((I \otimes A) + (A^T \otimes I)\right) \text{vec}(dA) -\]

-

We should then get the Jacobian we computed from the following:

-
-
@syms A[1:3, 1:3]::real
-using LinearAlgebra: I
-J = vec(A^2).jacobian(vec(A))
-JJ = kron(I(3), A) + kron(A', I(3))
-all(j == jj for (j,jj) in zip(J,JJ))
-
-
true
-
-
-

This technique can also be used with other powers, say \(f(A) = A^3\), where the resulting \(df = A^2 dA + A dA A + dA A^2\) is one answer that can be compared to a Jacobian through

-

\[ -\begin{align*} -df &= \text{vec}(A^2 dA I^T) + \text{vec}(A dA A) + \text{vec}(I dA A^2)\\ -&= (I \otimes A^2)\text{vec}(dA) + (A^T \otimes A) \text{vec}(dA) + ((A^T)^2 \otimes I) \text{vec}(dA) -\end{align*} -\]

-

The above shows how to relate the derivative of a matrix function to the Jacobian of a vectorized function, but only for illustration. It is decidely not necessary to express the derivative of \(f\) in terms of the derivative of its vectorized counterpart.

-
-
Example: derivative of the inverse
-

What is the derivative of \(f(A) = A^{-1}\). When \(A\) is a scalar, we related it to the reciprocal of the derivative of \(f\) at some other point. The same technique is available. Starting with \(I = AA^{-1}\) and noting \(dI\) is \(0\) we have

-

\[ -\begin{align*} -0 &= d(AA^{-1})\\ -&= dAA^{-1} + A d(A^{-1}) -\end{align*} -\]

-

So, \(d(A^{-1}) = -A^{-1} dA A^{-1}\).

-

This could be re-expressed as a linear operator through

-

\[ -\text{vec}(dA^{-1}) = -\left((A^{-1})^T \otimes A^{-1}\right) \text{vec}(dA) -= \left((A^T)^{-1} \otimes A^{-1}\right) \text{vec}(dA). -\]

-
-
-
Example: derivative of the determinant
-

Let \(f(A) = \text{det}(A)\). What is the derivative?

-

First, the determinant of a square, \(n\times n\), matrix \(A\) is a scalar summary of \(A\) with different means to compute it, but one recursive one in particular is helpful here:

-

\[ -\text{det}(A) = a_{1j}C_{1j} + a_{2j}C_{2j} + \cdots a_{nj}C_{nj} -\]

-

for any \(j\). The cofactor \(C_{ij}\) is the determinant of the \((n-1)\times(n-1)\) matrix with the \(i\)th row and \(j\)th column deleted times \((-1)^{i+j}\).

-

To find the gradient of \(f\), we differentiate by each of the \(A_{ij}\) variables, and so

-

\[ -\frac{\partial\text{det}(A)}{\partial A_{ij}} = -\frac{\partial (a_{1j}C_{1j} + a_{2j}C_{2j} + \cdots a_{nj}C_{nj})}{\partial A_{ij}} = -C_{ij}, -\]

-

as each cofactor in the expansion has no dependence on \(A_{ij}\) as the cofactor removes the \(i\)th row and \(j\)th column.

-

So the gradient is the matrix of cofactors.

-

@BrightEdelmanJohnson also give a different proof, starting with this observation

-

\[ -\text{det}(I + dA) - \text{det}(I) = \text{tr}(dA) -\]

-

Assuming that, then by the fact \(\text{det}(AB) = \text{det}(A)\text{det}(B)\):

-

\[ -\begin{align*} -\text{det}(A + A(A^{-1}dA)) - \text{det}(A) &= \text{det}(A)\cdot(\text{det}(I+ A^{-1}dA) - \text{det}(I)) \\ -&= \text{det}(A) \text{tr}(A^{-1}dA)\\ -&= \text{tr}(\text{det}(A)A^{-1}dA)\\ -\end{align*} -\]

-

This agrees through a formula to compute the inverse of a matrix through its cofactor matrix divided by its determinant.

-

That the trace gets involved, can be seen from this computation, which shows the only first-order terms are from the diagonal sum:

-
-
using LinearAlgebra
-@syms dA[1:2, 1:2]
-det(I + dA) - det(I)
-
-

\(dA₁_{₁} dA₂_{₂} + dA₁_{₁} - dA₁_{₂} dA₂_{₁} + dA₂_{₂}\)

-
-
-
-
-
-
-

The adjoint method

-

The chain rule brings about a series of products. The adjoint method illustrated below, shows how to approach the computation of the series in a direction that minimizes the computational cost, illustrating why reverse mode is preferred to forward mode when a scalar function of several variables is considered.

-

@BrightEdelmanJohnson consider the derivative of

-

\[ -g(p) = f(A(p)^{-1} b) -\]

-

This might arise from applying a scalar-valued \(f\) to the solution of \(Ax = b\), where \(A\) is parameterized by \(p\).

-

The chain rule gives the following computation to find the derivative (or gradient):

-

\[ -\begin{align*} -dg -&= f'(x)[dx]\\ -&= f'(x) [d(A(p)^{1} b)]\\ -&= f'(x)[-A(p)^{-1} dA A(p)^{-1} b + 0]\\ -&= -f'(x) A(p)^{-1} dA A(p)^{-1} b. -\end{align*} -\]

-

By writing \(dA = A'(p)[dp]\) and setting \(v^T = f'(x)A(p)^{-1}\) this becomes

-

\[ -dg = -v^T dA A(p)^{-1} b = -v^T dA x -\]

-

This product of three terms can be computed in two directions:

-

From left to right:

-

First \(v\) is found by solving \(v^T = f'(x) A^{-1}\) through the solving of \(v = (A^{-1})^T (f'(x))^T = (A^T)^{-1} \nabla(f)\) or by solving \(A^T v = \nabla f\). This is called the adjoint equation.

-

The partial derivatives in \(g\) is related to each partial derivative of \(dA\) through:

-

\[ -\frac{\partial g}{\partial p_k} = -v^T\frac{\partial A}{\partial p_k} x, -\]

-

as the scalar factor commutes through. With \(v\) and \(x\) solved for (via the adjoint equation and from solving \(Ax=b\)) the partials in \(p_k\) are computed with dot products. There are just two costly operations.

-

From right to left:

-

The value of \(x\) can be solved for, as above, but computing the value of

-

\[ -\frac{\partial g}{\partial p_k} = --f'(x) \left(A^{-1} \frac{\partial A}{\partial p_k} x \right) -\]

-

requires a costly solve for each \(p_k\), and \(p\) may have many components. As mentioned above, the reverse mode offers advantages when there are many input parameters (\(p\)) and a single output parameter.

-
-
Example
-

Suppose \(x(p)\) solves some system of equations \(h(x(p),p) = 0\) in \(R^n\) (\(n\) possibly just \(1\)) and \(g(p) = f(x(p))\) is some non-linear transformation of \(x\). What is the derivative of \(g\) in \(p\)?

-

Suppose the implicit function theorem applies to \(h(x,p) = 0\), that is – locally – there is an implicitly defined function \(x(p)\) with a derivative. Moreover by differentiating both sides it can be identified:

-

\[ -0 = \frac{\partial h}{\partial p} dp + \frac{\partial h}{\partial x} dx -\]

-

which can be solved for \(dx\) to give

-

\[ -dx = -\left(\frac{\partial h}{\partial x}\right)^{-1} \frac{\partial h}{\partial p} dp. -\]

-

The chain rule then gives

-

\[ -dg = f'(x) dx = -f'(x) \left(\frac{\partial h}{\partial x}\right)^{-1} \frac{\partial h}{\partial p} dp. -\]

-

This can be computed in two directions:

-

From left to right:

-

Call \(A =\left(\frac{\partial h}{\partial x}\right)^{-1}\). Then define \(v\) indirectly through \(v^T = f'(x) A^{-1}\). With this: \(v = (A^{-1})^T (f'(x))^T = (A^T)^{-1} \nabla{f}\) which is found by solving \(A^Tv = \nabla{f}\). Again, this is the adjoint equation.

-

The value of \(dA\) is related to each partial derivative for which

-

\[ -\frac{\partial g}{\partial p_k} = -v^T\frac{\partial A}{\partial p_k} x, -\]

-

as the scalar factor commutes through. With \(v\) and \(x\) solved for (via the adjoint equation and from solving \(Ax=b\)) the partials in \(p_k\) are computed with dot products.

-

However, from right to left, the value of \(x\) can be solved for, but computing the value of

-

\[ -\frac{\partial g}{\partial p_k} = --f'(x) -\left(A^{-1} \frac{\partial A}{\partial p_k} x \right) -\]

-

requires a costly solve for each \(p_k\), and \(p\) may have many components. The reverse mode offers advantages when there are many input parameters (\(p\)) and a single output parameter.

-
-
-
Example
-

Suppose \(x(p)\) solves some system of equations \(h(x(p),p) = 0\) in \(R^n\) (\(n\) possibly just 1$) and \(g(p) = f(x(p))\) is some non-linear transformation of \(x\). What is the derivative of \(g\) in \(p\)?

-

Suppose the implicit function theorem applies to \(h(x,p) = 0\), that is locally the response \(x(p)\) has a derivative, and moreover by the chain rule

-

\[ -0 = \frac{\partial h}{\partial p} dp + \frac{\partial h}{\partial x} dx. -\]

-

Solving the above for \(dx\) gives:

-

\[ -dx = -\left(\frac{\partial h}{\partial x}\right)^{-1} \frac{\partial h}{\partial p} dp. -\]

-

The chain rule applied to \(g(p) = f(x(p))\) then yields

-

\[ -dg = f'(x) dx = - f'(x) \left(\frac{\partial h}{\partial x}\right)^{-1} \frac{\partial h}{\partial p} dp. -\]

-

Setting

-

\[ -v^T = -f'(x) \left(\frac{\partial h}{\partial x}\right)^{-1} -\]

-

then \(v\) can be solved from taking adjoints (as before). Let \(A = \partial h/\partial x\), the \(v^T = -f'(x) A^{-1}\) or \(v = -(A^{-1})^T (f'(x))^t= -(A^T)^{-1} \nabla f\). As before it would take two solves to get both \(g\) and its gradient.

-
-
-
-

Second derivatives, Hessian

-

@CarlssonNikitinTroedssonWendt

-

We reference a theorem presented by Carlsson, Nikitin, Troedsson, and Wendt for exposition with some modification

-
-
-
- -
-
-

Theorem 1. Let \(f:X \rightarrow Y\), where \(X,Y\) are finite dimensional inner product spaces with elements in \(R\). Suppose \(f\) is smooth (a certain number of derivatives). Then for each \(x\) in \(X\) there exists a unique linear operator, \(f'(x)\), and a unique bilinear symmetric operator \(f'': X \oplus X \rightarrow Y\) such that

-

\[ -f(x + \delta x) = f(x) + f'(x)[\delta x] + \frac{1}{2}f''(x)[\delta x, \delta x] + \mathscr(||\delta x ||^2). -\]

-
-
-
-

New terms include bilinear, symmetric, and inner product. An operator (\(X\oplus X \rightarrow Y\)) is bilinear if it is a linear operator in each of its two arguments. Such an operator is symmetric if interchanging its two arguments makes no difference in its output. Finally, an inner product space is one with a generalization of the dot product. An inner product takes two vectors \(x\) and \(y\) and returns a scalar; it is denoted \(\langle x,y\rangle\); and has properties of symmetry, linearity, and non-negativity (\(\langle x,x\rangle \geq 0\), and equal \(0\) only if \(x\) is the zero vector.) Inner products can be used to form a norm (or length) for a vector through \(||x||^2 = \langle x,x\rangle\).

-

We reference this, as the values denoted \(f'\) and \(f''\) are unique. So if we identify them one way, we have identified them.

-

Specializing to \(X=R^n\) and \(Y=R^1\), we have, \(f'=\nabla f^T\) and \(f''\) is the Hessian.

-

Take \(n=2\). Previously we wrote a formula for Taylor’s theorem for \(f:R^n \rightarrow R\) that with \(n=2\) has with \(x=\langle x_1,x_2\rangle\):

-

\[ -\begin{align*} -f(x + dx) &= f(x) + -\frac{\partial f}{\partial x_1} dx_1 + \frac{\partial f}{\partial x_2} dx_2\\ -&+ \frac{1}{2}\left( -\frac{\partial^2 f}{\partial x_1^2}dx_1^2 + -\frac{\partial^2 f}{\partial x_1 \partial x_2}dx_1dx_2 + -\frac{\partial^2 f}{\partial x_2^2}dx_2^2 -\right) + \mathscr{o}(dx). -\end{align*} -\]

-

We can see \(\nabla{f} \cdot dx = f'(x) dx\) to tidy up part of the first line, and more over the second line can be seen to be a matrix product:

-

\[ -[dx_1 dx_2] -\begin{bmatrix} -\frac{\partial^2 f}{\partial x_1^2} & -\frac{\partial^2 f}{\partial x_1 \partial x_2}\\ -\frac{\partial^2 f}{\partial x_2 \partial x_1} & -\frac{\partial^2 f}{\partial x_2^2} -\end{bmatrix} -\begin{bmatrix} -dx_1\\ -dx_2 -\end{bmatrix} -= dx^T H dx, -\]

-

\(H\) being the Hessian with entries \(H_{ij} = \frac{\partial f}{\partial x_i \partial x_j}\).

-

This formula – \(f(x+dx)-f(x) \approx f'(x)dx + dx^T H dx\) – is valid for any \(n\), showing \(n=2\) was just for ease of notation when expressing in the coordinates and not as matrices.

-

By uniqueness, we have under these assumptions that the Hessian is symmetric and the expression \(dx^T H dx\) is a bilinear form, which we can identify as \(f''(x)[dx,dx]\).

-

That the Hessian is symmetric could also be derived under these assumptions by directly computing that the mixed partials can have their order exchanged. But in this framework, as explained by @BrightEdelmanJohnson it is a result of the underlying vector space having an addition that is commutative (e.g. \(u+v = v+u\)).

-

The mapping \((u,v) \rightarrow u^T A v\) for a matrix \(A\) is bilinear. For a fixed \(u\), it is linear as it can be viewed as \((u^TA)[v]\) and matrix multiplication is linear. Similarly for a fixed \(v\).

-

@BrightEdelmanJohnson extend this characterization to a broader setting. The second derivative can be viewed as expressing first-order change in \(f'(x)\), a linear operator. The value \(df'\) has the same shape as \(f'\), which is a linear operator, so \(df'\) acts on vectors, say \(dx\), then:

-

\[ -df'[dx] = f''(x)[dx'][dx] = f''(x)[dx', dx] -\]

-

The prime in \(dx'\) is just notation, not a derivative operation for \(dx\).

-

With this view, we can see that \(f''(x)\) has two vectors it acts on. By definition it is linear in \(dx\). However, as \(f'(x)\) is a linear operator and the sum and product rules apply to derivatives, this operator is linear in \(dx'\) as well. So \(f''(x)\) is bilinear and as mentioned earlier symmetric.

-
-

Polarization

-

@BrightEdelmanJohnson interpret \(f''\) by looking at the image under \(f\) of \(x + dx + dx'\). If \(x\) is a vector, then this has a geometrical picture, from vector addtion, relating \(x + dx\), \(x+dx'\), and \(x + dx + dx'\).

-

The image for \(x +dx\) is to second order \(f(x) + f'(x)[dx] + (1/2)f''(x)[dx, dx]\), similarly \(x + dx'\) is to second order \(f(x) + f'(x)[dx'] + (1/2)f''(x)[dx', dx']\). The key formula for \(f''(x)\) is

-

\[ -\begin{align*} -f(x + dx + dx') &= f(x) + f'(x)[dx + dx'] + \frac{1}{2}f''(x)[dx, dx']\\ -&= f(x) + f'(x)[dx] + (1/2)f''(x)[dx, dx] -&+ f(x) + f'(x)[dx] + (1/2)f''(x)[dx', dx'] -&+ f''(x)[dx, dx'] -\end{align} -\]

-

This gives a means to compute \(f''\) in terms of \(f''\) acting on diagonal terms, where the two vectors are equal:

-

\[ -f''(x)[dx, dx'] = \frac{1}{2} f''(x)[dx+dx',dx+dx'] - f''(x)[dx,dx] - f''(x)[dx',dx'] -\]

-
-
-

XXX does this fit in?

-

However, as a description of second-order change in \(f\), we recover the initial terms in the Taylor series

-

\[ -f(x + \delta x) = f(x) + f'(x)\delta x + (1/2) f''(x)[\delta x, \delta x] + \mathscr{o}(||\delta x||^2). -\]

-
-
-

Examples

-
-
Example: second derivative of \(x^TAx\)
-

Consider an expression from earlier \(f(x) = x^T A x\) for some constant \(A\). Then \(f''\) is found by noting that \(f' = (\nabla f)^T = x^T(A + A^T)\), or \(\nabla f = (A^T + A)x\) and \(f'' = H = A^T + A\) is the Jacobian of the gradient.

-

By rearranging terms, it can be shown that \(f(x) = 1/2 x^THx = 1/2 f''[x,x]\).

-
-
-
Example: second derivative of \(\text{det}(A)\)
-

Consider \(f(A) = \text{det}(A)\). We saw previously that:

-

\[ -\begin{align*} -\text{tr}(A + B) &= \text{tr}(A) + \text{tr}(B)\\ -\text{det}(A + dA') &= \text{det}(A) + \text{det}(A)\text{tr}(A^{-1}dA')\\ -(A + dA') &= A^{-1} - A^{-1} dA' A^{-1} -\end{align*} -\]

-

These are all used to simplify:

-

\[ -\begin{align*} -\text{det}(A+dA')&\text{tr}((A + dA')^{-1} dA) - \text{det}(A) \text{tr}(A^{-1}dA) \\ -&= \left( -\text{det}(A) + \text{det}(A)\text{tr}(A^{-1}dA') -\right) -\text{tr}((A^{-1} - A^{-1}dA' A^{-1})dA) - \text{det}(A) \text{tr}(A^{-1}dA) \\ -&= -\text{det}(A) \text{tr}(A^{-1}dA)\\ -&+ \text{det}(A)\text{tr}(A^{-1}dA')\text{tr}(A^{-1}dA) \\ -&- \text{det}(A)\text{tr}(A^{-1}dA' A^{-1}dA)\\ -&- \text{det}(A)\text{tr}(A^{-1}dA')\text{tr}(A^{-1}dA' A^{-1}dA)\\ -&- \text{det}(A) \text{tr}(A^{-1}dA) \\ -&= \text{det}(A)\text{tr}(A^{-1}dA')\text{tr}(A^{-1}dA) - \text{det}(A)\text{tr}(A^{-1}dA' A^{-1}dA)\\ -&+ \text{third order term} -\end{align*} -\]

-

So, after dropping the third-order term, we see: \[ -\begin{align*} -f''(A)&[dA,dA'] \\ -&= \text{det}(A)\text{tr}(A^{-1}dA')\text{tr}(A^{-1}dA) -- \text{det}(A)\text{tr}(A^{-1}dA' A^{-1}dA). -\end{align*} -\]

-
-
-
- -
- - -
- - - - - \ No newline at end of file From ad52202c922b44b6f9b7180c9bbb9fa2ad403355 Mon Sep 17 00:00:00 2001 From: jverzani Date: Fri, 9 May 2025 07:33:21 -0400 Subject: [PATCH 04/22] edits --- .../matrix_calculus_notes.qmd | 32 +++++++++---------- 1 file changed, 16 insertions(+), 16 deletions(-) diff --git a/quarto/differentiable_vector_calculus/matrix_calculus_notes.qmd b/quarto/differentiable_vector_calculus/matrix_calculus_notes.qmd index 8c5dab8..68449e6 100644 --- a/quarto/differentiable_vector_calculus/matrix_calculus_notes.qmd +++ b/quarto/differentiable_vector_calculus/matrix_calculus_notes.qmd @@ -176,7 +176,7 @@ $$ \begin{align*} df &= f(x + dx) - f(x)\\ &= (x + dx)^T A (x + dx) - x^TAx \\ -&= \textcolor{blue}{x^TAx} + dx^TA x + \textcolor{blue}{x^TAx} + \textcolor{red}{dx^T A dx} - \textcolor{blue}{x^TAx}\\ +&= \textcolor{blue}{x^TAx} + dx^TA x + x^TAdx + \textcolor{red}{dx^T A dx} - \textcolor{blue}{x^TAx}\\ &= dx^TA x + x^TAdx \\ &= (dx^TAx)^T + x^TAdx \\ &= x^T A^T dx + x^T A dx\\ @@ -246,7 +246,7 @@ df &= f(x + dx) - f(x) \\ \end{align*} $$ -Comparing we get $f'(x)dx = (g'(x) + h'(x))[dx]$ or $f'(x) = g'(x) + h'(x)$. +Comparing we get $f'(x)[dx] = (g'(x) + h'(x))[dx]$ or $f'(x) = g'(x) + h'(x)$. (The last two lines above show how the new linear operator $g'(x) + h'(x)$ is defined on a value, but adding the application for each. The sum rule has the same derivation as was done with univariate, scalar functions. Similarly for the product rule. @@ -256,10 +256,10 @@ $$ \begin{align*} df &= f(x + dx) - f(x) \\ &= g(x+dx)h(x + dx) - g(x) h(x)\\ -&= \left(g(x) + g'(x)dx\right)\left(h(x) + h'(x) dx\right) - \left(g(x) h(x)\right) \\ -&= \textcolor{blue}{g(x)h(x)} + g'(x) dx h(x) + g(x) h'(x) dx + \textcolor{red}{g'(x)dx h'(x) dx} - \textcolor{blue}{g(x) h(x)}\\ -&= g'(x)dxh(x) + g(x)h'(x) dx\\ -&= (g'(x)h(x) + g(x)h'(x)) dx +&= \left(g(x) + g'(x)[dx]\right)\left(h(x) + h'(x) [dx]\right) - g(x) h(x) \\ +&= \textcolor{blue}{g(x)h(x)} + g'(x) [dx] h(x) + g(x) h'(x) [dx] + \textcolor{red}{g'(x)[dx] h'(x) [dx]} - \textcolor{blue}{g(x) h(x)}\\ +&= \left(g'(x)[dx]\right)h(x) + g(x)\left(h'(x) [dx]\right)\\ +&= dg h + g dh \end{align*} $$ @@ -369,16 +369,18 @@ Multiplying left to right (the first) is called reverse mode; multiplying right * if $f:R^n \rightarrow R^m$ has $n=1$ and $m$ much bigger than one, the it is faster to do right to left multiplication (many outputs than inputs) -The basic idea comes down to the shape of the matrices. When $m=1$, the derviative is a product of matrices of size $n\times j$, $j\times k$, and $k \times 1$ yielding a matrix of size $n \times 1$ matching the function dimension. Matrix multiplication of an $m \times q$ matrix times a $q \times n$ matrix takes an order of $mqn$ operations. +The reason comes down to the shape of the matrices. To see, we need to know that matrix multiplication of an $m \times q$ matrix times a $q \times n$ matrix takes an order of $mqn$ operations. -The operations involved in multiplication of left to right can be quantified. The first operation takes $njk$ operation leaving an $n\times k$ matrix, the next multiplication then takes another $nk1$ operations or $njk + nk$ together. +When $m=1$, the derviative is a product of matrices of size $n\times j$, $j\times k$, and $k \times 1$ yielding a matrix of size $n \times 1$ matching the function dimension. + +The operations involved in multiplication from left to right can be quantified. The first operation takes $njk$ operation leaving an $n\times k$ matrix, the next multiplication then takes another $nk1$ operations or $njk + nk$ together. Whereas computing from the right to left is first $jk1$ operations leaving a $j \times 1$ matrix. The next operation would take another $nk1$ operations. In total: -* left to right is $njk + nk$ = $nk \cdot (1 + j)$. +* left to right is $njk + nk$ = $nk \cdot (j + 1)$. * right to left is $jk + jn = j\cdot (k+n)$. -When $j=k$, say, we can compare and see the second is a factor less in terms of operations. This can be quite significant in higher dimensions, whereas the dimensions of calculus (where $n$ and $m$ are $3$ or less) it is not an issue. +When $j=k$, say, we can compare and see the second is a factor less in terms of operations. This can be quite significant in higher dimensions, whereas the dimensions of calculus (where $n$ and $m$ are $3$ or less) it is not an issue. ##### Example @@ -401,17 +403,15 @@ Whereas the relationship is changed when the first matrix is skinny and the last @btime (A*B)*C setup=(A=rand(m,k); B=rand(k,j); C=rand(j,n)); ``` -##### Example +---- In calculus, we have $n$ and $m$ are $1$,$2$,or $3$. But that need not be the case, especially if differentiation is over a parameter space. -XXX insert example XXX - ## Derivatives of matrix functions What is the the derivative of $f(A) = A^2$? -The function $f$ takes a $n\times n$ matrix and returns a matrix of the same size. This innocuous question isn't directly handled, here, by the Jacobian, which is defined for vector-valued functions $f:R^n \rightarrow R^m$. +The function $f$ takes a $n\times n$ matrix and returns a matrix of the same size. This derivative can be derived directly from the *product rule*: @@ -422,7 +422,7 @@ df &= d(A^2) = d(AA)\\ \end{align*} $$ -That is $f'(A)$ is the operator $f'(A)[\delta A] = A \delta A + \delta A A$ and not $2A\delta A$, as $A$ may not commute with $\delta A$. +That is $f'(A)$ is the operator $f'(A)[\delta A] = A \delta A + \delta A A$. (This is not $2A\delta A$, as $A$ may not commute with $\delta A$.) ### Vectorization of a matrix @@ -463,7 +463,7 @@ J = vec(f(A)).jacobian(vec(A)) # jacobian of f̃ We do this via linear algebra first, then see a more elegant manner following the notes. -A basic course in linear algebra shows that any linear operator on a finite vector space can be represented as a matrix. The basic idea is to represent what the operator does to each *basis* element and put these values as columns of the matrix. +A course in linear algebra shows that any linear operator on a finite vector space can be represented as a matrix. The basic idea is to represent what the operator does to each *basis* element and put these values as columns of the matrix. In this $3 \times 3$ case, the linear operator works on an object with $9$ slots and returns an object with $9$ slots, so the matrix will be $9 \times 9$. From 4f60e9a41461d43775fb1dbc46bffc1cd0dbb0de Mon Sep 17 00:00:00 2001 From: jverzani Date: Sat, 14 Jun 2025 07:23:19 -0400 Subject: [PATCH 05/22] typos --- quarto/precalc/julia_overview.qmd | 4 ++-- quarto/precalc/numbers_types.qmd | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/quarto/precalc/julia_overview.qmd b/quarto/precalc/julia_overview.qmd index bac1c2d..86adc26 100644 --- a/quarto/precalc/julia_overview.qmd +++ b/quarto/precalc/julia_overview.qmd @@ -97,7 +97,7 @@ julia> 2 + 2 * An IDE. For programmers, an integrated development environment is often used to manage bigger projects. `Julia` has `Juno` and `VSCode`. - * A notebook. The [Project Juptyer](https://jupyter.org/) provides a notebook interface for interacting with `Julia` and a more `IDE` style `jupyterlab` interface. A jupyter notebook has cells where commands are typed and immediately following is the printed output returned by `Julia`. The output of a cell depends on the state of the kernel when the cell is computed, not the order of the cells in the notebook. Cells have a number attached, showing the execution order. The `Juypter` notebook is used by `binder` and can be used locally through the `IJulia` package. This notebook has the ability to display many different types of outputs in addition to plain text, such as images, marked up math text, etc. + * A notebook. The [Project Jupyter](https://jupyter.org/) provides a notebook interface for interacting with `Julia` and a more `IDE` style `jupyterlab` interface. A jupyter notebook has cells where commands are typed and immediately following is the printed output returned by `Julia`. The output of a cell depends on the state of the kernel when the cell is computed, not the order of the cells in the notebook. Cells have a number attached, showing the execution order. The `Juypter` notebook is used by `binder` and can be used locally through the `IJulia` package. This notebook has the ability to display many different types of outputs in addition to plain text, such as images, marked up math text, etc. * The [Pluto](https://github.com/fonsp/Pluto.jl) package provides a *reactive* notebook interface. Reactive means when one "cell" is modified and executed, the new values cascade to all other dependent cells which in turn are updated. This is very useful for exploring a parameter space, say. Pluto notebooks can be exported as HTML files which make them easy to read online and – by clever design – embed the `.jl` file that can run through `Pluto` if it is downloaded. @@ -280,7 +280,7 @@ Values will be promoted to a common type (or type `Any` if none exists). For exa (Vectors are used as a return type from some functions, as such, some familiarity is needed.) -Other common container types are variables of vectors (higher-dimensional arrarys, offset arrays, etc.) tuples (for heterogeneous, immutable, indexed values); named tuples (which add a name to each value in a tuple); and dictionaries (for associative relationships between a key and a value). +Other common container types are variables of vectors (higher-dimensional arrays, offset arrays, etc.) tuples (for heterogeneous, immutable, indexed values); named tuples (which add a name to each value in a tuple); and dictionaries (for associative relationships between a key and a value). Regular arithmetic sequences can be defined by either: diff --git a/quarto/precalc/numbers_types.qmd b/quarto/precalc/numbers_types.qmd index 929797a..454abfc 100644 --- a/quarto/precalc/numbers_types.qmd +++ b/quarto/precalc/numbers_types.qmd @@ -26,7 +26,7 @@ On top of these, we have special subsets, such as the natural numbers $\{1, 2, \ Mathematically, these number systems are naturally nested within each other as integers are rational numbers which are real numbers, which can be viewed as part of the complex numbers. -Calculators typically have just one type of number - floating point values. These model the real numbers. `Julia`, on the other hand, has a rich type system, and within that has serveral different number types. There are types that model each of the four main systems above, and within each type, specializations for how these values are stored. +Calculators typically have just one type of number - floating point values. These model the real numbers. `Julia`, on the other hand, has a rich type system, and within that has several different number types. There are types that model each of the four main systems above, and within each type, specializations for how these values are stored. Most of the details will not be of interest to all, and will be described later. From 580e87ccb2016b5c5499e9e4a9168c9ea98e0dbe Mon Sep 17 00:00:00 2001 From: jverzani Date: Fri, 27 Jun 2025 14:29:32 -0400 Subject: [PATCH 06/22] orthogonal; work around plotly() --- quarto/_quarto.yml | 9 +- quarto/adjust_plotly.jl | 38 +- quarto/alternatives.qmd | 4 +- quarto/alternatives/Project.toml | 7 +- quarto/alternatives/SciML.qmd | 2 +- quarto/alternatives/makie_plotting.qmd | 80 +- quarto/alternatives/plotly_plotting.qmd | 58 +- quarto/alternatives/symbolics.qmd | 9 +- .../matrix_calculus_notes.qmd | 16 +- quarto/integral_vector_calculus/Project.toml | 5 + quarto/integrals/make_pdf.jl | 3 +- quarto/integrals/orthogonal_polynomials.qmd | 724 ++++++++++++++++++ quarto/misc/Project.toml | 12 + 13 files changed, 853 insertions(+), 114 deletions(-) create mode 100644 quarto/integrals/orthogonal_polynomials.qmd create mode 100644 quarto/misc/Project.toml diff --git a/quarto/_quarto.yml b/quarto/_quarto.yml index f95426f..178bc7e 100644 --- a/quarto/_quarto.yml +++ b/quarto/_quarto.yml @@ -83,6 +83,7 @@ book: - integrals/volumes_slice.qmd - integrals/arc_length.qmd - integrals/surface_area.qmd + - integrals/orthogonal_polynomials.qmd - integrals/twelve-qs.qmd - part: ODEs.qmd @@ -115,7 +116,7 @@ book: chapters: - alternatives/symbolics.qmd - alternatives/SciML.qmd - # - alternatives/interval_arithmetic.qmd + #- alternatives/interval_arithmetic.qmd - alternatives/plotly_plotting.qmd - alternatives/makie_plotting.qmd @@ -157,5 +158,7 @@ format: execute: error: false -# freeze: false - freeze: auto + freeze: false +# freeze: auto +# cache: false +# enabled: true \ No newline at end of file diff --git a/quarto/adjust_plotly.jl b/quarto/adjust_plotly.jl index 5060e98..34a0d01 100644 --- a/quarto/adjust_plotly.jl +++ b/quarto/adjust_plotly.jl @@ -5,19 +5,39 @@ # This little script just adds a line *before* the require call # which seems to make it all work. The line number 83 might change. -f = "_book/alternatives/plotly_plotting.html" -lineno = 88 +#alternatives/plotly_plotting.html +function _add_plotly(f) + lineno = 117 str = """ - - """ -r = readlines(f) -open(f, "w") do io - for (i,l) ∈ enumerate(r) - i == lineno && println(io, str) - println(io, l) + r = readlines(f) + open(f, "w") do io + for (i,l) ∈ enumerate(r) + i == lineno && println(io, str) + println(io, l) + end end end + + + +function (@main)(args...) + for (root, dirs, files) in walkdir("_book") + for fᵢ ∈ files + f = joinpath(root, fᵢ) + if endswith(f, ".html") + _add_plotly(f) + end + end + end + + #f = "_book/integrals/center_of_mass.html" + #_add_plotly(f) + + return 1 +end + + ["ODEs", "alternatives", "derivatives", "differentiable_vector_calculus", "integral_vector_calculus", "integrals", "limits", "misc", "precalc", "site_libs"] diff --git a/quarto/alternatives.qmd b/quarto/alternatives.qmd index 23cb5d6..b917e5a 100644 --- a/quarto/alternatives.qmd +++ b/quarto/alternatives.qmd @@ -6,8 +6,8 @@ These notes use a particular selection of packages. This selection could have be * The finding of zeros of scalar-valued, univariate functions is done with `Roots`. The [NonlinearSolve](./alternatives/SciML.html#nonlinearsolve) package provides an alternative for univariate and multi-variate functions. -* The finding of minima and maxima was done mirroring the framework of a typical calculus class; the [Optimization](./alternatives/SciML.html#optimization-optimization.jl) provides an alternative. +* The finding of minima and maxima was done mirroring the framework of a typical calculus class; the [Optimization](./alternatives/SciML.html#optimization-optimization.jl) package provides an alternative. -* The computation of numeric approximations for definite integrals is computed with the `QuadGK` and `HCubature` packages. The [Integrals](./alternatives/SciML.html#integration-integrals.jl) package provides a unified interface for numeric to these two packages, among others. +* The computation of numeric approximations for definite integrals is computed with the `QuadGK` and `HCubature` packages. The [Integrals](./alternatives/SciML.html#integration-integrals.jl) package provides a unified interface for numeric integration, including these two packages, among others. * Plotting was done using the popular `Plots` package. The [Makie](./alternatives/makie_plotting.html) package provides a very powerful alternative. Whereas the [PlotlyLight](./alternatives/plotly_plotting.html) package provides a light-weight alternative using an open-source JavaScript library. diff --git a/quarto/alternatives/Project.toml b/quarto/alternatives/Project.toml index 4188acb..4c47c77 100644 --- a/quarto/alternatives/Project.toml +++ b/quarto/alternatives/Project.toml @@ -5,11 +5,13 @@ ForwardDiff = "f6369f11-7733-5829-9624-2563aa707210" GLMakie = "e9467ef8-e4e7-5192-8a1a-b1aee30e663a" GeometryBasics = "5c1252a2-5f33-56bf-86c9-59e7332b4326" IJulia = "7073ff75-c697-5162-941a-fcdaad2a7d2a" -Implicit3DPlotting = "d997a800-832a-4a4c-b340-7dddf3c1ad50" Integrals = "de52edbc-65ea-441a-8357-d3a637375a31" +LaTeXStrings = "b964fa9f-0449-5b57-a5c2-d3ea65f4040f" LinearAlgebra = "37e2e46d-f89d-539d-b4ee-838fcccc9c8e" +Meshes = "eacbb407-ea5a-433e-ab97-5258b1ca43fa" Meshing = "e6723b4c-ebff-59f1-b4b7-d97aa5274f73" ModelingToolkit = "961ee093-0014-501f-94e3-6117800e7a78" +Mustache = "ffc61752-8dc7-55ee-8c37-f3e9cdd09e70" NonlinearSolve = "8913a72c-1f9b-4ce2-8d82-65094dcecaec" Optimization = "7f7a1694-90dd-40f0-9382-eb1efda571ba" OptimizationOptimJL = "36348300-93cb-4f02-beb5-3c3902f8871e" @@ -19,6 +21,7 @@ PlotlyKaleido = "f2990250-8cf9-495f-b13a-cce12b45703c" PlotlyLight = "ca7969ec-10b3-423e-8d99-40f33abb42bf" Plots = "91a5bcdd-55d7-5caf-9e0b-520d859cae80" QuadGK = "1fd47b50-473d-5c70-9696-f719f8f3bcdc" +QuizQuestions = "612c44de-1021-4a21-84fb-7261cf5eb2d4" Roots = "f2b01f46-fcfa-551c-844a-d8ac1e96c665" SplitApplyCombine = "03a91e81-4c3e-53e1-a0a4-9c0c8f19dd66" StaticArrays = "90137ffa-7385-5640-81b9-e52037218182" @@ -26,3 +29,5 @@ SymPy = "24249f21-da20-56a4-8eb1-6a02cf4ae2e6" SymbolicLimits = "19f23fe9-fdab-4a78-91af-e7b7767979c3" SymbolicNumericIntegration = "78aadeae-fbc0-11eb-17b6-c7ec0477ba9e" Symbolics = "0c5d862f-8b57-4792-8d23-62f2024744c7" +Tables = "bd369af6-aec1-5ad0-b16a-f7cc5008161c" +TextWrap = "b718987f-49a8-5099-9789-dcd902bef87d" diff --git a/quarto/alternatives/SciML.qmd b/quarto/alternatives/SciML.qmd index 8092595..a6e23c1 100644 --- a/quarto/alternatives/SciML.qmd +++ b/quarto/alternatives/SciML.qmd @@ -250,7 +250,7 @@ With the system defined, we can pass this to `NonlinearProblem`, as was done wit ```{julia} -prob = NonlinearProblem(ns, [1.0], [α => 1.0]) +prob = NonlinearProblem(mtkcompile(ns), [1.0], Dict(α => 1.0)) ``` The problem is solved as before: diff --git a/quarto/alternatives/makie_plotting.qmd b/quarto/alternatives/makie_plotting.qmd index 1cca87b..0aea3b0 100644 --- a/quarto/alternatives/makie_plotting.qmd +++ b/quarto/alternatives/makie_plotting.qmd @@ -1,6 +1,5 @@ # Calculus plots with Makie - {{< include ../_common_code.qmd >}} The [Makie.jl webpage](https://github.com/JuliaPlots/Makie.jl) says @@ -36,8 +35,7 @@ using GLMakie import LinearAlgebra: norm ``` -The `Makie` developers have workarounds for the delayed time to first plot, but without utilizing these the time to load the package is lengthy. - +The package load time as of recent version of `Makie` is quite reasonable for a complicated project. (The time to first plot is under 3 seconds on a typical machine.) ## Points (`scatter`) @@ -158,7 +156,8 @@ A point is drawn with a "marker" with a certain size and color. These attributes ```{julia} scatter(xs, ys; - marker=[:x,:cross, :circle], markersize=25, + marker=[:x,:cross, :circle], + markersize=25, color=:blue) ``` @@ -176,7 +175,7 @@ A single value will be repeated. A vector of values of a matching size will spec ## Curves -The curves of calculus are lines. The `lines` command of `Makie` will render a curve by connecting a series of points with straight-line segments. By taking a sufficient number of points the connect-the-dot figure can appear curved. +A visualization of a curve in calculus is comprised of line segments. The `lines` command of `Makie` will render a curve by connecting a series of points with straight-line segments. By taking a sufficient number of points the connect-the-dot figure can appear curved. ### Plots of univariate functions @@ -304,6 +303,7 @@ current_figure() ### Text (`annotations`) +XXX FIX ME XXX Text can be placed at a point, as a marker is. To place text, the desired text and a position need to be specified along with any adjustments to the default attributes. @@ -315,25 +315,43 @@ For example: xs = 1:5 pts = Point2.(xs, xs) scatter(pts) -annotations!("Point " .* string.(xs), pts; - fontsize = 50 .- 2*xs, - rotation = 2pi ./ xs) +annotation!(pts; + text = "Point " .* string.(xs), + fontsize = 30 .- 5*xs) current_figure() ``` -The graphic shows that `fontsize` adjusts the displayed size and `rotation` adjusts the orientation. (The graphic also shows a need to manually override the limits of the `y` axis, as the `Point 5` is chopped off; the `ylims!` function to do so will be shown later.) +The graphic shows that `fontsize` adjusts the displayed size. Attributes for `text`, among many others, include: * `align` Specify the text alignment through `(:pos, :pos)`, where `:pos` can be `:left`, `:center`, or `:right`. - * `rotation` to indicate how the text is to be rotated * `fontsize` the font point size for the text * `font` to indicate the desired font +Annotations with an arrow can be useful to highlight a feature of a graph. This example is modified from the documentation and utilizes some interval functions to draw an arrow with an arc: + +```{julia} +g(x) = cos(6x) * exp(x) +xs = 0:0.01:4 + +_, ax, _ = lines(xs, g.(xs); axis = (; xgridvisible = false, ygridvisible = false)) + +annotation!(ax, 1, 20, 2.1, g(2.1), + text = "A relative maximum", + path = Ann.Paths.Arc(0.3), + style = Ann.Styles.LineArrow(), + labelspace = :data +) + +current_figure() +``` + + #### Line attributes @@ -666,6 +684,7 @@ A surface of revolution for $g(u)$ revolved about the $z$ axis can be visualized ```{julia} g(u) = u^2 * exp(-u) r(u,v) = (g(u)*sin(v), g(u)*cos(v), u) + us = range(0, 3, length=10) vs = range(0, 2pi, length=10) xs, ys, zs = parametric_grid(us, vs, r) @@ -681,6 +700,7 @@ A torus with big radius $2$ and inner radius $1/2$ can be visualized as follows ```{julia} r1, r2 = 2, 1/2 r(u,v) = ((r1 + r2*cos(v))*cos(u), (r1 + r2*cos(v))*sin(u), r2*sin(v)) + us = vs = range(0, 2pi, length=25) xs, ys, zs = parametric_grid(us, vs, r) @@ -696,6 +716,7 @@ A Möbius strip can be produced with: ws = range(-1/4, 1/4, length=8) thetas = range(0, 2pi, length=30) r(w, θ) = ((1+w*cos(θ/2))*cos(θ), (1+w*cos(θ/2))*sin(θ), w*sin(θ/2)) + xs, ys, zs = parametric_grid(ws, thetas, r) surface(xs, ys, zs) @@ -865,20 +886,19 @@ end #### Implicitly defined surfaces, $F(x,y,z)=0$ -To plot the equation $F(x,y,z)=0$, for $F$ a scalar-valued function, again the implicit function theorem says that, under conditions, near any solution $(x,y,z)$, $z$ can be represented as a function of $x$ and $y$, so the graph will look likes surfaces stitched together. The `Implicit3DPlotting` package takes an approach like `ImplicitPlots` to represent these surfaces. It replaces the `Contour` package computation with a $3$-dimensional alternative provided through the `Meshing` and `GeometryBasics` packages. +To plot the equation $F(x,y,z)=0$, for $F$ a scalar-valued function, again the implicit function theorem says that, under conditions, near any solution $(x,y,z)$, $z$ can be represented as a function of $x$ and $y$, so the graph will look like surfaces stitched together. -```{julia} -using Implicit3DPlotting -``` +With `Makie`, many implicitly defined surfaces can be adequately represented using `countour` with the attribute `levels=[0]`. We will illustrate this technique. +The `Implicit3DPlotting` package takes an approach like `ImplicitPlots` to represent these surfaces. It replaces the `Contour` package computation with a $3$-dimensional alternative provided through the `Meshing` and `GeometryBasics` packages. This package has a `plot_implicit_surface` function that does something similar to below. We don't illustrate it, as it *currently* doesn't work with the latest version of `Makie`. -This example, plotting an implicitly defined sphere, comes from the documentation of `Implicit3DPlotting`. The `f` to be plotted is a scalar-valued function of a vector: +To begin, we plot a sphere implicitly as a solution to $F(x,y,z) = x^2 + y^2 + z^2 - 1 = 0$> ```{julia} -f(x) = sum(x.^2) - 1 -xlims = ylims = zlims = (-5, 5) -plot_implicit_surface(f; xlims, ylims, zlims) +f(x,y,z) = x^2 + y^2 + z^2 - 1 +xs = ys = zs = range(-3/2, 3/2, 100) +contour(xs, ys, zs, f; levels=[0]) ``` @@ -887,11 +907,13 @@ Here we visualize an intersection of a sphere with another figure: ```{julia} -r₂(x) = sum(x.^2) - 5/4 # a sphere +r₂(x) = sum(x.^2) - 2 # a sphere r₄(x) = sum(x.^4) - 1 -xlims = ylims = zlims = (-2, 2) -p = plot_implicit_surface(r₂; xlims, ylims, zlims, color=:yellow) -plot_implicit_surface!(p, r₄; xlims, ylims, zlims, color=:red) +ϕ(x,y,z) = (x,y,z) + +xs = ys = zs = range(-2, 2, 100) +contour(xs, ys, zs, r₂∘ϕ; levels = [0], colormap=:RdBu) +contour!(xs, ys, zs, r₄∘ϕ; levels = [0], colormap=:viridis) current_figure() ``` @@ -900,11 +922,12 @@ This example comes from [Wikipedia](https://en.wikipedia.org/wiki/Implicit_surfa ```{julia} f(x,y,z) = 2y*(y^2 -3x^2)*(1-z^2) + (x^2 +y^2)^2 - (9z^2-1)*(1-z^2) -xlims = ylims = zlims = (-5/2, 5/2) -plot_implicit_surface(x -> f(x...); xlims, ylims, zlims) +xs = ys = zs = range(-5/2, 5/2, 100) +contour(xs, ys, zs, f; levels=[0], colormap=:RdBu) + ``` -(This figure does not render well through `contour(xs, ys, zs, f, levels=[0])`, as the hole is not shown.) +(This figure does not render well though, as the hole is not shown.) For one last example from Wikipedia, we have the Cassini oval which "can be defined as the point set for which the *product* of the distances to $n$ given points is constant." That is: @@ -915,8 +938,8 @@ function cassini(λ, ps = ((1,0,0), (-1, 0, 0))) n = length(ps) x -> prod(norm(x .- p) for p ∈ ps) - λ^n end -xlims = ylims = zlims = (-2, 2) -plot_implicit_surface(cassini(1.05); xlims, ylims, zlims) +xs = ys = zs = range(-2, 2, 100) +contour(xs, ys, zs, cassini(0.80) ∘ ϕ; levels=[0], colormap=:RdBu) ``` ## Vector fields. Visualizations of $f:R^2 \rightarrow R^2$ @@ -1064,7 +1087,7 @@ F ### Observables -The basic components of a plot in `Makie` can be updated [interactively](https://makie.juliaplots.org/stable/documentation/nodes/index.html#observables_interaction). `Makie` uses the `Observables` package which allows complicated interactions to be modeled quite naturally. In the following we give a simple example. +The basic components of a plot in `Makie` can be updated [interactively](https://makie.juliaplots.org/stable/documentation/nodes/index.html#observables_interaction). Historically `Makie` used the `Observables` package which allows complicated interactions to be modeled quite naturally. In the following we give a simple example, though newer versions of `Makie` rely on a different mechanism. In Makie, an `Observable` is a structure that allows its value to be updated, similar to an array. When changed, observables can trigger an event. Observables can rely on other observables, so events can be cascaded. @@ -1123,6 +1146,7 @@ end lines!(ax, xs, f) lines!(ax, points) +scatter!(ax, points; markersize=10) current_figure() ``` diff --git a/quarto/alternatives/plotly_plotting.qmd b/quarto/alternatives/plotly_plotting.qmd index 32b4254..139ae17 100644 --- a/quarto/alternatives/plotly_plotting.qmd +++ b/quarto/alternatives/plotly_plotting.qmd @@ -73,7 +73,6 @@ The `Config` constructor (from the `EasyConfig` package loaded with `PlotlyLight ```{julia} -#| hold: true cfg = Config() cfg.key1.key2.key3 = "value" cfg @@ -89,7 +88,6 @@ A basic scatter plot of points $(x,y)$ is created as follows: ```{julia} -#| hold: true xs = 1:5 ys = rand(5) data = Config(x = xs, @@ -113,7 +111,6 @@ A line plot is very similar, save for a different `mode` specification: ```{julia} -#| hold: true xs = 1:5 ys = rand(5) data = Config(x = xs, @@ -134,7 +131,6 @@ The line graph plays connect-the-dots with the points specified by paired `x` an ```{julia} -#| hold: true data = Config( x=[0,1,nothing,3,4,5], y = [0,1,2,3,4,5], @@ -149,7 +145,6 @@ More than one graph or layer can appear on a plot. The `data` argument can be a ```{julia} -#| hold: true data = [Config(x = 1:5, y = rand(5), type = "scatter", @@ -177,7 +172,6 @@ For example, here we plot the graphs of both the $\sin(x)$ and $\cos(x)$ over $[ ```{julia} -#| hold: true a, b = 0, 2pi xs, ys = PlotUtils.adapted_grid(sin, (a,b)) @@ -193,7 +187,6 @@ The values for `a` and `b` are used to generate the $x$- and $y$-values. These c ```{julia} -#| hold: true xs, ys = PlotUtils.adapted_grid(x -> x^5 - x - 1, (0, 2)) # answer is (0,2) p = Plot([Config(x=xs, y=ys, name="Polynomial"), Config(x=xs, y=0 .* ys, name="x-axis", mode="lines", line=Config(width=5))] @@ -232,7 +225,6 @@ A marker's attributes can be adjusted by values passed to the `marker` key. Labe ```{julia} -#| hold: true data = Config(x = 1:5, y = rand(5), mode="markers+text", @@ -251,40 +243,7 @@ The `text` mode specification is necessary to have text be displayed on the char #### RGB Colors -The `ColorTypes` package is the standard `Julia` package providing an `RGB` type (among others) for specifying red-green-blue colors. To make this work with `Config` and `JSON3` requires some type-piracy (modifying `Base.string` for the `RGB` type) to get, say, `RGB(0.5, 0.5, 0.5)` to output as `"rgb(0.5, 0.5, 0.5)"`. (RGB values in JavaScript are integers between $0$ and $255$ or floating point values between $0$ and $1$.) A string with this content can be specified. Otherwise, something like the following can be used to avoid the type piracy: - - -```{julia} -struct rgb - r - g - b -end -PlotlyLight.JSON3.StructTypes.StructType(::Type{rgb}) = PlotlyLight.JSON3.StructTypes.StringType() -Base.string(x::rgb) = "rgb($(x.r), $(x.g), $(x.b))" -``` - -With these defined, red-green-blue values can be used for colors. For example to give a range of colors, we might have: - - -```{julia} -#| hold: true -cols = [rgb(i,i,i) for i in range(10, 245, length=5)] -sizes = [12, 16, 20, 24, 28] -data = Config(x = 1:5, - y = rand(5), - mode="markers+text", - type="scatter", - name="scatter plot", - text = ["marker $i" for i in 1:5], - textposition = "top center", - marker = Config(size=sizes, color=cols) - ) -Plot(data) -``` - -The `opacity` key can be used to control the transparency, with a value between $0$ and $1$. - +The `ColorTypes` package is the standard `Julia` package providing an `RGB` type (among others) for specifying red-green-blue colors. To make this work with `Config` and `JSON3` requires some type-piracy (modifying `Base.string` for the `RGB` type) to get, say, `RGB(0.5, 0.5, 0.5)` to output as `"rgb(0.5, 0.5, 0.5)"`. (RGB values in JavaScript are integers between $0$ and $255$ or floating point values between $0$ and $1$.) A string with this content can be specified. #### Marker symbols @@ -293,7 +252,6 @@ The `marker_symbol` key can be used to set a marker shape, with the basic values ```{julia} -#| hold: true markers = ["circle", "square", "diamond", "cross", "x", "triangle", "pentagon", "hexagram", "star", "diamond", "hourglass", "bowtie", "asterisk", "hash", "y", "line"] @@ -327,7 +285,6 @@ The `shape` attribute determine how the points are connected. The default is `li ```{julia} -#| hold: true shapes = ["linear", "hv", "vh", "hvh", "vhv", "spline"] data = [Config(x = 1:5, y = 5*(i-1) .+ [1,3,2,3,1], mode="lines+markers", type="scatter", name=shape, @@ -358,7 +315,6 @@ In the following, to highlight the difference between $f(x) = \cos(x)$ and $p(x) ```{julia} -#| hold: true xs = range(-1, 1, 100) data = [ Config( @@ -381,7 +337,6 @@ The `toself` declaration is used below to fill in a polygon: ```{julia} -#| hold: true data = Config( x=[-1,1,1,-1,-1], y = [-1,1,-1,1,-1], fill="toself", @@ -399,7 +354,6 @@ The legend is shown when $2$ or more charts or specified, by default. This can b ```{julia} -#| hold: true data = Config(x=1:5, y=rand(5), type="scatter", mode="markers", name="legend label") lyt = Config(title = "Main chart title", xaxis = Config(title="x-axis label"), @@ -416,7 +370,6 @@ The aspect ratio of the chart can be set to be equal through the `scaleanchor` k ```{julia} -#| hold: true ts = range(0, 2pi, length=100) data = Config(x = sin.(ts), y = cos.(ts), mode="lines", type="scatter") lyt = Config(title = "A circle", @@ -434,7 +387,6 @@ Text annotations may be specified as part of the layout object. Annotations may ```{julia} -#| hold: true data = Config(x = [0, 1], y = [0, 1], mode="markers", type="scatter") layout = Config(title = "Annotations", xaxis = Config(title="x", @@ -452,7 +404,7 @@ Plot(data, layout) The following example is more complicated use of the elements previously described. It mimics an image from [Wikipedia](https://en.wikipedia.org/wiki/List_of_trigonometric_identities) for trigonometric identities. The use of `LaTeX` does not seem to be supported through the `JavaScript` interface; unicode symbols are used instead. The `xanchor` and `yanchor` keys are used to position annotations away from the default. The `textangle` key is used to rotate text, as desired. -```{julia, hold=true} +```{julia} alpha = pi/6 beta = pi/5 xₘ = cos(alpha)*cos(beta) @@ -569,7 +521,6 @@ Earlier, we plotted a two dimensional circle, here we plot the related helix. ```{julia} -#| hold: true helix(t) = [cos(t), sin(t), t] ts = range(0, 4pi, length=200) @@ -596,7 +547,6 @@ There is no `quiver` plot for `plotly` using JavaScript. In $2$-dimensions a tex ```{julia} -#| hold: true helix(t) = [cos(t), sin(t), t] helix′(t) = [-sin(t), cos(t), 1] ts = range(0, 4pi, length=200) @@ -642,7 +592,6 @@ A contour plot is created by the "contour" trace type. The data is prepared as a ```{julia} -#| hold: true f(x,y) = x^2 - 2y^2 xs = range(0,2,length=25) @@ -661,7 +610,6 @@ The same `zs` data can be achieved by broadcasting and then collecting as follow ```{julia} -#| hold: true f(x,y) = x^2 - 2y^2 xs = range(0,2,length=25) @@ -692,7 +640,6 @@ Surfaces defined through a scalar-valued function are drawn quite naturally, sav ```{julia} -#| hold: true peaks(x,y) = 3 * (1-x)^2 * exp(-(x^2) - (y+1)^2) - 10*(x/5 - x^3 - y^5) * exp(-x^2-y^2) - 1/3 * exp(-(x+1)^2 - y^2) @@ -713,7 +660,6 @@ For parametrically defined surfaces, the $x$ and $y$ values also correspond to m ```{julia} -#| hold: true r, R = 1, 5 X(theta,phi) = [(r*cos(theta)+R)*cos(phi), (r*cos(theta)+R)*sin(phi), diff --git a/quarto/alternatives/symbolics.qmd b/quarto/alternatives/symbolics.qmd index 06bd02b..f4a3225 100644 --- a/quarto/alternatives/symbolics.qmd +++ b/quarto/alternatives/symbolics.qmd @@ -601,14 +601,7 @@ det(N) det(collect(N)) ``` -Similarly, with `norm`: - - -```{julia} -norm(v) -``` - -and +Similarly, with `norm`, which returns a generator unless collected: ```{julia} diff --git a/quarto/differentiable_vector_calculus/matrix_calculus_notes.qmd b/quarto/differentiable_vector_calculus/matrix_calculus_notes.qmd index 68449e6..545c972 100644 --- a/quarto/differentiable_vector_calculus/matrix_calculus_notes.qmd +++ b/quarto/differentiable_vector_calculus/matrix_calculus_notes.qmd @@ -355,9 +355,15 @@ The operator $f'(x)= g'(h(x)) h'(x)$ is a product of matrices. ### Computational differences with expressions from the chain rule -Of note here is the application of the chain rule to three (or more compositions): +Of note here is the application of the chain rule to three (or more compositions) where $c:R^n \rightarrow R^j$, $b:R^j \rightarrow R^k$, and $a:R^k \rightarrow R^m$: -The derivative of $f(x) = a(x) b(x) c(x)$ can be expressed as +If $f(x) = a(b(c(x)))$ then the derivative is: + +$$ +f'(x) = a'(b(c(x))) b'(c(x)) c'(x), +$$ + +which can be expressed as three matrix multiplications two ways: $$ f' = (a'b')c' \text{ or } f' = a'(b'c') @@ -380,7 +386,7 @@ Whereas computing from the right to left is first $jk1$ operations leaving a $j * left to right is $njk + nk$ = $nk \cdot (j + 1)$. * right to left is $jk + jn = j\cdot (k+n)$. -When $j=k$, say, we can compare and see the second is a factor less in terms of operations. This can be quite significant in higher dimensions, whereas the dimensions of calculus (where $n$ and $m$ are $3$ or less) it is not an issue. +When $j=k$, say, we can compare and see the second is a factor less in terms of operations. This can be quite significant in higher dimensions. ##### Example @@ -405,7 +411,7 @@ Whereas the relationship is changed when the first matrix is skinny and the last ---- -In calculus, we have $n$ and $m$ are $1$,$2$,or $3$. But that need not be the case, especially if differentiation is over a parameter space. +In calculus, we typically have $n$ and $m$ are $1$, $2$,or $3$. But that need not be the case, especially if differentiation is over a parameter space. ## Derivatives of matrix functions @@ -546,7 +552,7 @@ Appropriate sizes for $A$, $B$, and $C$ are determined by the various products i If $A$ is $m \times n$ and $B$ is $r \times s$, then since $BC$ is defined, $C$ has $s$ rows, and since $CA^T$ is defined, $C$ must have $n$ columns, as $A^T$ is $n \times m$, so $C$ must be $s\times n$. Checking this is correct on the other side, $A \times B$ would be size $mr \times ns$ and $\vec{C}$ would be size $sn$, so that product works, size wise. -The referred to notes have an explanation for this formula, but we confirm with an example with $m=n=2$ and $r=s=3$: +The referred to notes have an explanation for this formula, but we only confirm it with an example using $m=n=2$ and $r=s=3$: ```{julia} @syms A[1:2, 1:2]::real B[1:3, 1:3]::real C[1:3, 1:2]::real diff --git a/quarto/integral_vector_calculus/Project.toml b/quarto/integral_vector_calculus/Project.toml index 7a49be7..818f132 100644 --- a/quarto/integral_vector_calculus/Project.toml +++ b/quarto/integral_vector_calculus/Project.toml @@ -4,9 +4,14 @@ ForwardDiff = "f6369f11-7733-5829-9624-2563aa707210" HCubature = "19dc6840-f33b-545b-b366-655c7e3ffd49" IJulia = "7073ff75-c697-5162-941a-fcdaad2a7d2a" ImplicitIntegration = "bc256489-3a69-4a66-afc4-127cc87e6182" +LaTeXStrings = "b964fa9f-0449-5b57-a5c2-d3ea65f4040f" +Mustache = "ffc61752-8dc7-55ee-8c37-f3e9cdd09e70" PlotlyBase = "a03496cd-edff-5a9b-9e67-9cda94a718b5" PlotlyKaleido = "f2990250-8cf9-495f-b13a-cce12b45703c" Plots = "91a5bcdd-55d7-5caf-9e0b-520d859cae80" QuadGK = "1fd47b50-473d-5c70-9696-f719f8f3bcdc" +QuizQuestions = "612c44de-1021-4a21-84fb-7261cf5eb2d4" Roots = "f2b01f46-fcfa-551c-844a-d8ac1e96c665" SymPy = "24249f21-da20-56a4-8eb1-6a02cf4ae2e6" +Tables = "bd369af6-aec1-5ad0-b16a-f7cc5008161c" +TextWrap = "b718987f-49a8-5099-9789-dcd902bef87d" diff --git a/quarto/integrals/make_pdf.jl b/quarto/integrals/make_pdf.jl index 876fd95..0d3af78 100644 --- a/quarto/integrals/make_pdf.jl +++ b/quarto/integrals/make_pdf.jl @@ -15,7 +15,8 @@ files = ( "center_of_mass", "volumes_slice", "arc_length", - "surface_area", + "surface_area", + "orthogonal_polynomials", "twelve-qs", ) diff --git a/quarto/integrals/orthogonal_polynomials.qmd b/quarto/integrals/orthogonal_polynomials.qmd new file mode 100644 index 0000000..3c07f02 --- /dev/null +++ b/quarto/integrals/orthogonal_polynomials.qmd @@ -0,0 +1,724 @@ +# Orthogonal polynomials + +{{< include ../_common_code.qmd >}} + +This section uses these add-on packages: + + +```{julia} +using SymPy +using QuadGK +using Roots +using ForwardDiff: derivative +``` + +This section takes a detour to give some background on why the underlying method of `quadgk` is more efficient than those of Riemann sums. Orthogonal polynomials play a key role. There are many families of such polynomials. We highlight two. + + +## Inner product + +Define an operation between two integrable, real-valued functions $f(x)$ and $g(x)$ by: + +$$ +\langle f, g \rangle = \int_{-1}^1 f(x)g(x) dx +$$ + +The properties of the integral mean this operation satisfies these three main properties: + +* symmetry: $\langle f, g \rangle = \langle g,f \rangle$ +* positive definiteness: $\langle f, f \rangle > 0$ *unless* $f(x)=0$. +* linearity: if $a$ and $b$ are scalars, then $\langle af + bg, h \rangle = a\langle f, h \rangle + b \langle g, h \rangle$. + + +The set of integrable functions forms a *vector space*, which simply means two such functions can be added to yield another integrable function and an integrable function times a scalar is still an integrable function. Many different collections of objects form a vector space. In particular, other sets of functions form a vector space, for example the collection of polynomials of degree $n$ or less or just the set of all polynomials. + +For a vector space, an operation like the above satisfying these three properties is called an *inner product*; the combination of an inner product and a vector space is called an *inner product space*. In the following, we assume $f$ and $g$ are from a vector space with a real-valued inner product. + +Inner products introduce a sense of size through a *norm*: +$\lVert f \rVert = \sqrt{\langle f, f\rangle }$. + +Norms satisfy two main properties: + +* scalar: $\lVert af \rVert = |a|\lVert f\rVert$ +* triangle inequality: $\lVert f + g \rVert \leq \lvert f \rVert + \lVert g \rVert$ + +Two elements of an inner product space, $f$ and $g$, are *orthogonal* if $\langle f, g \rangle = 0$. This is a generalization of perpendicular. The Pythagorean theorem for orthogonal elements holds: $\lVert f\rVert^2 + \lVert g\rVert^2 = \lVert f+g\rVert^2$. + +As we assume a real-valued inner product, the angle between two elements can be defined by: + +$$ +\angle(f,g) = \cos^{-1}\left(\frac{\langle f, g\rangle}{\lVert f \rVert \lVert g \rVert}\right). +$$ + +This says, the angle between two orthogonal elements is $90$ degrees (in some orientation) + +The Cauchy-Schwarz inequality, $|\langle f, g \rangle| \leq \lVert f \rVert \lVert g\rVert$, for an inner product space, ensures the argument to $\cos^{-1}$ is between $-1$ and $1$. + +These properties generalize two-dimensional vectors, with components $\langle x, y\rangle$. Recall, these can be visualized by placing a tail at the origin and a tip at the point $(x,y)$. Such vectors can be added by placing the tail of one at the tip of the other and using the vector from the other tail to the other tip. + +With this, we have a vector anchored at the origin can be viewed as a line segment with slope $y/x$ (rise over run). A perpendicular line segment would have slope $-x/y$ (the negative reciprocal) which would be associated with the vector $\langle y, -x \rangle$. The dot product is just the sum of the multiplied components, or for these two vectors $x\cdot y + y\cdot (-x)$, which is $0$, as the line segments are perpendicular (orthogonal). + +Consider now two vectors, say $f$, $g$. We can make a new vector that is orthogonal to $f$ by combining $g$ with a piece of $f$. But what piece? + +Consider this + +$$ +\begin{align*} +\langle f, g - \frac{\langle f,g\rangle}{\langle f, f\rangle} f \rangle +&= \langle f, g \rangle - \langle f, \frac{\langle f,g\rangle}{\langle f, f\rangle} f \rangle \\ +&= \langle f, g \rangle - \frac{\langle f,g\rangle}{\langle f, f\rangle}\langle f,f \rangle \\ +&= \langle f, g \rangle - \langle f, g \rangle = 0 +\end{align*} +$$ + +Define +$$ +proj_f(g) = \frac{\langle f,g\rangle }{\langle f, f\rangle} f, +$$ +then we have $u_1 = f$ and $u_2 = g-proj_f(g)$, $u_1$ and $u_2$ are orthogonal. + +A similar calculation shows if $h$ is added to the set of elements, then +$u_3 = h - proj_{u_1}(h) - proj_{u_2}(h)$ will be orthogonal to $u_1$ and $u_2$. etc. + +This process, called the [Gram-Schmidt](https://en.wikipedia.org/wiki/Gram%E2%80%93Schmidt_process) process, can turn any set of vectors into a set of orthogonal vectors, assuming they are all non zero and no non-trivial linear combination makes them zero. + +## Legendre + +Consider now polynomials of degree $n$ or less with the normalization that $p(1) = 1$. We begin with two such polynomials: $u_0(x) = 1$ and $u_1(x) = x$. + +These are orthogonal with respect to $\int_{-1}^1 f(x) g(x) dx$, as + +$$ +\int_{-1}^1 u_0(x) u_1(x) dx = +\int_{-1}^1 1 \cdot x dx = +x^2 \mid_{-1}^1 = 1^2 - (-1)^2 = 0. +$$ + +Now consider a quadratic polynomial, $u_2(x) = ax^2 + bx + c$, we want a polynomial which is orthogonal to $u_0$ and $u_1$ with the extra condition that $u_2(1) = c =1$ (or $c=1$.). We can do this using Gram-Schmidt as above, or as here through a system of two equations: + +```{julia} +@syms a b c d x +u0 = 1 +u1 = x +u2 = a*x^2 + b*x + c +eqs = (integrate(u0 * u2, (x, -1, 1)) ~ 0, + integrate(u1 * u2, (x, -1, 1)) ~ 0) +sols = solve(eqs, (a, b, c)) # b => 0, a => -3c +u2 = u2(sols...) +u2 = simplify(u2 / u2(x=>1)) # make u2(1) = 1 and fix c +``` + +The quadratic polynomial has $3$ unknowns and the orthgonality conditions give two equations. Solving these equations leaves one unknown (`c`). But the normalization condition (that $u_i(1) = 1$) allows `c` to be simplified out. + +We can do this again with $u_3$: + +```{julia} +u3 = a*x^3 + b*x^2 + c*x + d +eqs = (integrate(u0 * u3, (x, -1, 1)) ~ 0, + integrate(u1 * u3, (x, -1, 1)) ~ 0, + integrate(u2 * u3, (x, -1, 1)) ~ 0) +sols = solve(eqs, (a, b, c, d)) # a => -5c/3, b=>0, d=>0 +u3 = u3(sols...) +u3 = simplify(u3/u3(x=>1)) # make u3(1) = 1 +``` + +In theory, this can be continued up until any $n$. The resulting +polynomials are called the +[Legendre](https://en.wikipedia.org/wiki/Legendre_polynomials) +polynomials. + +Rather than continue this, we develop easier means to generate these polynomials. + +## General weight function + +Let $w(x)$ be some non-negative function and consider the new inner product between two polynomials: + +$$ +\langle p, q\rangle = \int_I p(x) q(x) w(x) dx +$$ + +where $I$ is an interval and $w(x)$ is called a weight function. In the above discussion $I=[-1,1]$ and $w(x) = 1$. + +Suppose we have *orthogonal* polynomials $p_i(x)$, $i=0,1, \dots, n$, where $p_i$ is a polynomial of degree $i$ ($p_i(x) = k_i x^i + \cdots$, where $k_i \neq 0$), and + +$$ +\langle p_m, p_n \rangle = +\int_I p_m(x) p_n(x) w(x) dx = +\begin{cases} +0 & m \neq n\\ +h_m & m = n +\end{cases} +$$ + +Unique elements can be defined by specifying some additional property. For Legendre, it was $p_n(1)=1$, for other orthogonal families this may be specified by having leading coefficient of $1$ (monic), or a norm of $1$ (orthonormal), etc. + +The above is the *absolutely continuous* case, generalizations of the integral allow this to be more general. + +Orthogonality can be extended: If $q(x)$ is any polynomial of degree $m < n$, then +$\langle q, p_n \rangle = \int_I q(x) p_n(x) w(x) dx = 0$. (See the questions for more detail.) + + +Some names used for the characterizing constants are: + +* $p_n(x) = k_n x^n + \cdots$ ($k_n$ is the leading term) +* $h_n = \langle p_n, p_n\rangle$ + + + +### Three-term reccurence + +Orthogonal polynomials, as defined above through a weight function, satisfy a *three-term recurrence*: + +$$ +p_{n+1}(x) = (A_n x + B_n) p_n(x) - C_n p_{n-1}(x), +$$ + +where $n \geq 0$ and $p_{n-1}(x) = 0$. + +(With this and knowledge of $A_n$, $B_n$, and $C_n$, the polynomials can be recursively generated from just specifying a value for the constant $p_0(x)$. + +First, $p_{n+1}$ has leading term $k_{n+1}x^{n+1}$. Looking on the right hand side for the coefficient of $x^{n+1}$ we find $A_n k_n$, so $A_n = k_{n+1}/k_n$. + +Next, we look at $u(x) = p_{n+1}(x) - A_n x p_n(x)$, a polynomial of degree $n$ or less. + +As this has degree $n$ or less, it can be expressed in terms of $p_0, p_1, \dots, p_n$. Write it as $u(x) = \sum_{j=0}^n d_j p_j(x)$. Now, take any $m < n-1$ and consider $p_m$. We consider the inner product of $u$ and $p_m$ two ways: + +$$ +\begin{align*} +\int_I p_m(x) u(x) w(x) dx &= +\int_I p_m(x) \sum_{j=0}^n p_j(x) w(x) dx \\ +&= \int_I p_m(x) \left(p_m(x) + \textcolor{red}{\sum_{j=0, j\neq m}^{n} p_j(x)}\right) w(x) dx \\ +&= \int_I p_m(x) p_m(x) w(x) dx = h_m +\end{align*} +$$ + +As well + +$$ +\begin{align*} +\int_I p_m(x) u(x) w(x) dx +&= \int_I p_m(x) (p_{n+1}(x) - A_n x p_n(x)) w(x) dx \\ +&= \int_I p_m(x) \textcolor{red}{p_{n+1}(x)} w(x) dx - \int_I p_m(x) A_n x p_n(x) w(x) dx\\ +&= 0 - A_n \int_I (\textcolor{red}{x p_m(x)}) p_n(x) w(x) dx\\ +&= 0 +\end{align*} +$$ + +The last integral being $0$ as $xp_m(x)$ has degree $n-1$ or less and hence is orthogonal to $p_n$. + +That is $p_{n+1} - A_n x p_n(x) = d_n p_n(x) + d_{n-1} p_{n-1}(x)$. Setting $B_n=d_n$ and $C_{n-1} = -d_{n-1}$ shows the three-term recurrence applies. + + +#### Example: Legendre polynomials + + +With this notation, the Legendre polynomials have: + +$$ +\begin{align*} +w(x) &= 1\\ +I &= [-1,1]\\ +A_n &= \frac{2n+1}{n+1}\\ +B_n &= 0\\ +C_n & = \frac{n}{n+1}\\ +k_{n+1} &= \frac{2n+1}{n+1}k_n - \frac{n}{n-1}k_{n-1}, k_1=k_0=1\\ +h_n &= \frac{2}{2n+1} +\end{align*} +$$ + + +#### Favard theorem + +In an efficient review of the subject, [Koornwinder](https://arxiv.org/pdf/1303.2825) states conditions on the recurrence that ensure that if a $n$-th degree polynomials $p_n$ satisfy a three-term recurrence, then there is an associated weight function (suitably generalized). The conditions use this form of a three-term recurrence: + +$$ +\begin{align*} +xp_n(x) &= a_n p_{n+1}(x) + b_n p_n(x) + c_n p_{n-1}(x),\quad (n > 0)\\ +xp_0(x) &= a_0 p_1(x) + b_0 p_0(x) +\end{align*} +$$ + +where the constants are real and $a_n c_{n+1} > 0$. These force $a_n = k_n/k_{n+1}$ and $c_n/h_{n+1} = a_n/h_n$ + + +#### Clenshaw algorithm + +When introducing polynomials, the synthetic division algorithm was given to compute $p(x) / (x-r)$. This same algorithm also computed $p(r)$ efficiently and is called Horner's method. The `evalpoly` method in `Julia`'s base implements this. + +For a set of polynomials $p_0(x), p_1(x), \dots, p_n(x)$ satisfying a three-term recurrence $p_{n+1}(x) = (A_n x + B_n) p_n(x) - C_n p_{n-1}(x)$, the Clenshaw algorithm gives an efficient means to compute an expression of a linear combination of the polynomials, $q(x) = a_0 p_0(x) + a_1 p_1(x) + \cdots + a_n p_n(x)$. + +The [method](https://en.wikipedia.org/wiki/Clenshaw_algorithm) uses a reverse recurrence formula starting with + +$$ +b_{n+1}(x) = b_{n+2}(x) = 0 +$$ + +and then computing for $k = n, n-1, \dots, 1$ + +$$ +b_k(x) = a_k + (A_k x + B_k) b_{k+1}(x) - C_k b_{k+2}(x) +$$ + + +Finally finishing by computing $a_0 p_0(x) + b_1 p_1(x) - C(1) p_0(x) b_2$. + +For example, with the Legendre polynomials, we have + +```{julia} +A(n) = (2n+1)//(n+1) +B(n) = 0 +C(n) = n // (n+1) +``` + +Say we want to compute $a_0 u_0(x) + a_1 u_1(x) + a_2 u_2(x) + a_3 u_3(x) + a_4 u_4(x)$. The necessary inputs are the coefficients, the value of $x$, and polynomials $p_0$ and $p_1$. + +```{julia} +function clenshaw(x, as, p0, p1) + n = length(as) - 1 + bn1, bn2 = 0, 0 + a(k) = as[k + 1] # offset + for k in n:-1:1 + bn1, bn2 = a(k) + (A(k) * x + B(k)) * bn1 - C(k+1) * bn2, bn1 + end + b1, b2 = bn1, bn2 + p0(x) * a(0) + p1(x) * b1 - C(1) * p0(x) * b2 +end +``` + +This function can be purposed to generate additional Legendre polynomials. For example, to compute $u_4$ we pass in a symbolic value for $x$ and mask out all by $a_4$ in our coefficients: + +```{julia} +p₀(x) = 1 +p₁(x) = x # Legendre +@syms x +clenshaw(x, (0,0,0,0,1), p₀, p₁) |> expand |> simplify +``` + +:::{.callout-note} +### Differential equations approach + +A different description of families of orthogonal polynomials is that they satisfy a differential equation of the type + +$$ +\sigma(x) y''(x) + \tau(x) y'(x) + \lambda_n y(x) = 0, +$$ + +where $\sigma(x) = ax^2 + bx + c$, $\tau(x) = dx + e$, and $\lambda_n = -(a\cdot n(n-1) + dn)$. + +With this parameterization, values for $A_n$, $B_n$, and $C_n$ can be given in terms of the leading coefficient, $k_n$ (cf. [Koepf and Schmersau](https://arxiv.org/pdf/math/9612224)): + +$$ +\begin{align*} +A_n &= \frac{k_{n+1}}{k_n}\\ +B_n &= \frac{k_{n+1}}{k_n} \frac{2bn(a(n-1)+d) + e(d-2a)}{(2a(n-1) + d)(2an+d)}\\ +C_n &= \frac{k_{n+1}}{k_{n-1}} +\frac{n(a(n-1) + d)(a(n-2)+d)(n(an+d))(4ac-b^2)+ae^2+cd^2-bde}{ +(a(n-1)+d)(a(2n-1)+d)(a(2n-3)+d)(2a(n-1)+d)^2} +\end{align*} +$$ + +There are other relations between derivatives and the orthogonal polynomials. For example, another three-term recurrence is: + +$$ +\sigma(x) p_n'(x) = (\alpha_n x + \beta_n)p_n(x) + \gamma_n p_{n-1}(x) +$$ + +The same reference has formulas for $\alpha$, $\beta$, and $\gamma$ in terms of $a,b,c,d$, and $e$ along with many others. +::: + + + +## Chebyshev + +The Chebyshev polynomials (of the first kind) satisfy the three-term recurrence + +$$ +T_{n+1}(x) = 2x T_n(x) - T_{n-1}(x) +$$ + +with $T_0(x)= 1$ and $T_1(x)=x$. + +These polynomials have domain $(-1,1)$ and weight function $(1-x^2)^{-1/2}$. + +(The Chebyshev polynomials of the second kind satisfy the same three-term recurrence but have $U_0(x)=1$ and $U_1(x)=2x$.) + + +These polynomials are related to trigonometry through + +$$ +T_n(\cos(\theta)) = \cos(n\theta) +$$ + + +This characterization makes it easy to find the zeros of the +polynomial $T_n$, as they happen when $\cos(n\theta)$ is $0$, or when +$n\theta = \pi/2 + k\pi$ for $k$ in $0$ to $n-1$. Solving for $\theta$ +and taking the cosine, we get the zeros of the $n$th degree polynomial +$T_n$ are $\cos(\pi(k + 1/2)/n)$ for $k$ in $0$ to $n-1$. + +These evenly spaced angles lead to roots more concentrated at the edges of the interval $(-1,1)$. + + +##### Example + +Chebyshev polynomials have a minimal property that makes them fundamental for use with interpolation. + +Define the *infinity* norm over $[-1,1]$ to be the maximum value of the absolute value of the function over these values. + +Let $f(x) = 2^{-n+1}T_n(x)$ be a monic version of the Chebyshev polynomial. + +> If $q(x)$ is any monic polynomial of degree $n$, then the infinity norm of $q(x)$ is greater than or equal to that of $f$. + +Using the trigonometric representation of $T_n$, we have + +* $f(x)$ has infinity norm of $2^{-n+1}$ and these maxima occur at $x=\cos((k\pi)/n)$, where $0 \leq k \leq n$. (There is a cosine curve with known peaks, oscillating between $-1$ and $1$.) +* $f(x) > 0$ at $x = \cos((2k\pi)/n)$ for $0 \leq 2k \leq n$ +* $f(x) < 0$ at $x = \cos(((2k+1)\pi)/n)$ for $0 \leq 2k+1 \leq n$ + +Suppose $w(x)$ is a monic polynomial of degree $n$ and suppose it has smaller infinity norm. Consider $u(x) = f(x) - w(x)$. At the extreme points of $f(x)$, we must have $|f(x)| \geq |w(x)|$. But this means + +* $u(x) > 0$ at $x = \cos((2k\pi)/n)$ for $0 \leq 2k \leq n$ +* $u(x) < 0$ at $x = \cos(((2k+1)\pi)/n)$ for $0 \leq 2k+1 \leq n$ + +As $u$ is continuous, this means there are at least $n$ sign changes, hence $n$ or more zeros. But as both $f$ and $w$ are monic, $u$ is of degree $n-1$, at most. This is a contradiction unless $u(x)$ is the zero polynomial, which it can't be by assumption. + + + + +### Integration + +Recall, a Riemann sum can be thought of in terms of weights, $w_i$ and nodes $x_i$ for which $\int_I f(x) dx \approx \sum_{i=0}^{n-1} w_i f(x_i)$. +For a right-Riemann sum with partition given by $a_0 < a_1 < \cdots < a_n$ the nodes are $x_i = a_i$ and the weights are $w_i = (a_i - a_{i-1})$ (or in the evenly spaced case, $w_i = (a_n - a_0)/n$. + +More generally, this type of expression can represent integrals of the type $\int_I f(x) w(x) dx$, with $w(x)$ as in an inner product. Call such a sum a Gaussian quadrature. + +We will see that the zeros of orthogonal polynomials have special properties as nodes. + +> For orthogonal polynomials over the interval $I$ with weight function $w(x)$, each $p_n$ has $n$ distinct real zeros in $I$. + +Suppose that $p_n$ had only $k 0 +$$ + +But, the product is of degree $k < n$, so by orthogonality must be $0$. Hence, it can't be that $k < n$, so there must be $n$ sign changes in $I$ by $p_n$. Each corresponds to a zero, as $p_n$ is continuous. + + +This next statement says that using the zeros of $p_n$ for the nodes of Gaussian quadrature and appropriate weights that the quadrature is exact for higher degree polynomials. + +> For a fixed $n$, suppose $p_0, p_1, \dots, p_n$ are orthogonal polynomials over $I$ with weight function $w(x)$. If the zeros of $p_n$ are the nodes $x_i$, then there exists $n$ weights so that the any polynomial of degree $2n-1$ or less, the Gaussian quadrature is exact. + +That is if $q(x)$ is a polynomial with degree $2n-1$ or less, we have for some choice of $w_i$: + +$$ +\int_I q(x) w(x) dx = \sum_{i=1}^n w_i q(x_i) +$$ + + +To compare, recall, Riemann sums ($1$-node) were exact for constant functions (degree $0$), the trapezoid rule ($2$-nodes) is exact for linear polynomials (degree $1$), and Simpson's rule ($3$ nodes) are exact for cubic polynomials (degree $3$). + + +We follow [Wikipedia](https://en.wikipedia.org/wiki/Gaussian_quadrature#Fundamental_theorem) to see this key fact. + + +Take $h(x)$ of degree $2n-1$ or less. Then by polynomial long division, there are polynomials $q(x)$ and $r(x)$ where + +$$ +h(x) = q(x) p_n(x) + r(x) +$$ + +and the degree of $r(x)$ is less than $n-1$, the degree of $p_n(x)$. Further, the degree of $q(x)$ is also less than $n-1$, as were it more, then the degree of $q(x)p_n(x)$ would be more than $n-1+n$ or $2n-1$. Let's note that if $x_i$ is a zero of $p_n(x)$ that $h(x_i)= r(x_i)$. + +So + +$$ +\begin{align*} +\int_I h(x) w(x) dx &= \int_I \textcolor{red}{q(x)} p_n(x) w(x) dx + \int_I r(x) w(x)dx\\ +&= 0 + \int r(x) w(x) dx. +\end{align*} +$$ + +Now consider the polynomials made from the zeros of $p_n(x)$ + +$$ +l_i(x) = \prod_{j \ne i} \frac{x - x_j}{x_i - x_j} +$$ + +These are called Lagrange interpolating polynomials and have the property that $l_i(x_i) = 1$ and $l_i(x_j) = 0$ if $i \neq j$. + +These allow the expression of + +$$ +\begin{align*} +r(x) &= l_1(x)r(x_1) + l_2(x) r(x_2) + \cdots + l_n(x) r(x_n) \\ +&= \sum_{i=1}^n l_i(x) r(x_i) +\end{align*} +$$ + +This isn't obviously true, but this expression agrees with an at-most degree $n-1$ polynomial ($r(x)$) at $n$ points hence it must be the same polynomial.) + +With this representation, the integral becomes + +$$ +\begin{align*} +\int_I h(x) w(x) dx &= \int_I r(x) w(x)dx \\ +&= \int_I \sum_{i=1}^n l_i(x) r(x_i) w(x) dx\\ +&= \sum_{i=1}^n r(x_i) \int_I l_i(x) w(x) dx \\ +&= \sum_{i=1}^n r(x_i) w_i\\ +&= \sum_{i=1}^n w_i h(x_i) +\end{align*} +$$ + + + +That is there are weights, $w_i = \int_I l_i(x) w(x) dx$, for which the integration is exactly found by Gaussian quadrature using the roots of $p_n$ as the nodes. + + +The general formula for the weights can be written in terms of the polynomials $p_i = k_ix^i + \cdots$: + +$$ +\begin{align*} +w_i &= \int_I l_i(x) w(x) dx \\ +&= \frac{k_n}{k_{n-1}} +\frac{\int_I p_{n-1}(x)^2 w(x) dx}{p'_n(x_i) p_{n-1}(x_i)}. +\end{align*} +$$ + +To see this, consider: + +$$ +\begin{align*} +\prod_{j \neq i} (x - x_j) &= +\frac{\prod_j (x-x_j)}{x-x_i} \\ +&= \frac{1}{k_n}\frac{k_n \prod_j (x - x_j)}{x - x_i} \\ +&= \frac{1}{k_n} \frac{p_n(x)}{x-x_i}\\ +&= \frac{1}{k_n} \frac{p_n(x) - p_n(x_i)}{x-x_i}\\ +&\rightarrow \frac{p'_n(x_i)}{k_n}, \text{ as } x \rightarrow x_i. +\end{align*} +$$ + +Thus + +$$ +\prod_{j \neq i} (x_i - x_j) = \frac{p'_n(x_i)}{k_n}. +$$ + +This gives + +$$ +\begin{align*} +w_i &= \int_i \frac{k_n \prod_j (x-x_j)}{p'_n(x_i)} w(x) dx\\ +&= \frac{1}{p'_n(x_i)} \int_i \frac{p_n(x)}{x-x_i} w(x) dx +\end{align*} +$$ + +To work on the last term, a trick (see the questions for detail) can show that for any $k \leq n$ that + +$$ +\int_I \frac{x^k p_n(x)}{x - x_i} w(x) dx += x_i^k \int_I \frac{p_n(x)}{x - x_i} w(x) dx +$$ + +Hence for any degree $n$ or less polynomial: we have + +$$ +q(x_i) \int_I \frac{p_n(x)}{x - x_i} w(x) dx = +\int_I \frac{q(x) p_n(x)}{x - x_i} w(x) dx +$$. + +We will use this for $p_{n-1}$. First, as $x_i$ is a zero of $p_n(x)$ we have + +$$ +\frac{p_n(x)}{x-x_i} = k_n x^{n-1}+ r(x), +$$ + +where $r(x)$ has degree $n-2$ at most. This is due to $p_n$ being divided by a monic polynomial, hence leaving a degree $n-1$ polynomial with leading coefficient $k_n$. + +But then + +$$ +\begin{align*} +w_i &= \frac{1}{p'_n(x_i)} \int_I \frac{p_n(x)}{x-x_i} w(x) dx \\ +&= \frac{1}{p'_n(x_i)} \frac{1}{p_{n-1}(x_i)} \int_I \frac{p_{n-1}(x) p_n(x)}{x - x_i} w(x) dx\\ +&= \frac{1}{p'_n(x_i)p_{n-1}(x_i)} \int_I p_{n-1}(x) +(k_n x^{n-1} + \textcolor{red}{r(x)}) w(x) dx\\ +&= \frac{k_n}{p'_n(x_i)p_{n-1}(x_i)} \int_I p_{n-1}(x) x^{n-1} w(x) dx\\ +&= \frac{k_n}{p'_n(x_i)p_{n-1}(x_i)} \int_I p_{n-1}(x) +\left( +\textcolor{red}{\left(x^{n-1} - \frac{p_{n-1}(x)}{k_{n-1}}\right) } ++ \frac{p_{n-1}(x)}{k_{n-1}}\right) w(x) dx\\ +&= \frac{k_n}{p'_n(x_i)p_{n-1}(x_i)} \int_I p_{n-1}(x)\frac{p_{n-1}(x)}{k_{n-1}} w(x) dx\\ +&= \frac{k_n}{k_{n-1}} \frac{1}{p'_n(x_i)p_{n-1}(x_i)} \int_I p_{n-1}(x)^2 w(x) dx. +\end{align*} +$$ + +### Examples of quadrature formula + +The `QuadGK` package uses a modification to Gauss quadrature to estimate numeric integrals. Let's see how. Behind the scenes, `quadgk` calls `kronrod` to compute nodes and weights. + +We have from earlier that + +```{julia} +u₃(x) = x*(5x^2 - 3)/2 +u₄(x) = 35x^4 / 8 - 15x^2 / 4 + 3/8 +``` + +```{julia} +xs = find_zeros(u₄, -1, 1) +``` + +From this we can compute the weights from the derived general formula: + +```{julia} +k₃, k₄ = 5/2, 35/8 +w(x) = 1 +I = first(quadgk(x -> u₃(x)^2 * w(x), -1, 1)) +ws = [k₄/k₃ * 1/(derivative(u₄,xᵢ) * u₃(xᵢ)) * I for xᵢ ∈ xs] +(xs, ws) +``` + +We compare now to the values returned by `kronrod` in `QuadGK` + +```{julia} +kxs, kwts, wts = kronrod(4, -1, 1) +[ws wts xs kxs[2:2:end]] +``` + +(The `kronrod` function computes $2n-1$ nodes and weights. The Gauss-Legendre nodes are $n$ of those, and extracted by taking the 2nd, 4th, etc.) + +To compare integrations of some smooth function we have + + +```{julia} +u(x) = exp(x) +GL = sum(wᵢ * u(xᵢ) for (xᵢ, wᵢ) ∈ zip(xs, ws)) +KL = sum(wᵢ * u(xᵢ) for (xᵢ, wᵢ) ∈ zip(kxs, kwts)) +QL, esterror = quadgk(u, -1, 1) +(; GL, KL, QL, esterror) +``` + +The first two are expected to not be as accurate, as they utilize a fixed number of nodes. + + +## Questions + +###### Question + +Let $p_i$ for $i$ in $0$ to $n$ be polynomials of degree $i$. It is true that for any polynomial $q(x)$ of degree $k \leq n$ that there is a linear combination such that $q(x) = a_0 p_0(x) + \cdots + a_k p_k(x)$. + +First it is enough to do this for a monic polynomial $x^k$, why? + +```{julia} +#| echo: false +choices = [raw"If you can do it for each $x^i$ then if $q(x) = b_0 + b_1x + b_2x^2 + \cdots + b_k x^k$ we just multiply the coefficients for each $x^i$ by $b_i$ and add.", + raw"It isn't true"] +radioq(choices, 1) +``` + +Suppose $p_0 = k_0$ and $p_1 = k_1x + a$. How would you make $x=x^1$? + +```{julia} +#| echo: false +choices = [raw"$(p1 - (a/k_0) p_0)/k_1$", + raw"$p1 - p0$"] +radioq(choices, 1) +``` + +Let $p_i = k_i x^i + \cdots$ ($k_i$ is the leading term) +To reduce $p_3 = k_3x^3 + a_2x^2 + a_1x^1 + a_0$ to $k_3x^3$ we could try: + +* form $q_3 = p_3 - p_2 (a_2/k_2)$. As $p_2$ is degree $2$, this leaves $k_3x^3$ alone, but it + +```{julia} +#| echo: false +choices = [raw"It leaves $0$ as the coefficient of $x^2$", + raw"It leaves all the other terms as $0$"] +radioq(choices, 1) +``` + +* We then use $p_1$ times some multiple $a/k_1$ to remove the $x$ term +* we then use $p_0$ times some multiple $a/k_0$ to remove the constant term + +Would this strategy work to reduce $p_n$ to $k_n x^n$? + +```{julia} +#| echo: false +radioq(["Yes", "No"], 1) +``` + + +###### Question + +Suppose $p(x)$ and $q(x)$ are polynomials of degree $n$ and there are $n+1$ points for which $p(x_i) = q(x_i)$. + +First, is it true or false that a polynomial of degree $n$ has *at most* n zeros? + +```{julia} +#| echo: false +radioq(["true, unless it is the zero polynomial", "false"], 1) +``` + +What is the degree of $h(x) = p(x) - q(x)$? + +```{julia} +#| echo: false +radioq([raw"At least $n+1$", raw"At most $n$"], 2) +``` + +At least how many zeros does the polynomial $h(x)$ have? + +```{julia} +#| echo: false +radioq([raw"At least $n+1$", raw"At most $n$"], 1) +``` + +Is $p(x) = q(x)$ with these assumptions? + + +```{julia} +#| echo: false +radioq(["yes", "no"], 1) +``` + + + + +###### Question + +We wish to show that for any $k \leq n$ that + +$$ +\int_I \frac{x^k p_n(x)}{x - x_i} w(x) dx += x_i^k \int_I \frac{p_n(x)}{x - x_i} w(x) dx +$$ + +We have for $u=x/x_i$ that + +$$ +\frac{1}{x - x_i} = \frac{1 - u^k}{x - x_i} + \frac{u^k}{x - x_i} +$$ + +But the first term, $(1-u^k)/(x-x_i)$ is a polynomial of degree $k-1$. Why? + +```{julia} +#| echo: false +choices = [raw""" +Because we can express this as $x_i^k - x^k$ which factors as $(x_i - x) \cdot u(x)$ where $u(x)$ has degree $k-1$, at most. +""", + raw""" +It isn't true, it clearly has degree $k$ +"""] +radioq(choices, 1) +``` + +This gives if $k \leq n$ and with $u=x/x_i$: + +$$ +\begin{align*} +\int_I \frac{p_n(x)}{x - x_i} w(x) dx +&= \int_I p_n(x) \left( \textcolor{red}{\frac{1 - u^k}{x - x_i}} + \frac{u^k}{x - x_i} \right) w(x) dx\\ +&= \int_I p_n(x) \frac{\frac{x^k}{x_i^k}}{x - x_i} w(x) dx\\ +&= \frac{1}{x_i^k} \int_I \frac{x^k p_n(x)}{x - x_i} w(x) dx +\end{align*} +$$ diff --git a/quarto/misc/Project.toml b/quarto/misc/Project.toml new file mode 100644 index 0000000..671818d --- /dev/null +++ b/quarto/misc/Project.toml @@ -0,0 +1,12 @@ +[deps] +CalculusWithJulia = "a2e0e22d-7d4c-5312-9169-8b992201a882" +ForwardDiff = "f6369f11-7733-5829-9624-2563aa707210" +HCubature = "19dc6840-f33b-545b-b366-655c7e3ffd49" +LaTeXStrings = "b964fa9f-0449-5b57-a5c2-d3ea65f4040f" +Mustache = "ffc61752-8dc7-55ee-8c37-f3e9cdd09e70" +Plots = "91a5bcdd-55d7-5caf-9e0b-520d859cae80" +QuadGK = "1fd47b50-473d-5c70-9696-f719f8f3bcdc" +QuizQuestions = "612c44de-1021-4a21-84fb-7261cf5eb2d4" +SymPy = "24249f21-da20-56a4-8eb1-6a02cf4ae2e6" +Tables = "bd369af6-aec1-5ad0-b16a-f7cc5008161c" +TextWrap = "b718987f-49a8-5099-9789-dcd902bef87d" From c4bd214ecdf2324f28ff3f6e610c1af33de8ab03 Mon Sep 17 00:00:00 2001 From: jverzani Date: Fri, 27 Jun 2025 19:21:20 -0400 Subject: [PATCH 07/22] WIP: Plots --- quarto/alternatives/Project.toml | 7 ++++- quarto/alternatives/SciML.qmd | 2 +- quarto/alternatives/makie_plotting.qmd | 36 +++++++++++++------------ quarto/alternatives/plotly_plotting.qmd | 35 +----------------------- quarto/alternatives/symbolics.qmd | 9 +------ 5 files changed, 28 insertions(+), 61 deletions(-) diff --git a/quarto/alternatives/Project.toml b/quarto/alternatives/Project.toml index 4188acb..1e24643 100644 --- a/quarto/alternatives/Project.toml +++ b/quarto/alternatives/Project.toml @@ -5,11 +5,13 @@ ForwardDiff = "f6369f11-7733-5829-9624-2563aa707210" GLMakie = "e9467ef8-e4e7-5192-8a1a-b1aee30e663a" GeometryBasics = "5c1252a2-5f33-56bf-86c9-59e7332b4326" IJulia = "7073ff75-c697-5162-941a-fcdaad2a7d2a" -Implicit3DPlotting = "d997a800-832a-4a4c-b340-7dddf3c1ad50" Integrals = "de52edbc-65ea-441a-8357-d3a637375a31" +LaTeXStrings = "b964fa9f-0449-5b57-a5c2-d3ea65f4040f" LinearAlgebra = "37e2e46d-f89d-539d-b4ee-838fcccc9c8e" +Makie = "ee78f7c6-11fb-53f2-987a-cfe4a2b5a57a" Meshing = "e6723b4c-ebff-59f1-b4b7-d97aa5274f73" ModelingToolkit = "961ee093-0014-501f-94e3-6117800e7a78" +Mustache = "ffc61752-8dc7-55ee-8c37-f3e9cdd09e70" NonlinearSolve = "8913a72c-1f9b-4ce2-8d82-65094dcecaec" Optimization = "7f7a1694-90dd-40f0-9382-eb1efda571ba" OptimizationOptimJL = "36348300-93cb-4f02-beb5-3c3902f8871e" @@ -19,6 +21,7 @@ PlotlyKaleido = "f2990250-8cf9-495f-b13a-cce12b45703c" PlotlyLight = "ca7969ec-10b3-423e-8d99-40f33abb42bf" Plots = "91a5bcdd-55d7-5caf-9e0b-520d859cae80" QuadGK = "1fd47b50-473d-5c70-9696-f719f8f3bcdc" +QuizQuestions = "612c44de-1021-4a21-84fb-7261cf5eb2d4" Roots = "f2b01f46-fcfa-551c-844a-d8ac1e96c665" SplitApplyCombine = "03a91e81-4c3e-53e1-a0a4-9c0c8f19dd66" StaticArrays = "90137ffa-7385-5640-81b9-e52037218182" @@ -26,3 +29,5 @@ SymPy = "24249f21-da20-56a4-8eb1-6a02cf4ae2e6" SymbolicLimits = "19f23fe9-fdab-4a78-91af-e7b7767979c3" SymbolicNumericIntegration = "78aadeae-fbc0-11eb-17b6-c7ec0477ba9e" Symbolics = "0c5d862f-8b57-4792-8d23-62f2024744c7" +Tables = "bd369af6-aec1-5ad0-b16a-f7cc5008161c" +TextWrap = "b718987f-49a8-5099-9789-dcd902bef87d" diff --git a/quarto/alternatives/SciML.qmd b/quarto/alternatives/SciML.qmd index 8092595..a6e23c1 100644 --- a/quarto/alternatives/SciML.qmd +++ b/quarto/alternatives/SciML.qmd @@ -250,7 +250,7 @@ With the system defined, we can pass this to `NonlinearProblem`, as was done wit ```{julia} -prob = NonlinearProblem(ns, [1.0], [α => 1.0]) +prob = NonlinearProblem(mtkcompile(ns), [1.0], Dict(α => 1.0)) ``` The problem is solved as before: diff --git a/quarto/alternatives/makie_plotting.qmd b/quarto/alternatives/makie_plotting.qmd index 1cca87b..61ed0b0 100644 --- a/quarto/alternatives/makie_plotting.qmd +++ b/quarto/alternatives/makie_plotting.qmd @@ -865,20 +865,19 @@ end #### Implicitly defined surfaces, $F(x,y,z)=0$ -To plot the equation $F(x,y,z)=0$, for $F$ a scalar-valued function, again the implicit function theorem says that, under conditions, near any solution $(x,y,z)$, $z$ can be represented as a function of $x$ and $y$, so the graph will look likes surfaces stitched together. The `Implicit3DPlotting` package takes an approach like `ImplicitPlots` to represent these surfaces. It replaces the `Contour` package computation with a $3$-dimensional alternative provided through the `Meshing` and `GeometryBasics` packages. +To plot the equation $F(x,y,z)=0$, for $F$ a scalar-valued function, again the implicit function theorem says that, under conditions, near any solution $(x,y,z)$, $z$ can be represented as a function of $x$ and $y$, so the graph will look like surfaces stitched together. -```{julia} -using Implicit3DPlotting -``` +With `Makie`, many implicitly defined surfaces can be adequately represented using `countour` with the attribute `levels=[0]`. We will illustrate this technique. +The `Implicit3DPlotting` package takes an approach like `ImplicitPlots` to represent these surfaces. It replaces the `Contour` package computation with a $3$-dimensional alternative provided through the `Meshing` and `GeometryBasics` packages. This package has a `plot_implicit_surface` function that does something similar to below. We don't illustrate it, as it *currently* doesn't work with the latest version of `Makie`. -This example, plotting an implicitly defined sphere, comes from the documentation of `Implicit3DPlotting`. The `f` to be plotted is a scalar-valued function of a vector: +To begin, we plot a sphere implicitly as a solution to $F(x,y,z) = x^2 + y^2 + z^2 - 1 = 0$> ```{julia} -f(x) = sum(x.^2) - 1 -xlims = ylims = zlims = (-5, 5) -plot_implicit_surface(f; xlims, ylims, zlims) +f(x,y,z) = x^2 + y^2 + z^2 - 1 +xs = ys = zs = range(-3/2, 3/2, 100) +contour(xs, ys, zs, f; levels=[0]) ``` @@ -887,11 +886,13 @@ Here we visualize an intersection of a sphere with another figure: ```{julia} -r₂(x) = sum(x.^2) - 5/4 # a sphere +r₂(x) = sum(x.^2) - 2 # a sphere r₄(x) = sum(x.^4) - 1 -xlims = ylims = zlims = (-2, 2) -p = plot_implicit_surface(r₂; xlims, ylims, zlims, color=:yellow) -plot_implicit_surface!(p, r₄; xlims, ylims, zlims, color=:red) +ϕ(x,y,z) = (x,y,z) + +xs = ys = zs = range(-2, 2, 100) +contour(xs, ys, zs, r₂∘ϕ; levels = [0], colormap=:RdBu) +contour!(xs, ys, zs, r₄∘ϕ; levels = [0], colormap=:viridis) current_figure() ``` @@ -900,11 +901,12 @@ This example comes from [Wikipedia](https://en.wikipedia.org/wiki/Implicit_surfa ```{julia} f(x,y,z) = 2y*(y^2 -3x^2)*(1-z^2) + (x^2 +y^2)^2 - (9z^2-1)*(1-z^2) -xlims = ylims = zlims = (-5/2, 5/2) -plot_implicit_surface(x -> f(x...); xlims, ylims, zlims) +xs = ys = zs = range(-5/2, 5/2, 100) +contour(xs, ys, zs, f; levels=[0], colormap=:RdBu) + ``` -(This figure does not render well through `contour(xs, ys, zs, f, levels=[0])`, as the hole is not shown.) +(This figure does not render well though, as the hole is not shown.) For one last example from Wikipedia, we have the Cassini oval which "can be defined as the point set for which the *product* of the distances to $n$ given points is constant." That is: @@ -915,8 +917,8 @@ function cassini(λ, ps = ((1,0,0), (-1, 0, 0))) n = length(ps) x -> prod(norm(x .- p) for p ∈ ps) - λ^n end -xlims = ylims = zlims = (-2, 2) -plot_implicit_surface(cassini(1.05); xlims, ylims, zlims) +xs = ys = zs = range(-2, 2, 100) +contour(xs, ys, zs, cassini(0.80) ∘ ϕ; levels=[0], colormap=:RdBu) ``` ## Vector fields. Visualizations of $f:R^2 \rightarrow R^2$ diff --git a/quarto/alternatives/plotly_plotting.qmd b/quarto/alternatives/plotly_plotting.qmd index 32b4254..c44546e 100644 --- a/quarto/alternatives/plotly_plotting.qmd +++ b/quarto/alternatives/plotly_plotting.qmd @@ -251,40 +251,7 @@ The `text` mode specification is necessary to have text be displayed on the char #### RGB Colors -The `ColorTypes` package is the standard `Julia` package providing an `RGB` type (among others) for specifying red-green-blue colors. To make this work with `Config` and `JSON3` requires some type-piracy (modifying `Base.string` for the `RGB` type) to get, say, `RGB(0.5, 0.5, 0.5)` to output as `"rgb(0.5, 0.5, 0.5)"`. (RGB values in JavaScript are integers between $0$ and $255$ or floating point values between $0$ and $1$.) A string with this content can be specified. Otherwise, something like the following can be used to avoid the type piracy: - - -```{julia} -struct rgb - r - g - b -end -PlotlyLight.JSON3.StructTypes.StructType(::Type{rgb}) = PlotlyLight.JSON3.StructTypes.StringType() -Base.string(x::rgb) = "rgb($(x.r), $(x.g), $(x.b))" -``` - -With these defined, red-green-blue values can be used for colors. For example to give a range of colors, we might have: - - -```{julia} -#| hold: true -cols = [rgb(i,i,i) for i in range(10, 245, length=5)] -sizes = [12, 16, 20, 24, 28] -data = Config(x = 1:5, - y = rand(5), - mode="markers+text", - type="scatter", - name="scatter plot", - text = ["marker $i" for i in 1:5], - textposition = "top center", - marker = Config(size=sizes, color=cols) - ) -Plot(data) -``` - -The `opacity` key can be used to control the transparency, with a value between $0$ and $1$. - +The `ColorTypes` package is the standard `Julia` package providing an `RGB` type (among others) for specifying red-green-blue colors. To make this work with `Config` and `JSON3` requires some type-piracy (modifying `Base.string` for the `RGB` type) to get, say, `RGB(0.5, 0.5, 0.5)` to output as `"rgb(0.5, 0.5, 0.5)"`. (RGB values in JavaScript are integers between $0$ and $255$ or floating point values between $0$ and $1$.) A string with this content can be specified. #### Marker symbols diff --git a/quarto/alternatives/symbolics.qmd b/quarto/alternatives/symbolics.qmd index 06bd02b..f4a3225 100644 --- a/quarto/alternatives/symbolics.qmd +++ b/quarto/alternatives/symbolics.qmd @@ -601,14 +601,7 @@ det(N) det(collect(N)) ``` -Similarly, with `norm`: - - -```{julia} -norm(v) -``` - -and +Similarly, with `norm`, which returns a generator unless collected: ```{julia} From 23e00863a5e3da78b53be13e2cd4437491565633 Mon Sep 17 00:00:00 2001 From: jverzani Date: Fri, 27 Jun 2025 19:21:39 -0400 Subject: [PATCH 08/22] Plots --- quarto/_quarto.yml | 4 ++-- quarto/integral_vector_calculus/Project.toml | 5 +++++ quarto/misc/Project.toml | 2 ++ 3 files changed, 9 insertions(+), 2 deletions(-) create mode 100644 quarto/misc/Project.toml diff --git a/quarto/_quarto.yml b/quarto/_quarto.yml index b1dd890..4379fab 100644 --- a/quarto/_quarto.yml +++ b/quarto/_quarto.yml @@ -156,5 +156,5 @@ format: execute: error: false -# freeze: false - freeze: auto + freeze: false +# freeze: auto diff --git a/quarto/integral_vector_calculus/Project.toml b/quarto/integral_vector_calculus/Project.toml index 7a49be7..818f132 100644 --- a/quarto/integral_vector_calculus/Project.toml +++ b/quarto/integral_vector_calculus/Project.toml @@ -4,9 +4,14 @@ ForwardDiff = "f6369f11-7733-5829-9624-2563aa707210" HCubature = "19dc6840-f33b-545b-b366-655c7e3ffd49" IJulia = "7073ff75-c697-5162-941a-fcdaad2a7d2a" ImplicitIntegration = "bc256489-3a69-4a66-afc4-127cc87e6182" +LaTeXStrings = "b964fa9f-0449-5b57-a5c2-d3ea65f4040f" +Mustache = "ffc61752-8dc7-55ee-8c37-f3e9cdd09e70" PlotlyBase = "a03496cd-edff-5a9b-9e67-9cda94a718b5" PlotlyKaleido = "f2990250-8cf9-495f-b13a-cce12b45703c" Plots = "91a5bcdd-55d7-5caf-9e0b-520d859cae80" QuadGK = "1fd47b50-473d-5c70-9696-f719f8f3bcdc" +QuizQuestions = "612c44de-1021-4a21-84fb-7261cf5eb2d4" Roots = "f2b01f46-fcfa-551c-844a-d8ac1e96c665" SymPy = "24249f21-da20-56a4-8eb1-6a02cf4ae2e6" +Tables = "bd369af6-aec1-5ad0-b16a-f7cc5008161c" +TextWrap = "b718987f-49a8-5099-9789-dcd902bef87d" diff --git a/quarto/misc/Project.toml b/quarto/misc/Project.toml new file mode 100644 index 0000000..43aec5b --- /dev/null +++ b/quarto/misc/Project.toml @@ -0,0 +1,2 @@ +[deps] +Plots = "91a5bcdd-55d7-5caf-9e0b-520d859cae80" From 50cc2b21930e654e622403ead6d399941cf9aae5 Mon Sep 17 00:00:00 2001 From: jverzani Date: Fri, 27 Jun 2025 19:22:38 -0400 Subject: [PATCH 09/22] xxx --- quarto/alternatives/makie_plotting.qmd | 2 -- 1 file changed, 2 deletions(-) diff --git a/quarto/alternatives/makie_plotting.qmd b/quarto/alternatives/makie_plotting.qmd index 0aea3b0..4b2c658 100644 --- a/quarto/alternatives/makie_plotting.qmd +++ b/quarto/alternatives/makie_plotting.qmd @@ -303,8 +303,6 @@ current_figure() ### Text (`annotations`) -XXX FIX ME XXX - Text can be placed at a point, as a marker is. To place text, the desired text and a position need to be specified along with any adjustments to the default attributes. From 2af5a6a213bacac16d449b8b4b87b4d1743e25e8 Mon Sep 17 00:00:00 2001 From: jverzani Date: Fri, 27 Jun 2025 19:23:48 -0400 Subject: [PATCH 10/22] WIP --- quarto/alternatives/makie_plotting.qmd | 29 ++++++++++++++++++++------ 1 file changed, 23 insertions(+), 6 deletions(-) diff --git a/quarto/alternatives/makie_plotting.qmd b/quarto/alternatives/makie_plotting.qmd index 61ed0b0..ed558e6 100644 --- a/quarto/alternatives/makie_plotting.qmd +++ b/quarto/alternatives/makie_plotting.qmd @@ -304,7 +304,6 @@ current_figure() ### Text (`annotations`) - Text can be placed at a point, as a marker is. To place text, the desired text and a position need to be specified along with any adjustments to the default attributes. @@ -315,25 +314,43 @@ For example: xs = 1:5 pts = Point2.(xs, xs) scatter(pts) -annotations!("Point " .* string.(xs), pts; - fontsize = 50 .- 2*xs, - rotation = 2pi ./ xs) +annotation!(pts; + text = "Point " .* string.(xs), + fontsize = 30 .- 5*xs) current_figure() ``` -The graphic shows that `fontsize` adjusts the displayed size and `rotation` adjusts the orientation. (The graphic also shows a need to manually override the limits of the `y` axis, as the `Point 5` is chopped off; the `ylims!` function to do so will be shown later.) +The graphic shows that `fontsize` adjusts the displayed size. Attributes for `text`, among many others, include: * `align` Specify the text alignment through `(:pos, :pos)`, where `:pos` can be `:left`, `:center`, or `:right`. - * `rotation` to indicate how the text is to be rotated * `fontsize` the font point size for the text * `font` to indicate the desired font +Annotations with an arrow can be useful to highlight a feature of a graph. This example is modified from the documentation and utilizes some interval functions to draw an arrow with an arc: + +```{julia} +g(x) = cos(6x) * exp(x) +xs = 0:0.01:4 + +_, ax, _ = lines(xs, g.(xs); axis = (; xgridvisible = false, ygridvisible = false)) + +annotation!(ax, 1, 20, 2.1, g(2.1), + text = "A relative maximum", + path = Ann.Paths.Arc(0.3), + style = Ann.Styles.LineArrow(), + labelspace = :data +) + +current_figure() +``` + + #### Line attributes From 145bc6043f7cb1a08362d13491cbaf08164477a9 Mon Sep 17 00:00:00 2001 From: jverzani Date: Fri, 27 Jun 2025 20:53:48 -0400 Subject: [PATCH 11/22] WIP --- quarto/misc/Project.toml | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/quarto/misc/Project.toml b/quarto/misc/Project.toml index 43aec5b..671818d 100644 --- a/quarto/misc/Project.toml +++ b/quarto/misc/Project.toml @@ -1,2 +1,12 @@ [deps] +CalculusWithJulia = "a2e0e22d-7d4c-5312-9169-8b992201a882" +ForwardDiff = "f6369f11-7733-5829-9624-2563aa707210" +HCubature = "19dc6840-f33b-545b-b366-655c7e3ffd49" +LaTeXStrings = "b964fa9f-0449-5b57-a5c2-d3ea65f4040f" +Mustache = "ffc61752-8dc7-55ee-8c37-f3e9cdd09e70" Plots = "91a5bcdd-55d7-5caf-9e0b-520d859cae80" +QuadGK = "1fd47b50-473d-5c70-9696-f719f8f3bcdc" +QuizQuestions = "612c44de-1021-4a21-84fb-7261cf5eb2d4" +SymPy = "24249f21-da20-56a4-8eb1-6a02cf4ae2e6" +Tables = "bd369af6-aec1-5ad0-b16a-f7cc5008161c" +TextWrap = "b718987f-49a8-5099-9789-dcd902bef87d" From 30300f295b25394bd4518d67789159ef01b5f100 Mon Sep 17 00:00:00 2001 From: jverzani Date: Fri, 27 Jun 2025 20:55:42 -0400 Subject: [PATCH 12/22] adjust plots --- adjust_plotly.jl | 43 +++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 43 insertions(+) create mode 100644 adjust_plotly.jl diff --git a/adjust_plotly.jl b/adjust_plotly.jl new file mode 100644 index 0000000..34a0d01 --- /dev/null +++ b/adjust_plotly.jl @@ -0,0 +1,43 @@ +# The issue with `PlotlyLight` appears to be that +# the `str` below is called *after* the inclusion of `require.min.js` +# (That str is included in the `.qmd` file to be included in the header +# but the order of inclusion appears not to be adjustable) +# This little script just adds a line *before* the require call +# which seems to make it all work. The line number 83 might change. + +#alternatives/plotly_plotting.html +function _add_plotly(f) + lineno = 117 + +str = """ + +""" + + r = readlines(f) + open(f, "w") do io + for (i,l) ∈ enumerate(r) + i == lineno && println(io, str) + println(io, l) + end + end +end + + + +function (@main)(args...) + for (root, dirs, files) in walkdir("_book") + for fᵢ ∈ files + f = joinpath(root, fᵢ) + if endswith(f, ".html") + _add_plotly(f) + end + end + end + + #f = "_book/integrals/center_of_mass.html" + #_add_plotly(f) + + return 1 +end + + ["ODEs", "alternatives", "derivatives", "differentiable_vector_calculus", "integral_vector_calculus", "integrals", "limits", "misc", "precalc", "site_libs"] From aa8e9e04ca151538394da95e1cbc811f662a9c7a Mon Sep 17 00:00:00 2001 From: jverzani Date: Fri, 27 Jun 2025 21:02:20 -0400 Subject: [PATCH 13/22] script to adjust plotly position in loading of javascript --- quarto/adjust_plotly.jl | 39 ++++++++++++++++++++++++++++++--------- 1 file changed, 30 insertions(+), 9 deletions(-) diff --git a/quarto/adjust_plotly.jl b/quarto/adjust_plotly.jl index 5060e98..c646e5e 100644 --- a/quarto/adjust_plotly.jl +++ b/quarto/adjust_plotly.jl @@ -5,19 +5,40 @@ # This little script just adds a line *before* the require call # which seems to make it all work. The line number 83 might change. -f = "_book/alternatives/plotly_plotting.html" -lineno = 88 +#alternatives/plotly_plotting.html +function _add_plotly(f) + lineno = 117 str = """ - - """ -r = readlines(f) -open(f, "w") do io - for (i,l) ∈ enumerate(r) - i == lineno && println(io, str) - println(io, l) + r = readlines(f) + open(f, "w") do io + for (i,l) ∈ enumerate(r) + i == lineno && println(io, str) + println(io, l) + end end end + + + +function (@main)(args...) + for (root, dirs, files) in walkdir("_book") + for fᵢ ∈ files + f = joinpath(root, fᵢ) + if endswith(f, ".html") + @show :adjust, f + _add_plotly(f) + end + end + end + + #f = "_book/integrals/center_of_mass.html" + #_add_plotly(f) + + return 1 +end + + ["ODEs", "alternatives", "derivatives", "differentiable_vector_calculus", "integral_vector_calculus", "integrals", "limits", "misc", "precalc", "site_libs"] From 50132119542ca03cba828e2ff7d508b42c611bf3 Mon Sep 17 00:00:00 2001 From: jverzani Date: Wed, 2 Jul 2025 06:25:10 -0400 Subject: [PATCH 14/22] work on better figures --- quarto/_common_code.qmd | 22 + quarto/_quarto.yml | 4 +- quarto/adjust_plotly.jl | 14 +- quarto/index.qmd | 2 +- quarto/integrals/area_between_curves.qmd | 82 +++- quarto/integrals/volumes_slice.qmd | 399 ++++++++++++++++++- quarto/limits/intermediate_value_theorem.qmd | 69 +++- quarto/limits/limits.qmd | 75 ++-- quarto/precalc/functions.qmd | 203 +++++++++- quarto/precalc/inversefunctions.qmd | 176 +++++++- quarto/precalc/plotting.qmd | 104 ++++- quarto/references.bib | 9 + 12 files changed, 1098 insertions(+), 61 deletions(-) diff --git a/quarto/_common_code.qmd b/quarto/_common_code.qmd index fcc1575..db01bfd 100644 --- a/quarto/_common_code.qmd +++ b/quarto/_common_code.qmd @@ -2,6 +2,28 @@ #| output: false #| echo: false +# Some style choices for `Plots.jl` + +empty_style = (xaxis=([], false), + yaxis=([], false), + framestyle=:origin, + legend=false) +axis_style = (arrow=true, side=:head, line=(:gray, 1)) +text_style = (10,) +fn_style = (;line=(:black, 3)) +fn2_style = (;line=(:red, 4)) +mark_style = (;line=(:gray, 1, :dot)) + +domain_style = (;fill=(:orange, 0.35)) +range_style = (; fill=(:blue, 0.35)) + +nothing +``` + +```{julia} +#| output: false +#| echo: false + ## Formatting options are included here; not in CalculusWithJulia.WeaveSupport using QuizQuestions nothing diff --git a/quarto/_quarto.yml b/quarto/_quarto.yml index 178bc7e..ddd335a 100644 --- a/quarto/_quarto.yml +++ b/quarto/_quarto.yml @@ -158,7 +158,7 @@ format: execute: error: false - freeze: false -# freeze: auto +# freeze: false + freeze: auto # cache: false # enabled: true \ No newline at end of file diff --git a/quarto/adjust_plotly.jl b/quarto/adjust_plotly.jl index 34a0d01..ec06770 100644 --- a/quarto/adjust_plotly.jl +++ b/quarto/adjust_plotly.jl @@ -7,16 +7,19 @@ #alternatives/plotly_plotting.html function _add_plotly(f) - lineno = 117 + #lineno = 117 -str = """ - -""" r = readlines(f) + inserted = false open(f, "w") do io for (i,l) ∈ enumerate(r) - i == lineno && println(io, str) + if contains(l, "require.min.js") + !inserted && println(io, """ + +""") + inserted = true + end println(io, l) end end @@ -29,6 +32,7 @@ function (@main)(args...) for fᵢ ∈ files f = joinpath(root, fᵢ) if endswith(f, ".html") + dirname(f) == "_book" && continue _add_plotly(f) end end diff --git a/quarto/index.qmd b/quarto/index.qmd index 46d6eca..0529f15 100644 --- a/quarto/index.qmd +++ b/quarto/index.qmd @@ -59,7 +59,7 @@ in a spirit similar to a section of a book. Just like a book, there are try-it-yourself questions at the end of each page. All have a limited number of self-graded answers. These notes borrow ideas from many sources, for example @Strang, @Knill, @Schey, @Thomas, -@RogawskiAdams, several Wikipedia pages, and other sources. +@RogawskiAdams, @Angenent, several Wikipedia pages, and other sources. These notes are accompanied by a `Julia` package `CalculusWithJulia` that provides some simple functions to streamline some common tasks diff --git a/quarto/integrals/area_between_curves.qmd b/quarto/integrals/area_between_curves.qmd index cce4ab5..b279dea 100644 --- a/quarto/integrals/area_between_curves.qmd +++ b/quarto/integrals/area_between_curves.qmd @@ -64,7 +64,88 @@ $$ #### Examples +Find the area between +$$ +\begin{align*} +f(x) &= \frac{x^3 \cdot (2-x)}{2} \text{ and } \\ +g(x) &= e^{x/3} + (1-\frac{x}{1.7})^6 - 0.6 +\end{align*} +$$ + +over the interval $[0.2, 1.7]$. The area is illustrated in the figure below. + +```{julia} +f(x) = x^3*(2-x)/2 +g(x) = exp(x/3) + (1 - (x/1.7))^6 - 0.6 +a, b = 0.2, 1.7 +h(x) = g(x) - f(x) +answer, _ = quadgk(h, a, b) +answer +``` + + +::: {#fig-area-between-f-g} + +```{julia} +#| echo: false +p = let + gr() + # area between graphs + # https://github.com/SigurdAngenent/WisconsinCalculus/blob/master/figures/221/09areabetweengraphs.pdf + + f(x) = 1/6+x^3*(2-x)/2 + g(x) = 1/6+exp(x/3)+(1-x/1.7)^6-0.6 + a,b =0.2, 1.7 + A, B = 0, 2 + A′, B′ = A + .1, B - .1 + n = 20 + + plot(; empty_style..., aspect_ratio=:equal, xlims=(A,B)) + plot!(f, A′, B′; fn_style...) + plot!(g, A′, B′; fn_style...) + + xp = range(a, b, n) + marked = n ÷ 2 + for i in 1:n-1 + x0, x1 = xp[i], xp[i+1] + mpt = (x0 + x1)/2 + R = Shape([x0,x1,x1,x0], [f(mpt),f(mpt),g(mpt),g(mpt)]) + color = i == marked ? :gray : :white + plot!(R; fill=(color, 0.5), line=(:black, 1)) + end + + # axis + plot!([(A,0),(B,0)]; axis_style...) + + # hightlight + x0, x1 = xp[marked], xp[marked+1] + _style = (;line=(:gray, 1, :dash)) + plot!([(a,0), (a, f(a))]; _style...) + plot!([(b,0), (b,f(b))]; _style...) + plot!([(x0,0), (x0, f(x0))]; _style...) + plot!([(x1,0), (x1, f(x1))]; _style...) + + annotate!([ + (B′, f(B′), text(L"f(x)", 10, :left,:top)), + (B′, g(B′), text(L"g(x)", 10, :left, :bottom)), + (a, 0, text(L"a=x_0", 10, :top, :left)), + (b, 0, text(L"b=x_n", 10, :top, :left)), + (x0, 0, text(L"x_i", 10, :top)), + (x1, 0, text(L"x_{i+1}", 10, :top,:left)) + ]) + + current() +end + +plotly() +p +``` + +Illustration of a Riemann sum approximation to estimate the area between $f(x)$ and $g(x)$ over an interval $[a,b]$. (Figure follows one by @Angenent.) +::: + +##### Example Find the area bounded by the line $y=2x$ and the curve $y=2 - x^2$. @@ -970,4 +1051,3 @@ choices = ["The two enclosed areas should be equal", "The two enclosed areas are clearly different, as they do not overap"] radioq(choices, 1) ``` - diff --git a/quarto/integrals/volumes_slice.qmd b/quarto/integrals/volumes_slice.qmd index 2a891ff..4b4d0d7 100644 --- a/quarto/integrals/volumes_slice.qmd +++ b/quarto/integrals/volumes_slice.qmd @@ -20,6 +20,7 @@ using SymPy #| echo: false #| results: "hidden" import LinearAlgebra: norm +using SplitApplyCombine nothing ``` @@ -122,6 +123,145 @@ The formula is for a rotation around the $x$-axis, but can easily be generalized ::: +::: {#fig-solid-of-revolution} + +```{julia} +#| echo: false +plt = let + gr() + # Follow lead of # https://github.com/SigurdAngenent/WisconsinCalculus/blob/master/figures/221/09surf_of_rotation2.py + # plot surface of revolution around x axis between [0, 3] + # best if r(t) descreases + + rad(x) = 2/(1 + exp(x))#2/(2.0+x) + viewp = [2,-2,1] + + ## + unitize(x) = x / norm(x) + + """Orthogonal projection along the vector viewp""" + function make_Pmat(viewp) + a = unitize( [-viewp[2], viewp[1], 0] ) + b = unitize( [-viewp[3]*viewp[1], + -viewp[3]*viewp[2], + viewp[1]^2 + viewp[2]^2] ) + collect(zip(a,b)) + end + + #linear projection of R^3 onto R^2 + function proj(X, viewp) + Pmat = make_Pmat(viewp) + x=sum([Pmat[i][1]*X[i] for i in 1:3]) + y=sum([Pmat[i][2]*X[i] for i in 1:3]) + (x, y) # a point + end + proj(X) = proj(X, viewp) + + # discrete determinant of Jacobian; area multiplier? + function jac(X, u, v) + ϵ = 0.000001 + A = map((p,q) -> (p-q)/ϵ, X(u+ϵ/2, v), X(u-ϵ/2, v)) + B = map((p,q) -> (p-q)/ϵ, X(u, v+ϵ/2), X(u, v-ϵ/2)) + return A[1]*B[2]-A[2]*B[1] + end + + + # surface of revolution + surf(t, z) = [t, rad(t)*cos(z), rad(t)*sin(z)] + + # project the surface at (t, a=theta) + psurf(t,z) = proj(surf(t,z)) + + + bisect(f, a, b) = find_zero(f, (a,b), Bisection()) + _fold(t, zmin, zmax) = bisect(z -> jac(psurf, t, z), zmin, zmax) + + # create shape holding project disc + drawdiscF(t) = Shape(invert([psurf(t, 2*i*pi/100) for i in 1:101])...) + + # project a line between two points + pline!(p,q; kwargs...) = plot!([proj(p),proj(q)]; + line_style..., kwargs...) + α = 1.0 + line_style = (; line=(:black, 1)) + + plot(; empty_style..., aspect_ratio=:equal) + + + + # by layering, we get x-axis as desired + pline!([-1,0,0], [0,0,0]) + plot!(drawdiscF(0); fill =(:lightgray, α)) + pline!([0,0,0], [1,0,0]) + plot!(drawdiscF(1); fill =(:black, α)) # black to lightgray gives thickness + plot!(drawdiscF(1.1); fill=(:lightgray, α)) + pline!([1.1,0,0], [2,0,0]) + plot!(drawdiscF(2); fill=(:lightgray, α)) + pline!([2,0,0], [3,0,0]) + plot!(drawdiscF(3); fill=(:lightgray, α)) + pline!([3,0,0], [4,0,0]; arrow=true, side=:head) + pline!([0,0,0], [0,0,1.25]; arrow=true, side=:head) + + + tt = range(0, pi, 30) + curve = [psurf(t, pi/2) for t in tt] + plot!(curve; line=(:black, 2)) + + f1 = [[t, _fold(t, 0, pi)] for t in tt] + curve = [psurf(f[1], f[2]) for f in f1] + plot!(curve; line=(:black,)) + + f2 = [[t, _fold(t, pi, 2*pi)] for t in tt] + curve = [psurf(f[1], f[2]) for f in f2] + plot!(curve; line=(:black,)) + + + + tt= [0.025*i for i in 1:121] + f1 = [[t, _fold(t, pi, 2*pi)] for t in tt] + for f in f1 + plot!([psurf(f[1], f[2]-k*0.01*(4-f[1])) for k in 1:21]; + line=(:black,1)) + end + tt= [0.05*i for i in 1:61] + f1 = [[t, _fold(t, pi, 2*pi)] for t in tt] + for f in f1 + plot!([psurf( f[1], f[2]-k*0.01*(6-f[1]) ) + for k in 1:21]; line=(:black, 1)) + end +#= + ts = 0:.1:2.95 + θs = range(pi + pi/4, 3pi/2, 25) + for ti in ts + plot!([psurf(ti,θ) for θ in θs]; line=(:black, 1)) + end + θs = range(pi + pi/6, 3pi/2, 25) + for ti in ts + plot!([psurf(ti +0.05,θ) for θ in θs]; line=(:black, 1)) + end +=# + current() + + #ts = range(0, 3, 100) + #θs = range(0, 2pi, 100) + #contour(ts, θs, (t,z) -> jac(psurf,t,z), levels=[0]) + +end +plt +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + + +Illustration of a figure being rotated around the $x$-axis. The discs have approximate volume given by the area of the base times the height or $\pi r(x)^2 \Delta x$. (Figure ported from @Angenent.) + +::: + + For a numeric example, we consider the original Red [Solo](http://en.wikipedia.org/wiki/Red_Solo_Cup) Cup. The dimensions of the cup were basically: a top diameter of $d_1 = 3~ \frac{3}{4}$ inches, a bottom diameter of $d_0 = 2~ \frac{1}{2}$ inches and a height of $h = 4~ \frac{3}{4}$ inches. @@ -352,6 +492,136 @@ $$ V = \int_a^b \pi \cdot (R(x)^2 - r(x)^2) dx. $$ +::: {#fig-washer-illustration} +```{julia} +#| echo: false +plt = let + gr() + # Follow lead of # https://github.com/SigurdAngenent/WisconsinCalculus/blob/master/figures/221/09surf_of_rotation2.py + # plot surface of revolution around x axis between [0, 3] + # best if r(t) descreases + + rad(x) = 2/(1 + exp(x))#2/(2.0+x) + viewp = [2,-2,1] + + ## + unitize(x) = x / norm(x) + + """Orthogonal projection along the vector viewp""" + function make_Pmat(viewp) + a = unitize( [-viewp[2], viewp[1], 0] ) + b = unitize( [-viewp[3]*viewp[1], + -viewp[3]*viewp[2], + viewp[1]^2 + viewp[2]^2] ) + collect(zip(a,b)) + end + + #linear projection of R^3 onto R^2 + function proj(X, viewp) + Pmat = make_Pmat(viewp) + x=sum([Pmat[i][1]*X[i] for i in 1:3]) + y=sum([Pmat[i][2]*X[i] for i in 1:3]) + (x, y) # a point + end + proj(X) = proj(X, viewp) + + # discrete determinant of Jacobian; area multiplier? + function jac(X, u, v) + ϵ = 0.000001 + A = map((p,q) -> (p-q)/ϵ, X(u+ϵ/2, v), X(u-ϵ/2, v)) + B = map((p,q) -> (p-q)/ϵ, X(u, v+ϵ/2), X(u, v-ϵ/2)) + return A[1]*B[2]-A[2]*B[1] + end + + + # surface of revolution + surf(t, z) = [t, rad(t)*cos(z), rad(t)*sin(z)] + surf2(t, z) = (t, rad(t)*cos(z)/2, rad(t)*sin(z)/2) + + # project the surface at (t, a=theta) + psurf(t,z) = proj(surf(t,z)) + psurf2(t, z) = proj(surf2(t,z)) + + bisect(f, a, b) = find_zero(f, (a,b), Bisection()) + _fold(t, zmin, zmax) = bisect(z -> jac(psurf, t, z), zmin, zmax) + + # create shape holding project disc + drawdiscF(t) = Shape(invert([psurf(t, 2*i*pi/100) for i in 1:101])...) + drawdiscI(t) = Shape([psurf2(t, 2*i*pi/100) for i in 1:101]) + + # project a line between two points + pline!(p,q; kwargs...) = plot!([proj(p),proj(q)]; + line_style..., kwargs...) + α = 1.0 + line_style = (; line=(:black, 1)) + + plot(; empty_style..., aspect_ratio=:equal) + + + + # by layering, we get x-axis as desired + pline!([-1,0,0], [0,0,0]) + plot!(drawdiscF(0); fill =(:lightgray, α)) + plot!(drawdiscI(0); fill=(:white, .5)) + pline!([0,0,0], [1,0,0]) + plot!(drawdiscF(1); fill =(:black, α)) # black to lightgray gives thickness + plot!(drawdiscI(1); fill=(:white, .5)) + plot!(drawdiscF(1.1); fill=(:lightgray, α)) + plot!(drawdiscI(1.1); fill=(:white, .5)) + pline!([1.1,0,0], [2,0,0]) + plot!(drawdiscF(2); fill=(:lightgray, α)) + plot!(drawdiscI(2); fill=(:white, .5)) + pline!([2,0,0], [3,0,0]) + plot!(drawdiscF(3); fill=(:lightgray, α)) + plot!(drawdiscI(3); fill=(:white, .5)) + pline!([3,0,0], [4,0,0]; arrow=true, side=:head) + pline!([0,0,0], [0,0,1.25]; arrow=true, side=:head) + + + tt = range(0, pi, 30) + curve = [psurf(t, pi/2) for t in tt] + plot!(curve; line=(:black, 2)) + + f1 = [[t, _fold(t, 0, pi)] for t in tt] + curve = [psurf(f[1], f[2]) for f in f1] + plot!(curve; line=(:black,)) + + f2 = [[t, _fold(t, pi, 2*pi)] for t in tt] + curve = [psurf(f[1], f[2]) for f in f2] + plot!(curve; line=(:black,)) + + + + tt= [0.025*i for i in 1:121] + f1 = [[t, _fold(t, pi, 2*pi)] for t in tt] + for f in f1 + plot!([psurf(f[1], f[2]-k*0.01*(4-f[1])) for k in 1:21]; + line=(:black,1)) + end + tt= [0.05*i for i in 1:61] + f1 = [[t, _fold(t, pi, 2*pi)] for t in tt] + for f in f1 + plot!([psurf( f[1], f[2]-k*0.01*(6-f[1]) ) + for k in 1:21]; line=(:black, 1)) + end + + current() + + +end +plt +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + +Modification of earlier figure to show washer method. The interior volumn would be given by $\int_a^b \pi r(x)^2 dx$, the entire volume by $\int_a^b \pi R(x)^2 dx$. The difference then is the volume computed by the washer method. + +::: + ##### Example @@ -412,19 +682,21 @@ Let $h$ be the distance from the apex to the base. Consider cones with the prope ```{julia} -#| hold: true #| echo: false -h = 5 -R, r, rho = 1, 1/4, 1/4 -f(t) = (R-r) * cos(t) + rho * cos((R-r)/r * t) -g(t) = (R-r) * sin(t) - rho * sin((R-r)/r * t) -ts = range(0, 2pi, length=100) +plt = let + h = 5 + R, r, rho = 1, 1/4, 1/4 + f(t) = (R-r) * cos(t) + rho * cos((R-r)/r * t) + g(t) = (R-r) * sin(t) - rho * sin((R-r)/r * t) + ts = range(0, 2pi, length=100) -p = plot(f.(ts), g.(ts), zero.(ts), legend=false) -for t ∈ range(0, 2pi, length=25) - plot!(p, [0,f(t)], [0,g(t)], [h, 0], linecolor=:red) + plot(f.(ts), g.(ts), zero.(ts), legend=false) + for t ∈ range(0, 2pi, length=25) + plot!([0,f(t)], [0,g(t)], [h, 0], linecolor=:red) + end + current() end -p +plot ``` A right circular cone is one where this shape is a circle. This definition can be more general, as a square-based right pyramid is also such a cone. After possibly reorienting the cone in space so the base is at $u=0$ and the apex at $u=h$ the volume of the cone can be found from: @@ -450,6 +722,72 @@ $$ This gives a general formula for the volume of such cones. +::: {#fig-cross-sections} + +```{julia} +#| echo: false +plt = let + gr() + # sections + # https://github.com/SigurdAngenent/WisconsinCalculus/blob/master/figures/221/09Xsections.py + + x(h,z) = 0.3*h^2+(0.6-0.2*h)*cos(z) + y(h,z) = h+(0.3-0.2*h)*sin(z)+0.05*sin(4*z) + r(h,z) = (x(h,z), y(h,z)) + r1(h,z) = (2,0) .+ r(h,z) + + Nh=30 + heights = range(-1/2, 1/2, Nh) + h0=heights[Nh ÷ 2] + h1=heights[Nh ÷ 2 + 1] + hs = [heights[1], h0, h1, heights[end]] + ts = range(0, 2pi, 300) + + plot(; empty_style..., aspect_ratio=:equal) + + # stack the curves + for h in heights + curve = r.(h, ts) + plot!(Shape(curve); fill=(:white, 1.0), line=(:black, 1)) + end + + # shape pull outs; use black to give thickness + for (h, color) in zip(hs, (:white, :black, :white, :white)) + curve = r1.(h,ts) + plot!(Shape(curve); fill=(color,1.0), line=(:black, 1,)) + end + + # axis with marked points + plot!([(-1,-1), (-1, 1)]; axis_style...) + pts = [(-1, y(h, pi)) for h in hs] + scatter!(pts, marker=(5, :circle)) + + # connect with dashes + for h in hs + plot!([(-1, y(h, pi)), r(h,pi)]; line=(:black, 1, :dash)) + plot!([r(h,0), r1(h,pi)]; line=(:black, 1, :dash)) + end + + + + + + current() + + +end +plt +``` + +This figure shows the volume of a figure being comprised of slices. A discrete approximation would be found by estimating the volume of each slice by the cross sectional area times a small $\Delta h$. +(This figure was ported from @Angenent.) +::: + +```{julia} +#| echo: false +plotly() +nothing +``` ### Cavalieri's method @@ -457,6 +795,47 @@ This gives a general formula for the volume of such cones. [Cavalieri's](http://tinyurl.com/oda9xd9) Principle is "Suppose two regions in three-space (solids) are included between two parallel planes. If every plane parallel to these two planes intersects both regions in cross-sections of equal area, then the two regions have equal volumes." (Wikipedia). +::: {#fig-Cavalieris-first} + +```{julia} +#| echo: false +plt = let + gr() + + x(h,z) = (0.6-0.2*h) * cos(z) + y(h,z) = h + (0.2-0.15*h) * sin(z) + 0.01 * sin(4*z) + xa(h,z) = 2 + 0.1 * cos(7*pi*h) + (0.6-0.2*h)*cos(z) + + + heights = range(-1/2, 1/2, 50) + ts = range(0, 2pi, 300) + h0 = heights[25] + h1 = heights[26] + + plot(; empty_style..., aspect_ratio=:equal) + + for h in heights + curve=[(x(h, t), y(h, t)) for t in ts] + plot!(Shape(curve); fill=(:white,), line=(:black,1)) + + curve=[(xa(h, t), y(h, t)) for t in ts] + plot!(Shape(curve); fill=(:white,), line=(:black,1)) + end + current() +end +plt +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + +Illustration of Cavalieri's first principle. The discs from the left are moved around to form the left volume, but as the volumes of each cross-sectional disc remains the same, the two valumes are equally approximated. (This figure ported from @Angenent.) + +::: + With the formula for the volume of solids based on cross sections, this is a trivial observation, as the functions giving the cross-sectional area are identical. Still, it can be surprising. Consider a sphere with an interior cylinder bored out of it. (The [Napkin](http://tinyurl.com/o237v83) ring problem.) The bore has height $h$ - for larger radius spheres this means very wide bores. diff --git a/quarto/limits/intermediate_value_theorem.qmd b/quarto/limits/intermediate_value_theorem.qmd index ab07071..5562eeb 100644 --- a/quarto/limits/intermediate_value_theorem.qmd +++ b/quarto/limits/intermediate_value_theorem.qmd @@ -32,12 +32,69 @@ If $f$ is continuous on $[a,b]$ with, say, $f(a) < f(b)$, then for any $y$ with ::: +::: {#fig-IVT} ```{julia} #| hold: true #| echo: false #| cache: true ### {{{IVT}}} -gr() +plt = let + gr() + # IVT + empty_style = (xaxis=([], false), + yaxis=([], false), + framestyle=:origin, + legend=false) + axis_style = (arrow=true, side=:head, line=(:gray, 1)) + text_style = (10,) + fn_style = (;line=(:black, 3)) + fn2_style = (;line=(:red, 4)) + mark_style = (;line=(:gray, 1, :dot)) + domain_style = (;fill=(:orange, 0.35), line=nothing) + range_style = (; fill=(:blue, 0.35), line=nothing) + + f(x) = x + sinpi(3x) + 5sin(2x) + 3cospi(2x) + a, b = -1, 5 + xs = range(a, b, 251) + ys = f.(xs) + y0, y1 = extrema(ys) + + plot(; empty_style...) + plot!(f, a, b; fn_style...) + + plot!([a-.2, b + .2],[0,0]; axis_style...) + plot!([a-.1, a-.1], [y0-2, y1+2]; axis_style...) + + plot!([(a,0),(a,f(a))]; line=(:black, 1, :dash)) + plot!([(b,0),(b,f(b))]; line=(:black, 1, :dash)) + + m = f(a/2 + b/2) + plot!([a, b], [m,m]; line=(:black, 1, :dashdot)) + + δx = 0.03 + plot!(Shape([a,b,b,a], 4*δx*[-1,-1,1,1]); + domain_style...) + plot!(Shape((a-.1) .+ 2δx * [-1,1,1,-1], [f(a),f(a),f(b), f(b)]); + range_style...) + plot!(Shape((a-.1) .+ δx/2 * [-1,1,1,-1], [y0,y0,y1,y1]); + range_style...) + + zs = find_zeros(f, (a,b)) + c = zs[2] + plot!([(c,0), (c,f(c))]; line=(:black, 1, :dashdot)) + + annotate!([ + (a, 0, text(L"a", 12, :bottom)), + (b, 0, text(L"b", 12, :top)), + (c, 0, text(L"c", 12, :top)), + (a-.1, f(a), text(L"f(a)", 12, :right)), + (a-.1, f(b), text(L"f(b)", 12, :right)), + (b, m, text(L"y", 12, :left)), + ]) +end +plt + +#= function IVT_graph(n) f(x) = sin(pi*x) + 9x/10 a,b = [0,3] @@ -76,7 +133,17 @@ with $f(x)=y$. plotly() ImageFile(imgfile, caption) +=# ``` +```{julia} +#| echo: false +plotly() +nothing +``` + +Illustration of the intermediate value theorem. The theorem implies that any randomly chosen $y$ value between $f(a)$ and $f(b)$ will have at least one $c$ in $[a,b]$ with $f(c)=y$. This graphic shows one of several possible values for the given choice of $y$. + +::: In the early years of calculus, the intermediate value theorem was intricately connected with the definition of continuity, now it is a consequence. diff --git a/quarto/limits/limits.qmd b/quarto/limits/limits.qmd index 59c24b4..614319d 100644 --- a/quarto/limits/limits.qmd +++ b/quarto/limits/limits.qmd @@ -218,35 +218,64 @@ This bounds the expression $\sin(x)/x$ between $1$ and $\cos(x)$ and as $x$ gets The above bound comes from this figure, for small $x > 0$: +::: {#fig-sin-cos-bound} ```{julia} #| hold: true #| echo: false -gr() -p = plot(x -> sqrt(1 - x^2), 0, 1, legend=false, aspect_ratio=:equal, - linewidth=3, color=:black) -θ = π/6 -y,x = sincos(θ) -col=RGBA(0.0,0.0,1.0, 0.25) -plot!(range(0,x, length=2), zero, fillrange=u->y/x*u, color=col) -plot!(range(x, 1, length=50), zero, fillrange = u -> sqrt(1 - u^2), color=col) -plot!([x,x],[0,y], linestyle=:dash, linewidth=3, color=:black) -plot!([x,1],[y,0], linestyle=:dot, linewidth=3, color=:black) -plot!([1,1], [0,y/x], linewidth=3, color=:black) -plot!([0,1], [0,y/x], linewidth=3, color=:black) -plot!([0,1], [0,0], linewidth=3, color=:black) -Δ = 0.05 -annotate!([(0,0+Δ,"A"), (x-Δ,y+Δ/4, "B"), (1+Δ/2,y/x, "C"), - (1+Δ/2,0+Δ/2,"D")]) -annotate!([(.2*cos(θ/2), 0.2*sin(θ/2), "θ")]) -imgfile = tempname() * ".png" -savefig(p, imgfile) -caption = "Triangle ``ABD`` has less area than the shaded wedge, which has less area than triangle ``ACD``. Their respective areas are ``(1/2)\\sin(\\theta)``, ``(1/2)\\theta``, and ``(1/2)\\tan(\\theta)``. The inequality used to show ``\\sin(x)/x`` is bounded below by ``\\cos(x)`` and above by ``1`` comes from a division by ``(1/2) \\sin(x)`` and taking reciprocals. -" -plotly() -ImageFile(imgfile, caption) +plt = let + gr() + + empty_style = (xaxis=([], false), + yaxis=([], false), + framestyle=:origin, + legend=false) + axis_style = (arrow=true, side=:head, line=(:gray, 1)) + text_style = (10,) + fn_style = (;line=(:black, 3)) + fn2_style = (;line=(:red, 4)) + mark_style = (;line=(:gray, 1, :dot)) + domain_style = (;fill=(:orange, 0.35), line=nothing) + range_style = (; fill=(:blue, 0.35), line=nothing) + + plot(; empty_style..., aspect_ratio=:equal) + plot!(x -> sqrt(1 - x^2), 0, 1; line=(:black, 2)) + + + θ = π/6 + y,x = sincos(θ) + col=RGBA(0.0,0.0,1.0, 0.25) + + plot!(range(0,x, length=2), zero, fillrange=u->y/x*u, color=col) + plot!(range(x, 1, length=50), zero, fillrange = u -> sqrt(1 - u^2), color=col) + + plot!([x,x],[0,y], line=(:dash, 2, :black)) + plot!([x,1],[y,0], line=(:dot, 2, :black)) + plot!([1,1], [0,y/x], line=(2, :black)) + plot!([0,1], [0,y/x], line=(2, :black)) + plot!([0,1], [0,0], line=(2, :black)) + Δ = 0.05 + annotate!([(0,0+Δ, text(L"A", 10)), + (x-Δ,y+Δ/4, text(L"B",10)), + (1+Δ/2,y/x, text(L"C", 10)), + (1+Δ/2,0+Δ/2, text(L"D", 10)), + (0.2*cos(θ/2), 0.2*sin(θ/2), text(L"\theta", 12)) + ]) + current() +end +plt ``` +```{julia} +#| echo: false +plotly() +nothing +``` + +Triangle $\triangle ABD$ has less area than the shaded wedge, which has less area than triangle $\triangle ACD$. Their respective areas are $(1/2)\sin(\theta)$, $(1/2)\theta$, and $(1/2)\tan(\theta)$. The inequality used to show $\sin(x)/x$ is bounded below by $\cos(x)$ and above by $1$ comes from a division by $(1/2) \sin(x)$ and taking reciprocals. + +::: + To discuss the case of $(1+x)^{1/x}$ it proved convenient to assume $x = 1/m$ for integer values of $m$. At the time of Cauchy, log tables were available to identify the approximate value of the limit. Cauchy computed the following value from logarithm tables: diff --git a/quarto/precalc/functions.qmd b/quarto/precalc/functions.qmd index b51b5a2..2df9f4b 100644 --- a/quarto/precalc/functions.qmd +++ b/quarto/precalc/functions.qmd @@ -42,6 +42,101 @@ For these examples, the domain of both $f(x)$ and $g(x)$ is all real values of $ In general the range is harder to identify than the domain, and this is the case for these functions too. For $f(x)$ we may know the $\cos$ function is trapped in $[-1,1]$ and it is intuitively clear than all values in that set are possible. The function $h(x)$ would have range $[0,\infty)$. The $s(x)$ function is either $-1$ or $1$, so only has two possible values in its range. What about $g(x)$? It is a parabola that opens upward, so any $y$ values below the $y$ value of its vertex will not appear in the range. In this case, the symmetry indicates that the vertex will be at $(1/2, -1/4)$, so the range is $[-1/4, \infty)$. +::: {#fig-domain-range layout-nrow=2} +```{julia} +#| echo: false +plt = let + gr() + # domain/range shade + λ = 1.2 + a, b = .1, 3 + f(x) = (x-1/2) + sin((x-1)^2) + + empty_style = (xaxis=([], false), + yaxis=([], false), + framestyle=:origin, + legend=false) + axis_style = (arrow=true, side=:head, line=(:gray, 1)) + text_style = (10,) + fn_style = (;line=(:black, 3)) + fn2_style = (;line=(:red, 4)) + mark_style = (;line=(:gray, 1, :dot)) + + domain_style = (;fill=(:orange, 0.35)) + range_style = (; fill=(:blue, 0.35)) + + xs = range(a,b, 1000) + y0,y1 = extrema(f.(xs)) + Δy = (y1-y0)/60 + Δx = (b - a)/75 + + plot(; aspect_ratio=:equal, empty_style...) + plot!([-.25,3.25],[0,0]; axis_style...) + plot!([0,0], [min(-2Δy, y0 - Δy), y1 + 4Δy]; axis_style... ) + + plot!(f, a, b; fn_style...) + plot!(Shape([a,b,b,a], Δy * [-1,-1,1,1] ); domain_style...) + plot!(Shape(Δx*[-1,1,1,-1], [y0,y0,y1,y1]); range_style...) + + plot!([a,a], [0, f(a)]; mark_style...) + plot!([b,b], [0, f(b)]; mark_style...) + plot!([a, b], [f(a), f(a)]; mark_style...) + plot!([a, b], [f(b), f(b)]; mark_style...) +end +plot +``` + +```{julia} +#| echo: false +plt = let + a, b = 0, 2pi + λ = 1.1 + Δx, Δy = .033, .1 + δx = 0.05 + f(x) = sec(x) + g(x) = abs(f(x)) < 5 ? f(x) : NaN + + empty_style = (xaxis=([], false), + yaxis=([], false), + framestyle=:origin, + legend=false) + axis_style = (arrow=true, side=:head, line=(:gray, 1)) + text_style = (10,) + fn_style = (;line=(:black, 3)) + fn2_style = (;line=(:red, 4)) + mark_style = (;line=(:gray, 1, :dot)) + + domain_style = (;fill=(:orange, 0.35)) + range_style = (; fill=(:blue, 0.35)) + + + plot(; empty_style...) + plot!(g, a+0.1, b; fn_style...) + plot!(λ*[a,b],[0,0]; axis_style...) + plot!([0,0], λ*[-5,5+1/2]; axis_style...) + vline!([pi/2, 3pi/2]; line=(:gray, :dash)) + + plot!(Shape([a,a+pi/2-δx,a+pi/2-δx,a], Δy*[-1,-1,1,1]); domain_style...) + plot!(Shape([a+pi/2+δx, a+pi/2+pi-δx,a+pi/2+pi-δx,a+pi/2+δx], + Δy*[-1,-1,1,1]); domain_style...) + plot!(Shape([3pi/2 + δx, 2pi, 2pi, 3pi/2+δx], + Δy*[-1,-1,1,1]); domain_style...) + plot!(Shape(Δx*[-1,1,1,-1], [-5, -5,-1,-1]); range_style...) + plot!(Shape(Δx*[-1,1,1,-1], [ 5, 5,1,1]); range_style...) +end +plot +``` + + +```{julia} +#| echo: false +plotly() +nothing +``` + +The top figure shows the domain and range of the function as highlighted intervals. The bottom figure shows that the domain may be a collection of intervals. (In this case the $\sec$ function is not defined at $\pi/2 + k \pi$ for integer $k$) and the range may be a collection of intervals. (In this case, the $\sec$ function never have a value in $(-1,1)$. +::: + :::{.callout-note} ## Note @@ -90,7 +185,7 @@ Whereas function definitions and usage in `Julia` mirrors standard math notation ::: -### The domain of a function +### The domain of a function in Julia Functions in `Julia` have an implicit domain, just as they do mathematically. In the case of $f(x)$ and $g(x)$, the right-hand side is defined for all real values of $x$, so the domain is all $x$. For $h(x)$ this isn't the case, of course. Trying to call $h(x)$ when $x < 0$ will give an error: @@ -347,10 +442,27 @@ In our example, we see that in trying to find an answer to $f(x) = 0$ ( $\sqrt{2 ```{julia} #| echo: false -plot(q, a, b, linewidth=5, legend=false) -plot!(zero, a, b) -plot!([a, b], q.([a, b])) -scatter!([c], [q(c)]) +plt = let +gr() +plt = plot(q, a, b, linewidth=5, legend=false) +plot!(plt, zero, a, b) +plot!(plt, [a, b], q.([a, b])) +scatter!(plt, [c], [q(c)]; marker=(:circle,)) +scatter!(plt, [a,b], [q(a), q(b)]; marker=(:square,)) + +annotate!(plt, [ +(a, 0, text(L"a", 10,:top)), +(b, 0, text(L"b", 10, :top)), +(c, 0, text(L"c", 10, :bottom)) +]) +end +plt +``` + +```{julia} +#| echo: false +plotly() +nothing ``` Still, `q(c)` is not really close to $0$: @@ -398,6 +510,7 @@ c = secant_intersection(f, a, b) p = plot(f, a, b, linewidth=5, legend=false) plot!(p, zero, a, b) +scatter!([a,b], [f(a), f(b)]; marker=(:square,)) plot!(p, [a,b], f.([a,b])); scatter!(p, [c], [f(c)]) @@ -918,6 +1031,86 @@ answ = 1 radioq(choices, answ) ``` +##### Question + +::: {#fig-floor-function} + +```{julia} +#| echo: false +plt = let + gr() + empty_style = (xaxis=([], false), + yaxis=([], false), + framestyle=:origin, + legend=false) + axis_style = (arrow=true, side=:head, line=(:gray, 1)) + text_style = (10,) + fn_style = (;line=(:black, 3)) + fn2_style = (;line=(:red, 4)) + mark_style = (;line=(:gray, 1, :dot)) + domain_style = (;fill=(:orange, 0.35), line=nothing) + range_style = (; fill=(:blue, 0.35), line=nothing) + + ts = range(0, 2pi, 100) + xys = sincos.(ts) + xys = [.1 .* xy for xy in xys] + + plot(; empty_style..., aspect_ratio=:equal) + plot!([-4.25,4.25], [0,0]; axis_style...) + plot!([0,0], [-4.25, 4.25]; axis_style...) + + for k in -4:4 + P,Q = (k,k),(k+1,k) + plot!([P,Q], line=(:black,1)) + S = Shape([k .+ xy for xy in xys]) + plot!(S; fill=(:black,)) + S = Shape([(k+1,k) .+ xy for xy in xys]) + plot!(S; fill=(:white,), line=(:black,1)) + end + current() +end +plotly() +plt +``` + +The `floor` function rounds down. For example, any value in $[k,k+1)$ rounds to $k$ for integer $k$. +::: + +The figure shows the `floor` function which is useful in programming. It rounds down to the first integer value. + + +From the graph, what is the domain of the function? + + +```{julia} +#| hold: true +#| echo: false +choices = [ + "The entire real line", + "The entire real line except for integer values", + "The integers" +] +answer = 1 +radioq(choices, answ) +``` + +From the graph, what is the range of the function? + + +```{julia} +#| hold: true +#| echo: false +choices = [ + "The entire real line", + "The entire real line except for integer values", + "The integers" +] +answer = 3 +radioq(choices, answ) +``` + +(This graphic uses the convention that a filled in point is present, but an open point is not, hence each bar represents some $[k, k+1)$.) + ###### Question diff --git a/quarto/precalc/inversefunctions.qmd b/quarto/precalc/inversefunctions.qmd index 324a4da..822b3cf 100644 --- a/quarto/precalc/inversefunctions.qmd +++ b/quarto/precalc/inversefunctions.qmd @@ -30,20 +30,57 @@ Why is this useful? When available, it can help us solve equations. If we can wr Let's explore when we can "solve" for an inverse function. -Consider the graph of the function $f(x) = 2^x$: - +Consider this graph of the function $f(x) = 2^x$ ```{julia} -#| hold: true -f(x) = 2^x -plot(f, 0, 4, legend=false) -plot!([2,2,0], [0,f(2),f(2)]) +#| echo: false +p = let + gr() + empty_style = (xaxis=([], false), + yaxis=([], false), + framestyle=:origin, + legend=false) + axis_style = (arrow=true, side=:head, line=(:gray, 1)) + text_style = (10,) + fn_style = (;line=(:black, 3)) + fn2_style = (;line=(:red, 4)) + mark_style = (;line=(:gray, 1, :dot)) + domain_style = (;fill=(:orange, 0.35), line=nothing) + range_style = (; fill=(:blue, 0.35), line=nothing) + + f(x) = 2^x + a, b = 0, 2 + plot(; empty_style...) + xs = range(a, b, 200) + ys = f.(xs) + + plot!( xs, ys; fn_style...) + plot!([a-1/4, b+.2], [0,0]; axis_style...) + plot!([0, 0], [-.1, f(2.1)]; axis_style...) + + x = 1 + y = (f(b)+f(a))/2 + plot!([x,x,0],[0,f(x),f(x)]; line=(:black, 1, :dash), arrow=true, side=:head) + plot!([0,log2(y), log2(y)], [y,y,0]; line=(:black,1,:dash), arrow=true, side=:head) + + annotate!([ + (x, 0, text(L"c", 10, :top)), + (0,f(x), text(L"f(c)", 10, :right)), + (0, y, text(L"y=f(d)", 10, :right)), + (log2(y), 0, text(L"d", 10, :top)) + + ]) + + +end +plotly() +p ``` The graph of a function is a representation of points $(x,f(x))$, so to *find* $y = f(c)$ from the graph, we begin on the $x$ axis at $c$, move vertically to the graph (the point $(c, f(c))$), and then move horizontally to the $y$ axis, intersecting it at $y = f(c)$. The figure shows this for $c=2$, from which we can read that $f(c)$ is about $4$. This is how an $x$ is associated to a single $y$. -If we were to *reverse* the direction, starting at $y = f(c)$ on the $y$ axis and then moving horizontally to the graph, and then vertically to the $x$-axis we end up at a value $c$ with the correct $f(c)$. This allows solving for $x$ knowing $y$ in $y=f(x)$. +If we were to *reverse* the direction, starting at $y = f(d)$ on the $y$ axis and then moving horizontally to the graph, and then vertically to the $x$-axis we end up at a value $d$ with the correct $f(d)$. This allows solving for $x$ knowing $y$ in $y=f(x)$. The operation described will form a function **if** the initial movement horizontally is guaranteed to find *no more than one* value on the graph. That is, to have an inverse function, there can not be two $x$ values corresponding to a given $y$ value. This observation is often visualized through the "horizontal line test" - the graph of a function with an inverse function can only intersect a horizontal line at most in one place. @@ -309,6 +346,131 @@ What do we see? In blue, we can see the familiar square root graph along with a This is reminiscent of the formula for the slope of a perpendicular line, $-1/m$, but quite different, as this formula implies the two lines have either both positive slopes or both negative slopes, unlike the relationship in slopes between a line and a perpendicular line. +::: {#fig-inverse-normal layout-ncol=1} +```{julia} +#| echo: false +# inverse function slope +gr() +p1 = let + f(x) = x^2 + df(x) = 2x + + empty_style = (xaxis=([], false), + yaxis=([], false), + framestyle=:origin, + legend=false) + axis_style = (arrow=true, side=:head, line=(:gray, 1)) + text_style = (10,) + fn_style = (;line=(:black, 3)) + fn2_style = (;line=(:red, 4)) + mark_style = (;line=(:gray, 1, :dot)) + + plot(; aspect_ratio=:equal, empty_style...) + xs = range(0, 1.25, 100) + plot!(xs,f.(xs); fn_style...) + plot!(f.(xs), xs; fn2_style...) + plot!(identity, -1/4, 2; line=(:gray, 1, :dot)) + + #plot!([-.1, 1.35],[0,0]; axis_style...) + #plot!([0,0], [-0.1, f(1.3)]; axis_style...) + + c = .4 + m = df(c) + tl(x) = f(c) + df(c)*(x-c) + plot!(tl; line=(:black, 1, :dash)) + d = c + .6 + p1, p2, p3 = (c, tl(c)), (d, tl(c)), (d, tl(d)) + q1, q2, q3 = (tl(c),c), (tl(c),d), (tl(d), d) + + plot!([p1, p2, p3]; line=(:black, 1, :dot)) + + + tl1(x) = c + (1/m)*(x - f(c)) + plot!(tl1; line=(:red, 1, :dash)) + plot!([q1, q2, q3]; line=(:red, 1, :dot)) + + annotate!([ + ((c+d)/2, f(c), text(L"\Delta x", 10, :top, :black)), + (d, (tl(c)+tl(d))/2, text(L"\Delta y", 10, :left, :black)), + (f(c), (c+d)/2, text(L"\Delta x", 10, :right, :red)), + ((tl(c)+tl(d))/2, d, text(L"\Delta y", 10, :bottom, :red)), + (d, tl(d), + text(L"rise/run = $m = \Delta y / \Delta x$", 10, :top, :left, + rotation= rad2deg(atan(m)))), + (tl(d), d, + text(L"rise/run = $\Delta x / \Delta y = 1/m$", 10, :bottom, :left, rotation=rad2deg(atan(1/m)))), + (1.9, 1.9, text(L"y=x", 10, :top, rotation=45)) + + + + ]) + current() +end + +# normal line + +p2 = let + f(x) = 4 - (x-2)^2 + df(x) = -2*(x-2) + + empty_style = (xaxis=([], false), + yaxis=([], false), + framestyle=:origin, + legend=false) + axis_style = (arrow=true, side=:head, line=(:gray, 1)) + text_style = (10,) + fn_style = (;line=(:black, 3)) + fn2_style = (;line=(:red, 4)) + mark_style = (;line=(:gray, 1, :dot)) + + plot(; aspect_ratio=:equal, empty_style..., + xlims=(1, 2.5), ylims=(3, 4.5)) + xs = range(.99, 2.01, 100) + plot!(xs,f.(xs); fn_style...) + + c = 1.5 + tl(x) = f(c) + df(c)*(x-c) + nl(x) = f(c) - (x-c)/df(c) + + xs = range(1, 2, 10) + plot!(xs, tl.(xs); fn2_style...) + + xs = range(.9, 2, 10) + plot!(xs, nl.(xs); fn2_style...) + + ylims!((f(.85), nl(.95))) + + o = 1/3 + plot!([c,c+o, c+o], [tl(c), tl(c), tl(c+o)]; mark_style...) + m = (tl(c+o) - tl(c)) + plot!([c,c,c+m], [nl(c),nl(c + m),nl(c+m)]; mark_style...) + + theta = rad2deg(atan(tl(c+o)-tl(c), o)) + annotate!([ + (c + o/2, f(c), text(L"1", :top, 10)), + (c + o, (f(c)+f(c+o))/2, text(L"m", :right, 10)), + (c, (nl(c) + nl(c+m))/2, text(L"-1", :right, 10)), + (c+m/2, nl(c+m), text(L"m", :top, 10)), + (c + o/2, tl(c+o), text(L"rise/run $=m/1$", 10, :top, + rotation=theta)), + (c + 1.1*o, nl(c+1.1*o), text(L"rise/run $=(-1)/m$", 10, :bottom, + rotation=theta-90)) + ]) + current() +end + +plot(p1, p2) +``` + +The inverse function has slope at a corresponding point that is the *reciprocal*; the "normal line" for a function at a point has slope that is the *negative reciprocal* of the "tangent line" at a point. + +::: + +```{julia} +#| echo: false +plotly() +nothing +``` The key here is that the shape of $f(x)$ near $x=c$ is somewhat related to the shape of $f^{-1}(x)$ at $f(c)$. In this case, if we use the tangent line as a fill in for how steep a function is, we see from the relationship that if $f(x)$ is "steep" at $x=c$, then $f^{-1}(x)$ will be "shallow" at $x=f(c)$. diff --git a/quarto/precalc/plotting.qmd b/quarto/precalc/plotting.qmd index c98c2dd..4218c76 100644 --- a/quarto/precalc/plotting.qmd +++ b/quarto/precalc/plotting.qmd @@ -108,11 +108,6 @@ Plotting a function is then this simple: `plot(f, xmin, xmax)`. -:::{.callout-note} -## Note -The time to first plot can feel sluggish, but subsequent plots will be speedy. See the technical note at the end of this section for an explanation. - -::: Let's see some other graphs. @@ -193,7 +188,70 @@ Some types we will encounter, such as the one for symbolic values or the special ::: ---- +:::{.callout-note} +## Viewing window + +The default style for `Plots.jl` is to use a frame style where the viewing window is emphasized. This is a rectangular region, $[x_0, x_1] \times [y_0, y_1]$, which is seen through the tick labeling, the bounding scales on the left and bottom, and emphasized through the grid. + +This choices does *not* show the $x-y$ axes. As such, we might layer on the axes when these are of interest. + +To emphasize concepts, we may stylize a function graph, rather than display the basic graphic. For example, in this graphic highlighting the amount the function goes up as it moves from $1$ to $x$: + +```{julia} +gr() +#| echo: false +let + f(x) = x^2 + + empty_style = (xaxis=([], false), + yaxis=([], false), + framestyle=:origin, + legend=false) + axis_style = (arrow=true, side=:head, line=(:gray, 1)) + text_style = (10,) + fn_style = (;line=(:black, 3)) + fn2_style = (;line=(:red, 4)) + mark_style = (;line=(:gray, 1, :dot)) + + plot(; empty_style..., aspect_ratio=:equal) + a, b = 0, 1.25 + x = 1.15 + + plot!(f, a, b; fn_style...) + plot!([-.1, 1.5], [0,0]; axis_style...) + plot!([0,0], [-.1, f(1.35)]; axis_style...) + + plot!([1,x,x], [f(1),f(1),f(x)]; line=(:black, 1)) + plot!([1,1],[0,f(1)]; mark_style...) + plot!([x,x],[0,f(1)]; mark_style...) + plot!([0,1],[f(1),f(1)]; mark_style...) + plot!([0,x],[f(x),f(x)]; mark_style...) + + annotate!([ + (1, 0, text(L"1", 10, :top)), + (x, 0, text(L"x", 10, :top)), + (0, f(1), text(L"1", 10, :right)), + (0, f(x), text(L"x^2", 10, :right)), + (1, f(1), text(L"P", 10, :right, :bottom)), + (x, f(x), text(L"Q", 10, :right, :bottom)), + ((1 + x)/2, f(1), text(L"\Delta x", 10, :top)), + (x, (f(1) + f(x))/2, text(L"\Delta y", 10, :left)) + ]) + current() +end +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + +::: + + + +---- Making a graph with `Plots` is easy, but producing a graph that is informative can be a challenge, as the choice of a viewing window can make a big difference in what is seen. For example, trying to make a graph of $f(x) = \tan(x)$, as below, will result in a bit of a mess - the chosen viewing window crosses several places where the function blows up: @@ -699,6 +757,40 @@ plot(f, g, 0, max((R-r)/r, r/(R-r))*2pi) In the above, one can fix $R=1$. Then different values for `r` and `rho` will produce different graphs. These graphs will be periodic if $(R-r)/r$ is a rational. (Nothing about these equations requires $\rho < r$.) +## Points, lines, polygons + +Two basic objects to graph are points and lines. + +A point in two-dimensional space has two coordinates, often denoted by $(x,y)$. In `Julia`, the same notation produces a `tuple`. Using square brackets, as in `[x,y]`, produces a vector. Vectors are usually used, as we have seen there are algebraic operations defined for them. However, tuples have other advantages and are how `Plots` designates a point. + +The plot command `plot(xs, ys)` plots the points $(x_1,y_1), \dots, (x_n, y_n)$ and then connects adjacent points with with lines. The command `scatter(xs, ys)` just plots the points. + +However, the points might be more naturally specified as coordinate pairs. If tuples are used to pair them off, then `Plots` will plot a vector of tuples as a sequence of points: + +```{julia} +pts = [(1, 0), (1/4, 1/4), (0, 1), (-1/4, 1/4), (-1, 0), + (-1/4, -1/4), (0, -1), (1/4, -1/4)] +scatter(pts; legend=false) +``` + +A line segment simply connects two points. While these can be specified as vectors of $x$ and $y$ values, again it may be more convenient to use coordinate pairs to specify the points. Continuing the above, we can connect adjacent points with line segments: + +```{julia} +plot!(pts; line=(:gray, 0.5, :dash)) +``` + +This uses the shorthand notation of `Plots` to specify `linecolor=:gray, linealpha=0.5, linestyle=:dash`. To plot just a line segment, just specifying two points suffices. + +The four-pointed star is not closed off, as there isn't a value from the last point to the first point. A polygon closes itself off. The `Shape` function can take a vector of points or a pair of `xs` and `ys` to specify a polygon. When these are plotted, the arguments to `fill` describe the interior of the polygon, the arguments to `line` the boundary: + + +```{julia} +plot(Shape(pts); fill=(:gray, 0.25), line=(:black, 2), legend=false) +scatter!(pts) +``` + + + ## Questions diff --git a/quarto/references.bib b/quarto/references.bib index b096c05..a324c0b 100644 --- a/quarto/references.bib +++ b/quarto/references.bib @@ -18,6 +18,15 @@ +@Misc{Angenent, + key = {WisconsinCalculus}, + author = {Sigurd Angenent}, + title = {Wisconsin Calculus}, + howpublished = {https://github.com/SigurdAngenent/WisconsinCalculus/tree/master}, + year = 2012, + note = {GNU Free Documentation License, Version 1.2} + } + @Book{Schey, author = {H.M. Schey}, title = {Div, Grad, Curl, and all that}, From 6c108037122ced1e653ac8aef89898f058821458 Mon Sep 17 00:00:00 2001 From: jverzani Date: Wed, 2 Jul 2025 11:09:37 -0400 Subject: [PATCH 15/22] modify adjust_plotly --- quarto/adjust_plotly.jl | 15 +++++++++------ 1 file changed, 9 insertions(+), 6 deletions(-) diff --git a/quarto/adjust_plotly.jl b/quarto/adjust_plotly.jl index c646e5e..ec06770 100644 --- a/quarto/adjust_plotly.jl +++ b/quarto/adjust_plotly.jl @@ -7,16 +7,19 @@ #alternatives/plotly_plotting.html function _add_plotly(f) - lineno = 117 + #lineno = 117 -str = """ - -""" r = readlines(f) + inserted = false open(f, "w") do io for (i,l) ∈ enumerate(r) - i == lineno && println(io, str) + if contains(l, "require.min.js") + !inserted && println(io, """ + +""") + inserted = true + end println(io, l) end end @@ -29,7 +32,7 @@ function (@main)(args...) for fᵢ ∈ files f = joinpath(root, fᵢ) if endswith(f, ".html") - @show :adjust, f + dirname(f) == "_book" && continue _add_plotly(f) end end From c38a7c9f1d1587457899c3356ba9dc8eea5027b4 Mon Sep 17 00:00:00 2001 From: jverzani Date: Wed, 2 Jul 2025 14:05:09 -0400 Subject: [PATCH 16/22] WIP; better figures --- quarto/adjust_plotly.jl | 1 - quarto/integrals/arc_length.qmd | 31 ++- quarto/integrals/area.qmd | 69 ++++++- quarto/integrals/twelve-qs.qmd | 40 ++-- quarto/integrals/volumes_slice.qmd | 290 ++++++++++++++++++++++++++--- 5 files changed, 380 insertions(+), 51 deletions(-) diff --git a/quarto/adjust_plotly.jl b/quarto/adjust_plotly.jl index ec06770..91bac71 100644 --- a/quarto/adjust_plotly.jl +++ b/quarto/adjust_plotly.jl @@ -9,7 +9,6 @@ function _add_plotly(f) #lineno = 117 - r = readlines(f) inserted = false open(f, "w") do io diff --git a/quarto/integrals/arc_length.qmd b/quarto/integrals/arc_length.qmd index a3d6e5b..e74509a 100644 --- a/quarto/integrals/arc_length.qmd +++ b/quarto/integrals/arc_length.qmd @@ -76,21 +76,34 @@ To see why, any partition of the interval $[a,b]$ by $a = t_0 < t_1 < \cdots < t ## {{{arclength_graph}}} gr() function make_arclength_graph(n) + x(t) = cos(t)/t + y(t) = sin(t)/t + a, b = 1, 4pi + ns = [10,15,20, 30, 50] + empty_style = (xaxis=([], false), + yaxis=([], false), + framestyle=:origin, + legend=false) - ns = [10,15,20, 30, 50] + plot(; empty_style..., aspect_ratio=:equal, size=fig_size) + title!("Approximate arc length with $(ns[n]) points") - g(t) = cos(t)/t - f(t) = sin(t)/t + ts = range(a, b, 250) + plot!(x.(ts), y.(ts); line=(:black,2)) + pttn = range(a, b, ns[n]) + plot!(x.(pttn), y.(pttn); line=(:red, 2)) - ts = range(1, stop=4pi, length=200) - tis = range(1, stop=4pi, length=ns[n]) + ts = range(0, 2pi, 100) - p = plot(g, f, 1, 4pi, legend=false, size=fig_size, - title="Approximate arc length with $(ns[n]) points") - plot!(p, map(g, tis), map(f, tis), color=:orange) + λ = 0.005 + cs = [λ .* xys for xys ∈ sincos.(ts)] - p + for v ∈ zip(x.(pttn), y.(pttn)) + S = Shape([v .+ xy for xy in cs]) + plot!(S; fill=(:white,), line=(:black,2)) + end + current() end n = 5 diff --git a/quarto/integrals/area.qmd b/quarto/integrals/area.qmd index 8cbe09d..b5701a1 100644 --- a/quarto/integrals/area.qmd +++ b/quarto/integrals/area.qmd @@ -190,6 +190,40 @@ Clearly for a given partition and choice of $c_i$, the above can be computed. Ea #| hold: true #| echo: false gr() + +function left_riemann(n) + empty_style = (xaxis=([], false), + yaxis=([], false), + framestyle=:origin, + legend=false) + axis_style = (arrow=true, side=:head, line=(:gray, 1)) + rectangle = (x, y, w, h) -> Shape(x .+ [0,w,w,0], y .+ [0,0,h,h]) + + f = x -> -(x+1/2)*(x-1)*(x-3) + 1 + a, b= 1, 3 + + plot(; empty_style...) + plot!(f, a, b; line=(:black, 3)) + plot!([a-.25, b+.25], [0,0]; axis_style...) + plot!([a-.1, a-.1], [-.25, .5 + f(a/2 +b/2)]; axis_style...) + + Δ = (b-a)/n + for i ∈ 0:n-1 + xᵢ = a + i*Δ + plot!(rectangle(xᵢ, 0, Δ, f(xᵢ)), opacity=0.5, color=:red) + end + area = round(sum(f(a + i*Δ)*Δ for i ∈ 0:n-1), digits=3) + + annotate!([ + (a, 0, text(L"a", :top)), + (b, 0, text(L"b", :top)), + (a, f(a/2+b/2), text("\$L_{$n} = $area\$", :left)) + ]) + + current() +end + +#= rectangle(x, y, w, h) = Shape(x .+ [0,w,w,0], y .+ [0,0,h,h]) function ₙ(j) a = ("₋","","","₀","₁","₂","₃","₄","₅","₆","₇","₈","₉") @@ -210,6 +244,7 @@ function left_riemann(n) title!("L$(ₙ(n)) = $a") p end +=# anim = @animate for i ∈ (2,4,8,16,32,64) left_riemann(i) @@ -631,19 +666,19 @@ Before continuing, we define a function to compute Riemann sums for us with an ```{julia} #| eval: false -riemann(f, a, b, n; method="right") = riemann(f, range(a,b,n+1); method=method) function riemann(f, xs; method="right") - Ms = (left = (f,a,b) -> f(a), - right= (f,a,b) -> f(b), + Ms = (left = (f,a,b) -> f(a), + right = (f,a,b) -> f(b), trapezoid = (f,a,b) -> (f(a) + f(b))/2, - simpsons = (f,a,b) -> (c = a/2 + b/2; (1/6) * (f(a) + 4*f(c) + f(b))), + simpsons = (f,a,b) -> (c = a/2 + b/2; (1/6) * (f(a) + 4*f(c) + f(b))) ) - _riemann(Ms[Symbol(method)], f, xs) -end -function _riemann(M, f, xs) + M = Ms[Symbol(method)} xs′ = zip(xs[1:end-1], xs[2:end]) sum(M(f, a, b) * (b-a) for (a,b) ∈ xs′) end + +riemann(f, a, b, n; method="right") = + riemann(f, range(a,b,n+1); method) ``` (This function is defined in `CalculusWithJulia` and need not be copied over if that package is loaded.) @@ -699,7 +734,7 @@ Consider a function $g(x)$ defined through its piecewise linear graph: ```{julia} #| echo: false g(x) = abs(x) > 2 ? 1.0 : abs(x) - 1.0 -plot(g, -3,3) +plot(g, -3,3; legend=false) plot!(zero) ``` @@ -710,6 +745,24 @@ plot!(zero) We could add the signed area over $[0,1]$ to the above, but instead see a square of area $1$, a triangle with area $1/2$ and a triangle with signed area $-1$. The total is then $1/2$. +This figure may make the above decomposition more clear: + +```{julia} +#| echo: false +let + g(x) = abs(x) > 2 ? 1.0 : abs(x) - 1.0 + xs = [ -3, -2, -1, 1, 2, 3] + plot(; legend=false, aspect_ratio=:equal) + plot!(Shape([-3,-2,-2,-3], [0,0,1,1]); fill=(:gray,)) + plot!(Shape([-2,-1,-1,-2], [0,0,0,1]); fill=(:gray10,)) + plot!(Shape([-1,0,1], [0,-1,0]); fill=(:gray90,)) + plot!(Shape([1,2,2,1], [0,1,0,0]); fill=(:gray10,)) + plot!(Shape([2,3,3,2], [0,0,1,1]); fill=(:gray,)) +end +``` + + + * Compute $\int_{-3}^{3} g(x) dx$: diff --git a/quarto/integrals/twelve-qs.qmd b/quarto/integrals/twelve-qs.qmd index 4622360..f8a9485 100644 --- a/quarto/integrals/twelve-qs.qmd +++ b/quarto/integrals/twelve-qs.qmd @@ -6,7 +6,12 @@ This section uses these packages: using SymPy using Plots using Roots -plotly() +``` + +```{julia} +#| echo: false +using LaTeXStrings +gr(); ``` ---- @@ -15,21 +20,31 @@ In the March 2003 issue of the College Mathematics Journal, Leon M Hall posed 12 ```{julia} #| echo: false -f(x) = x^2 -fp(x) = 2x a₀ = 7/8 q₀ = -a₀ - 1/(2a₀) +f(x) = x^2 +fp(x) = 2x tangent(x) = f(a₀) + fp(a₀) * (x - a₀) normal(x) = f(a₀) - (1 / fp(a₀)) * (x - a₀) -function make_plot(a₀=7/8, q₀=-a₀ - 1/2a₀) - plt = plot(; xlim=(-2,2), ylim=(-1, (1.5)^2), - xticks=nothing, yticks=nothing, - aspect_ratio=:equal, border=:none, legend=false) + +function make_plot() + empty_style = (xaxis=([], false), + yaxis=([], false), + framestyle=:origin, + legend=false) + axis_style = (arrow=true, side=:head, line=(:gray, 1)) + + tangent(x) = f(a₀) + fp(a₀) * (x - a₀) + normal(x) = f(a₀) - (1 / fp(a₀)) * (x - a₀) + + plt = plot(; empty_style..., + xlims=(-2,2), ylims=(-1, (1.5)^2)) + f(x) = x^2 fp(x) = 2x - plot!(f, -1.5, 1.5, line=(1, :royalblue)) - plot!(zero, line=(1, :black)) + plot!(f, -1.5, 1.5, line=(2, :black)) + plot!([-1.6, 1.6], [0,0]; axis_style...) tl = x -> f(a₀) + fp(a₀) * (x-a₀) nl = x -> f(a₀) - 1/(fp(a₀)) * (x-a₀) @@ -40,9 +55,10 @@ function make_plot(a₀=7/8, q₀=-a₀ - 1/2a₀) # add in right triangle scatter!([a₀, q₀], f.([a₀, q₀]), markersize=5) Δ = 0.01 - annotate!([(a₀ + Δ, nl(a₀+Δ), "P", :bottom), - (q₀ - Δ, nl(q₀-Δ), "Q", :top)]) - plt + annotate!([(a₀ + Δ, nl(a₀+Δ), text(L"P", :top)), + (q₀ - Δ, nl(q₀-Δ), text(L"Q", :bottom, :left)) + ]) + current() end make_plot() ``` diff --git a/quarto/integrals/volumes_slice.qmd b/quarto/integrals/volumes_slice.qmd index 4b4d0d7..80d466f 100644 --- a/quarto/integrals/volumes_slice.qmd +++ b/quarto/integrals/volumes_slice.qmd @@ -680,9 +680,93 @@ For a general cone, we use this [definition](http://en.wikipedia.org/wiki/Cone): Let $h$ be the distance from the apex to the base. Consider cones with the property that all planes parallel to the base intersect the cone with the same shape, though perhaps a different scale. This figure shows an example, with the rays coming from the apex defining the volume. +::: {#fig-generic-cone} +```{julia} +#| echo: false +plt = let + gr() + rad(t) = 3/2 - t + viewp = [2,-1/1.5,1/2+.2] + + empty_style = (xaxis=([], false), + yaxis=([], false), + framestyle=:origin, + legend=false) + axis_style = (arrow=true, side=:head, line=(:gray, 1)) + unitize(x) = x / norm(x) + + """Orthogonal projection along the vector viewp""" + function make_Pmat(viewp) + a = unitize( [-viewp[2], viewp[1], 0] ) + b = unitize( [-viewp[3]*viewp[1], + -viewp[3]*viewp[2], + viewp[1]^2 + viewp[2]^2] ) + collect(zip(a,b)) + end + + #linear projection of R^3 onto R^2 + function proj(X, viewp) + Pmat = make_Pmat(viewp) + x=sum([Pmat[i][1]*X[i] for i in 1:3]) + y=sum([Pmat[i][2]*X[i] for i in 1:3]) + (x, y) # a point + end + proj(X) = proj(X, viewp) + + drawdiscF(t) = Shape([psurf(t, 2*i*pi/100) for i in 1:101]) + + # discrete determinant of Jacobian; area multiplier? + function jac(X, u, v) + ϵ = 0.000001 + A = map((p,q) -> (p-q)/ϵ, X(u+ϵ/2, v), X(u-ϵ/2, v)) + B = map((p,q) -> (p-q)/ϵ, X(u, v+ϵ/2), X(u, v-ϵ/2)) + return A[1]*B[2]-A[2]*B[1] + end + + + # our surface + R, r, rho = 1, 1/4, 1/4 + f(t) = (R-r) * cos(t) + rho * cos((R-r)/r * t) + g(t) = (R-r) * sin(t) - rho * sin((R-r)/r * t) + surf(t, θ) = (rad(t)*f(θ), rad(t)*g(θ), t) + psurf(t,θ) = proj(surf(t,θ)) + + plot(; empty_style..., aspect_ratio=:equal) + for (i,t) in enumerate(range(0, 3/2, 30)) + plot!(drawdiscF(t); fill=(:gray,1), line=(:black,1)) + end + + + θ = 0; plot!([psurf(0, θ), psurf(3/2, θ)]; line=(:black, 2)) + θ = pi/2; plot!([psurf(0, θ), psurf(3/2, θ)]; line=(:black, 1)) + θ = 3pi/2; plot!([psurf(0, θ), psurf(3/2, θ)]; line=(:black, 1)) + + + current() +end +plt +``` ```{julia} #| echo: false +plotly() +nothing +``` + + +A "cone" formed from the parameterized curve +$r(t) = \langle +(R-r) \cdot \cos(t) + \rho \cdot \cos((R-r)/r \cdot t), +(R-r) \cdot \sin(t) - \rho \cdot \sin((R-r)/r \cdot t) +\rangle$ with apex at the point $[0,0,3/2]$ and rays extending down through the origin following $3/2-z$. + +::: + + + +```{julia} +#| echo: false +#| eval: false plt = let h = 5 R, r, rho = 1, 1/4, 1/4 @@ -696,7 +780,7 @@ plt = let end current() end -plot +plt ``` A right circular cone is one where this shape is a circle. This definition can be more general, as a square-based right pyramid is also such a cone. After possibly reorienting the cone in space so the base is at $u=0$ and the apex at $u=h$ the volume of the cone can be found from: @@ -736,6 +820,11 @@ plt = let r(h,z) = (x(h,z), y(h,z)) r1(h,z) = (2,0) .+ r(h,z) + empty_style = (xaxis=([], false), + yaxis=([], false), + framestyle=:origin, + legend=false) + Nh=30 heights = range(-1/2, 1/2, Nh) h0=heights[Nh ÷ 2] @@ -779,7 +868,8 @@ end plt ``` -This figure shows the volume of a figure being comprised of slices. A discrete approximation would be found by estimating the volume of each slice by the cross sectional area times a small $\Delta h$. +This figure shows the volume of a figure being comprised of slices. A discrete approximation would be found by estimating the volume of each slice by the cross sectional area times a small $\Delta h$. This leads to a formula +$V = \int_a^b A(h)dh$, where $A$ computes the cross sectional area. (This figure was ported from @Angenent.) ::: @@ -836,39 +926,197 @@ Illustration of Cavalieri's first principle. The discs from the left are moved a ::: -With the formula for the volume of solids based on cross sections, this is a trivial observation, as the functions giving the cross-sectional area are identical. Still, it can be surprising. Consider a sphere with an interior cylinder bored out of it. (The [Napkin](http://tinyurl.com/o237v83) ring problem.) The bore has height $h$ - for larger radius spheres this means very wide bores. +With the formula for the volume of solids based on cross sections, this is a trivial observation, as the functions giving the cross-sectional area are identical. Still, it can be surprising. +Consider a sphere with an interior cylinder bored out of it. (The [Napkin](http://tinyurl.com/o237v83) ring problem.) The bore has height $h$ - for larger radius spheres this means very wide bores. + +::: {#fig-napkin-ring-1} +```{julia} +#| echo: false +plt = let + # Follow lead of # https://github.com/SigurdAngenent/WisconsinCalculus/blob/master/figures/221/09surf_of_rotation2.py + # plot surface of revolution around x axis between [0, 3] + # best if r(t) descreases + + rad(t) = (t = clamp(t, -1, 1); sqrt(1 - t^2)) + rad2(t) = 1/2 + viewp = [2,-2,1] + + ## + unitize(x) = x / norm(x) + + """Orthogonal projection along the vector viewp""" + function make_Pmat(viewp) + a = unitize( [-viewp[2], viewp[1], 0] ) + b = unitize( [-viewp[3]*viewp[1], + -viewp[3]*viewp[2], + viewp[1]^2 + viewp[2]^2] ) + collect(zip(a,b)) + end + + #linear projection of R^3 onto R^2 + function proj(X, viewp) + Pmat = make_Pmat(viewp) + x=sum([Pmat[i][1]*X[i] for i in 1:3]) + y=sum([Pmat[i][2]*X[i] for i in 1:3]) + (x, y) # a point + end + proj(X) = proj(X, viewp) + + # discrete determinant of Jacobian; area multiplier? + function jac(X, u, v) + ϵ = 0.000001 + A = map((p,q) -> (p-q)/ϵ, X(u+ϵ/2, v), X(u-ϵ/2, v)) + B = map((p,q) -> (p-q)/ϵ, X(u, v+ϵ/2), X(u, v-ϵ/2)) + return A[1]*B[2]-A[2]*B[1] + end + + + # surface of revolution about the z axis + + surf(t, z) = (rad(t)*cos(z), rad(t)*sin(z), t) + surf2(t, z) = (rad2(t)*cos(z), rad2(t)*sin(z), t) + + # project the surface at (t, a=theta) + psurf(t,z) = proj(surf(t,z)) + psurf2(t, z) = proj(surf2(t,z)) + + bisect(f, a, b) = find_zero(f, (a,b), Bisection()) + _fold(t, zmin, zmax) = bisect(z -> jac(psurf, t, z), zmin, zmax) + _foldz(z, tmin, tmax) = bisect(t -> jac(psurf, t, z), tmin, tmax) + # create shape holding project disc + drawdiscF(t) = Shape([psurf(t, 2*i*pi/100) for i in 1:101]) + drawdiscI(t) = Shape([psurf2(t, 2*i*pi/100) for i in 1:101]) + + # project a line between two points + pline!(p,q; kwargs...) = plot!([proj(p),proj(q)]; + line_style..., kwargs...) + α = 1.0 + line_style = (; line=(:black, 1)) + + plot(; empty_style..., aspect_ratio=:equal) + + # washer + t0 = sqrt(3/4) + + Δ = .03 + δ = 0.785398 + 0.05 + + x₀ = -.25 + plot!(drawdiscF(x₀-Δ); fill=(:black,), line=(:black,1)) + plot!(drawdiscF(x₀); fill=(:orange,), line=(:black,1)) + plot!(drawdiscI(x₀); fill=(:white,1.0), line=(:black,1)) + + x₀ = 0.35 + plot!(drawdiscF(x₀-Δ); fill=(:black,), line=(:black,1)) + plot!(drawdiscF(x₀); fill=(:orange,), line=(:black,1)) + plot!(drawdiscI(x₀); fill=(:white,1.0), line=(:black,1)) + + z0 = 3pi/2 - δ + pline!(surf(t0, z0), surf(-t0, z0); line=(:black, 1)) + pline!(surf(t0, z0+pi), surf(-t0, z0+pi); line=(:black, 1)) + + + # boundary of sphere + z0 = 3pi/2 - δ + curve = [psurf(t, z0) for t in range(-t0, t0, 100)] + plot!(curve; line=(:black,3)) + z0 = 3pi/2 - δ + pi + curve = [psurf(t, z0) for t in range(-t0, t0, 100)] + plot!(curve; line=(:black,3)) + + # caps + curve = [psurf(t0, θ) for θ in range(0, 2pi, 100)] + plot!(curve, line=(:black, 2)) + curve = [psurf(-t0, θ) for θ in range(0, 2pi, 100)] + plot!(curve, line=(:black, 2)) + + + # Shade lines + δ = pi/6 + Δₜ = (4pi/2 - (3pi/2 - δ))/(2*25) + for θ ∈ range(3pi/2-δ, 4pi/2, 25) + curve = [psurf(t, θ) for t in + range(-t0, max(-t0, -t0 + 1/2*sin(θ+δ+pi/2 + pi/2)), 20)] + plot!(curve, line=(:black, 1)) + curve = [psurf(t, θ+Δₜ) for t in + range(-t0, max(-t0, -t0 + 1/3*sin(θ+δ+pi/2 + pi/2)), 20)] + plot!(curve, line=(:black, 1)) + end + + #= + f1 = [[t, _fold(t, 0, pi/2)] for t in range(-0.5, -0.1, 26)] + for f in f1 + plot!([psurf( f[1], f[2]-k*0.01*(6-f[1]) ) + for k in 1:21]; line=(:black, 1)) + end + =# + + current() +end +plt +``` + +Figure showing sphere with interior cylinder bored out. +::: + + +This cross-sectional figure is used to better understand the key dimensions. + +::: {#fig-napkin-ring-2} + ```{julia} #| hold: true #| echo: false -#The following illustrates $R=5$ and $h=8$. +#The following illustrates $R=1$ and $h=2sqrt(3/4$. +plt = let + gr() + empty_style = (xaxis=([], false), + yaxis=([], false), + framestyle=:origin, + legend=false) -R =5; h1 = 2*4 + R = 1; h = 2*sqrt(3/4) -theta = asin(h1/2/R) -thetas = range(-theta, stop=theta, length=100) -ts = range(-pi, stop=pi, length=100) -y = h1/4 + θ = theta = asin(h/2/R) + thetas = range(-theta, stop=theta, length=100) + ts = range(-pi, stop=pi, length=100) + y = h/4 -p = plot(legend=false, aspect_ratio=:equal); -plot!(p, R*cos.(ts), R*sin.(ts)); -plot!(p, R*cos.(thetas), R*sin.(thetas), color=:orange); + plot(; empty_style..., aspect_ratio=:equal) + plot!(R*cos.(ts), R*sin.(ts); line=(:black,)); + plot!(R*cos.(thetas), R*sin.(thetas), line=(:orange,1)); -plot!(p, [R*cos.(theta), R*cos.(theta)], [h1/2, -h1/2], color=:orange); -plot!(p, [R*cos.(theta), sqrt(R^2 - y^2)], [y, y], color=:orange) + plot!([R*cos.(theta), R*cos.(theta)], [h/2, -h/2]; color=:orange); + plot!([R*cos.(theta), sqrt(R^2 - y^2)], [y, y]; line=(:orange,3)) -plot!(p, [0, R*cos.(theta)], [0,0], color=:red); -plot!(p,[ 0, R*cos.(theta)], [0,h1/2], color=:red); + plot!([0, R*cos.(theta)], [0,0], color=:red); + plot!([ 0, R*cos.(theta)], [0,h/2], color=:red); -annotate!(p, [(.5, -2/3, "sqrt(R²- (h/2)²)"), - (R*cos.(theta)-.6, h1/4, "h/2"), - (1.5, 1.75*tan.(theta), "R")]) - -p + x₀ = sqrt(R^2 - (h/2)^2) + annotate!( [ + (x₀/2, 0, text(L"\sqrt{R^2- (\frac{h}{2})^2}",10, :top)), + (x₀, h/4, text(L"\frac{h}{2}",:right)), + (R/2*cos(θ),R/2*sin(θ), text(L"R", :bottom; rotation=rad2deg(θ))) + ]) + current() +end +plt ``` +```{julia} +#| echo: false +plotly() +nothing +``` + +Side view illustrating key dimensions of napkin ring problem with $R$ being the radius of the sphere and $h$ being the height of the resulting interior cylinder. + +::: + + The small orange line is rotated, so using the washer method we get the cross sections given by $\pi(r_0^2 - r_i^2)$, the outer and inner radii, as a function of $y$. From c3a94878f3477c5692a0f5dbb5289d6a735bea8f Mon Sep 17 00:00:00 2001 From: jverzani Date: Wed, 23 Jul 2025 08:05:43 -0400 Subject: [PATCH 17/22] edits --- quarto/.gitignore | 1 + quarto/_make_pdf.jl | 7 +- quarto/derivatives/derivatives.qmd | 75 +- quarto/derivatives/figures/lhopital-144.png | Bin 0 -> 38273 bytes .../derivatives/first_second_derivatives.qmd | 129 ++- .../derivatives/implicit_differentiation.qmd | 108 ++- quarto/derivatives/lhospitals_rule.qmd | 14 +- quarto/derivatives/linearization.qmd | 90 +- quarto/derivatives/mean_value_theorem.qmd | 80 +- quarto/derivatives/more_zeros.qmd | 17 +- quarto/derivatives/newtons_method.qmd | 6 +- quarto/derivatives/optimization.qmd | 57 +- quarto/derivatives/related_rates.qmd | 57 +- .../derivatives/taylor_series_polynomials.qmd | 85 +- quarto/differentiable_vector_calculus.qmd | 4 + .../Project.toml | 1 + .../make_pdf.jl | 1 + .../vector_fields.qmd | 225 ++++- quarto/integrals.qmd | 4 + quarto/integrals/_sa_bone_pile.qmd | 41 + quarto/integrals/arc_length.qmd | 21 +- quarto/integrals/area.qmd | 502 +++++++++-- quarto/integrals/area_between_curves.qmd | 3 +- quarto/integrals/figures/jigsaw.png | Bin 0 -> 744433 bytes quarto/integrals/ftc.qmd | 302 ++++++- quarto/integrals/improper_integrals.qmd | 101 ++- quarto/integrals/integration_by_parts.qmd | 97 ++- quarto/integrals/mean_value_theorem.qmd | 4 +- quarto/integrals/partial_fractions.qmd | 16 +- quarto/integrals/substitution.qmd | 4 +- quarto/integrals/surface_area.qmd | 406 +++++++-- quarto/integrals/volumes_slice.qmd | 378 ++++---- quarto/limits/continuity.qmd | 120 ++- quarto/limits/intermediate_value_theorem.qmd | 414 ++++++--- quarto/limits/limits.qmd | 250 ++++-- quarto/limits/limits_extensions.qmd | 46 +- quarto/precalc.qmd | 4 + quarto/precalc/Project.toml | 1 + quarto/precalc/functions.qmd | 145 +++- quarto/precalc/inversefunctions.qmd | 54 +- quarto/precalc/numbers_types.qmd | 38 +- quarto/precalc/plotting.qmd | 173 ++-- quarto/precalc/polynomial.qmd | 71 +- quarto/precalc/polynomial_roots.qmd | 6 +- quarto/precalc/polynomials_package.qmd | 13 +- quarto/precalc/ranges.qmd | 19 +- quarto/precalc/rational_functions.qmd | 6 +- quarto/precalc/transformations.qmd | 4 +- quarto/precalc/trig_functions.qmd | 91 +- quarto/precalc/vectors.qmd | 805 ++++++++++-------- 50 files changed, 3711 insertions(+), 1385 deletions(-) create mode 100644 quarto/derivatives/figures/lhopital-144.png create mode 100644 quarto/integrals/_sa_bone_pile.qmd create mode 100644 quarto/integrals/figures/jigsaw.png diff --git a/quarto/.gitignore b/quarto/.gitignore index e5b5ac0..b19a2da 100644 --- a/quarto/.gitignore +++ b/quarto/.gitignore @@ -3,5 +3,6 @@ /_freeze/ /*/*_files/ /*/*.ipynb/ +/*/bonepile.qmd /*/references.bib weave_support.jl \ No newline at end of file diff --git a/quarto/_make_pdf.jl b/quarto/_make_pdf.jl index 6dfc3e3..c20f6f2 100644 --- a/quarto/_make_pdf.jl +++ b/quarto/_make_pdf.jl @@ -11,7 +11,7 @@ typst_tpl = mt""" --- title: {{:title}} date: today -jupyter: julia-1.11 +engine: julia execute: daemon: false format: @@ -25,6 +25,11 @@ format: #set figure(placement: auto) bibliography: references.bib --- +```{julia} +#| echo: false +import Plots; Plots.plotly() = Plots.gr(); +nothing +``` """ index = "_pdf_index" diff --git a/quarto/derivatives/derivatives.qmd b/quarto/derivatives/derivatives.qmd index 95e3798..327aa22 100644 --- a/quarto/derivatives/derivatives.qmd +++ b/quarto/derivatives/derivatives.qmd @@ -230,13 +230,22 @@ function secant_line_tangent_line_graph(n) xs = range(0, stop=pi, length=50) fig_size=(800, 600) - plt = plot(f, 0, pi, legend=false, size=fig_size, - line=(2,), - axis=([],false), + plt = plot(; + xaxis=([], false), + yaxis=([], false), + framestyle=:origin, + legend=false, + ylims=(-.1,1.5) ) - plot!([0, 1.1* pi],[0,0], line=(3, :black)) - plot!([0, 0], [0,2*1], line=(3, :black)) + + plot!(f, 0, pi/2; line=(:black, 2)) + plot!(f, pi/2, pi/2 + pi/5; line=(:black, 2, 1/4)) + plot!(f, pi/2 + pi/5, pi; line=(:black, 2)) + + + plot!(0.1 .+ [0,0],[-.1, 1.5]; line=(:gray,1), arrow=true, side=:head) + plot!([-0.2, 3.4], [.1, .1]; line=(:gray, 1), arrow=true, side=:head) plot!(plt, xs, f(c) .+ cos(c)*(xs .- c), color=:orange) plot!(plt, xs, f(c) .+ m*(xs .- c), color=:black) @@ -244,8 +253,10 @@ function secant_line_tangent_line_graph(n) plot!(plt, [c, c+h, c+h], [f(c), f(c), f(c+h)], color=:gray30) - annotate!(plt, [(c+h/2, f(c), text("h", :top)), - (c + h + .05, (f(c) + f(c + h))/2, text("f(c+h) - f(c)", :left)) + annotate!(plt, [(c+h/2, f(c), text(L"h", :top)), + (c + h + .05, (f(c) + f(c + h))/2, text(L"f(c+h) - f(c)", :left)), + + ]) plt @@ -258,7 +269,7 @@ The slope of each secant line represents the *average* rate of change between $c -n = 5 +n = 6 anim = @animate for i=0:n secant_line_tangent_line_graph(i) end @@ -279,11 +290,59 @@ $$ We will define the tangent line at $(c, f(c))$ to be the line through the point with the slope from the limit above - provided that limit exists. Informally, the tangent line is the line through the point that best approximates the function. +::: {#fig-tangent_line_approx_graph} + +```{julia} +#| echo: false +gr() +let + function make_plot(Δ) + f(x) = 1 + sin(x-c) + df(x) = cos(x-c) + plt = plot(; + #xaxis=([], false), + yaxis=([], false), + aspect_ratio=:equal, + legend=false, + + ) + + c = 1 + xticks!([c-Δ, c, c+Δ], [latexstring("c-$Δ"), L"c", latexstring("c-$Δ")]) + y₀ = f(c) - 2/3 * Δ + tl(x) = f(c) + df(c) * (x-c) + + plot!(f, c - Δ, c + Δ; line=(:black, 2)) + plot!(tl, c - Δ, c + Δ; line=(:red, 2)) + plot!([c,c], [tl(c-Δ), f(c)]; line=(:gray, :dash, 1)) + #plot!([c-1.1*Δ, c+1.1*Δ], y₀ .+ [0,0]; line=(:gray, 1), arrow=true) + + + current() + end + + ps = make_plot.((1.5, 1.0, 0.5, 0.1)) + plot(ps...) + + +end +``` + +Illustration that the tangent line is the best linear approximation *near* $c$. +::: + +```{julia} +#| echo: false + +plotly() +nothing +``` ```{julia} #| hold: true #| echo: false #| cache: true +#| eval: false gr() function line_approx_fn_graph(n) f(x) = sin(x) diff --git a/quarto/derivatives/figures/lhopital-144.png b/quarto/derivatives/figures/lhopital-144.png new file mode 100644 index 0000000000000000000000000000000000000000..788421e8e3e5407198de097b4c54deb5cbc30007 GIT binary patch literal 38273 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 af&T%L_)9pdQri*$0000!Gmc4 literal 0 HcmV?d00001 diff --git a/quarto/derivatives/first_second_derivatives.qmd b/quarto/derivatives/first_second_derivatives.qmd index 4e8f857..b60cde9 100644 --- a/quarto/derivatives/first_second_derivatives.qmd +++ b/quarto/derivatives/first_second_derivatives.qmd @@ -50,7 +50,7 @@ A parallel definition with $a < b$ implying $f(a) > f(b)$ would be used for a *s We can try and prove these properties for a function algebraically – we'll see both are related to the zeros of some function. However, before proceeding to that it is usually helpful to get an idea of where the answer is using exploratory graphs. -We will use a helper function, `plotif(f, g, a, b)` that plots the function `f` over `[a,b]` highlighting the regions in the domain when `g` is non-negative. Such a function is defined for us in the accompanying `CalculusWithJulia` package, which has been previously been loaded. +We will use a helper function, `plotif(f, g, a, b)` that plots the function `f` over `[a,b]` highlighting the regions in the domain when `g` is non-negative. Such a function is defined for us in the accompanying `CalculusWithJulia` package, which has been previously loaded. To see where a function is positive, we simply pass the function object in for *both* `f` and `g` above. For example, let's look at where $f(x) = \sin(x)$ is positive: @@ -475,22 +475,22 @@ Let's look at the function $x^2 \cdot e^{-x}$ for positive $x$. A quick graph sh ```{julia} -h(x) = x^2 * exp(-x) -plotif(h, h'', 0, 8) +g(x) = x^2 * exp(-x) +plotif(g, g'', 0, 8) ``` From the graph, we would expect that the second derivative - which is continuous - would have two zeros on $[0,8]$: ```{julia} -ips = find_zeros(h'', 0, 8) +ips = find_zeros(g'', 0, 8) ``` As well, between the zeros we should have the sign pattern `+`, `-`, and `+`, as we verify: ```{julia} -sign_chart(h'', 0, 8) +sign_chart(g'', 0, 8) ``` ### Second derivative test @@ -744,6 +744,90 @@ choices=[ answ = 3 radioq(choices, answ) ``` +###### Question + +The function + +$$ +f(x) = +\begin{cases} +\frac{x}{2} + x^2 \sin(\frac{\pi}{x}) & x \neq 0\\ +0 & x = 0 +\end{cases} +$$ + +is graphed below over $[-1/3, 1/3]$. + + +```{julia} +#| echo: false +plt = let + gr() + empty_style = (xaxis=([], false), + yaxis=([], false), + framestyle=:origin, + legend=false) + axis_style = (arrow=true, side=:head, line=(:gray, 1)) + + ## f'(0) > 0 but not increasing + f(x) = x/2 + x^2 * sinpi(1/x) + g(x) = x/2 - x^2 + a, b = -1/3, 1/3 + xs = range(a, b, 10_000) + ys = f.(xs) + y0,y1 = extrema(ys) + plot(; empty_style..., aspect_ratio=:equal) + plot!([a,b],[0,0]; axis_style...) + plot!([0,0], [y0,y1]; axis_style...) + plot!(xs, f.(xs); line=(:black, 1)) + + plot!(xs, x -> x/2 + x^2; line=(:gray, 1, :dot)) + plot!(xs, x -> x/2 - x^2; line=(:gray, 1, :dot)) + plot!(xs, x -> x/2; line=(:gray, 1)) + + a1 = (1/4 + 1/5)/2 + a2 = -(1*1/3 + 4*1/4)/5 + annotate!([ + (a1, g(a1), text(L"\frac{x}{2} - x^2", 10, :top)), + (a1, f(a1), text(L"\frac{x}{2} + x^2", 10, :bottom)), + (-1/6, f(1/6), text(L"\frac{x}{2} + x^2\sin(\frac{\pi}{x})", 10, :bottom)) + ]) + plot!([-1/6, -1/13.5], [f(1/6), f(-1/13.5)]; axis_style...) +end +plt +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + +This function has a derivative at $0$ that is *positive* + +```{julia} +f(x) = x == 0 ? 0 : x/2 + x^2 * sinpi(1/x) +@syms h +limit((f(0+h) - f(0))/h, h=>0; dir="+-") +``` + +Is the function increasing **around** $0$? + +(The derivative away from $0$ is given by: + +```{julia} +@syms x +diff(f(x), x) +``` + +```{julia} +#| echo: false +choices = ["Yes", "No"] +answer = 1 +buttonq(choices, answer; explanation=raw""" +The slope of the tangent line away from $0$ oscillates from positive to negative at every rational number of the form $1/n$ due to the $\cos(\pi/x)$ term, so it is neither going just up or down around $0$. (This example comes from @Angenent.) +""") +``` ###### Question @@ -779,21 +863,30 @@ Consider the following figure of a graph of $f$: ```{julia} #| echo: false -ex(x) = x * tanh(exp(x)) -a, b = -5, 1 -plot(ex, a, b, legend=false, - axis=([], false), - color = :royalblue - ) -plot!([a-.1, b+.1], [0,0], line=(3, :black)) +let + gr() + ex(x) = x * tanh(exp(x)) + a, b = -5, 1 + plot(ex, a, b, legend=false, + axis=([], false), + line=(:black, 2) + ) + plot!([a-.1, b+.1], [0,0], line=(:gray,1), arrow=true, side=:head) -zs = find_zeros(ex, (a, b)) -cps = find_zeros(ex', (a, b)) -ips = find_zeros(ex'', (a, b)) + zs = find_zeros(ex, (a, b)) + cps = find_zeros(ex', (a, b)) + ips = find_zeros(ex'', (a, b)) -scatter!(zs, ex.(zs), marker=(5, "black", :circle)) -scatter!(cps, ex.(cps), marker=(5, "forestgreen", :diamond)) -scatter!(ips, ex.(ips), marker=(5, :brown3, :star5)) + scatter!(zs, ex.(zs), fill=(:black,), marker=(8, :circle)) + scatter!(cps, ex.(cps), fill=(:green,), marker=(8, :diamond)) + scatter!(ips, ex.(ips), fill=(:brown3,), marker=(8,:star5)) +end +``` + +```{julia} +#| echo: false +plotly() +nothing ``` The black circle denotes what? diff --git a/quarto/derivatives/implicit_differentiation.qmd b/quarto/derivatives/implicit_differentiation.qmd index b5514fa..52ab46e 100644 --- a/quarto/derivatives/implicit_differentiation.qmd +++ b/quarto/derivatives/implicit_differentiation.qmd @@ -93,6 +93,42 @@ In general though, we may not be able to solve for $y$ in terms of $x$. What the The idea is to *assume* that $y$ is representable by some function of $x$. This makes sense, moving on the curve from $(x,y)$ to some nearby point, means changing $x$ will cause some change in $y$. This assumption is only made *locally* - basically meaning a complicated graph is reduced to just a small, well-behaved, section of its graph. +::: {#fig-well-behaved-section} + +```{julia} +#| echo: false +let + gr() + a = 1 + k = 2 + F(x,y) = a * (x + a)*(x^2 + y^2) - k*x^2 + xs = range(-3/2, 3/2, 100) + ys = range(-2, 2, 100) + contour(xs, ys, F; levels=[0], + axis=([], nothing), + line=(:black, 1), + framestyle=:none, legend=false) + + x₀, y₀ = 3/4, -0.2834733547569205 + m = (-a^2*x₀ - 3*a*x₀^2/2 - a*y₀^2/2 + k*x₀)/(a*(a + x₀)*y₀) + + plot!(x -> y₀ + m*(x - x₀), x₀-0.5, x₀ + 0.5; line=(:gray, 2)) + plot!(x -> -x*sqrt(-(a^2 + a*x - k)/(a*(a + x))), -1/8,0.99; + line=(:black,4)) + scatter!([x₀], [y₀]; marker=(:circle,5,:yellow)) +end +``` + + +```{julia} +#| echo: false +plotly() +nothing +``` + +Graph of an equation with a well behaved section emphasized. The tangent line can be found by finding a formula for this well behaved section and differentiating *or* by implicit differentiation simply by assuming a form for the implicit function. +::: + With this assumption, asking what $dy/dx$ is has an obvious meaning - what is the slope of the tangent line to the graph at $(x,y)$. (The assumption eliminates the question of what a tangent line would mean when a graph self intersects.) @@ -120,7 +156,7 @@ This says the slope of the tangent line depends on the point $(x,y)$ through the As a check, we compare to what we would have found had we solved for $y= \sqrt{1 - x^2}$ (for $(x,y)$ with $y \geq 0$). We would have found: $dy/dx = 1/2 \cdot 1/\sqrt{1 - x^2} \cdot (-2x)$. Which can be simplified to $-x/y$. This should show that the method above - assuming $y$ is a function of $x$ and differentiating - is not only more general, but can even be easier. -The name - *implicit differentiation* - comes from the assumption that $y$ is implicitly defined in terms of $x$. According to the [Implicit Function Theorem](http://en.wikipedia.org/wiki/Implicit_function_theorem) the above method will work provided the curve has sufficient smoothness near the point $(x,y)$. +The name - *implicit differentiation* - comes from the assumption that $y$ is implicitly defined in terms of $x$. According to the [Implicit Function Theorem](http://en.wikipedia.org/wiki/Implicit_function_theorem) the above method will work provided the curve has sufficient smoothness near the point $(x,y)$. (Continuously differentiable and non vanishing derivative in $y$.) ##### Examples @@ -140,10 +176,16 @@ For $a = 2, b=1$ we have the graph: #| hold: true a, b = 2, 1 f(x,y) = x^2*y + a * b * y - a^2 * x -implicit_plot(f) +implicit_plot(f; legend=false) + +x₀, y₀ = 0, 0 +m = (a^2 - 2x₀*y₀) / (a*b + x₀^2) +plot!(x -> y₀ + m*(x - x₀), -1, 1) ``` -We can see that at each point in the viewing window the tangent line exists due to the smoothness of the curve. Moreover, at a point $(x,y)$ the tangent will have slope $dy/dx$ satisfying: +To the plot we added a tangent line at $(0,0)$. + +We can see that at each point in the viewing window the tangent line exists due to the smoothness of the curve. To find the slope of the tangent line at a point $(x,y)$ the tangent line will have slope $dy/dx$ satisfying: $$ @@ -177,7 +219,7 @@ A graph for $a=3$ shows why it has the name it does: #| hold: true a = 3 f(x,y) = x^4 - a^2*(x^2 - y^2) -implicit_plot(f) +implicit_plot(f; xticks=-5:5) ``` The tangent line at $(x,y)$ will have slope, $dy/dx$ satisfying: @@ -341,7 +383,7 @@ The next step is solve for $dy/dx$ - the lone answer to the linear equation - wh ```{julia} dydx = diff(u(x), x) -ex3 = solve(ex2, dydx)[1] # pull out lone answer with [1] indexing +ex3 = only(solve(ex2, dydx)) # pull out the only answer ``` As this represents an answer in terms of `u(x)`, we replace that term with the original variable: @@ -369,9 +411,9 @@ Let $a = b = c = d = 1$, then $(1,4)$ is a point on the curve. We can draw a tan ```{julia} H = ex(a=>1, b=>1, c=>1, d=>1) x0, y0 = 1, 4 -𝒎 = dydx₁(x=>1, y=>4, a=>1, b=>1, c=>1, d=>1) +m = dydx₁(x=>1, y=>4, a=>1, b=>1, c=>1, d=>1) implicit_plot(lambdify(H); xlims=(-5,5), ylims=(-5,5), legend=false) -plot!(y0 + 𝒎 * (x-x0)) +plot!(y0 + m * (x-x0)) ``` Basically this includes all the same steps as if done "by hand." Some effort could have been saved in plotting, had values for the parameters been substituted initially, but not doing so shows their dependence in the derivative. @@ -379,7 +421,7 @@ Basically this includes all the same steps as if done "by hand." Some effort cou :::{.callout-warning} ## Warning -The use of `lambdify(H)` is needed to turn the symbolic expression, `H`, into a function. +The use of `lambdify(H)` is needed to turn the symbolic expression, `H`, into a function for plotting purposes. ::: @@ -517,15 +559,9 @@ This could have been made easier, had we leveraged the result of the previous ex #### Example: from physics - -Many problems are best done with implicit derivatives. A video showing such a problem along with how to do it analytically is [here](http://ocw.mit.edu/courses/mathematics/18-01sc-single-variable-calculus-fall-2010/unit-2-applications-of-differentiation/part-b-optimization-related-rates-and-newtons-method/session-32-ring-on-a-string/). - - -This video starts with a simple question: - - > If you have a rope and heavy ring, where will the ring position itself due to gravity? +This problem illustrates one best done with implicit derivatives. A video showing this problem along with how to do it analytically is [here](http://ocw.mit.edu/courses/mathematics/18-01sc-single-variable-calculus-fall-2010/unit-2-applications-of-differentiation/part-b-optimization-related-rates-and-newtons-method/session-32-ring-on-a-string/). Well, suppose you hold the rope in two places, which we can take to be $(0,0)$ and $(a,b)$. Then let $(x,y)$ be all the possible positions of the ring that hold the rope taught. Then we have this picture: @@ -534,19 +570,41 @@ Well, suppose you hold the rope in two places, which we can take to be $(0,0)$ a ```{julia} #| hold: true #| echo: false - +let + gr() P = (4,1) Q = (1, -3) - scatter([0,4], [0,1], legend=false, xaxis=nothing, yaxis=nothing) - plot!([0,1,4],[0,-3,1]) - 𝑎, 𝑏= .05, .25 + plot(; + axis=([],false), + legend=false) + + scatter!([0,4], [0,1]) + plot!([0,1,4],[0,-3,1]; line=(:black,2)) + a, b = .05, .25 ts = range(0, 2pi, length=100) - plot!(1 .+ 𝑎*sin.(ts), -3 .+ 𝑏*cos.(ts), color=:gold) - annotate!((4-0.3,1,"(a,b)")) - plot!([0,1,1],[0,0,-3], color=:gray, alpha=0.25) - plot!([1,1,4],[0,1,1], color=:gray, alpha=0.25) - Δ = 0.15 - annotate!([(1/2, 0-Δ, "x"), (5/2, 1 - Δ, "a-x"), (1-Δ, -1, "|y|"), (1+Δ, -1, "b-y")]) + plot!(1 .+ a*sin.(ts), -3 .+ b*cos.(ts), line=(:gold,2)) + + plot!([0,1,1],[0,0,-3], color=:gray, alpha=0.75) + plot!([1,1,4],[0,1,1], color=:gray, alpha=0.75) + Δ = 0.05 + + annotate!([ + (0,0, text(L"(0,0)",:bottom)), + (4,1, text(L"(a,b)",:bottom)), + (1/2, 0, text(L"x",:top)), + (5/2, 1, text(L"a-x", :top)), + (1, -1, text(L"|y|",:right)), + (1+Δ, -1, text(L"b-y",:left)), + (1+2a, -3, text(L"(x,y)",:left)) + ]) + current() +end +``` + +```{julia} +#| echo: false +plotly() +nothing ``` Since the length of the rope does not change, we must have for any admissible $(x,y)$ that: diff --git a/quarto/derivatives/lhospitals_rule.qmd b/quarto/derivatives/lhospitals_rule.qmd index 611f4da..13b7867 100644 --- a/quarto/derivatives/lhospitals_rule.qmd +++ b/quarto/derivatives/lhospitals_rule.qmd @@ -28,17 +28,17 @@ We know this is $1$ using a bound from geometry, but might also guess this is on $$ -\sin(x) = x - \sin(\xi)x^2/2, \quad 0 < \xi < x. +\sin(x) = x - \sin(\xi)\frac{x^2}{2}, \quad 0 < \xi < x. $$ This would yield: $$ -\lim_{x \rightarrow 0} \frac{\sin(x)}{x} = \lim_{x\rightarrow 0} \frac{x -\sin(\xi) x^2/2}{x} = \lim_{x\rightarrow 0} 1 - \sin(\xi) \cdot x/2 = 1. +\lim_{x \rightarrow 0} \frac{\sin(x)}{x} = \lim_{x\rightarrow 0} \frac{x -\sin(\xi) \frac{x^2}{2}}{x} = \lim_{x\rightarrow 0} 1 - \sin(\xi) \cdot \frac{x}{2} = 1. $$ -This is because we know $\sin(\xi) x/2$ has a limit of $0$, when $|\xi| \leq |x|$. +This is because we know $\sin(\xi) \frac{x}{2}$ has a limit of $0$, when $|\xi| \leq |x|$. That doesn't look any easier, as we worried about the error term, but if just mentally replaced $\sin(x)$ with $x$ - which it basically is near $0$ - then we can see that the limit should be the same as $x/x$ which we know is $1$ without thinking. @@ -384,10 +384,10 @@ the first equality by L'Hospital's rule, as the second limit exists. Indeterminate forms of the type $0 \cdot \infty$, $0^0$, $\infty^\infty$, $\infty - \infty$ can be re-expressed to be in the form $0/0$ or $\infty/\infty$ and then L'Hospital's theorem can be applied. -###### Example: rewriting $0 \cdot \infty$ +##### Example: rewriting $0 \cdot \infty$ -What is the limit $x \log(x)$ as $x \rightarrow 0+$? The form is $0\cdot \infty$, rewriting, we see this is just: +What is the limit of $x \log(x)$ as $x \rightarrow 0+$? The form is $0\cdot \infty$, rewriting, we see this is just: $$ @@ -401,10 +401,10 @@ $$ \lim_{x \rightarrow 0+}\frac{1/x}{-1/x^2} = \lim_{x \rightarrow 0+} -x = 0. $$ -###### Example: rewriting $0^0$ +##### Example: rewriting $0^0$ -What is the limit $x^x$ as $x \rightarrow 0+$? The expression is of the form $0^0$, which is indeterminate. (Even though floating point math defines the value as $1$.) We can rewrite this by taking a log: +What is the limit of $x^x$ as $x \rightarrow 0+$? The expression is of the form $0^0$, which is indeterminate. (Even though floating point math defines the value as $1$.) We can rewrite this by taking a log: $$ diff --git a/quarto/derivatives/linearization.qmd b/quarto/derivatives/linearization.qmd index 9924919..81f9c4f 100644 --- a/quarto/derivatives/linearization.qmd +++ b/quarto/derivatives/linearization.qmd @@ -186,25 +186,45 @@ In each of these cases, a more complicated non-linear function is well approxim #| echo: false #| label: fig-tangent-dy-dx #| fig-cap: "Graph with tangent line layered on" -f(x) = sin(x) -a, b = -1/4, pi/2 +let + gr() + f(x) = sin(x) + a, b = -1/4, pi/2 -p = plot(f, a, b, legend=false, - line=(3, :royalblue), - axis=([], false) - ); + p = plot(f, a, b, legend=false, + line=(3, :royalblue), + axis=([], false) + ); -plot!(p, x->x, a, b); -plot!(p, [0,1,1], [0, 0, 1], color=:brown); + plot!(p, x->x, a, b); + plot!(p, [0,1,1], [0, 0, 1], color=:brown); -plot!(p, [1,1], [0, sin(1)], color=:green, linewidth=4); + plot!(p, [1,1], [0, sin(1)], color=:green, linewidth=4); -scatter!([0], [0], marker=(5, :mediumorchid3)) -annotate!(p, [(0, f(0), text("(c,f(c))", :bottom,:right))]) -annotate!(p, collect(zip([1/2, 1+.075, 1/2-1/8], - [.05, sin(1)/2, .75], - ["Δx", "Δy", "m=dy/dx"]))); + x₀ = 1.15 + δ = 0.1 + plot!(p, [x₀,x₀,1], [sin(1)/2-δ,0,0], line=(:black, 1, :dash), arrow=true) + plot!(p, [x₀,x₀,1], [sin(1)/2+δ,1, 1], line=(:black, 1, :dash), arrow=true) + plot!(p, [1/2 - 0.8δ, 0], [-δ, -δ]*3/4, line=(:black, 1, :dash), arrow=true) + plot!(p, [1/2 + 0.8δ, 1], [-δ, -δ]*3/4, line=(:black, 1, :dash), arrow=true) + scatter!([0], [0], marker=(5, :mediumorchid3)) + annotate!(p, [ + (0, f(0), text(L"(c, f(c))", :bottom, :right)), + (1/2, 0, text(L"\Delta x", :bottom)), + (1/2, 0, text(L"dx", :top)), + (1-0.02, sin(1)/2, text(L"Δ y", :right)), + (x₀, sin(1)/2, text(L"dy")), + (2/3, 2/3, text(L"m = \frac{dy}{dx} \approx \frac{\Delta y}{\Delta x}", + :bottom, rotation=33)) # why 33 and not 45? +]) p +end +``` + +```{julia} +#| echo: false +plotly() +nothing ``` The plot in @fig-tangent-dy-dx shows a tangent line with slope $dy/dx$ and the actual change in $y$, $\Delta y$, for some specified $\Delta x$ at a point $(c,f(c))$. The small gap above the sine curve is the error were the value of the sine approximated using the drawn tangent line. We can see that approximating the value of $\Delta y = \sin(c+\Delta x) - \sin(c)$ with the often easier to compute $(dy/dx) \cdot \Delta x = f'(c)\Delta x$ - for small enough values of $\Delta x$ - is not going to be too far off provided $\Delta x$ is not too large. @@ -480,7 +500,8 @@ To see formally why the remainder is as it is, we recall the mean value theorem $$ -\text{error} = h(x) - h(0) = (g(x) - g(0)) \frac{h'(e)}{g'(e)} = x^2 \cdot \frac{1}{2} \cdot \frac{f'(e) - f'(0)}{e} = +\text{error} = h(x) - h(0) = (g(x) - g(0)) \frac{h'(e)}{g'(e)} = +(x^2 - 0) \cdot \frac{f'(e) - f'(0)}{2e} = x^2 \cdot \frac{1}{2} \cdot f''(\xi). $$ @@ -551,7 +572,8 @@ Is it a coincidence that a basic algebraic operation with tangent lines approxim $$ \begin{align*} f(x) \cdot g(x) &= [f(c) + f'(c)(x-c) + \mathcal{O}((x-c)^2)] \cdot [g(c) + g'(c)(x-c) + \mathcal{O}((x-c)^2)]\\ -&=[f(c) + f'(c)(x-c)] \cdot [g(c) + g'(c)(x-c)] + (f(c) + f'(c)(x-c)) \cdot \mathcal{O}((x-c)^2) + (g(c) + g'(c)(x-c)) \cdot \mathcal{O}((x-c)^2) + [\mathcal{O}((x-c)^2)]^2\\ +&=[f(c) + f'(c)(x-c)] \cdot [g(c) + g'(c)(x-c)] \\ +&+ (f(c) + f'(c)(x-c)) \cdot \mathcal{O}((x-c)^2) + (g(c) + g'(c)(x-c)) \cdot \mathcal{O}((x-c)^2) + [\mathcal{O}((x-c)^2)]^2\\ &= [f(c) + f'(c)(x-c)] \cdot [g(c) + g'(c)(x-c)] + \mathcal{O}((x-c)^2)\\ &= f(c) \cdot g(c) + [f'(c)\cdot g(c) + f(c)\cdot g'(c)] \cdot (x-c) + [f'(c)\cdot g'(c) \cdot (x-c)^2 + \mathcal{O}((x-c)^2)] \\ &= f(c) \cdot g(c) + [f'(c)\cdot g(c) + f(c)\cdot g'(c)] \cdot (x-c) + \mathcal{O}((x-c)^2) @@ -630,7 +652,7 @@ Automatic differentiation (forward mode) essentially uses this technique. A "dua ```{julia} -Dual(0, 1) +x = Dual(0, 1) ``` Then what is $x$? It should reflect both $(\sin(0), \cos(0))$ the latter being the derivative of $\sin$. We can see this is *almost* what is computed behind the scenes through: @@ -638,11 +660,13 @@ Then what is $x$? It should reflect both $(\sin(0), \cos(0))$ the latter being t ```{julia} #| hold: true -x = Dual(0, 1) @code_lowered sin(x) ``` -This output of `@code_lowered` can be confusing, but this simple case needn't be. Working from the end we see an assignment to a variable named `%3` of `Dual(%6, %12)`. The value of `%6` is `sin(x)` where `x` is the value `0` above. The value of `%12` is `cos(x)` *times* the value `1` above (the `xp`), which reflects the *chain* rule being used. (The derivative of `sin(u)` is `cos(u)*du`.) So this dual number encodes both the function value at `0` and the derivative of the function at `0`. +This output of `@code_lowered` can be confusing, but this simple case needn't be, as we know what to look for: we need to evaluate `sin` at `1` and carry along the derivative `cos(x)` **times** the derivative at `x`. + +The `sin` is computed in `%6` and is passed to `Dual` in `%13` as the first arguments. The `cos` is computed in `%11` and then *multiplied* in `%` by `xp`, which holds the derivative information about `x`. This is passed as the second argument to `Dual` in `%13`. + Similarly, we can see what happens to `log(x)` at `1` (encoded by `Dual(1,1)`): @@ -654,14 +678,15 @@ x = Dual(1, 1) @code_lowered log(x) ``` -We can see the derivative again reflects the chain rule, it being given by `1/x * xp` where `xp` acts like `dx` (from assignments `%9` and `%8`). Comparing the two outputs, we see only the assignment to `%9` differs, it reflecting the derivative of the function. +We again see `log(x)` being evaluated in line `%6`. The derivative evaluated at `x` is done in line `%11` and this is multiplied by `xp` in line `%12`. + ## Curvature The curvature of a function will be a topic in a later section on differentiable vector calculus, but the concept of linearization can be used to give an earlier introduction. -The tangent line linearizes the function, it begin the best linear approximation to the graph of the function at the point. The slope of the tangent line is the limit of the slopes of different secant lines. Consider now, the orthogonal concept, the *normal line* at a point. This is a line perpendicular to the tangent line that goes through the point on the curve. +The tangent line linearizes the function, it being the best linear approximation to the graph of the function at the point. The slope of the tangent line is the limit of the slopes of different secant lines. Consider now, the orthogonal concept, the *normal line* at a point. This is a line perpendicular to the tangent line that goes through the point on the curve. At a point $(c,f(c))$ the slope of the normal line is $-1/f'(c)$. @@ -692,6 +717,7 @@ Call $R$ the intersection point of the two normal lines: #| echo: false using Roots let + gr() f(x) = x^4 fp(x) = 4x^3 c = 1/4 @@ -706,12 +732,17 @@ let Rx = find_zero(x -> nlc(x) - nlch(x), (-10, 10)) scatter!([c,c+h], f.([c, c+h])) scatter!([Rx], [nlc(Rx)]) - annotate!([(c, f(c), "(c,f(c))",:top), - (c+h, f(c+h), "(c+h, f(c+h))",:bottom), - (Rx, nlc(Rx), "R",:left)]) + annotate!([(c, f(c), L"(c,f(c))",:top), + (c+h, f(c+h), L"(c+h, f(c+h))",:bottom), + (Rx, nlc(Rx), L"R",:left)]) end ``` +```{julia} +#| echo: false +plotly() +nothing +``` What happens to $R$ as $h \rightarrow 0$? @@ -760,6 +791,7 @@ This formula for $r$ is known as the radius of curvature of $f$ -- the radius of ```{julia} #| echo: false let + gr() f(x) = x^4 fp(x) = 4x^3 fpp(x) = 12x^2 @@ -779,8 +811,8 @@ let scatter!([c], f.([c])) scatter!([Rx], [nlc(Rx)]) - annotate!([(c, f(c), "(c,f(c))",:top), - (Rx, nlc(Rx), "R",:left)]) + annotate!([(c, f(c), L"(c,f(c))",:top), + (Rx, nlc(Rx), L"R",:left)]) Delta = pi/10 @@ -801,6 +833,12 @@ let end ``` +```{julia} +#| echo: false +plotly() +nothing +``` + ## Questions diff --git a/quarto/derivatives/mean_value_theorem.qmd b/quarto/derivatives/mean_value_theorem.qmd index 286e52a..c2f2d06 100644 --- a/quarto/derivatives/mean_value_theorem.qmd +++ b/quarto/derivatives/mean_value_theorem.qmd @@ -277,6 +277,13 @@ For $f$ differentiable on $(a,b)$ and continuous on $[a,b]$, if $f(a)=f(b)$, the ::: +::: {#fig-l-hospital-144} + +![Figure from L'Hospital's calculus book](figures/lhopital-144.png) + +Figure from L'Hospital's calculus book showing Rolle's theorem where $c=E$ in the labeling. +::: + This modest observation opens the door to many relationships between a function and its derivative, as it ties the two together in one statement. @@ -323,46 +330,71 @@ The mean value theorem is a direct generalization of Rolle's theorem. ::: {.callout-note icon=false} ## Mean value theorem -Let $f(x)$ be differentiable on $(a,b)$ and continuous on $[a,b]$. Then there exists a value $c$ in $(a,b)$ where $f'(c) = (f(b) - f(a)) / (b - a)$. +Let $f(x)$ be differentiable on $(a,b)$ and continuous on $[a,b]$. Then there exists a value $c$ in $(a,b)$ where + +$$ +f'(c) = (f(b) - f(a)) / (b - a). +$$ ::: This says for any secant line between $a < b$ there will be a parallel tangent line at some $c$ with $a < c < b$ (all provided $f$ is differentiable on $(a,b)$ and continuous on $[a,b]$). -Figure @fig-mean-value-theorem illustrates the theorem. The blue line is the secant line. A parallel line tangent to the graph is guaranteed by the mean value theorem. In this figure, there are two such lines, rendered using brown. +@fig-mean-value-theorem illustrates the theorem. The secant line between $a$ and $b$ is dashed. For this function there are two values of $c$ where the slope of the tangent line is seen to be the same as the slope of this secant line. At least one is guaranteed by the theorem. ```{julia} #| hold: true #| echo: false #| label: fig-mean-value-theorem -f(x) = x^3 - x -a, b = -2, 1.75 -m = (f(b) - f(a)) / (b-a) -cps = find_zeros(x -> f'(x) - m, a, b) +let + # mean value theorem + gr() + f(x) = x^3 -4x^2 + 3x - 1 + a, b = -3/4, 3+3/4 + plot(; axis=([], nothing), + legend=false, + xlims=(-1.1,4), + framestyle=:none) + y₀ = 0.3 + f(-1) -p = plot(f, a-0.75, b+1, - color=:mediumorchid3, - linewidth=3, legend=false, - axis=([],false), - ) + plot!(f, -1, 4; line=(:black, 2)) + plot!([-1.1, 4], y₀*[1,1]; line=(:black, 1), arrow=true, head=:top) + p,q = (a,f(a)), (b, f(b)) + scatter!([p,q]; marker=(:circle, 4, :red)) + plot!([p,q]; line=(:gray, 2, :dash)) + m = (f(b) - f(a))/(b-a) + c₁, c₂ = find_zeros(x -> f'(x) - m, (a,b)) + + Δ = 2/3 + for c ∈ (c₁, c₂) + plot!(tangent(f,c), c-Δ, c+Δ; line=(:gray, 2)) + plot!([(c, y₀), (c, f(c))]; line=(:gray, 1, :dash)) + end + + for c ∈ (a,b) + plot!([(c, y₀), (c, f(c))]; line=(:gray, 1)) + end + + annotate!([ + (a, y₀, text(L"a", :top)), + (b, y₀, text(L"b", :top)), + (c₁, y₀, text(L"c_1", :top)), + (c₂, y₀, text(L"c_2", :top)), + + ]) + + current() -plot!(x -> f(a) + m*(x-a), a-1, b+1, linewidth=5, color=:royalblue) -scatter!([a,b], [f(a), f(b)]) -annotate!([(a, f(a), text("a", :bottom)), - (b, f(b), text("b", :bottom))]) - - -for cp in cps - plot!(x -> f(cp) + f'(cp)*(x-cp), a-1, b+1, color=:brown3) end +``` -scatter!(cps, f.(cps)) -subsscripts = collect("₀₁₂₃₄₅₆₇₈₉") -annotate!([(cp, f(cp), text("c"*subsscripts[i], :bottom)) for (i,cp) ∈ enumerate(cps)]) -p +```{julia} +#| echo: false +plotly() +nothing ``` Like Rolle's theorem this is a guarantee that something exists, not a recipe to find it. In fact, the mean value theorem is just Rolle's theorem applied to: @@ -506,7 +538,7 @@ function parametric_fns_graph(n) xlim=(-1.1,1.1), ylim=(-pi/2-.1, pi/2+.1)) scatter!(plt, [f(ts[end])], [g(ts[end])], color=:orange, markersize=5) val = @sprintf("% 0.2f", ts[end]) - annotate!(plt, [(0, 1, "t = $val")]) + annotate!(plt, [(0, 1, L"t = %$val")]) end caption = L""" diff --git a/quarto/derivatives/more_zeros.qmd b/quarto/derivatives/more_zeros.qmd index 81b6c9b..4d05d14 100644 --- a/quarto/derivatives/more_zeros.qmd +++ b/quarto/derivatives/more_zeros.qmd @@ -315,10 +315,10 @@ One way to think about this is the difference between `x` and the next largest f For the specific example, `abs(b-a) <= 2eps(m)` means that the gap between `a` and `b` is essentially 2 floating point values from the $x$ value with the smallest $f(x)$ value. -For bracketing methods that is about as good as you can get. However, once floating values are understood, the absolute best you can get for a bracketing interval would be +For bracketing methods that is about as good as you can get. However, once floating point values are understood, the absolute best you can get for a bracketing interval would be - * along the way, a value `f(c)` is found which is *exactly* `0.0` + * along the way, a value `f(c)` is found which evaluates *exactly* to `0.0` * the endpoints of the bracketing interval are *adjacent* floating point values, meaning the interval can not be bisected and `f` changes sign between the two values. @@ -334,6 +334,8 @@ chandrapatla(fu, -9, 1, λ3) Here the issue is `abs(b-a)` is tiny (of the order `1e-119`) but `eps(m)` is even smaller. +> For checking if $x_n \approx x_{n+1}$ both a relative and absolute error should be used unless something else is known. + For non-bracketing methods, like Newton's method or the secant method, different criteria are useful. There may not be a bracketing interval for `f` (for example `f(x) = (x-1)^2`) so the second criteria above might need to be restated in terms of the last two iterates, $x_n$ and $x_{n-1}$. Calling this difference $\Delta = |x_n - x_{n-1}|$, we might stop if $\Delta$ is small enough. As there are scenarios where this can happen, but the function is not at a zero, a check on the size of $f$ is needed. @@ -347,7 +349,7 @@ First if `f(x_n)` is `0.0` then it makes sense to call `x_n` an *exact zero* of However, there may never be a value with `f(x_n)` exactly `0.0`. (The value of `sin(1pi)` is not zero, for example, as `1pi` is an approximation to $\pi$, as well the `sin` of values adjacent to `float(pi)` do not produce `0.0` exactly.) -Suppose `x_n` is the closest floating number to $\alpha$, the zero. Then the relative rounding error, $($ `x_n` $- \alpha)/\alpha$, will be a value $\delta$ with $\delta$ less than `eps()`. +Suppose `x_n` is the closest floating point number to $\alpha$, the zero. Then the relative rounding error, $($ `x_n` $- \alpha)/\alpha$, will be a value $\delta$ with $\delta$ less than `eps()`. How far then can `f(x_n)` be from $0 = f(\alpha)$? @@ -364,10 +366,11 @@ $$ f(x_n) \approx f(\alpha) + f'(\alpha) \cdot (\alpha\delta) = f'(\alpha) \cdot \alpha \delta $$ -So we should consider `f(x_n)` an *approximate zero* when it is on the scale of $f'(\alpha) \cdot \alpha \delta$. +So we should consider `f(x_n)` an *approximate zero* when it is on the scale of $f'(\alpha) \cdot \alpha \delta$. That $\alpha$ factor means we consider a *relative* tolerance for `f`. +> For checking if $f(x_n) \approx 0$ both a relative and absolute error should be used--the relative error involving the size of $x_n$. -That $\alpha$ factor means we consider a *relative* tolerance for `f`. Also important – when `x_n` is close to `0`, is the need for an *absolute* tolerance, one not dependent on the size of `x`. So a good condition to check if `f(x_n)` is small is +A good condition to check if `f(x_n)` is small is `abs(f(x_n)) <= abs(x_n) * rtol + atol`, or `abs(f(x_n)) <= max(abs(x_n) * rtol, atol)` @@ -396,7 +399,7 @@ It is not uncommon to assign `rtol` to have a value like `sqrt(eps())` to accoun In Part III of @doi:10.1137/1.9781611977165 we find language of numerical analysis useful to formally describe the zero-finding problem. Key concepts are errors, conditioning, and stability. These give some theoretical justification for the tolerances above. -Abstractly a *problem* is a mapping, $F$, from a domain $X$ of data to a range $Y$ of solutions. Both $X$ and $Y$ have a sense of distance given by a *norm*. A norm is a generalization of the absolute value and gives quantitative meaning to terms like small and large. +Abstractly a *problem* is a mapping, $F$, from a domain $X$ of data to a range $Y$ of solutions. Both $X$ and $Y$ have a sense of distance given by a *norm*. A norm (denoted with $\lVert\cdot\rVert$) is a generalization of the absolute value and gives quantitative meaning to terms like small and large. > A *well-conditioned* problem is one with the property that all small perturbations of $x$ lead to only small changes in $F(x)$. @@ -435,7 +438,7 @@ $$ \tilde{F}(x) = F(\tilde{x}) $$ -for some $\tilde{x}$ with $\lVert\tilde{x} - x\rVert/\lVert x\rVert$ is small. +for some $\tilde{x}$ where $\lVert\tilde{x} - x\rVert/\lVert x\rVert$ is small. > "A backward stable algorithm gives exactly the right answer to nearly the right question." diff --git a/quarto/derivatives/newtons_method.qmd b/quarto/derivatives/newtons_method.qmd index 9b978b7..c09812e 100644 --- a/quarto/derivatives/newtons_method.qmd +++ b/quarto/derivatives/newtons_method.qmd @@ -413,7 +413,7 @@ To machine tolerance the answer is a zero, even though the exact answer is irrat The first example by Newton of applying the method to a non-polynomial function was solving an equation from astronomy: $x - e \sin(x) = M$, where $e$ is an eccentric anomaly and $M$ a mean anomaly. Newton used polynomial approximations for the trigonometric functions, here we can solve directly. -Let $e = 1/2$ and $M = 3/4$. With $f(x) = x - e\sin(x) - M$ then $f'(x) = 1 - e cos(x)$. Starting at 1, Newton's method for 3 steps becomes: +Let $e = 1/2$ and $M = 3/4$. With $f(x) = x - e\sin(x) - M$ then $f'(x) = 1 - e \cos(x)$. Starting at 1, Newton's method for 3 steps becomes: ```{julia} ec, M = 0.5, 0.75 @@ -490,7 +490,7 @@ $$ x_{i+1} = x_i - (1/x_i - q)/(-1/x_i^2) = -qx^2_i + 2x_i. $$ -Now for $q$ in the interval $[1/2, 1]$ we want to get a *good* initial guess. Here is a claim. We can use $x_0=48/17 - 32/17 \cdot q$. Let's check graphically that this is a reasonable initial approximation to $1/q$: +Now for $q$ in the interval $[1/2, 1]$ we want to get a *good* initial guess. Here is a claim: we can use $x_0=48/17 - 32/17 \cdot q$. Let's check graphically that this is a reasonable initial approximation to $1/q$: ```{julia} @@ -865,7 +865,7 @@ The function $f(x) = x^{20} - 1$ has two bad behaviours for Newton's method: for $x < 1$ the derivative is nearly $0$ and for $x>1$ the second derivative is very big. In this illustration, we have an initial guess of $x_0=8/9$. As the tangent line is fairly flat, the -next approximation is far away, $x_1 = 1.313\dots$. As this guess +next approximation is far away, $x_1 = 1.313\dots$. As this guess is much bigger than $1$, the ratio $f(x)/f'(x) \approx x^{20}/(20x^{19}) = x/20$, so $x_i - f(x_i)/f'(x_i) \approx (19/20)x_i$ yielding slow, linear convergence until $f''(x_i)$ is moderate. For diff --git a/quarto/derivatives/optimization.qmd b/quarto/derivatives/optimization.qmd index 406d3b7..ce7f820 100644 --- a/quarto/derivatives/optimization.qmd +++ b/quarto/derivatives/optimization.qmd @@ -70,7 +70,7 @@ function perimeter_area_graphic_graph(n) size=fig_size, xlim=(0,10), ylim=(0,10)) scatter!(plt, [w], [h], color=:orange, markersize=5) - annotate!(plt, [(w/2, h/2, "Area=$(round(w*h,digits=1))")]) + annotate!(plt, [(w/2, h/2, L"Area$=\; %$(round(w*h,digits=1))$")]) plt end @@ -79,7 +79,7 @@ caption = """ Some possible rectangles that satisfy the constraint on the perimeter and their area. """ -n = 6 +n = 5 anim = @animate for i=1:n perimeter_area_graphic_graph(i-1) end @@ -187,8 +187,11 @@ ts = range(0, stop=pi, length=50) x1,y1 = 4, 4.85840 x2,y2 = 3, 6.1438 delta = 4 -p = plot(delta .+ x1*[0, 1,1,0], y1*[0,0,1,1], linetype=:polygon, fillcolor=:blue, legend=false) -plot!(p, x2*[0, 1,1,0], y2*[0,0,1,1], linetype=:polygon, fillcolor=:blue) +p = plot(delta .+ x1*[0, 1,1,0], y1*[0,0,1,1]; + linetype=:polygon, fillcolor=:blue, legend=false, + aspect_ratio=:equal) +plot!(p, x2*[0, 1,1,0], y2*[0,0,1,1]; + linetype=:polygon, fillcolor=:blue) plot!(p, delta .+ x1/2 .+ x1/2*cos.(ts), y1.+x1/2*sin.(ts), linetype=:polygon, fillcolor=:red) plot!(p, x2/2 .+ x2/2*cos.(ts), y2 .+ x2/2*sin.(ts), linetype=:polygon, fillcolor=:red) @@ -308,7 +311,7 @@ A₀ = w₀ * h₀ + pi * (w₀/2)^2 / 2 Perim = 2*h₀ + w₀ + pi * w₀/2 h₁ = solve(Perim - 20, h₀)[1] A₁ = A₀(h₀ => h₁) -w₁ = solve(diff(A₁,w₀), w₀)[1] +w₁ = solve(diff(A₁,w₀) ~ 0, w₀)[1] ``` We know that `w₀` is the maximum in this example from our previous work. We shall see soon, that just knowing that the second derivative is negative at `w₀` would suffice to know this. Here we check that condition: @@ -392,14 +395,29 @@ The figure shows a ladder of length $l_1 + l_2$ that got stuck - it was too long ```{julia} #| hold: true #| echo: false -p = plot([0, 0, 15], [15, 0, 0], color=:blue, legend=false) -plot!(p, [5, 5, 15], [15, 8, 8], color=:blue) -plot!(p, [0,14.53402874075368], [12.1954981558864, 0], linewidth=3) -plot!(p, [0,5], [8,8], color=:orange) -plot!(p, [5,5], [0,8], color=:orange) -annotate!(p, [(13, 1/2, "θ"), - (2.5, 11, "l₂"), (10, 5, "l₁"), (2.5, 7.0, "l₂ ⋅ cos(θ)"), - (5.1, 4, "l₁ ⋅ sin(θ)")]) +let + gr() + p = plot([0, 0, 15], [15, 0, 0], + xticks = [0,5, 15], + yticks = [0,8, 12], + line=(:blue, 2), + legend=false) + plot!(p, [5, 5, 15], [15, 8, 8]; line=(:blue,2)) + plot!(p, [0,14.53402874075368], [12.1954981558864, 0], linewidth=3) + plot!(p, [0,5], [8,8], color=:orange) + plot!(p, [5,5], [0,8], color=:orange) + annotate!(p, [(13, 1/2, L"\theta"), + (2.5, 11, L"l_2"), + (10, 5, L"l_1"), + (2.5, 7.0, L"l_2 \cos(\theta)"), + (5.1, 4, text(L"l_1 \sin(\theta)", :top,rotation=90))]) +end +``` + +```{julia} +#| echo: false +plotly() +nothing ``` We approach this problem in reverse. It is easy to see when a ladder is too long. It gets stuck at some angle $\theta$. So for each $\theta$ we find that ladder length that is just too long. Then we find the minimum length of all these ladders that are too long. If a ladder is this length or more it will get stuck for some angle. However, if it is less than this length it will not get stuck. So to maximize a ladder length, we minimize a different function. Neat. @@ -834,10 +852,12 @@ A rancher with $10$ meters of fence wishes to make a pen adjacent to an existing ```{julia} #| hold: true #| echo: false -p = plot(; legend=false, aspect_ratio=:equal, axis=nothing, border=:none) + p = plot(; legend=false, aspect_ratio=:equal, axis=nothing, border=:none) + plot!([0,10, 10, 0, 0], [0,0,10,10,0]; linewidth=3) plot!(p, [10,14,14,10], [2, 2, 8,8]; linewidth = 1) -annotate!(p, [(15, 5, "x"), (12,1, "y")]) + +annotate!(p, [(14-0.1, 5, text("x", :right)), (12,2, text("y",:bottom))]) p ``` @@ -1353,7 +1373,12 @@ p = 1/2 x = a/p plot!(plt, [0, b*(1+p), 0, 0], [0, 0, a+x, 0]) plot!(plt, [b,b,0,0],[0,a,a,0]) -annotate!(plt, [(b/2,0, "b"), (0,a/2,"a"), (0,a+x/2,"x"), (b+b*p/2,0,"bp")]) +annotate!(plt, [ + (b/2,0, text("b",:top)), + (0,a/2, text("a",:right)), + (0,a+x/2, text("x",:right)), + (b+b*p/2,0, text("bp",:top)) +]) plt ``` diff --git a/quarto/derivatives/related_rates.qmd b/quarto/derivatives/related_rates.qmd index f6fd169..8997205 100644 --- a/quarto/derivatives/related_rates.qmd +++ b/quarto/derivatives/related_rates.qmd @@ -18,7 +18,7 @@ using SymPy --- -Related rates problems involve two (or more) unknown quantities that are related through an equation. As the two variables depend on each other, also so do their rates - change with respect to some variable which is often time, though exactly how remains to be discovered. Hence the name "related rates." +Related rates problems involve two (or more) unknown quantities that are related through an equation. As the two variables depend on each other, also so do their rates - change with respect to some variable which is often time. Exactly how remains to be discovered. Hence the name "related rates." #### Examples @@ -27,7 +27,7 @@ Related rates problems involve two (or more) unknown quantities that are related The following is a typical "book" problem: -> A screen saver displays the outline of a $3$ cm by $2$ cm rectangle and then expands the rectangle in such a way that the $2$ cm side is expanding at the rate of $4$ cm/sec and the proportions of the rectangle never change. How fast is the area of the rectangle increasing when its dimensions are $12$ cm by $8$ cm? [Source.](http://oregonstate.edu/instruct/mth251/cq/Stage9/Practice/ratesProblems.html) +> A *vintage* screen saver displays the outline of a $3$ cm by $2$ cm rectangle and then expands the rectangle in such a way that the $2$ cm side is expanding at the rate of $4$ cm/sec and the proportions of the rectangle never change. How fast is the area of the rectangle increasing when its dimensions are $12$ cm by $8$ cm? [Source.](http://oregonstate.edu/instruct/mth251/cq/Stage9/Practice/ratesProblems.html) @@ -125,7 +125,7 @@ w(t) = 2 + 4*t ```{julia} -h(t) = 3/2 * w(t) +h(t) = 3 * w(t) / 2 ``` This means again that area depends on $t$ through this formula: @@ -198,6 +198,50 @@ A ladder, with length $l$, is leaning against a wall. We parameterize this probl If the ladder starts to slip away at the base, but remains in contact with the wall, express the rate of change of $h$ with respect to $t$ in terms of $db/dt$. +```{julia} +#| echo: false +let + gr() + l = 12 + b = 6 + h = sqrt(l^2 - b^2) + + plot(; + axis=([],false), + legend=false, + aspect_ratio=:equal) + P,Q = (0,h),(b,0) + w = 0.2 + S = Shape([-w,0,0,-w],[0,0,h+1,h+1]) + plot!(S; fillstyle=:/, fillcolor=:gray80, fillalpha=0.5) + R = Shape([-w,b+2,b+2,-w],[-w,-w,0,0]) + plot!(R, fill=(:gray, 0.25)) + + plot!([P,Q]; line=(:black, 2)) + scatter!([P,Q]) + b′ = b + 3/2 + h′ = sqrt(l^2 - b′^2) + + plot!([b,b′],[0,0]; arrow=true, side=:head, line=(:blue, 3)) + plot!([0,0], [h,h′]; arrow=true, side=:head, line=(:blue, 3)) + + annotate!([ + (b,-w,text(L"(b(t),0)",:top)), + (-w, h, text(L"(0,h(t))", :bottom, rotation=90)), + (b/2, h/2, text(L"L", rotation = -atand(h,b), :bottom)) + + ]) + current() +end +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + + We have from implicitly differentiating in $t$ the equation $l^2 = h^2 + b^2$, noting that $l$ is a constant, that: @@ -236,7 +280,7 @@ As $b$ goes to $l$, $h$ goes to $0$, so $b/h$ blows up. Unless $db/dt$ goes to $ :::{.callout-note} ## Note -Often, this problem is presented with $db/dt$ having a constant rate. In this case, the ladder problem defies physics, as $dh/dt$ eventually is faster than the speed of light as $h \rightarrow 0+$. In practice, were $db/dt$ kept at a constant, the ladder would necessarily come away from the wall. The trajectory would follow that of a tractrix were there no gravity to account for. +Often, this problem is presented with $db/dt$ having a constant rate. In this case, the ladder problem defies physics, as $dh/dt$ eventually is faster than the speed of light as $h \rightarrow 0+$. In practice, were $db/dt$ kept at a constant, the ladder would necessarily come away from the wall. ::: @@ -247,12 +291,15 @@ Often, this problem is presented with $db/dt$ having a constant rate. In this ca ```{julia} #| hold: true #| echo: false +#| eval: false caption = "A man and woman walk towards the light." imgfile = "figures/long-shadow-noir.png" ImageFile(:derivatives, imgfile, caption) ``` +![A man and woman walk towards the light](./figures/long-shadow-noir.png) + Shadows are a staple of film noir. In the photo, suppose a man and a woman walk towards a street light. As they approach the light the length of their shadow changes. @@ -340,7 +387,7 @@ This can be solved for the unknown: $dx/dt = 50/20$. A batter hits a ball toward third base at $75$ ft/sec and runs toward first base at a rate of $24$ ft/sec. At what rate does the distance between the ball and the batter change when $2$ seconds have passed? -We will answer this with `SymPy`. First we create some symbols for the movement of the ball towards third base, `b(t)`, the runner toward first base, `r(t)`, and the two velocities. We use symbolic functions for the movements, as we will be differentiating them in time: +We will answer this symbolically. First we create some symbols for the movement of the ball towards third base, `b(t)`, the runner toward first base, `r(t)`, and the two velocities. We use symbolic functions for the movements, as we will be differentiating them in time: ```{julia} diff --git a/quarto/derivatives/taylor_series_polynomials.qmd b/quarto/derivatives/taylor_series_polynomials.qmd index e44f00c..2b95e42 100644 --- a/quarto/derivatives/taylor_series_polynomials.qmd +++ b/quarto/derivatives/taylor_series_polynomials.qmd @@ -42,12 +42,14 @@ gr() taylor(f, x, c, n) = series(f, x, c, n+1).removeO() function make_taylor_plot(u, a, b, k) k = 2k - plot(u, a, b, title="plot of T_$k", linewidth=5, legend=false, size=fig_size, ylim=(-2,2.5)) - if k == 1 - plot!(zero, range(a, stop=b, length=100)) - else - plot!(taylor(u, x, 0, k), range(a, stop=b, length=100)) - end + plot(u, a, b; + title = L"plot of $T_{%$k}$", + line = (:black, 3), + legend = false, + size = fig_size, + ylim = (-2,2.5)) + fn = k == 1 ? zero : taylor(u, x, 0, k) + plot!(fn, range(a, stop=b, length=100); line=(:red,2)) end @@ -76,7 +78,7 @@ ImageFile(imgfile, caption) ## The secant line and the tangent line -We approach this general problem **much** more indirectly than is needed. We introduce notations that are attributed to Newton and proceed from there. By leveraging `SymPy` we avoid tedious computations and *hopefully* gain some insight. +Heads up: we approach this general problem **much** more indirectly than is needed by introducing notations that are attributed to Newton and proceed from there. By leveraging `SymPy` we avoid tedious computations and *hopefully* gain some insight. Suppose $f(x)$ is a function which is defined in a neighborhood of $c$ and has as many continuous derivatives as we care to take at $c$. @@ -102,7 +104,10 @@ $$ tl(x) = f(c) + f'(c) \cdot(x - c). $$ -The key is the term multiplying $(x-c)$ for the secant line this is an approximation to the related term for the tangent line. That is, the secant line approximates the tangent line, which is the linear function that best approximates the function at the point $(c, f(c))$. This is quantified by the *mean value theorem* which states under our assumptions on $f(x)$ that there exists some $\xi$ between $x$ and $c$ for which: +The key is the term multiplying $(x-c)$ -- for the secant line this is an approximation to the related term for the tangent line. That is, the secant line approximates the tangent line, which is the linear function that best approximates the function at the point $(c, f(c))$. + + +This is quantified by the *mean value theorem* which states under our assumptions on $f(x)$ that there exists some $\xi$ between $x$ and $c$ for which: $$ @@ -189,7 +194,7 @@ function divided_differences(f, x, xs...) end ``` -In the following, by adding a `getindex` method, we enable the `[]` notation of Newton to work with symbolic functions, like `u()` defined below, which is used in place of $f$: +In the following--even though it is *type piracy*--by adding a `getindex` method, we enable the `[]` notation of Newton to work with symbolic functions, like `u()` defined below, which is used in place of $f$: ```{julia} @@ -199,48 +204,38 @@ Base.getindex(u::SymFunction, xs...) = divided_differences(u, xs...) ex = u[c, c+h] ``` -We can take a limit and see the familiar (yet differently represented) value of $u'(c)$: +A limit as $h\rightarrow 0$ would show a value of $u'(c)$. -```{julia} -limit(ex, h => 0) -``` - -The choice of points is flexible. Here we use $c-h$ and $c+h$: - - -```{julia} -limit(u[c-h, c+h], h=>0) -``` Now, let's look at: ```{julia} ex₂ = u[c, c+h, c+2h] -simplify(ex₂) ``` -Not so bad after simplification. The limit shows this to be an approximation to the second derivative divided by $2$: - +If multiply by $2$ and simplify, a discrete approximation for the second derivative--the second order forward [difference equation](http://tinyurl.com/n4235xy)--is seen: ```{julia} -limit(ex₂, h => 0) +simplify(2ex₂) ``` -(The expression is, up to a divisor of $2$, the second order forward [difference equation](http://tinyurl.com/n4235xy), a well-known approximation to $f''$.) - This relationship between higher-order divided differences and higher-order derivatives generalizes. This is expressed in this [theorem](http://tinyurl.com/zjogv83): -> Suppose $m=x_0 < x_1 < x_2 < \dots < x_n=M$ are distinct points. If $f$ has $n$ continuous derivatives then there exists a value $\xi$, where $m < \xi < M$, satisfying: +:::{.callout-note} +## Mean value theorem for Divided differences + +Suppose $m=x_0 < x_1 < x_2 < \dots < x_n=M$ are distinct points. If $f$ has $n$ continuous derivatives then there exists a value $\xi$, where $m < \xi < M$, satisfying: $$ f[x_0, x_1, \dots, x_n] = \frac{1}{n!} \cdot f^{(n)}(\xi). $$ +::: This immediately applies to the above, where we parameterized by $h$: $x_0=c, x_1=c+h, x_2 = c+2h$. For then, as $h$ goes to $0$, it must be that $m, M \rightarrow c$, and so the limit of the divided differences must converge to $(1/2!) \cdot f^{(2)}(c)$, as $f^{(2)}(\xi)$ converges to $f^{(2)}(c)$. @@ -496,16 +491,20 @@ f[x_0] &+ f[x_0,x_1] \cdot (x - x_0) + f[x_0, x_1, x_2] \cdot (x - x_0)\cdot(x-x $$ -and taking $x_i = c + i\cdot h$, for a given $n$, we have in the limit as $h > 0$ goes to zero that coefficients of this polynomial converge to the coefficients of the *Taylor Polynomial of degree n*: - +and taking $x_i = c + i\cdot h$, for a given $n$, we have in the limit as $h > 0$ goes to zero that coefficients of this polynomial converge: +:::{.callout-note} +## Taylor polynomial of degree $n$ +Suppose $f(x)$ has $n+1$ derivatives (continuous on $c$ and $x$), then $$ -f(c) + f'(c)\cdot(x-c) + \frac{f''(c)}{2!}(x-c)^2 + \cdots + \frac{f^{(n)}(c)}{n!} (x-c)^n. +T_n(x) = f(c) + f'(c)\cdot(x-c) + \frac{f''(c)}{2!}(x-c)^2 + \cdots + \frac{f^{(n)}(c)}{n!} (x-c)^n, $$ -This polynomial will be the best approximation of degree $n$ or less to the function $f$, near $c$. The error will be given - again by an application of the Cauchy mean value theorem: +will be the best approximation of degree $n$ or less to $f$, near $c$. + +The error will be given - again by an application of the Cauchy mean value theorem: $$ @@ -513,9 +512,10 @@ $$ $$ for some $\xi$ between $c$ and $x$. +::: -The Taylor polynomial for $f$ about $c$ of degree $n$ can be computed by taking $n$ derivatives. For such a task, the computer is very helpful. In `SymPy` the `series` function will compute the Taylor polynomial for a given $n$. For example, here is the series expansion to 10 terms of the function $\log(1+x)$ about $c=0$: +The Taylor polynomial for $f$ about $c$ of degree $n$ can be computed by taking $n$ derivatives. For such a task, the computer is very helpful. In `SymPy` the `series` function will compute the Taylor polynomial for a given $n$. For example, here is the series expansion to $10$ terms of the function $\log(1+x)$ about $c=0$: ```{julia} @@ -803,15 +803,7 @@ Now, $2s = m - s\cdot m$, so the above can be reworked to be $\log(1+m) = m - s\ (For larger values of $m$, a similar, but different approximation, can be used to minimize floating point errors.) -How big can the error be between this *approximations* and $\log(1+m)$? We plot to see how big $s$ can be: - - -```{julia} -@syms v -plot(v/(2+v), sqrt(2)/2 - 1, sqrt(2)-1) -``` - -This shows, $s$ is as big as +How big can the error be between this *approximations* and $\log(1+m)$? The expression $m/(2+m)$ increases for $m > 0$, so, on this interval $s$ is as big as ```{julia} @@ -822,17 +814,17 @@ The error term is like $2/19 \cdot \xi^{19}$ which is largest at this value of ```{julia} -(2/19)*Max^19 +(2/19) * Max^19 ``` Basically that is machine precision. Which means, that as far as can be told on the computer, the value produced by $2s + s \cdot p$ is about as accurate as can be done. -To try this out to compute $\log(5)$. We have $5 = 2^2(1+0.25)$, so $k=2$ and $m=0.25$. +We try this out to compute $\log(5)$. We have $5 = 2^2(1+ 1/4)$, so $k=2$ and $m=1/4$. ```{julia} -k, m = 2, 0.25 +k, m = 2, 1/4 s = m / (2+m) pₗ = 2 * sum(s^(2i)/(2i+1) for i in 1:8) # where the polynomial approximates the logarithm... @@ -1209,7 +1201,10 @@ $$ $$ -h(x)=b_0 + b_1 (x-x_n) + b_2(x-x_n)(x-x_{n-1}) + \cdots + b_n (x-x_n)(x-x_{n-1})\cdot\cdots\cdot(x-x_1). +\begin{align*} +h(x)&=b_0 + b_1 (x-x_n) + b_2(x-x_n)(x-x_{n-1}) + \cdots \\ +&+ b_n (x-x_n)(x-x_{n-1})\cdot\cdots\cdot(x-x_1). +\end{align*} $$ These two polynomials are of degree $n$ or less and have $u(x) = h(x)-g(x)=0$, by uniqueness. So the coefficients of $u(x)$ are $0$. We have that the coefficient of $x^n$ must be $a_n-b_n$ so $a_n=b_n$. Our goal is to express $a_n$ in terms of $a_{n-1}$ and $b_{n-1}$. Focusing on the $x^{n-1}$ term, we have: diff --git a/quarto/differentiable_vector_calculus.qmd b/quarto/differentiable_vector_calculus.qmd index 7e8464c..b9cb392 100644 --- a/quarto/differentiable_vector_calculus.qmd +++ b/quarto/differentiable_vector_calculus.qmd @@ -1,3 +1,7 @@ +--- +engine: julia +--- + # Differential vector calculus This section discussions generalizations of the derivative to functions which have more than one input and/or one output. diff --git a/quarto/differentiable_vector_calculus/Project.toml b/quarto/differentiable_vector_calculus/Project.toml index 37832cc..abc5574 100644 --- a/quarto/differentiable_vector_calculus/Project.toml +++ b/quarto/differentiable_vector_calculus/Project.toml @@ -18,6 +18,7 @@ QuadGK = "1fd47b50-473d-5c70-9696-f719f8f3bcdc" QuizQuestions = "612c44de-1021-4a21-84fb-7261cf5eb2d4" Roots = "f2b01f46-fcfa-551c-844a-d8ac1e96c665" ScatteredInterpolation = "3f865c0f-6dca-5f4d-999b-29fe1e7e3c92" +SplitApplyCombine = "03a91e81-4c3e-53e1-a0a4-9c0c8f19dd66" SymPy = "24249f21-da20-56a4-8eb1-6a02cf4ae2e6" Tables = "bd369af6-aec1-5ad0-b16a-f7cc5008161c" TextWrap = "b718987f-49a8-5099-9789-dcd902bef87d" diff --git a/quarto/differentiable_vector_calculus/make_pdf.jl b/quarto/differentiable_vector_calculus/make_pdf.jl index b90af2c..8e48237 100644 --- a/quarto/differentiable_vector_calculus/make_pdf.jl +++ b/quarto/differentiable_vector_calculus/make_pdf.jl @@ -9,6 +9,7 @@ files = ( "scalar_functions", "scalar_functions_applications", "vector_fields", + "matrix_calculus_notes.qmd", "plots_plotting", ) diff --git a/quarto/differentiable_vector_calculus/vector_fields.qmd b/quarto/differentiable_vector_calculus/vector_fields.qmd index 3e3363f..8f612e7 100644 --- a/quarto/differentiable_vector_calculus/vector_fields.qmd +++ b/quarto/differentiable_vector_calculus/vector_fields.qmd @@ -80,6 +80,7 @@ Vector fields are also useful for other purposes, such as transformations, examp For transformations, a useful visualization is to plot curves where one variables is fixed. Consider the transformation from polar coordinates to cartesian coordinates $F(r, \theta) = r \langle\cos(\theta),\sin(\theta)\rangle$. The following plot will show in blue fixed values of $r$ (circles) and in red fixed values of $\theta$ (rays). +::: {#fig-transformation-partial-derivative} ```{julia} #| hold: true @@ -97,10 +98,21 @@ pt = [1, pi/4] J = ForwardDiff.jacobian(F, pt) arrow!(F(pt...), J[:,1], linewidth=5, color=:red) arrow!(F(pt...), J[:,2], linewidth=5, color=:blue) + +pt = [0.5, pi/8] +J = ForwardDiff.jacobian(F, pt) +arrow!(F(pt...), J[:,1], linewidth=5, color=:red) +arrow!(F(pt...), J[:,2], linewidth=5, color=:blue) + ``` +Plot of a vector field from $R^2 \rightarrow R^2$ illustrated by drawing curves with fixed $r$ and $\theta$. The partial derivatives are added as layers. +::: + To the plot, we added the partial derivatives with respect to $r$ (in red) and with respect to $\theta$ (in blue). These are found with the soon-to-be discussed Jacobian. From the graph, you can see that these vectors are tangent vectors to the drawn curves. +The curves form a non-rectangular grid. Were the cells exactly parallelograms, the area would be computed taking into account the length of the vectors and the angle between them -- the same values that come out of a cross product. + ## Parametrically defined surfaces @@ -138,7 +150,7 @@ When a surface is described as a level curve, $f(x,y,z) = c$, then the gradient When a surface is described parametrically, there is no "gradient." The *partial* derivatives are of interest, e.g., $\partial{F}/\partial{\theta}$ and $\partial{F}/\partial{\phi}$, vectors defined componentwise. These will be lie in the tangent plane of the surface, as they can be viewed as tangent vectors for parametrically defined curves on the surface. Their cross product will be *normal* to the surface. The magnitude of the cross product, which reflects the angle between the two partial derivatives, will be informative as to the surface area. -### Plotting parametrized surfaces in `Julia` +### Plotting parameterized surfaces in `Julia` Consider the parametrically described surface above. How would it be plotted? Using the `Plots` package, the process is quite similar to how a surface described by a function is plotted, but the $z$ values must be computed prior to plotting. @@ -236,6 +248,217 @@ arrow!(Phi(pt...), out₁[:,1], linewidth=3) arrow!(Phi(pt...), out₁[:,2], linewidth=3) ``` +##### Example: A detour into plotting + + +The presentation of a 3D figure in a 2D format requires the use of linear perspective. The `Plots` package adds lighting effects, to nicely render a surface, as seen. + +In this example, we see some of the mathematics behind how drawing a surface can be done more primitively to showcase some facts about vectors. We follow a few techniques learned from @Angenent. + +```{julia} +#| echo: false +gr() +nothing +``` + +For our purposes we wish to mathematically project a figure onto a 2D plane. + + + +The plane here is described by a view point in 3D space, $\vec{v}$. Taking this as one vector in an orthogonal coordinate system, the other two can be easily produced, the first by switching two coordinates, as would be done in 2D; the second through the cross product: + +```{julia} +function projection_plane(v) + vx, vy, vz = v + a = [-vy, vx, 0] # v ⋅ a = 0 + b = v × a # so v ⋅ b = 0 + return (a/norm(a), b/norm(b)) +end +``` + +Using these two unit vectors to describe the plane, the projection of a point onto the plane is simply found by taking dot products: + +```{julia} +function project(x, v) + â, b̂ = projection_plane(v) + (x ⋅ â, x ⋅ b̂) # (x ⋅ â) â + (x ⋅ b̂) b̂ +end +``` + +Let's see this in action by plotting a surface of revolution given by + +```{julia} +radius(t) = 1 / (1 + exp(t)) +t₀, tₙ = 0, 3 +surf(t, θ) = [t, radius(t)*cos(θ), radius(t)*sin(θ)] +``` + +We begin by fixing a view point and plotting the projected axes. We do the latter with a function for re-use. + +```{julia} +v = [2, -2, 1] +function plot_axes() + empty_style = (xaxis = ([], false), + yaxis = ([], false), + legend=false) + + plt = plot(; empty_style...) + + axis_values = [[(0,0,0), (3.5,0,0)], # x axis + [(0,0,0), (0, 2.0 * radius(0), 0)], # yaxis + [(0,0,0), (0, 0, 1.5 * radius(0))]] # z axis + + for (ps, ax) ∈ zip(axis_values, ("x", "y", "z")) + p0, p1 = ps + a, b = project(p0, v), project(p1, v) + annotate!([(b...,text(ax, :bottom))]) + plot!([a, b]; arrow=true, head=:tip, line=(:gray, 1)) # gr() allows arrows + end + + plt +end + +plt = plot_axes() +``` + +We are using the vector of tuples interface (representing points) to specify the curve to draw. + +Now we add on some curves for fixed $t$ and then fixed $\theta$ utilizing the fact that `project` returns a tuple of $x$--$y$ values to display. + +```{julia} +for t in range(t₀, tₙ, 20) + curve = [project(surf(t, θ), v) for θ in range(0, 2pi, 100)] + plot!(curve; line=(:black, 1)) +end +for θ in range(0, 2pi, 60) + curve = [project(surf(t, θ), v) for t in range(t₀, tₙ, 20)] + plot!(curve; line=(:black, 1)) +end +plt +``` + + + +The graphic is a little busy! + + +Let's focus on the cells layering the surface. These have equal size in the $t \times \theta$ range, but unequal area on the screen. Where they parallellograms, the area could be found by taking the 2-dimensional cross product of the two partial derivatives, resulting in a formula like: $a_x b_y - a_y b_x$. +When we discuss integrals related to such figures, this amount of area will be characterized by a computation involving the determinant of the upcoming Jacobian function. + +We make a function to close over the viewpoint vector that can be passed to `ForwardDiff`, as it will return a vector and not a tuple. + +```{julia} +function psurf(v) + (t,θ) -> begin + v1, v2 = project(surf(t, θ), v) + [v1, v2] # or call collect to make a tuple into a vector + end +end +``` + +The function returned by `psurf` is from $R^2 \rightarrow R^2$. With such a function, the computation of this approximate area becomes: + +```{julia} +function detJ(F, t, θ) + ∂θ = ForwardDiff.derivative(θ -> F(t, θ), θ) + ∂t = ForwardDiff.derivative(t -> F(t, θ), t) + (ax, ay), (bx, by) = ∂θ, ∂t + ax * by - ay * bx +end +``` + +For our purposes, we are interested in the sign of the returned value. Plotting, we can see that some "area" is positive, some "negative": + +```{julia} +t = 1 +G = psurf(v) +plot(θ -> detJ(G, t, θ), 0, 2pi) +``` + +With this parameterization and viewpoint, the positive area for the surface is when the normal vector points towards the viewing point. In the following, we only plot such values: + +```{julia} +plt = plot_axes() + +function I(F, t, θ) + x, y = F(t, θ) + detJ(F, t, θ) >= 0 ? (x, y) : (x, NaN) # use NaN for y value +end + +for t in range(t₀, tₙ, 20) + curve = [I(G, t, θ) for θ in range(0, 2pi, 100)] + plot!(curve; line=(:gray, 1)) +end + +for θ in range(0, 2pi, 60) + curve = [I(G, t, θ) for t in range(t₀, tₙ, 20)] + plot!(curve; line=(:gray, 1)) +end + +plt +``` + +The values for which `detJ` is zero form the visible boundary of the object. We can plot just those to get an even less busy view. We identify them by finding the value of $\theta$ in $[0,\pi]$ and $[\pi,2\pi]$ that makes the `detJ` function zero: + +```{julia} +fold(F, t, θmin, θmax) = find_zero(θ -> detJ(F, t, θ), (θmin, θmax)) + +ts = range(t₀, tₙ, 100) +back_edge = fold.(G, ts, 0, pi) +front_edge = fold.(G, ts, pi, 2pi) + +plt = plot_axes() +plot!(project.(surf.(ts, back_edge), (v,)); line=(:black, 1)) +plot!(project.(surf.(ts, front_edge), (v,)); line=(:black, 1)) +``` + +Adding caps makes the graphic stand out. The caps are just discs (fixed values of $t$) which are filled in with gray using a transparency so that the axes aren't masked. + +```{julia} +θs = range(0, 2pi, 100) +S = Shape(project.(surf.(t₀, θs), (v,))) +plot!(S; fill=(:gray, 0.33)) + +S = Shape(project.(surf.(tₙ, θs), (v,))) +plot!(S; fill=(:gray, 0.33)) +``` + + +Finally, we introduce some shading using the same technique but assuming the light comes from a different position. + +```{julia} +lightpt = [2, -2, 5] # from further above +H = psurf(lightpt) +light_edge = fold.(H, ts, pi, 2pi); +``` + +Angles between the light edge and the front edge would be in shadow. We indicate this by drawing lines for fixed $t$ values. As denser lines indicate more shadow, we feather how these are drawn: + +```{julia} +for (i, (t, top, bottom)) in enumerate(zip(ts, light_edge, front_edge)) + λ = iseven(i) ? 1.0 : 0.8 + top = bottom + λ*(top - bottom) + curve = [project(surf(t, θ), v) for θ in range(bottom, top, 20)] + plot!(curve, line=(:black, 1)) +end +plt +``` + +We can compare to the graph produced by `surface` for the same function: + +```{julia} +ts = range(t₀, tₙ, 50) +θs = range(0, 2pi, 100) +surface(unzip(surf.(ts, θs'))...; legend=false) +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + + ## The total derivative diff --git a/quarto/integrals.qmd b/quarto/integrals.qmd index 1e48fbf..6bbc73a 100644 --- a/quarto/integrals.qmd +++ b/quarto/integrals.qmd @@ -1,3 +1,7 @@ +--- +engine: julia +--- + # Integrals Identifying the area under a curve between two values is an age-old problem. In this chapter we see that for many case the Fundamental Theorem of Calculus can be used to identify the area. When not applicable, we will see how such areas may be accurately estimated. diff --git a/quarto/integrals/_sa_bone_pile.qmd b/quarto/integrals/_sa_bone_pile.qmd new file mode 100644 index 0000000..7b42098 --- /dev/null +++ b/quarto/integrals/_sa_bone_pile.qmd @@ -0,0 +1,41 @@ +# Appendix + +```{julia} +#| hold: true +#| echo: false +gr() +## For **some reason** having this in the natural place messes up the plots. +## {{{approximate_surface_area}}} + +xs,ys = range(-1, stop=1, length=50), range(-1, stop=1, length=50) +f(x,y)= 2 - (x^2 + y^2) + +dr = [1/2, 3/4] +df = [f(dr[1],0), f(dr[2],0)] + +function sa_approx_graph(i) + p = plot(xs, ys, f, st=[:surface], legend=false) + for theta in range(0, stop=i/10*2pi, length=10*i ) + path3d!(p,sin(theta)*dr, cos(theta)*dr, df) + end + p +end +n = 10 + +anim = @animate for i=1:n + sa_approx_graph(i) +end + +imgfile = tempname() * ".gif" +gif(anim, imgfile, fps = 1) + + +caption = L""" + +Surface of revolution of $f(x) = 2 - x^2$ about the $y$ axis. The lines segments are the images of rotating the secant line connecting $(1/2, f(1/2))$ and $(3/4, f(3/4))$. These trace out the frustum of a cone which approximates the corresponding surface area of the surface of revolution. In the limit, this approximation becomes exact and a formula for the surface area of surfaces of revolution can be used to compute the value. + +""" + +plotly() +ImageFile(imgfile, caption) +``` diff --git a/quarto/integrals/arc_length.qmd b/quarto/integrals/arc_length.qmd index e74509a..0e9d159 100644 --- a/quarto/integrals/arc_length.qmd +++ b/quarto/integrals/arc_length.qmd @@ -95,11 +95,11 @@ function make_arclength_graph(n) ts = range(0, 2pi, 100) - λ = 0.005 - cs = [λ .* xys for xys ∈ sincos.(ts)] + λ = 0.01 + C = Plots.scale(Shape(:circle), λ) - for v ∈ zip(x.(pttn), y.(pttn)) - S = Shape([v .+ xy for xy in cs]) + for (u,v) ∈ zip(x.(pttn), y.(pttn)) + S = Plots.translate(C, u,v) plot!(S; fill=(:white,), line=(:black,2)) end @@ -555,7 +555,9 @@ plot(t -> g(𝒔(t)), t -> f(𝒔(t)), 0, sinv(2*pi)) Following (faithfully) [Kantorwitz and Neumann](https://www.researchgate.net/publication/341676916_The_English_Galileo_and_His_Vision_of_Projectile_Motion_under_Air_Resistance), we consider a function $f(x)$ with the property that **both** $f$ and $f'$ are strictly concave down on $[a,b]$ and suppose $f(a) = f(b)$. Further, assume $f'$ is continuous. We will see this implies facts about arc-length and other integrals related to $f$. -The following figure is clearly of a concave down function. The asymmetry about the critical point will be seen to be a result of the derivative also being concave down. This asymmetry will be characterized in several different ways in the following including showing that the arc length from $(a,0)$ to $(c,f(c))$ is longer than from $(c,f(c))$ to $(b,0)$. +@fig-kantorwitz-neumann is clearly of a concave down function. The asymmetry about the critical point will be seen to be a result of the derivative also being concave down. This asymmetry will be characterized in several different ways in the following including showing that the arc length from $(a,0)$ to $(c,f(c))$ is longer than from $(c,f(c))$ to $(b,0)$. + +::: {#@fig-kantorwitz-neumann} ```{julia} @@ -590,7 +592,12 @@ plot!(zero) annotate!([(0, 𝒚, "a"), (152, 𝒚, "b"), (u, 𝒚, "u"), (v, 𝒚, "v"), (c, 𝒚, "c")]) ``` -Take $a < u < c < v < b$ with $f(u) = f(v)$ and $c$ a critical point, as in the picture. There must be a critical point by Rolle's theorem, and it must be unique, as the derivative, which exists by the assumptions, must be strictly decreasing due to concavity of $f$ and hence there can be at most $1$ critical point. +Graph of function $f(x)$ with both $f$ and $f'$ strictly concave down. +::: + +By Rolle's theorem there exists $c$ in $(a,b)$, a critical point, as in the picture. There must be a critical point by Rolle's theorem, and it must be unique, as the derivative, which exists by the assumptions, must be strictly decreasing due to concavity of $f$ and hence there can be at most $1$ critical point. + +Take $a < u < c < v < b$ with $f(u) = f(v)$. Some facts about this picture can be proven from the definition of concavity: @@ -653,7 +660,7 @@ By the fundamental theorem of calculus: $$ -(f_1^{-1}(y) + f_2^{-1}(y))\big|_\alpha^\beta > 0 +(f_1^{-1}(y) + f_2^{-1}(y))\Big|_\alpha^\beta > 0 $$ On rearranging: diff --git a/quarto/integrals/area.qmd b/quarto/integrals/area.qmd index b5701a1..7882d29 100644 --- a/quarto/integrals/area.qmd +++ b/quarto/integrals/area.qmd @@ -16,6 +16,8 @@ using Roots --- +![A jigsaw puzzle needs a certain amount of area to complete. For a traditional rectangular puzzle, this area is comprised of the sum of the areas for each piece. Decomposing a total area into the sum of smaller, known, ones--even if only approximate--is the basis of definite integration.](figures/jigsaw.png) + The question of area has long fascinated human culture. As children, we learn early on the formulas for the areas of some geometric figures: a square is $b^2$, a rectangle $b\cdot h$, a triangle $1/2 \cdot b \cdot h$ and for a circle, $\pi r^2$. The area of a rectangle is often the intuitive basis for illustrating multiplication. The area of a triangle has been known for ages. Even complicated expressions, such as [Heron's](http://tinyurl.com/mqm9z) formula which relates the area of a triangle with measurements from its perimeter have been around for 2000 years. The formula for the area of a circle is also quite old. Wikipedia dates it as far back as the [Rhind](http://en.wikipedia.org/wiki/Rhind_Mathematical_Papyrus) papyrus for 1700 BC, with the approximation of $256/81$ for $\pi$. @@ -81,39 +83,46 @@ gr() f(x) = x^2 colors = [:black, :blue, :orange, :red, :green, :orange, :purple] -## Area of parabola - ## Area of parabola function make_triangle_graph(n) title = "Area of parabolic cup ..." - n==1 && (title = "Area = 1/2") - n==2 && (title = "Area = previous + 1/8") - n==3 && (title = "Area = previous + 2*(1/8)^2") - n==4 && (title = "Area = previous + 4*(1/8)^3") - n==5 && (title = "Area = previous + 8*(1/8)^4") - n==6 && (title = "Area = previous + 16*(1/8)^5") - n==7 && (title = "Area = previous + 32*(1/8)^6") + n==1 && (title = L"Area $= 1/2$") + n==2 && (title = L"Area $=$ previous $+\; \frac{1}{8}$") + n==3 && (title = L"Area $=$ previous $+\; 2\cdot\frac{1}{8^2}$") + n==4 && (title = L"Area $=$ previous $+\; 4\cdot\frac{1}{8^3}$") + n==5 && (title = L"Area $=$ previous $+\; 8\cdot\frac{1}{8^4}$") + n==6 && (title = L"Area $=$ previous $+\; 16\cdot\frac{1}{8^5}$") + n==7 && (title = L"Area $=$ previous $+\; 32\cdot\frac{1}{8^6}$") - plt = plot(f, 0, 1, legend=false, size = fig_size, linewidth=2) - annotate!(plt, [(0.05, 0.9, text(title,:left))]) # if in title, it grows funny with gr - n >= 1 && plot!(plt, [1,0,0,1, 0], [1,1,0,1,1], color=colors[1], linetype=:polygon, fill=colors[1], alpha=.2) - n == 1 && plot!(plt, [1,0,0,1, 0], [1,1,0,1,1], color=colors[1], linewidth=2) + plt = plot(f, 0, 1; + legend=false, + size = fig_size, + linewidth=2) + annotate!(plt, [ + (0.05, 0.9, text(title,:left)) + ]) # if in title, it grows funny with gr + n >= 1 && plot!(plt, [1,0,0,1, 0], [1,1,0,1,1]; + color=colors[1], linetype=:polygon, + fill=colors[1], alpha=.2) + n == 1 && plot!(plt, [1,0,0,1, 0], [1,1,0,1,1]; + color=colors[1], linewidth=2) for k in 2:n xs = range(0, stop=1, length=1+2^(k-1)) - ys = map(f, xs) - k < n && plot!(plt, xs, ys, linetype=:polygon, fill=:black, alpha=.2) + ys = f.(xs) + k < n && plot!(plt, xs, ys; + linetype=:polygon, fill=:black, alpha=.2) if k == n - plot!(plt, xs, ys, color=colors[k], linetype=:polygon, fill=:black, alpha=.2) - plot!(plt, xs, ys, color=:black, linewidth=2) + plot!(plt, xs, ys; + color=colors[k], linetype=:polygon, fill=:black, alpha=.2) + plot!(plt, xs, ys; + color=:black, linewidth=2) end end plt end - - n = 7 anim = @animate for i=1:n make_triangle_graph(i) @@ -183,7 +192,7 @@ $$ S_n = f(c_1) \cdot (x_1 - x_0) + f(c_2) \cdot (x_2 - x_1) + \cdots + f(c_n) \cdot (x_n - x_{n-1}). $$ -Clearly for a given partition and choice of $c_i$, the above can be computed. Each term $f(c_i)\cdot(x_i-x_{i-1})$ can be visualized as the area of a rectangle with base spanning from $x_{i-1}$ to $x_i$ and height given by the function value at $c_i$. The following visualizes left Riemann sums for different values of $n$ in a way that makes Beekman's intuition plausible – that as the number of rectangles gets larger, the approximate sum will get closer to the actual area. +Clearly for a given partition and choice of $c_i$, the above can be computed. Each term $f(c_i)\cdot(x_i-x_{i-1}) = f(c_i)\Delta_i$ can be visualized as the area of a rectangle with base spanning from $x_{i-1}$ to $x_i$ and height given by the function value at $c_i$. The following visualizes left Riemann sums for different values of $n$ in a way that makes Beekman's intuition plausible – that as the number of rectangles gets larger, the approximate sum will get closer to the actual area. ```{julia} @@ -354,7 +363,7 @@ When the integral exists, it is written $V = \int_a^b f(x) dx$. :::{.callout-note} ## History note -The expression $V = \int_a^b f(x) dx$ is known as the *definite integral* of $f$ over $[a,b]$. Much earlier than Riemann, Cauchy had defined the definite integral in terms of a sum of rectangular products beginning with $S=(x_1 - x_0) f(x_0) + (x_2 - x_1) f(x_1) + \cdots + (x_n - x_{n-1}) f(x_{n-1})$ (the left Riemann sum). He showed the limit was well defined for any continuous function. Riemann's formulation relaxes the choice of partition and the choice of the $c_i$ so that integrability can be better understood. +The expression $V = \int_a^b f(x) dx$ is known as the *definite integral* of $f$ over $[a,b]$. Much earlier than Riemann, Cauchy had defined the definite integral in terms of a sum of rectangular products beginning with $S=f(x_0) \cdot (x_1 - x_0) + f(x_1) \cdot (x_2 - x_1) + \cdots + f(x_{n-1}) \cdot (x_n - x_{n-1}) $ (the left Riemann sum). He showed the limit was well defined for any continuous function. Riemann's formulation relaxes the choice of partition and the choice of the $c_i$ so that integrability can be better understood. ::: @@ -364,18 +373,6 @@ The expression $V = \int_a^b f(x) dx$ is known as the *definite integral* of $f$ The following formulas are consequences when $f(x)$ is integrable. These mostly follow through a judicious rearranging of the approximating sums. -The area is $0$ when there is no width to the interval to integrate over: - - -> $$ -> \int_a^a f(x) dx = 0. -> $$ - - - -Even our definition of a partition doesn't really apply, as we assume $a < b$, but clearly if $a=x_0=x_n=b$ then our only"approximating" sum could be $f(a)(b-a) = 0$. - - The area under a constant function is found from the area of rectangle, a special case being $c=0$ yielding $0$ area: @@ -388,16 +385,65 @@ The area under a constant function is found from the area of rectangle, a specia For any partition of $a < b$, we have $S_n = c(x_1 - x_0) + c(x_2 -x_1) + \cdots + c(x_n - x_{n-1})$. By factoring out the $c$, we have a *telescoping sum* which means the sum simplifies to $S_n = c(x_n-x_0) = c(b-a)$. Hence any limit must be this constant value. -Scaling the $y$ axis by a constant can be done before or after computing the area: + +::: {#fig-consequence-rectangle-area} +```{julia} +#| echo: false +gr() +let + c = 1 + a,b = 0.5, 1.5 + f(x) = c + Δ = 0.1 + plt = plot(; + xaxis=([], false), + yaxis=([], false), + legend=false, + ) + + plot!(f, a, b; line=(:black, 2)) + plot!([a-Δ, b + Δ], [0,0]; line=(:gray, 1), arrow=true, side=:head) + plot!([a-Δ/2, a-Δ/2], [-Δ, c + Δ]; line=(:gray, 1), arrow=true, side=:head) + plot!([a,a],[0,f(a)]; line=(:black, 1, :dash)) + plot!([b,b],[0,f(b)]; line=(:black, 1, :dash)) + + annotate!([ + (a, 0, text(L"a", :top, :right)), + (b, 0, text(L"b", :top, :left)), + (a-Δ/2-0.01, c, text(L"c", :right)) + ]) + current() +end +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + +Illustration that the area under a constant function is that of a rectangle +::: + + + + + +The area is $0$ when there is no width to the interval to integrate over: > $$ -> \int_a^b cf(x) dx = c \int_a^b f(x) dx. +> \int_a^a f(x) dx = 0. > $$ -Let $a=x_0 < x_1 < \cdots < x_n=b$ be any partition. Then we have $S_n= cf(c_1)(x_1-x_0) + \cdots + cf(c_n)(x_n-x_{n-1})$ $=$ $c\cdot\left[ f(c_1)(x_1 - x_0) + \cdots + f(c_n)(x_n - x_{n-1})\right]$. The "limit" of the left side is $\int_a^b c f(x) dx$. The "limit" of the right side is $c \cdot \int_a^b f(x)$. We call this a "sketch" as a formal proof would show that for any $\epsilon$ we could choose a $\delta$ so that any partition with norm $\delta$ will yield a sum less than $\epsilon$. Here, then our "any" partition would be one for which the $\delta$ on the left hand side applies. The computation shows that the same $\delta$ would apply for the right hand side when $\epsilon$ is the same. +Even our definition of a partition doesn't really apply, as we assume $a < b$, but clearly if $a=x_0=x_n=b$ then our only"approximating" sum could be $f(a)(b-a) = 0$. + + +#### Shifts + +A jigsaw puzzle piece will have the same area if it is moved around on the table or flipped over. Similarly some shifts preserve area under a function. The area is invariant under shifts left or right. @@ -424,6 +470,59 @@ $$ The left side will have a limit of $\int_a^b f(x-c) dx$ the right would have a "limit" of $\int_{a-c}^{b-c}f(x)dx$. + + +::: {#fig-consequence-rectangle-area} +```{julia} +#| echo: false +gr() +let + f(x) = 2 + cospi(x^2/10)*sinpi(x) + plt = plot(; + xaxis=([], false), + yaxis=([], false), + legend=false, + ) + + a, b = 0,4 + c = 5 + plot!(f, a, b; line=(:black, 2)) + plot!(x -> f(x-c), a+c, b+c; line=(:red, 2)) + + plot!([-1, b+c + 1], [0,0]; line=(:gray, 2), arrow=true, side=:head) + for x ∈ (a,b) + plot!([x,x],[0,f(x)]; line=(:black,1, :dash)) + end + + for x ∈ (a+c,b+c) + plot!([x,x],[0,f(x)]; line=(:red,1, :dash)) + end + + annotate!([ + (a+c,0, text(L"a", :top)), + (b+c,0, text(L"b", :top)), + (a,0, text(L"a-c", :top)), + (b,0, text(L"b-c", :top)), + (1.0, 3, text(L"f(x)",:left)), + (1.0+c, 3, text(L"f(x-c)",:left)), + + + ]) + current() +end +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + +Illustration that the area under shift remains the same +::: + + + Similarly, reflections don't effect the area under the curve, they just require a new parameterization: @@ -433,7 +532,79 @@ Similarly, reflections don't effect the area under the curve, they just require -The scaling operation $g(x) = f(cx)$ has the following: +::: {#fig-consequence-reflect-area} +```{julia} +#| echo: false +gr() +let + f(x) = 2 + cospi(x^2/10)*sinpi(x) + g(x) = f(-x) + plt = plot(; + xaxis=([], false), + yaxis=([], false), + legend=false, + ) + + a, b = 1,4 + + plot!(f, a, b; line=(:black, 2)) + plot!(g, -b, -a; line=(:red, 2)) + + plot!([-5, 5], [0,0]; line=(:gray,1), arrow=true, side=:head) + plot!([0,0], [-0.1,3.15]; line=(:gray,1), arrow=true, side=:head) + + for x in (a,b) + plot!([x,x], [0,f(x)]; line=(:black,1,:dash)) + plot!([-x,-x], [0,g(-x)]; line=(:red,1,:dash)) + end + + annotate!([ + (a,0, text(L"a", :top)), + (b,0, text(L"b", :top)), + (-a,0, text(L"-a", :top)), + (-b,0, text(L"-b", :top)), + + ]) + current() +end +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + +Illustration that the area remains constant under reflection through $y$ axis. +::: + + + +The "reversed" area is the same, only accounted for with a minus sign. + + +> $$ +> \int_a^b f(x) dx = -\int_b^a f(x) dx. +> $$ + + + +#### Scaling + +Scaling the $y$ axis by a constant can be done before or after computing the area: + + +> $$ +> \int_a^b cf(x) dx = c \int_a^b f(x) dx. +> $$ + + + +Let $a=x_0 < x_1 < \cdots < x_n=b$ be any partition. Then we have $S_n= cf(c_1)(x_1-x_0) + \cdots + cf(c_n)(x_n-x_{n-1})$ $=$ $c\cdot\left[ f(c_1)(x_1 - x_0) + \cdots + f(c_n)(x_n - x_{n-1})\right]$. The "limit" of the left side is $\int_a^b c f(x) dx$. The "limit" of the right side is $c \cdot \int_a^b f(x)$. We call this a "sketch" as a formal proof would show that for any $\epsilon$ we could choose a $\delta$ so that any partition with norm $\delta$ will yield a sum less than $\epsilon$. Here, then our "any" partition would be one for which the $\delta$ on the left hand side applies. The computation shows that the same $\delta$ would apply for the right hand side when $\epsilon$ is the same. + + + +The scaling operation on the $x$ axis, $g(x) = f(cx)$, has the following property: > $$ @@ -448,8 +619,9 @@ The scaling operation shifts $a$ to $ca$ and $b$ to $cb$ so the limits of integr Combining two operations above, the operation $g(x) = \frac{1}{h}f(\frac{x-c}{h})$ will leave the area between $a$ and $b$ under $g$ the same as the area under $f$ between $(a-c)/h$ and $(b-c)/h$. ---- +#### Area is additive +When two jigsaw pieces interlock their combined area is that of each added. This also applies to areas under functions. The area between $a$ and $b$ can be broken up into the sum of the area between $a$ and $c$ and that between $c$ and $b$. @@ -463,35 +635,185 @@ The area between $a$ and $b$ can be broken up into the sum of the area between $ For this, suppose we have a partition for both the integrals on the right hand side for a given $\epsilon/2$ and $\delta$. Combining these into a partition of $[a,b]$ will mean $\delta$ is still the norm. The approximating sum will combine to be no more than $\epsilon/2 + \epsilon/2$, so for a given $\epsilon$, this $\delta$ applies. -This is due to the area on the left and right of $0$ being equivalent. + +::: {#fig-consequence-additive-area} +```{julia} +#| echo: false +gr() +let +f(x) = 2 + cospi(x^2/7)*sinpi(x) + a,b,c = 0.1, 8, 3 + xs = range(a,c,100) + A1 = Shape(vcat(xs,c,a), vcat(f.(xs), 0, 0)) + + xs = range(c,b,100) + A2 = Shape(vcat(xs,b,c), vcat(f.(xs), 0, 0)) -The "reversed" area is the same, only accounted for with a minus sign. + plt = plot(; + xaxis=([], false), + yaxis=([], false), + legend=false, + ) + plot!([0,0] .- 0.1,[-.1,3]; line=(:gray, 1), arrow=true, side=:head) + plot!([0-.2, b+0.5],[0,0]; line=(:gray, 1), arrow=true, side=:head) + + plot!(A1; fill=(:gray60,1), line=(nothing,)) + plot!(A2; fill=(:gray90,1), line=(nothing,)) + plot!(f, a, b; line=(:black, 2)) + for x in (a,b,c) + plot!([x,x], [0, f(x)]; line=(:black, 1, :dash)) + end -> $$ -> \int_a^b f(x) dx = -\int_b^a f(x) dx. -> $$ + annotate!([(x,0,text(latexstring("$y"),:top)) for (x,y) in zip((a,b,c),("a","b","c"))]) +end +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + +Illustration that the area between $a$ and $b$ can be computed as area between $a$ and $c$ and then $c$ and $b$. +::: + + A consequence of the last few statements is: -> If $f(x)$ is an even function, then $\int_{-a}^a f(x) dx = 2 \int_0^a f(x) dx$. If $f(x)$ is an odd function, then $\int_{-a}^a f(x) dx = 0$. +> If $f(x)$ is an even function, then $\int_{-a}^a f(x) dx = 2 \int_0^a f(x) dx$. + +> If $f(x)$ is an odd function, then $\int_{-a}^a f(x) dx = 0$. + + +Additivity works in the $y$ direction as well. + +If $f(x)$ and $g(x)$ are two functions then + +> $$ +> \int_a^b (f(x) + g(x)) dx = \int_a^b f(x) dx + \int_a^b g(x) dx +> $$ + + +For any partitioning with $x_i, x_{i-1}$ and $c_i$ this holds: + +$$ +(f(c_i) + g(c_i)) \cdot (x_i - x_{i-1}) = +f(c_i) \cdot (x_i - x_{i-1}) + g(c_i) \cdot (x_i - x_{i-1}) +$$ + +This leads to the same statement for the areas under the curves. + + +The *linearity* of the integration operation refers to this combination of the above: + +> $$ +> \int_a^b (cf(x) + dg(x)) dx = c\int_a^b f(x) dx + d \int_a^b g(x)dx +> $$ + +The integral of a shifted function satisfies: + +> $$ +> \int_a^b \left(D + C\cdot f(\frac{x - B}{A})\right) dx = D\cdot(b-a) + C \cdot A \int_{\frac{a-B}{A}}^{\frac{b-B}{A}} f(x) dx +> $$ + + +This follows from a few of the statements above: + +$$ +\begin{align*} +\int_a^b \left(D + C\cdot f(\frac{x - B}{A})\right) dx &= +\int_a^b D dx + C \int_a^b f(\frac{x-B}{A}) dx \\ +&= D\cdot(b-a) + C\cdot A \int_{\frac{a-B}{A}}^{\frac{b-B}{A}} f(x) dx +\end{align*} +$$ + + +#### Inequalities + +Area under a non-negative function is non-negative + +> $$ +> \int_a^b f(x) dx \geq 0,\quad\text{when } a < b, \text{ and } f(x) \geq 0 +> $$ + +Under this assumption, for any partitioning with $x_i, x_{i-1}$ and $c_i$ it holds the $f(c_i)\cdot(x_i - x_{i-1}) \geq 0$. So any sum of non-negative values can only be non-negative, even in the limit. -If $g$ bounds $f$ then the area under $g$ will bound the area under $f$, in particular if $f(x)$ is non negative, so will the area under $f$ be non negative for any $a < b$. (This assumes that $g$ and $f$ are integrable.) - - -> If $0 \leq f(x) \leq g(x)$ then $\int_a^b f(x) dx \leq \int_a^b g(x) dx.$ +If $g$ bounds $f$ then the area under $g$ will bound the area under $f$. +> $$ +> $\int_a^b f(x) dx \leq \int_a^b g(x) dx \quad\text{when } a < b\text{ and } 0 \leq f(x) \leq g(x) +> $$ For any partition of $[a,b]$ and choice of $c_i$, we have the term-by-term bound $f(c_i)(x_i-x_{i-1}) \leq g(c_i)(x_i-x_{i-1})$ So any sequence of partitions that converges to the limits will have this inequality maintained for the sum. + + +::: {#fig-consequence-0-area} +```{julia} +#| echo: false +gr() +let + f(x) = 1/6+x^3*(2-x)/2 + g(x) = 1/6+exp(x/3)+(1-x/1.7)^6-0.6 + a, b = 0, 2 + + plot(; empty_style...) + plot!([a-.5, b+.25], [0,0]; line=(:gray, 1), arrow=true, side=:head) + plot!([0,0] .- 0.25, [-0.25, 1.8]; line=(:gray, 1), arrow=true, side=:head) + + xs = range(a,b,250) + S = Shape(vcat(xs, reverse(xs)), vcat(f.(xs), g.(reverse(xs)))) + plot!(S; fill=(:gray70, 0.3), line=(nothing,)) + + S = Shape(vcat(xs, reverse(xs)), vcat(zero.(xs), f.(reverse(xs)))) + plot!(S; fill=(:gray90, 0.3), line=(nothing,)) + + + plot!(f, a, b; line=(:black, 4)) + plot!(g, a, b; line=(:black, 2)) + + + for x in (a,b) + plot!([x,x], [0, g(x)]; line=(:black,1,:dash)) + end + + annotate!([(x,0,text(t, :top)) for (x,t) in zip((a,b),(L"a", L"b"))]) + + + current() +end +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + +Illustration that if $f(x) \le g(x)$ on $[a,b]$ then the integrals share the same property. The excess area is clearly positive. +::: + +(This also follows by considering $h(x) = g(x) - f(x) \geq 0$ by assumption, so $\int_a^b h(x) dx \geq 0$.) + + +For non-negative functions, integrals over larger domains are bigger + +> $$ +> \int_a^c f(x) dx \le \int_a^b f(x) dx,\quad\text{when } c < b \text{ and } f(x) \ge 0 +> $$ + +This follows as $\int_c^b f(x) dx$ is non-negative under these assumptions. + ### Some known integrals @@ -606,7 +928,7 @@ The main idea behind this is that the difference between the maximum and minimum For example, the function $f(x) = 1$ for $x$ in $[0,1]$ and $0$ otherwise will be integrable, as it is continuous at all but two points, $0$ and $1$, where it jumps. -* Some functions can have infinitely many points of discontinuity and still be integrable. The example of $f(x) = 1/q$ when $x=p/q$ is rational, and $0$ otherwise is often used as an example. +* Some functions can have infinitely many points of discontinuity and still be integrable. The example of $f(x) = 1/q$ when $x=p/q$ is rational, and $0$ otherwise is often used to illustrate this. ## Numeric integration @@ -636,11 +958,11 @@ deltas = diff(xs) # forms x2-x1, x3-x2, ..., xn-xn-1 cs = xs[1:end-1] # finds left-hand end points. xs[2:end] would be right-hand ones. ``` -Now to multiply the values. We want to sum the product `f(cs[i]) * deltas[i]`, here is one way to do so: +We want to sum the products $f(c_i)\Delta_i$. Here is one way to do so using `zip` to iterate over the paired off values in `cs` and `deltas`. ```{julia} -sum(f(cs[i]) * deltas[i] for i in 1:length(deltas)) +sum(f(ci)*Δi for (ci, Δi) in zip(cs, deltas)) ``` Our answer is not so close to the value of $1/3$, but what did we expect - we only used $n=5$ intervals. Trying again with $50,000$ gives us: @@ -652,7 +974,7 @@ n = 50_000 xs = a:(b-a)/n:b deltas = diff(xs) cs = xs[1:end-1] -sum(f(cs[i]) * deltas[i] for i in 1:length(deltas)) +sum(f(ci)*Δi for (ci, Δi) in zip(cs, deltas)) ``` This value is about $10^{-5}$ off from the actual answer of $1/3$. @@ -745,7 +1067,7 @@ plot!(zero) We could add the signed area over $[0,1]$ to the above, but instead see a square of area $1$, a triangle with area $1/2$ and a triangle with signed area $-1$. The total is then $1/2$. -This figure may make the above decomposition more clear: +This figure--using equal sized axes--may make the above decomposition more clear: ```{julia} #| echo: false @@ -758,6 +1080,7 @@ let plot!(Shape([-1,0,1], [0,-1,0]); fill=(:gray90,)) plot!(Shape([1,2,2,1], [0,1,0,0]); fill=(:gray10,)) plot!(Shape([2,3,3,2], [0,0,1,1]); fill=(:gray,)) + plot!([0,0], [0, g(0)]; line=(:black,1,:dash)) end ``` @@ -844,7 +1167,9 @@ We have the well-known triangle [inequality](http://en.wikipedia.org/wiki/Triang This suggests that the following inequality holds for integrals: -> $\lvert \int_a^b f(x) dx \rvert \leq \int_a^b \lvert f(x) \rvert dx$. +> $$ +> \lvert \int_a^b f(x) dx \rvert \leq \int_a^b \lvert f(x) \rvert dx$. +> $$ @@ -922,7 +1247,8 @@ The formulas for an approximation to the integral $\int_{-1}^1 f(x) dx$ discusse $$ \begin{align*} S &= f(x_1) \Delta_1 + f(x_2) \Delta_2 + \cdots + f(x_n) \Delta_n\\ - &= w_1 f(x_1) + w_2 f(x_2) + \cdots + w_n f(x_n). + &= w_1 f(x_1) + w_2 f(x_2) + \cdots + w_n f(x_n)\\ + &= \sum_{i=1}^n w_i f(x_i). \end{align*} $$ @@ -957,7 +1283,7 @@ f(x) = x^5 - x + 1 quadgk(f, -2, 2) ``` -The error term is $0$, answer is $4$ up to the last unit of precision (1 ulp), so any error is only in floating point approximations. +The error term is $0$, the answer is $4$ up to the last unit of precision (1 ulp), so any error is only in floating point approximations. For the numeric computation of definite integrals, the `quadgk` function should be used over the Riemann sums or even Simpson's rule. @@ -1476,6 +1802,70 @@ val, _ = quadgk(f, a, b) numericq(val) ``` + +###### Question + +Let $A=1.98$ and $B=1.135$ and + +$$ +f(x) = \frac{1 - e^{-Ax}}{B\sqrt{\pi}x} e^{-x^2}. +$$ + +Find $\int_0^1 f(x) dx$ + +```{julia} +#| echo: false +let + A,B = 1.98, 1.135 + f(x) = (1 - exp(-A*x))*exp(-x^2)/(B*sqrt(pi)*x) + val,_ = quadgk(f, 0, 1) + numericq(val) +end +``` + +###### Question + +A bound for the complementary error function ( positive function) is + +$$ +\text{erfc}(x) \leq \frac{1}{2}e^{-2x^2} + \frac{1}{2}e^{-x^2} \leq e^{-x^2} +\quad x \geq 0. +$$ + +Let $f(x)$ be the first bound, $g(x)$ the second. +Assuming this is true, confirm numerically using `quadgk` that + +$$ +\int_0^3 f(x) dx \leq \int_0^3 g(x) dx +$$ + + +The value of $\int_0^3 f(x) dx$ is + +```{julia} +#| echo: false +let + f(x) = 1/2 * exp(-2x^2) + 1/2 * exp(-x^2) + val,_ = quadgk(f, 0, 3) + numericq(val) +end +``` + +The value of $\int_0^3 g(x) dx$ is + +```{julia} +#| echo: false +let + g(x) = exp(-x^2) + val,_ = quadgk(g, 0, 3) + numericq(val) +end +``` + + + + + ###### Question diff --git a/quarto/integrals/area_between_curves.qmd b/quarto/integrals/area_between_curves.qmd index b279dea..471322f 100644 --- a/quarto/integrals/area_between_curves.qmd +++ b/quarto/integrals/area_between_curves.qmd @@ -32,7 +32,7 @@ can be interpreted as the "signed" area between $f(x)$ and $g(x)$ over $[a,b]$. ```{julia} #| hold: true #| echo: false -#| label: fig-area-between-f-g +#| label: fig-area-between-f-g-shade #| fig-cap: "Area between two functions" f1(x) = x^2 g1(x) = sqrt(x) @@ -626,7 +626,6 @@ Each term describes the area of a trapezoid, possibly signed. This figure illustrates for a simple case: ```{julia} -using Plots xs = [1, 3, 4, 2, 1] # n = 4 to give 5=n+1 values ys = [1, 1, 2, 3, 1] p = plot(xs, ys; line=(3, :black), ylims=(0,4), legend=false) diff --git a/quarto/integrals/figures/jigsaw.png b/quarto/integrals/figures/jigsaw.png new file mode 100644 index 0000000000000000000000000000000000000000..4486d41897b6744104eeaaf401d3db083dfbd500 GIT binary patch literal 744433 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≻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


@3P9Q&(9QW|{Q7TNf4ZWi6^=BsC)O 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≦U9CWZBm@#_F?O8>*M1~e_n{xK7Wty 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 z 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�kjwvOMUctZZBXRNghm`9& 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�k2<*odhPSWHr%n6>=woGW8 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⪙eJIB3CXk60gG1DyH!!K|XZtBkDr=hbR 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 0$, consider the function + +$$ +F(x) = \frac{2}{\sqrt{\pi}} \int_{-x}^0 \exp(-t^2) dt +$$ + +Why is $F'(x) = \text{erf}'(x)$? + +```{julia} +#| echo: false +choices = ["The integrand is an *even* function so the itegral from ``0`` to ``x`` is the same as the integral from ``-x`` to ``0``", + "This isn't true"] +radioq(choices, 1; keep_order=true) +``` + + +Consider the function + +$$ +F(x) = \frac{2}{\sqrt{\pi}} \int_0^{\sqrt{x}} \exp(-t^2) dt, \quad x \geq 0 +$$ + + + + + +What is the derivative of $F$? + +```{julia} +#| echo: false +choices = [L"\exp(-x^2)", + L"\frac{2}{\sqrt{\pi}} \exp(-x^2)", + L"\frac{2}{\sqrt{\pi}} \exp(-x^2) \cdot (-2x)"] +radioq(choices, 3; keep_order=true, explanation="Don't forget to apply the chain rule, as ``F(x) = \\text{erf}(\\sqrt{x})``") +``` + +###### Question + +Define two function through the integrals: + +$$ +\begin{align*} +S(x) &= \int_0^x \sin(t^2) dt\\ +C(x) &= \int_0^x \cos(t^2) dt +\end{align*} +$$ + +These are called *Fresnel Integrals*. + +A non-performant implementation might look like: + +```{julia} +S(x) = first(quadgk(t -> sin(t^2), 0, x)) +``` + +Define a similar function for $C(x)$ and them make a parametric plot for $0 \le t \le 5$. + +Describe the shape. + +```{julia} +#| echo: false +choices = ["It makes a lovely star shape", + "It makes a lovely spiral shape", + "It makes a lovely circle"] +radioq(choices, 2; keep_order=true) +``` + + +What is the value of $S'(x)^2 + C'(x)^2$ when $x=\pi$? + +```{julia} +#| echo: false +numericq(1) +``` + +###### Question + +Define a function with parameter $\alpha \geq 1$ by: + +$$ +\gamma(x; \alpha) = \int_0^x \exp(-t) t^{\alpha-1} dt, \quad x > 0 +$$ + +What is the ratio of $\gamma'(2; 3) / \gamma'(2; 4)$? + +```{julia} +#| echo: false +df(x,alpha) = exp(-x)*x^(alpha -1) +numericq(df(2,3)/df(2,4)) +``` + +###### Question + +Define a function + +$$ +i(x) = \int_0^{x^2} \exp(-t) t^{1/2} dt +$$ + +What is the derivative if $i$? + +```{julia} +#| echo: false +choices = [L"\exp(-x) x^{1/2}", + L"\exp(-x) x^{1/2} \cdot 2x", + L"\exp(-x^2) (x^2)^{1/2}", + L"\exp(-x^2) (x^2)^{1/2}\cdot 2x"] +radioq(choices, 4; keep_order=true) +``` + +###### Question + +The function `sinint` from `SpecialFunctions` computes + +$$ +F(x) = \int_0^x \frac{\sin(t)}{t} dt = \int_0^x \phi(t) dt, +$$ + +Where we define $\phi$ above to be $1$ when $t=0$, so that it will be continuous over $[0,x]$. + +A related integral might be: + +$$ +G(x) = \int_0^x \frac{\sin(\pi t)}{\pi t} dt = \int_0^x \phi(\pi t) dt +$$ + +As this is an integral involving a simple transformation of $\phi(x)$, we can see that $G(x) = (1/\pi) F(\pi x)$. What is the derivative of $G$? + +```{julia} +#| echo: false +choices = [ +L"\phi(x)", +L"\phi(\pi x)", +L"\pi \phi(\pi x)" +] +radioq(choices, 2; keep_order=true) +``` + + + ###### Question @@ -1144,12 +1401,14 @@ radioq(choices, answ, keep_order=true) Barrow presented a version of the fundamental theorem of calculus in a 1670 volume edited by Newton, Barrow's student (cf. [Wagner](http://www.maa.org/sites/default/files/0746834234133.di020795.02p0640b.pdf)). His version can be stated as follows (cf. [Jardine](http://www.maa.org/publications/ebooks/mathematical-time-capsules)): -Consider the following figure where $f$ is a strictly increasing function with $f(0) = 0$. and $x > 0$. The function $A(x) = \int_0^x f(u) du$ is also plotted. The point $Q$ is $f(x)$, and the point $P$ is $A(x)$. The point $T$ is chosen to so that the length between $T$ and $x$ times the length between $Q$ and $x$ equals the length from $P$ to $x$. ($\lvert Tx \rvert \cdot \lvert Qx \rvert = \lvert Px \rvert$.) Barrow showed that the line segment $PT$ is tangent to the graph of $A(x)$. This figure illustrates the labeling for some function: +Consider the following figure where $f$ is a strictly increasing function with $f(0) = 0$. and $x > 0$. The function $A(x) = \int_0^x f(u) du$ is also plotted with a dashed red line. The point $Q$ is $f(x)$, and the point $P$ is $A(x)$. The point $T$ is chosen to so that the length between $T$ and $x$ times the length between $Q$ and $x$ equals the length from $P$ to $x$. ($\lvert Tx \rvert \cdot \lvert Qx \rvert = \lvert Px \rvert$.) Barrow showed that the line segment $PT$ is tangent to the graph of $A(x)$. This figure illustrates the labeling for some function: ```{julia} #| hold: true #| echo: false +let + gr() f(x) = x^(2/3) x = 2 A(x) = quadgk(f, 0, x)[1] @@ -1160,14 +1419,21 @@ P = A(x) secpt = u -> 0 + P/(x-T) * (u-T) xs = range(0, stop=x+1/4, length=50 ) -p = plot(f, 0, x + 1/4, legend=false) -plot!(p, A, 0, x + 1/4, color=:red) +p = plot(f, 0, x + 1/4, legend=false, line=(:black,2)) +plot!(p, A, 0, x + 1/4, line=(:red, 2,:dash)) scatter!(p, [T, x, x, x], [0, 0, Q, P], color=:orange) -annotate!(p, collect(zip([T, x, x+.1, x+.1], [0-.15, 0-.15, Q-.1, P], ["T", "x", "Q", "P"]))) +annotate!(p, collect(zip([T, x, x+.1, x+.1], [0-.15, 0-.15, Q-.1, P], [L"T", L"x", L"Q", L"P"]))) plot!(p, [T-1/4, x+1/4], map(secpt, [T-1/4, x + 1/4]), color=:orange) plot!(p, [T, x, x], [0, 0, P], color=:green) -p + p +end +``` + +```{julia} +#| echo: false +plotly() +nothing ``` The fact that $\lvert Tx \rvert \cdot \lvert Qx \rvert = \lvert Px \rvert$ says what in terms of $f(x)$, $A(x)$ and $A'(x)$? diff --git a/quarto/integrals/improper_integrals.qmd b/quarto/integrals/improper_integrals.qmd index 057ad5b..2534f08 100644 --- a/quarto/integrals/improper_integrals.qmd +++ b/quarto/integrals/improper_integrals.qmd @@ -33,20 +33,26 @@ function make_sqrt_x_graph(n) b = 1 a = 1/2^n - xs = range(1/2^8, stop=b, length=250) - x1s = range(a, stop=b, length=50) + xs = range(1/2^n, stop=b, length=1000) + x1s = range(a, stop=b, length=1000) @syms x f(x) = 1/sqrt(x) val = N(integrate(f(x), (x, 1/2^n, b))) - title = "area under f over [1/$(2^n), $b] is $(rpad(round(val, digits=2), 4))" - plt = plot(f, range(a, stop=b, length=251), xlim=(0,b), ylim=(0, 15), legend=false, size=fig_size, title=title) - plot!(plt, [b, a, x1s...], [0, 0, map(f, x1s)...], linetype=:polygon, color=:orange) + title = L"area under $f$ over $[2^{-%$n}, %$b]$ is $%$(rpad(round(val, digits=2), 4))$" + + + plt = plot(f, range(a, stop=b, length=1000); + xlim=(0,b), ylim=(0, 15), + legend=false, + title=title) + plot!(plt, [b, a, x1s...], [0, 0, map(f, x1s)...]; + linetype=:polygon, color=:orange) plt - end + caption = L""" Area under $1/\sqrt{x}$ over $[a,b]$ increases as $a$ gets closer to $0$. Will it grow unbounded or have a limit? @@ -133,7 +139,7 @@ The limit is infinite, so does not exist except in an extended sense. Before showing this, we recall the fundamental theorem of calculus. The limit existing is the same as saying the limit of $F(M) - F(a)$ exists for an antiderivative of $f(x)$. -For this particular problem, it can be shown by integration by parts that for positive, integer values of $n$ that an antiderivative exists of the form $F(x) = p(x)e^{-x}$, where $p(x)$ is a polynomial of degree $n$. But we've seen that for any $n>0$, $\lim_{x \rightarrow \infty} x^n e^{-x} = 0$, so the same is true for any polynomial. So, $\lim_{M \rightarrow \infty} F(M) - F(1) = -F(1)$. +For this particular problem, it can be shown with integration by parts that for positive, integer values of $n$ that an antiderivative exists of the form $F(x) = p(x)e^{-x}$, where $p(x)$ is a polynomial of degree $n$. But we've seen that for any $n>0$, $\lim_{x \rightarrow \infty} x^n e^{-x} = 0,$ so the same is true for any polynomial. So, $\lim_{M \rightarrow \infty} F(M) - F(1) = -F(1)$. * The function $e^x$ is integrable over $(-\infty, a]$ but not @@ -175,6 +181,87 @@ As $M$ goes to $\infty$, this will converge to $1$. limit(sympy.Si(M), M => oo) ``` + +##### Example + +To formally find the limit as $x\rightarrow \infty$ of + +$$ +\text{Si}(x) = \int_0^\infty \frac{\sin(t)}{t} dt +$$ + +we introduce a trick and rely on some theorems that have not been discussed. + +First, we notice that $\Si(x)$ is the value of $I(\alpha)$ when $\alpha=0$ where + +$$ +I(\alpha) = \int_0^\infty \exp(-\alpha t) \frac{\sin(t)}{t} dt +$$ + +We differentiate $I$ in $\alpha$ to get: + +$$ +\begin{align*} +I'(\alpha) &= \frac{d}{d\alpha} \int_0^\infty \exp(-\alpha t) \frac{\sin(t)}{t} dt \\ +&= \int_0^\infty \frac{d}{d\alpha} \exp(-\alpha t) \frac{\sin(t)}{t} dt \\ +&= \int_0^\infty (-t) \exp(-\alpha t) \frac{\sin(t)}{t} dt \\ +&= -\int_0^\infty \exp(-\alpha t) \sin(t) dt \\ +\end{align*} +$$ + +As illustrated previously, this integral can be integrated by parts, though here we have infinite limits and have adjusted for the minus sign: + +$$ +\begin{align*} +-I'(\alpha) &= \int_0^\infty \exp(-\alpha t) \sin(t) dt \\ +&=\sin(t) \frac{-\exp(-\alpha t)}{\alpha} \Big|_0^\infty - +\int_0^\infty \frac{-\exp(-\alpha t)}{\alpha} \cos(t) dt \\ +&= 0 + \frac{1}{\alpha} \cdot \int_0^\infty \exp(-\alpha t) \cos(t) dt \\ +&= \frac{1}{\alpha} \cdot \cos(t)\frac{-\exp(-\alpha t)}{\alpha} \Big|_0^\infty - +\frac{1}{\alpha} \cdot \int_0^\infty \frac{-\exp(-\alpha t)}{\alpha} (-\sin(t)) dt \\ +&= \frac{1}{\alpha^2} - \frac{1}{\alpha^2} \cdot \int_0^\infty \exp(-\alpha t) \sin(t) dt +\end{align*} +$$ + +Combining gives: + +$$ +\left(1 + \frac{1}{\alpha^2}\right) \int_0^\infty \exp(-\alpha t) \sin(t) dt = \frac{1}{\alpha^2} +$$ + +Solving gives the desired integral as + +$$ +I'(\alpha) = -\frac{1}{\alpha^2} / (1 + \frac{1}{\alpha^2}) = -\frac{1}{1 + \alpha^2}. +$$ + + +This has a known antiderivative: $I(\alpha) = -\tan^{-1}(\alpha) + C$. As $\alpha \rightarrow \infty$ *if* we can pass the limit *inside* the integral, then $I(\alpha) \rightarrow 0$. So $\lim_{x \rightarrow \infty} -\tan^{-1}(x) + C = 0$ or $C = \pi/2$. + +As our question is answered by $I(0)$, we get $I(0) = \tan^{-1}(0) + C = C = \pi/2$. + +The above argument requires two places where a *limit* is passed inside the integral. The first involved the derivative. The [Leibniz integral rule](https://en.wikipedia.org/wiki/Leibniz_integral_rule) can be used to verify the first use is valid: + +:::{.callout-note icon=false} +## Leibniz integral rule +If $f(x,t)$ and the derivative in $x$ for a fixed $t$ is continuous (to be discussed later) in a region containing $a(x) \leq t \leq b(x)$ and $x_0 < x < x_1$ and both $a(x)$ and $b(x)$ are continuously differentiable, then + +$$ +\frac{d}{dx}\int_{a(x)}^{b(x)} f(x, t) dt = +\int_{a(x)}^{b(x)} \frac{d}{dx}f(x,t) dt + +f(x, b(x)) \frac{d}{dx}b(x) - f(x, a(x)) \frac{d}{dx}a(x). +$$ + +::: + +This extends the fundamental theorem of calculus for cases where the integrand also depends on $x$. In our use, both $a'(x)$ and $b'(x)$ are $0$. + +[Uniform convergence](https://en.wikipedia.org/wiki/Uniform_convergence) can be used to establish the other. + + + + + ### Numeric integration diff --git a/quarto/integrals/integration_by_parts.qmd b/quarto/integrals/integration_by_parts.qmd index dcbff56..9271f72 100644 --- a/quarto/integrals/integration_by_parts.qmd +++ b/quarto/integrals/integration_by_parts.qmd @@ -39,13 +39,13 @@ Now we turn our attention to the implications of the *product rule*: $[uv]' = u' By the fundamental theorem of calculus: $$ -[u(x)\cdot v(x)]\big|_a^b = \int_a^b [u(x) v(x)]' dx = \int_a^b u'(x) \cdot v(x) dx + \int_a^b u(x) \cdot v'(x) dx. +[u(x)\cdot v(x)]\Big|_a^b = \int_a^b [u(x) v(x)]' dx = \int_a^b u'(x) \cdot v(x) dx + \int_a^b u(x) \cdot v'(x) dx. $$ Or, $$ -\int_a^b u(x) v'(x) dx = [u(x)v(x)]\big|_a^b - \int_a^b v(x) u'(x)dx. +\int_a^b u(x) v'(x) dx = [u(x)v(x)]\Big|_a^b - \int_a^b v(x) u'(x)dx. $$ ::: @@ -58,16 +58,16 @@ The following visually illustrates integration by parts: #| label: fig-integration-by-parts #| fig-cap: "Integration by parts figure ([original](http://en.wikipedia.org/wiki/Integration_by_parts#Visualization))" let -## parts picture + ## parts picture + gr() u(x) = sin(x*pi/2) v(x) = x xs = range(0, stop=1, length=50) a,b = 1/4, 3/4 -p = plot(u, v, 0, 1, legend=false, axis=([], false)) -plot!([0, u(1)], [0,0], line=(:black, 3)) -plot!([0, 0], [0, v(1) ], line=(:black, 3)) -plot!(p, zero, 0, 1) +p = plot(u, v, 0, 1; legend=false, axis=([], false), line=(:black,2)) +plot!([0, u(1)], [0,0]; line=(:gray, 1), arrow=true, side=:head) +plot!([0, 0], [0, v(1) ]; line=(:gray, 1), arrow=true, side=:head) xs′ = range(a, b, length=50) plot!(Shape(vcat(u.(xs′), reverse(u.(xs′))), @@ -81,21 +81,28 @@ plot!(p, [u(a),u(a),0, 0, u(b),u(b),u(a)], [0, v(a), v(a), v(b), v(b), 0, 0], linetype=:polygon, fill=(:brown3, 0.25)) -annotate!(p, [(0.65, .25, "A"), - (0.4, .55, "B"), - (u(a),v(a) + .08, "(u(a),v(a))"), - (u(b),v(b)+.08, "(u(b),v(b))"), - (u(a),0, "u(a)",:top), - (u(b),0, "u(b)",:top), - (0, v(a), "v(a) ",:right), - (0, v(b), "v(b) ",:right) +annotate!(p, [(0.65, .25, text(L"A")), + (0.4, .55, text(L"B")), + (u(a),v(a), text(L"(u(a),v(a))", :bottom, :right)), + (u(b),v(b), text(L"(u(b),v(b))", :bottom, :right)), + (u(a),0, text(L"u(a)", :top)), + (u(b),0, text(L"u(b)", :top)), + (0, v(a), text(L"v(a)", :right)), + (0, v(b), text(L"v(b)", :right)), + (0,0, text(L"(0,0)", :top)) ]) end ``` +```{julia} +#| echo: false +plotly() +nothing +``` + @fig-integration-by-parts shows a parametric plot of $(u(t),v(t))$ for $a \leq t \leq b$.. -The total shaded area, a rectangle, is $u(b)v(b)$, the area of $A$ and $B$ combined is just $u(b)v(b) - u(a)v(a)$ or $[u(x)v(x)]\big|_a^b$. We will show that $A$ is $\int_a^b v(x)u'(x)dx$ and $B$ is $\int_a^b u(x)v'(x)dx$ giving the formula. +The total shaded area, a rectangle, is $u(b)v(b)$, the area of $A$ and $B$ combined is just $u(b)v(b) - u(a)v(a)$ or $[u(x)v(x)]\Big|_a^b$. We will show that $A$ is $\int_a^b v(x)u'(x)dx$ and $B$ is $\int_a^b u(x)v'(x)dx$ giving the formula. We can compute $A$ by a change of variables with $x=u^{-1}(t)$ (so $u'(x)dx = dt$): @@ -109,6 +116,7 @@ $$ $B$ is similar with the roles of $u$ and $v$ reversed. +---- Informally, the integration by parts formula is sometimes seen as $\int udv = uv - \int v du$, as well can be somewhat confusingly written as: @@ -131,10 +139,10 @@ Consider the integral $\int_0^\pi x\sin(x) dx$. If we let $u=x$ and $dv=\sin(x) $$ \begin{align*} \int_0^\pi x\sin(x) dx &= \int_0^\pi u dv\\ -&= uv\big|_0^\pi - \int_0^\pi v du\\ -&= x \cdot (-\cos(x)) \big|_0^\pi - \int_0^\pi (-\cos(x)) dx\\ +&= uv\Big|_0^\pi - \int_0^\pi v du\\ +&= x \cdot (-\cos(x)) \Big|_0^\pi - \int_0^\pi (-\cos(x)) dx\\ &= \pi (-\cos(\pi)) - 0(-\cos(0)) + \int_0^\pi \cos(x) dx\\ -&= \pi + \sin(x)\big|_0^\pi\\ +&= \pi + \sin(x)\Big|_0^\pi\\ &= \pi. \end{align*} $$ @@ -166,8 +174,8 @@ Putting together gives: $$ \begin{align*} \int_1^2 x \log(x) dx -&= (\log(x) \cdot \frac{x^2}{2}) \big|_1^2 - \int_1^2 \frac{x^2}{2} \frac{1}{x} dx\\ -&= (2\log(2) - 0) - (\frac{x^2}{4})\big|_1^2\\ +&= (\log(x) \cdot \frac{x^2}{2}) \Big|_1^2 - \int_1^2 \frac{x^2}{2} \frac{1}{x} dx\\ +&= (2\log(2) - 0) - (\frac{x^2}{4})\Big|_1^2\\ &= 2\log(2) - (1 - \frac{1}{4}) \\ &= 2\log(2) - \frac{3}{4}. \end{align*} @@ -204,7 +212,7 @@ Were this a definite integral problem, we would have written: $$ -\int_a^b \log(x) dx = (x\log(x))\big|_a^b - \int_a^b dx = (x\log(x) - x)\big|_a^b. +\int_a^b \log(x) dx = (x\log(x))\Big|_a^b - \int_a^b dx = (x\log(x) - x)\Big|_a^b. $$ ##### Example @@ -214,14 +222,14 @@ Sometimes integration by parts is used two or more times. Here we let $u=x^2$ an $$ -\int_a^b x^2 e^x dx = (x^2 \cdot e^x)\big|_a^b - \int_a^b 2x e^x dx. +\int_a^b x^2 e^x dx = (x^2 \cdot e^x)\Big|_a^b - \int_a^b 2x e^x dx. $$ But we can do $\int_a^b x e^xdx$ the same way: $$ -\int_a^b x e^x = (x\cdot e^x)\big|_a^b - \int_a^b 1 \cdot e^xdx = (xe^x - e^x)\big|_a^b. +\int_a^b x e^x = (x\cdot e^x)\Big|_a^b - \int_a^b 1 \cdot e^xdx = (xe^x - e^x)\Big|_a^b. $$ Combining gives the answer: @@ -229,8 +237,8 @@ Combining gives the answer: $$ \int_a^b x^2 e^x dx -= (x^2 \cdot e^x)\big|_a^b - 2( (xe^x - e^x)\big|_a^b ) = -e^x(x^2 - 2x + 2) \big|_a^b. += (x^2 \cdot e^x)\Big|_a^b - 2( (xe^x - e^x)\Big|_a^b ) = +e^x(x^2 - 2x + 2) \Big|_a^b. $$ In fact, it isn't hard to see that an integral of $x^m e^x$, $m$ a positive integer, can be handled in this manner. For example, when $m=10$, `SymPy` gives: @@ -247,14 +255,29 @@ The general answer is $\int x^n e^xdx = p(x) e^x$, where $p(x)$ is a polynomial ##### Example -The same technique is attempted for this integral, but ends differently. First in the following we let $u=\sin(x)$ and $dv=e^x dx$: +The same technique is attempted for the integral of $e^x\sin(x)$, but ends differently. + +First we let $u=\sin(x)$ and $dv=e^x dx$, then + +$$ +du = \cos(x)dx \quad \text{and}\quad v = e^x. +$$ + +So: $$ \int e^x \sin(x)dx = \sin(x) e^x - \int \cos(x) e^x dx. $$ -Now we let $u = \cos(x)$ and again $dv=e^x dx$: +Now we let $u = \cos(x)$ and again $dv=e^x dx$, then + + +$$ +du = -\sin(x)dx \quad \text{and}\quad v = e^x. +$$ + +So: $$ @@ -301,7 +324,7 @@ $$ This is called a reduction formula as it reduces the problem from an integral with a power of $n$ to one with a power of $n - 2$, so could be repeated until the remaining indefinite integral required knowing either $\int \cos(x) dx$ (which is $-\sin(x)$) or $\int \cos(x)^2 dx$, which by a double angle formula application, is $x/2 + \sin(2x)/4$. -`SymPy` is quite able to do this repeated bookkeeping. For example with $n=10$: +`SymPy` is able and willing to do this repeated bookkeeping. For example with $n=10$: ```{julia} @@ -350,7 +373,7 @@ Using right triangles to simplify, the last value $\cos(\sin^{-1}(x))$ can other The [trapezoid](http://en.wikipedia.org/wiki/Trapezoidal_rule) rule is an approximation to the definite integral like a Riemann sum, only instead of approximating the area above $[x_i, x_i + h]$ by a rectangle with height $f(c_i)$ (for some $c_i$), it uses a trapezoid formed by the left and right endpoints. That is, this area is used in the estimation: $(1/2)\cdot (f(x_i) + f(x_i+h)) \cdot h$. -Even though we suggest just using `quadgk` for numeric integration, estimating the error in this approximation is still of some theoretical interest. +Even though we suggest just using `quadgk` for numeric integration, estimating the error in this approximation is of theoretical interest. Recall, just using *either* $x_i$ or $x_{i-1}$ for $c_i$ gives an error that is "like" $1/n$, as $n$ gets large, though the exact rate depends on the function and the length of the interval. @@ -359,18 +382,18 @@ Recall, just using *either* $x_i$ or $x_{i-1}$ for $c_i$ gives an error that is This [proof](http://www.math.ucsd.edu/~ebender/20B/77_Trap.pdf) for the error estimate is involved, but is reproduced here, as it nicely integrates many of the theoretical concepts of integration discussed so far. -First, for convenience, we consider the interval $x_i$ to $x_i+h$. The actual answer over this is just $\int_{x_i}^{x_i+h}f(x) dx$. By a $u$-substitution with $u=x-x_i$ this becomes $\int_0^h f(t + x_i) dt$. For analyzing this we integrate once by parts using $u=f(t+x_i)$ and $dv=dt$. But instead of letting $v=t$, we choose to add - as is our prerogative - a constant of integration $A$, so $v=t+A$: +First, for convenience, we consider the interval $x_i$ to $x_i+h$. The actual answer over this is just $\int_{x_i}^{x_i+h}f(x) dx$. By a $u$-substitution with $u=x-x_i$ this becomes $\int_0^h f(t + x_i) dt$. For analyzing this we integrate once by parts using $u=f(t+x_i)$ and $dv=dt$. But instead of letting $v=t$, we choose to add--as is our prerogative--a constant of integration $A$, so $v=t+A$: $$ \begin{align*} -\int_0^h f(t + x_i) dt &= uv \big|_0^h - \int_0^h v du\\ -&= f(t+x_i)(t+A)\big|_0^h - \int_0^h (t + A) f'(t + x_i) dt. +\int_0^h f(t + x_i) dt &= uv \Big|_0^h - \int_0^h v du\\ +&= f(t+x_i)(t+A)\Big|_0^h - \int_0^h (t + A) f'(t + x_i) dt. \end{align*} $$ -We choose $A$ to be $-h/2$, any constant is possible, for then the term $f(t+x_i)(t+A)\big|_0^h$ becomes $(1/2)(f(x_i+h) + f(x_i)) \cdot h$, or the trapezoid approximation. This means, the error over this interval - actual minus estimate - satisfies: +We choose $A$ to be $-h/2$, any constant is possible, for then the term $f(t+x_i)(t+A)\Big|_0^h$ becomes $(1/2)(f(x_i+h) + f(x_i)) \cdot h$, or the trapezoid approximation. This means, the error over this interval - actual minus estimate - satisfies: $$ @@ -392,7 +415,7 @@ Again we added a constant of integration, $B$, to $v$. The error becomes: $$ -\text{error}_i = -(\frac{(t+A)^2}{2} + B)f'(t+x_i)\big|_0^h + \int_0^h (\frac{(t+A)^2}{2} + B) \cdot f''(t+x_i) dt. +\text{error}_i = -\left(\frac{(t+A)^2}{2} + B\right)f'(t+x_i)\Big|_0^h + \int_0^h \left(\frac{(t+A)^2}{2} + B\right) \cdot f''(t+x_i) dt. $$ With $A=-h/2$, $B$ is chosen so $(t+A)^2/2 + B = 0$ at endpoints, or $B=-h^2/8$. The error becomes @@ -406,14 +429,14 @@ Now, we assume the $\lvert f''(t)\rvert$ is bounded by $K$ for any $a \leq t \le $$ -\lvert \text{error}_i \rvert \leq K \int_0^h \lvert (\frac{(t-h/2)^2}{2} - \frac{h^2}{8}) \rvert dt. +\lvert \text{error}_i \rvert \leq K \int_0^h \lVert \left(\frac{(t-h/2)^2}{2} - \frac{h^2}{8}\right) \rVert dt. $$ But what is the function in the integrand? Clearly it is a quadratic in $t$. Expanding gives $1/2 \cdot (t^2 - ht)$. This is negative over $[0,h]$ (and $0$ at these endpoints, so the integral above is just: $$ -\frac{1}{2}\int_0^h (ht - t^2)dt = \frac{1}{2} (\frac{ht^2}{2} - \frac{t^3}{3})\big|_0^h = \frac{h^3}{12} +\frac{1}{2}\int_0^h (ht - t^2)dt = \frac{1}{2} \left(\frac{ht^2}{2} - \frac{t^3}{3}\right)\Big|_0^h = \frac{h^3}{12} $$ This gives the bound: $\vert \text{error}_i \rvert \leq K h^3/12$. The *total* error may be less, but is not more than the value found by adding up the error over each of the $n$ intervals. As our bound does not depend on the $i$, we have this sum satisfies: diff --git a/quarto/integrals/mean_value_theorem.qmd b/quarto/integrals/mean_value_theorem.qmd index b673ebd..527a05d 100644 --- a/quarto/integrals/mean_value_theorem.qmd +++ b/quarto/integrals/mean_value_theorem.qmd @@ -144,10 +144,10 @@ $$ So in particular $K$ is in $[m, M]$. But $m$ and $M$ correspond to values of $f(x)$, so by the intermediate value theorem, $K=f(c)$ for some $c$ that must lie in between $c_m$ and $c_M$, which means as well that it must be in $[a,b]$. -##### Proof of second part of Fundamental Theorem of Calculus +##### Proof of the second part of the Fundamental Theorem of Calculus -The mean value theorem is exactly what is needed to prove formally the second part of the Fundamental Theorem of Calculus. Again, suppose $f(x)$ is continuous on $[a,b]$ with $a < b$. For any $a < x < b$, we define $F(x) = \int_a^x f(u) du$. Then the derivative of $F$ exists and is $f$. +The mean value theorem is exactly what is needed to formally prove the second part of the Fundamental Theorem of Calculus. Again, suppose $f(x)$ is continuous on $[a,b]$ with $a < b$. For any $a < x < b$, we define $F(x) = \int_a^x f(u) du$. Then the derivative of $F$ exists and is $f$. Let $h>0$. Then consider the forward difference $(F(x+h) - F(x))/h$. Rewriting gives: diff --git a/quarto/integrals/partial_fractions.qmd b/quarto/integrals/partial_fractions.qmd index 3fcfdae..0af818f 100644 --- a/quarto/integrals/partial_fractions.qmd +++ b/quarto/integrals/partial_fractions.qmd @@ -14,7 +14,9 @@ using SymPy Integration is facilitated when an antiderivative for $f$ can be found, as then definite integrals can be evaluated through the fundamental theorem of calculus. -However, despite differentiation being an algorithmic procedure, integration is not. There are "tricks" to try, such as substitution and integration by parts. These work in some cases. However, there are classes of functions for which algorithms exist. For example, the `SymPy` `integrate` function mostly implements an algorithm that decides if an elementary function has an antiderivative. The [elementary](http://en.wikipedia.org/wiki/Elementary_function) functions include exponentials, their inverses (logarithms), trigonometric functions, their inverses, and powers, including $n$th roots. Not every elementary function will have an antiderivative comprised of (finite) combinations of elementary functions. The typical example is $e^{x^2}$, which has no simple antiderivative, despite its ubiquitousness. +However, despite differentiation being an algorithmic procedure, integration is not. There are "tricks" to try, such as substitution and integration by parts. These work in some cases--but not all! + +However, there are classes of functions for which algorithms exist. For example, the `SymPy` `integrate` function mostly implements an algorithm that decides if an elementary function has an antiderivative. The [elementary](http://en.wikipedia.org/wiki/Elementary_function) functions include exponentials, their inverses (logarithms), trigonometric functions, their inverses, and powers, including $n$th roots. Not every elementary function will have an antiderivative comprised of (finite) combinations of elementary functions. The typical example is $e^{x^2}$, which has no simple antiderivative, despite its ubiquitousness. There are classes of functions where an (elementary) antiderivative can always be found. Polynomials provide a case. More surprisingly, so do their ratios, *rational functions*. @@ -238,7 +240,11 @@ $$ #### Examples -Find an antiderivative for $1/(x\cdot(x^2+1)^2)$. +Find an antiderivative for + +$$ +\frac{1}{x\cdot(x^2+1)^2}. +$$ We have a partial fraction decomposition is: @@ -259,7 +265,11 @@ integrate(1/q, x) --- -Find an antiderivative of $1/(x^2 - 2x-3)$. +Find an antiderivative of + +$$ +\frac{1}{x^2 - 2x-3}. +$$ We again just let `SymPy` do the work. A partial fraction decomposition is given by: diff --git a/quarto/integrals/substitution.qmd b/quarto/integrals/substitution.qmd index 62d08f2..48a5e31 100644 --- a/quarto/integrals/substitution.qmd +++ b/quarto/integrals/substitution.qmd @@ -289,11 +289,11 @@ where $u = (x-\mu)/\sigma$, so $du = (1/\sigma) dx$. This shows that integrals involving a normal density with parameters $\mu$ and $\sigma$ can be computed using the *standard* normal density with $\mu=0$ and $\sigma=1$. Unfortunately, there is no elementary antiderivative for $\exp(-u^2/2)$, so integrals for the standard normal must be numerically approximated. -There is a function `erf` in the `SpecialFunctions` package (which is loaded by `CalculusWithJulia`) that computes: +There is a function `erf` in the `SpecialFunctions` package (which is loaded by `CalculusWithJulia`) defined by: $$ -\int_0^x \frac{2}{\sqrt{\pi}} \exp(-t^2) dt +\text{erf}(x) = \frac{2}{\sqrt{\pi}}\int_0^x \exp(-t^2) dt $$ A further change of variables by $t = u/\sqrt{2}$ (with $\sqrt{2}dt = du$) gives: diff --git a/quarto/integrals/surface_area.qmd b/quarto/integrals/surface_area.qmd index eed1d73..088dd89 100644 --- a/quarto/integrals/surface_area.qmd +++ b/quarto/integrals/surface_area.qmd @@ -63,7 +63,7 @@ $$ If the curve is parameterized by $(g(t), f(t))$ between $a$ and $b$ then the surface area is $$ -\int_a^b 2\pi f(t) \cdot \sqrt{g'(t)^2 + f'(t)^2} dx. +\int_a^b 2\pi f(t) \cdot \sqrt{g'(t)^2 + f'(t)^2} dt. $$ These formulas do not add in the surface area of either of the ends. @@ -90,11 +90,343 @@ To see why this formula is as it is, we look at the parameterized case, the firs Let a partition of $[a,b]$ be given by $a = t_0 < t_1 < t_2 < \cdots < t_n =b$. This breaks the curve into a collection of line segments. Consider the line segment connecting $(g(t_{i-1}), f(t_{i-1}))$ to $(g(t_i), f(t_i))$. Rotating this around the $x$ axis will generate something approximating a disc, but in reality will be the frustum of a cone. What will be the surface area? +::: {#fig-surface-area} +```{julia} +#| echo: false +let + gr() + function projection_plane(v) + vx, vy, vz = v + a = [-vy, vx, 0] # v ⋅ a = 0 + b = v × a # so v ⋅ b = 0 + return (a/norm(a), b/norm(b)) + end + function project(x, v) + â, b̂ = projection_plane(v) + (x ⋅ â, x ⋅ b̂) # (x ⋅ â) â + (x ⋅ b̂) b̂ + end + radius(t) = 1 / (1 + exp(t)) + t₀, tₙ = 0, 3 + surf(t, θ) = [t, radius(t)*cos(θ), radius(t)*sin(θ)] -Consider a right-circular cone parameterized by an angle $\theta$ and the largest radius $r$ (so that the height satisfies $r/h=\tan(\theta)$). If this cone were made of paper, cut up a side, and laid out flat, it would form a sector of a circle, whose area would be $R^2\gamma/2$ where $R$ is the radius of the circle (also the side length of our cone), and $\gamma$ an angle that we can figure out from $r$ and $\theta$. To do this, we note that the arc length of the circle's edge is $R\gamma$ and also the circumference of the bottom of the cone so $R\gamma = 2\pi r$. With all this, we can solve to get $A = \pi r^2/\sin(\theta)$. But we have a frustum of a cone with radii $r_0$ and $r_1$, so the surface area is a difference: $A = \pi (r_1^2 - r_0^2) /\sin(\theta)$. + v = [2, -2, 1] + function plot_axes() + empty_style = (xaxis = ([], false), + yaxis = ([], false), + legend=false) + + plt = plot(; empty_style...) + + axis_values = [[(0,0,0), (3.5,0,0)], # x axis + [(0,0,0), (0, 2.0 * radius(0), 0)], # yaxis + [(0,0,0), (0, 0, 1.5 * radius(0))]] # z axis + + for (ps, ax) ∈ zip(axis_values, ("x", "y", "z")) + p0, p1 = ps + a, b = project(p0, v), project(p1, v) + annotate!([(b...,text(ax, :bottom))]) + plot!([a, b]; arrow=true, head=:tip, line=(:gray, 1)) # gr() allows arrows + end + + plt + end + + function psurf(v) + (t,θ) -> begin + v1, v2 = project(surf(t, θ), v) + [v1, v2] # or call collect to make a tuple into a vector + end + end + + function detJ(F, t, θ) + ∂θ = ForwardDiff.derivative(θ -> F(t, θ), θ) + ∂t = ForwardDiff.derivative(t -> F(t, θ), t) + (ax, ay), (bx, by) = ∂θ, ∂t + ax * by - ay * bx + end + + function cap!(t, v; kwargs...) + θs = range(0, 2pi, 100) + S = Shape(project.(surf.(t, θs), (v,))) + plot!(S; kwargs...) + end + ## ---- + + G = psurf(v) + fold(F, t, θmin, θmax) = find_zero(θ -> detJ(F, t, θ), (θmin, θmax)) + + plt = plot_axes() -Relating this to our values in terms of $f$ and $g$, we have $r_1=f(t_i)$, $r_0 = f(t_{i-1})$, and $\sin(\theta) = \Delta f / \sqrt{(\Delta g)^2 + (\Delta f)^2}$, where $\Delta f = f(t_i) - f(t_{i-1})$ and similarly for $\Delta g$. + ts = range(t₀, tₙ, 100) + back_edge = fold.(G, ts, 0, pi) + front_edge = fold.(G, ts, pi, 2pi) + db = Dict(t => v for (t,v) in zip(ts, back_edge)) + df = Dict(t => v for (t,v) in zip(ts, front_edge)) + + # basic shape + plt = plot_axes() + plot!(project.(surf.(ts, back_edge), (v,)); line=(:black, 1)) + plot!(project.(surf.(ts, front_edge), (v,)); line=(:black, 1)) + + # add caps + cap!(t₀, v; fill=(:gray, 0.33)) + cap!(tₙ, v; fill=(:gray, 0.33)) + + + # add rotated surface segment + i,j = 33,38 + a = ts[i] + θs = range(db[ts[i]], df[ts[i]], 100) + θ′s = reverse(range(db[ts[j]], df[ts[j]], 100)) + function 𝐺(t,θ) + v1, v2 = G(t, θ) + (v1, v2) + end + S = Shape(vcat(𝐺.(ts[i], θs), 𝐺.(ts[j], θ′s))) + plot!(S) + + θs = range(df[ts[i]], 2pi + db[ts[i]], 100) + plot!([𝐺(ts[i], θ) for θ in θs]; line=(:black, 1, :dash)) + + θs = range(df[ts[j]], 2pi + db[ts[j]], 100) + plot!([𝐺(ts[j], θ) for θ in θs]; line=(:black, 1)) + + plot!([project((ts[i], 0,0),v), 𝐺(ts[i],db[ts[i]])]; line=(:black, 1, :dot), arrow=true) + plot!([project((ts[j], 0,0),v), 𝐺(ts[j],db[ts[j]])]; line=(:black, 1, :dot), arrow=true) + + # add shading + lightpt = [2, -2, 5] # from further above + H = psurf(lightpt) + light_edge = fold.(H, ts, pi, 2pi); + + for (i, (t, top, bottom)) in enumerate(zip(ts, light_edge, front_edge)) + λ = iseven(i) ? 1.0 : 0.8 + top = bottom + λ*(top - bottom) + curve = [project(surf(t, θ), v) for θ in range(bottom, top, 20)] + plot!(curve, line=(:black, 1)) + end + + # annotations + _x, _y, _z = surf(ts[i],db[ts[i]]) + __x, __y = project((_x, _y/2, _z/2), v) + _x, _y, _z = surf(ts[j],db[ts[j]]) + __x′, __y′ = project((_x, _y/2, _z/2), v) + # annotations + annotate!([ + (__x, __y, text(L"r_i", :left, :top)), + (__x′, __y′, text(L"r_{i+1}",:left, :top)), + ]) + + current() +end +``` + + +```{julia} +#| echo: false +plotly() +nothing +``` + +Illustration of function $(g(t), f(t))$ rotated about the $x$ axis with a section shaded. +::: + + + +Consider a right-circular cone parameterized by an angle $\theta$ which at a given height has radius $r$ and slant height $l$ (so that the height satisfies $r/l=\sin(\theta)$). If this cone were made of paper, cut up a side, and laid out flat, it would form a sector of a circle, as illustrated below: + +::: {#fig-frustum-cone-area} + +```{julia} +#| echo: false +p1 = let + gr() +function projection_plane(v) + vx, vy, vz = v + a = [-vy, vx, 0] # v ⋅ a = 0 + b = v × a # so v ⋅ b = 0 + return (a/norm(a), b/norm(b)) +end +function project(x, v) + â, b̂ = projection_plane(v) + (x ⋅ â, x ⋅ b̂) # (x ⋅ â) â + (x ⋅ b̂) b̂ +end + +function plot_axes(v) + empty_style = (xaxis = ([], false), + yaxis = ([], false), + legend=false) + + plt = plot(; empty_style..., aspect_ratio=:equal) + + a,b,c,d,e = project.([(0,0,2), (0,0,3), surf(3, 3pi/2), surf(2, 3pi/2),(0,0,0)], (v,)) + pts = [a,b,c,d,a]#project.([a,b,c,d,a], (v,)) + plot!(pts; line=(:gray, 1)) + plot!([c,d]; line=(:black, 2)) + plot!([d, e,a]; line=(:gray, 1,1)) + #plot!(project.([e,a,d,e],(v,)); line=(:gray, 1)) + + + + plt +end + + function psurf(v) + (t,θ) -> begin + v1, v2 = project(surf(t, θ), v) + [v1, v2] # or call collect to make a tuple into a vector + end + end +function detJ(F, t, θ) + ∂θ = ForwardDiff.derivative(θ -> F(t, θ), θ) + ∂t = ForwardDiff.derivative(t -> F(t, θ), t) + (ax, ay), (bx, by) = ∂θ, ∂t + ax * by - ay * bx +end + + function cap!(t, v; kwargs...) + θs = range(0, 2pi, 100) + S = Shape(project.(surf.(t, θs), (v,))) + plot!(S; kwargs...) + end + + function fold(F, t, θmin, θmax) + 𝐹(θ) = detJ(F, t, θ) + 𝐹(θmin) * 𝐹(θmax) <= 0 || return NaN + find_zero(𝐹, (θmin, θmax)) + end + + + + radius(t) = t/2 + t₀, tₙ = 0, 3 + surf(t, θ) = [radius(t)*cos(θ), radius(t)*sin(θ), t] # z axis + + v = [2, -2, 1] + G = psurf(v) + + + ts = range(t₀, tₙ, 100) + back_edge = fold.(G, ts, 0, pi) + front_edge = fold.(G, ts, pi, 2pi) + db = Dict(t => v for (t,v) in zip(ts, back_edge)) + df = Dict(t => v for (t,v) in zip(ts, front_edge)) + + plt = plot_axes(v) + plot!(project.(surf.(ts, back_edge), (v,)); line=(:black, 1)) + plot!(project.(surf.(ts, front_edge), (v,)); line=(:black, 1)) + + + cap!(tₙ, v; fill=(:gray80, 0.33)) + i = 67 + tᵢ = ts[i] # tᵢ = 2.0 + plot!(project.([surf.(tᵢ, θ) for θ in range(df[tᵢ], 2pi + db[tᵢ], 100)], (v,))) + + + + # add surface to rotate + + ## add light + lightpt = [2, -2, 5] # from further above + H = psurf(lightpt) + light_edge = fold.(H, ts, pi, 2pi); + + for (i, (t, top, bottom)) in enumerate(zip(ts, light_edge, front_edge)) + λ = iseven(i) ? 1.0 : 0.8 + (isnan(top) || isnan(bottom)) && continue + top = bottom + λ*(top - bottom) + curve = [project(surf(t, θ), v) for θ in range(bottom, top, 20)] + #plot!(curve, line=(:black, 1)) + end + + a,b,c = project(surf(tₙ, 3pi/2), v), project(surf(2, 3pi/2),v), project((0,0,0), v) + #plot!([a,b], line=(:black, 3)) + #plot!([b,c]; line=(:black,2)) + + # annotations + _x,_y,_z = surf(tₙ, 3pi/2) + r1 = project((_x/2, _y/2, _z), v) + _x,_y,_z = surf(2, 3pi/2) + r2 = project((_x/2, _y/2, _z), v) + _x, _y, _z = surf(1/2, 3pi/2) + theta = project((_x/2, _y/2, _z), v) + + a, b = project.((surf(3, 3pi/2), surf(2, 3pi/2)), (v,)) + + annotate!([ + (r1..., text(L"r_2",:bottom)), + (r2..., text(L"r_1",:bottom)), + (theta..., text(L"\theta")), + (a..., text(L"l_2",:right, :top)), + (b..., text(L"l_1", :right, :top)) + ]) + + current() +end + +p2 = let + θ = 2pi - pi/3 + + θs = range(2pi-θ, 2pi, 100) + r1, r2 = 2, 3 + + + empty_style = (xaxis = ([], false), + yaxis = ([], false), + legend=false, + aspect_ratio=:equal) + + plt = plot(; empty_style...) + + plot!(r1.*cos.(θs), r1 .* sin.(θs); line=(:black, 1)) + plot!(r2.*cos.(θs), r2 .* sin.(θs); line=(:black, 1)) + + plot!([(0,0),(r1,0)]; line=(:gray, 1, :dash)) + plot!([(r1,0),(r2,0)]; line=(:black, 1)) + + s, c = sincos(2pi-θ) + plot!([(0,0),(r1,0)]; line=(:gray, 1, :dash)) + plot!([(0,0), (r1*c, r1*s)]; line=(:gray, 1, :dash)) + plot!([(r1,0),(r2,0)]; line=(:black, 1)) + plot!([(r1*c, r1*s), (r2*c, r2*s)]; line=(:black, 2)) + + s′,c′ = sincos((2pi - θ)/2) + annotate!([ + (1/2*c′, 1/2*s′, text(L"\gamma")), + (r1*c, r1*s, text(L"l_1",:left, :top)), + (r2*c, r2*s, text(L"l_2", :left, :top)), + ]) + + #= + δ = pi/8 + scs = reverse(sincos.(range(2pi-θ, 2pi - θ + pi - δ,100))) + plot!([1/2 .* (c,s) for (s,c) in scs]; line=(:gray, 1,:dash), arrow=true, side=:head) + scs = sincos.(range(2pi - θ + pi + δ, 2pi,100)) + plot!([1/2 .* (c,s) for (s,c) in scs]; line=(:gray, 1,:dash), arrow=true, side=:head) + =# +end + +plot(p1, p2) +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + + + +The surface of a frustum of a cone and the same area spread out flat. Angle $\gamma = 2\pi(1 - \sin(\theta)$. + +::: + + + By comparing circumferences, it is seen that the angles $\theta$ and $\gamma$ are related by $\gamma = 2\pi(1 - \sin(\theta))$ (as $2\pi r_2 = 2\pi l_2\sin(\theta) = (2\pi-\gamma)/(2\pi) \cdot 2\pi l_2$). The values $l_i$ and $r_i$ are related by $r_i = l_i \sin(\theta)$. The area in both pictures is: $(\pi l_2^2 - \pi l_1^2) \cdot (2\pi-\gamma)/(2\pi)$ which simplifies to $\pi (l_2 + l_1) \cdot \sin(\theta) \cdot (l_2 - l_1)$ or $2\pi \cdot (r_2 - r_1)/2 \cdot \text{slant height}$. + + +Relating this to our values in terms of $f$ and $g$, we have $r_1=f(t_i)$, $r_0 = f(t_{i-1})$, and the slant height is related by $(l_2-l_1)^2 = (g(t_2)-g(t_1))^2 + (f(t_2) - f(t_1))^2$. Putting this altogether we get that the surface area generarated by rotating the line segment around the $x$ axis is @@ -102,7 +434,7 @@ Putting this altogether we get that the surface area generarated by rotating the $$ \text{sa}_i = \pi (f(t_i)^2 - f(t_{i-1})^2) \cdot \sqrt{(\Delta g)^2 + (\Delta f)^2} / \Delta f = -\pi (f(t_i) + f(t_{i-1})) \cdot \sqrt{(\Delta g)^2 + (\Delta f)^2}. +2\pi \frac{f(t_i) + f(t_{i-1})}{2} \cdot \sqrt{(\Delta g)^2 + (\Delta f)^2}. $$ (This is $2 \pi$ times the average radius times the slant height.) @@ -122,7 +454,9 @@ $$ \text{SA} = \int_a^b 2\pi f(t) \sqrt{g'(t)^2 + f'(t)^2} dt. $$ -If we assume integrability of the integrand, then as our partition size goes to zero, this approximate surface area converges to the value given by the limit. (As with arc length, this needs a technical adjustment to the Riemann integral theorem as here we are evaluating the integrand function at four points ($t_i$, $t_{i-1}$, $\xi$ and $\psi$) and not just at some $c_i$. An figure appears at the end. +If we assume integrability of the integrand, then as our partition size goes to zero, this approximate surface area converges to the value given by the limit. (As with arc length, this needs a technical adjustment to the Riemann integral theorem as here we are evaluating the integrand function at four points ($t_i$, $t_{i-1}$, $\xi$ and $\psi$) and not just at some $c_i$. + + #### Examples @@ -176,7 +510,7 @@ F(1) - F(0) ### Plotting surfaces of revolution -The commands to plot a surface of revolution will be described more clearly later; for now we present them as simply a pattern to be followed in case plots are desired. Suppose the curve in the $x-y$ plane is given parametrically by $(g(u), f(u))$ for $a \leq u \leq b$. +The commands to plot a surface of revolution will be described more clearly later; for now we present them as simply a pattern to be followed in case plots are desired. Suppose the curve in the $x-z$ plane is given parametrically by $(g(u), f(u))$ for $a \leq u \leq b$. To be concrete, we parameterize the circle centered at $(6,0)$ with radius $2$ by: @@ -195,14 +529,14 @@ The plot of this curve is: #| hold: true us = range(a, b, length=100) plot(g.(us), f.(us), xlims=(-0.5, 9), aspect_ratio=:equal, legend=false) -plot!([0,0],[-3,3], color=:red, linewidth=5) # y axis emphasis -plot!([3,9], [0,0], color=:green, linewidth=5) # x axis emphasis +plot!([(0, -3), (0, 3)], line=(:red, 5)) # z axis emphasis +plot!([(3, 0), (9, 0)], line=(:green, 5)) # x axis emphasis ``` -Though parametric plots have a convenience constructor, `plot(g, f, a, b)`, we constructed the points with `Julia`'s broadcasting notation, as we will need to do for a surface of revolution. The `xlims` are adjusted to show the $y$ axis, which is emphasized with a layered line. The line is drawn by specifying two points, $(x_0, y_0)$ and $(x_1, y_1)$ in the form `[x0,x1]` and `[y0,y1]`. +Though parametric plots have a convenience constructor, `plot(g, f, a, b)`, we constructed the points with `Julia`'s broadcasting notation, as we will need to do for a surface of revolution. The `xlims` are adjusted to show the $y$ axis, which is emphasized with a layered line. The line is drawn by specifying two points, $(x_0, y_0)$ and $(x_1, y_1)$ using tuples and wrapping in a vector. -Now, to rotate this about the $y$ axis, creating a surface plot, we have the following pattern: +Now, to rotate this about the $z$ axis, creating a surface plot, we have the following pattern: ```{julia} S(u,v) = [g(u)*cos(v), g(u)*sin(v), f(u)] @@ -210,23 +544,22 @@ us = range(a, b, length=100) vs = range(0, 2pi, length=100) ws = unzip(S.(us, vs')) # reorganize data surface(ws..., zlims=(-6,6), legend=false) -plot!([0,0], [0,0], [-3,3], color=:red, linewidth=5) # y axis emphasis +plot!([(0,0,-3), (0,0,3)], line=(:red, 5)) # z axis emphasis ``` -The `unzip` function is not part of base `Julia`, rather part of `CalculusWithJulia` (it is really `SplitApplyCombine`'s `invert` function). This function rearranges data into a form consumable by the plotting methods like `surface`. In this case, the result of `S.(us,vs')` is a grid (matrix) of points, the result of `unzip` is three grids of values, one for the $x$ values, one for the $y$ values, and one for the $z$ values. A manual adjustment to the `zlims` is used, as `aspect_ratio` does not have an effect with the `plotly()` backend and errors on 3d graphics with `pyplot()`. +The `unzip` function is not part of base `Julia`, rather part of `CalculusWithJulia` (it is really `SplitApplyCombine`'s `invert` function). This function rearranges data into a form consumable by the plotting methods like `surface`. In this case, the result of `S.(us,vs')` is a grid (matrix) of points, the result of `unzip` is three grids of values, one for the $x$ values, one for the $y$ values, and one for the $z$ values. A manual adjustment to the `zlims` is used, as `aspect_ratio` does not have an effect with the `plotly()` backend. To rotate this about the $x$ axis, we have this pattern: ```{julia} -#| hold: true S(u,v) = [g(u), f(u)*cos(v), f(u)*sin(v)] us = range(a, b, length=100) vs = range(0, 2pi, length=100) ws = unzip(S.(us,vs')) -surface(ws..., legend=false) -plot!([3,9], [0,0],[0,0], color=:green, linewidth=5) # x axis emphasis +plot([(3,0,0), (9,0,0)], line=(:green,5)) # x axis emphasis +surface!(ws..., legend=false) ``` The above pattern covers the case of rotating the graph of a function $f(x)$ of $a,b$ by taking $g(t)=t$. @@ -551,46 +884,3 @@ a, b = 0, pi val, _ = quadgk(t -> 2pi* f(t) * sqrt(g'(t)^2 + f'(t)^2), a, b) numericq(val) ``` - - -# Appendix - -```{julia} -#| hold: true -#| echo: false -gr() -## For **some reason** having this in the natural place messes up the plots. -## {{{approximate_surface_area}}} - -xs,ys = range(-1, stop=1, length=50), range(-1, stop=1, length=50) -f(x,y)= 2 - (x^2 + y^2) - -dr = [1/2, 3/4] -df = [f(dr[1],0), f(dr[2],0)] - -function sa_approx_graph(i) - p = plot(xs, ys, f, st=[:surface], legend=false) - for theta in range(0, stop=i/10*2pi, length=10*i ) - path3d!(p,sin(theta)*dr, cos(theta)*dr, df) - end - p -end -n = 10 - -anim = @animate for i=1:n - sa_approx_graph(i) -end - -imgfile = tempname() * ".gif" -gif(anim, imgfile, fps = 1) - - -caption = L""" - -Surface of revolution of $f(x) = 2 - x^2$ about the $y$ axis. The lines segments are the images of rotating the secant line connecting $(1/2, f(1/2))$ and $(3/4, f(3/4))$. These trace out the frustum of a cone which approximates the corresponding surface area of the surface of revolution. In the limit, this approximation becomes exact and a formula for the surface area of surfaces of revolution can be used to compute the value. - -""" - -plotly() -ImageFile(imgfile, caption) -``` diff --git a/quarto/integrals/volumes_slice.qmd b/quarto/integrals/volumes_slice.qmd index 80d466f..72a5f5b 100644 --- a/quarto/integrals/volumes_slice.qmd +++ b/quarto/integrals/volumes_slice.qmd @@ -19,11 +19,47 @@ using SymPy ```{julia} #| echo: false #| results: "hidden" -import LinearAlgebra: norm +import LinearAlgebra: norm, cross using SplitApplyCombine nothing ``` + +```{julia} +#| echo: false +# commands used for plotting from https://github.com/SigurdAngenent/WisconsinCalculus/blob/master/figures/221/09surf_of_rotation2.py + + +#linear projection of R^3 onto R^2 +function _proj(X, v) + # a is ⟂ to v and b is v × a + vx, vy, vz = v + a = [-vy, vx, 0] + b = cross([vx,vy,vz], a) + a′, b′ = a/norm(a), b/norm(b) + return (a′ ⋅ X, b′ ⋅ X) +end + +# project a curve in R3 onto R2 +pline(viewp, ps...) = [_proj(p, viewp) for p in ps] + +# determinant of Jacobian; area multiplier +# det(J); used to identify folds +function jac(X, u, v) + return det(ForwardDiff.jacobian(xs -> collect(X(xs...)), [u,v])) +end + +function _fold(F, t, θmin, θmax) + λ = θ -> jac(F, t, θ) # F is projected surface, psurf + iszero(λ(θmin)) && return θmin + iszero(λ(θmax)) && return θmax + + return solve(ZeroProblem(λ, (θmin, θmax))) +end + +nothing +``` + --- @@ -133,38 +169,13 @@ plt = let # plot surface of revolution around x axis between [0, 3] # best if r(t) descreases - rad(x) = 2/(1 + exp(x))#2/(2.0+x) - viewp = [2,-2,1] - - ## - unitize(x) = x / norm(x) - - """Orthogonal projection along the vector viewp""" - function make_Pmat(viewp) - a = unitize( [-viewp[2], viewp[1], 0] ) - b = unitize( [-viewp[3]*viewp[1], - -viewp[3]*viewp[2], - viewp[1]^2 + viewp[2]^2] ) - collect(zip(a,b)) - end - - #linear projection of R^3 onto R^2 - function proj(X, viewp) - Pmat = make_Pmat(viewp) - x=sum([Pmat[i][1]*X[i] for i in 1:3]) - y=sum([Pmat[i][2]*X[i] for i in 1:3]) - (x, y) # a point - end - proj(X) = proj(X, viewp) - - # discrete determinant of Jacobian; area multiplier? - function jac(X, u, v) - ϵ = 0.000001 - A = map((p,q) -> (p-q)/ϵ, X(u+ϵ/2, v), X(u-ϵ/2, v)) - B = map((p,q) -> (p-q)/ϵ, X(u, v+ϵ/2), X(u, v-ϵ/2)) - return A[1]*B[2]-A[2]*B[1] - end + rad(x) = 2/(1 + exp(x)) + trange = (0,3) + θrange = (0, 2pi) + viewp = [2,-2, 1] + ## + proj(X) = _proj(X, viewp) # surface of revolution surf(t, z) = [t, rad(t)*cos(z), rad(t)*sin(z)] @@ -172,80 +183,63 @@ plt = let # project the surface at (t, a=theta) psurf(t,z) = proj(surf(t,z)) - - bisect(f, a, b) = find_zero(f, (a,b), Bisection()) - _fold(t, zmin, zmax) = bisect(z -> jac(psurf, t, z), zmin, zmax) - # create shape holding project disc drawdiscF(t) = Shape(invert([psurf(t, 2*i*pi/100) for i in 1:101])...) - # project a line between two points - pline!(p,q; kwargs...) = plot!([proj(p),proj(q)]; - line_style..., kwargs...) - α = 1.0 - line_style = (; line=(:black, 1)) + α = 1.0 # opacity + line_style = (; line=(:black, 1)) plot(; empty_style..., aspect_ratio=:equal) # by layering, we get x-axis as desired - pline!([-1,0,0], [0,0,0]) + plot!(pline(viewp, [-1,0,0], [0,0,0]); line_style...) plot!(drawdiscF(0); fill =(:lightgray, α)) - pline!([0,0,0], [1,0,0]) + + plot!(pline(viewp, [0,0,0], [1,0,0]); line_style...) plot!(drawdiscF(1); fill =(:black, α)) # black to lightgray gives thickness plot!(drawdiscF(1.1); fill=(:lightgray, α)) - pline!([1.1,0,0], [2,0,0]) + + plot!(pline(viewp, [1.1,0,0], [2,0,0]); line_style...) plot!(drawdiscF(2); fill=(:lightgray, α)) - pline!([2,0,0], [3,0,0]) + + plot!(pline(viewp, [2,0,0], [3,0,0]); line_style...) plot!(drawdiscF(3); fill=(:lightgray, α)) - pline!([3,0,0], [4,0,0]; arrow=true, side=:head) - pline!([0,0,0], [0,0,1.25]; arrow=true, side=:head) + + plot!(pline(viewp, [3,0,0], [4,0,0]); line_style..., arrow=true, side=:head) + plot!(pline(viewp, [0,0,0], [0,0,1.25]); line_style..., arrow=true, side=:head) - tt = range(0, pi, 30) - curve = [psurf(t, pi/2) for t in tt] + tt = range(trange..., 30) + + curve = psurf.(tt, pi/2) plot!(curve; line=(:black, 2)) - f1 = [[t, _fold(t, 0, pi)] for t in tt] + f1 = [(t, _fold(psurf, t, 0, pi)) for t in tt] curve = [psurf(f[1], f[2]) for f in f1] - plot!(curve; line=(:black,)) + plot!(curve; line=(:black,1)) - f2 = [[t, _fold(t, pi, 2*pi)] for t in tt] + f2 = [(t, _fold(psurf, t, pi, 2*pi)) for t in tt] curve = [psurf(f[1], f[2]) for f in f2] - plot!(curve; line=(:black,)) + plot!(curve; line=(:black,1)) - tt= [0.025*i for i in 1:121] - f1 = [[t, _fold(t, pi, 2*pi)] for t in tt] - for f in f1 - plot!([psurf(f[1], f[2]-k*0.01*(4-f[1])) for k in 1:21]; - line=(:black,1)) + ## find bottom edge (t,θ) again + tt = range(0, 3, 120) + f1 = [(t, _fold(psurf, t, pi, 2*pi)) for t in range(trange..., 100)] + + # shade bottom by adding bigger density of lines near bottom + for (i,f) ∈ enumerate(f1) + λ = iseven(i) ? 6 : 4 # adjust density by have some lines only extend to 6 + isnan(f[1]) || isnan(f[2]) && continue + curve = [psurf(f[1], θ) for θ in range(f[2] - 0.2*(λ - f[1]), f[2], 20)] + plot!(curve; line=(:black, 1)) end - tt= [0.05*i for i in 1:61] - f1 = [[t, _fold(t, pi, 2*pi)] for t in tt] - for f in f1 - plot!([psurf( f[1], f[2]-k*0.01*(6-f[1]) ) - for k in 1:21]; line=(:black, 1)) - end -#= - ts = 0:.1:2.95 - θs = range(pi + pi/4, 3pi/2, 25) - for ti in ts - plot!([psurf(ti,θ) for θ in θs]; line=(:black, 1)) - end - θs = range(pi + pi/6, 3pi/2, 25) - for ti in ts - plot!([psurf(ti +0.05,θ) for θ in θs]; line=(:black, 1)) - end -=# + current() - #ts = range(0, 3, 100) - #θs = range(0, 2pi, 100) - #contour(ts, θs, (t,z) -> jac(psurf,t,z), levels=[0]) - end plt ``` @@ -501,37 +495,13 @@ plt = let # plot surface of revolution around x axis between [0, 3] # best if r(t) descreases - rad(x) = 2/(1 + exp(x))#2/(2.0+x) + rad(x) = 2/(1 + exp(x)) + trange = (0, 3) + θrange = (0, 2pi) viewp = [2,-2,1] ## - unitize(x) = x / norm(x) - - """Orthogonal projection along the vector viewp""" - function make_Pmat(viewp) - a = unitize( [-viewp[2], viewp[1], 0] ) - b = unitize( [-viewp[3]*viewp[1], - -viewp[3]*viewp[2], - viewp[1]^2 + viewp[2]^2] ) - collect(zip(a,b)) - end - - #linear projection of R^3 onto R^2 - function proj(X, viewp) - Pmat = make_Pmat(viewp) - x=sum([Pmat[i][1]*X[i] for i in 1:3]) - y=sum([Pmat[i][2]*X[i] for i in 1:3]) - (x, y) # a point - end - proj(X) = proj(X, viewp) - - # discrete determinant of Jacobian; area multiplier? - function jac(X, u, v) - ϵ = 0.000001 - A = map((p,q) -> (p-q)/ϵ, X(u+ϵ/2, v), X(u-ϵ/2, v)) - B = map((p,q) -> (p-q)/ϵ, X(u, v+ϵ/2), X(u, v-ϵ/2)) - return A[1]*B[2]-A[2]*B[1] - end + proj(X) = _proj(X, viewp) # surface of revolution @@ -542,67 +512,64 @@ plt = let psurf(t,z) = proj(surf(t,z)) psurf2(t, z) = proj(surf2(t,z)) - bisect(f, a, b) = find_zero(f, (a,b), Bisection()) - _fold(t, zmin, zmax) = bisect(z -> jac(psurf, t, z), zmin, zmax) - # create shape holding project disc drawdiscF(t) = Shape(invert([psurf(t, 2*i*pi/100) for i in 1:101])...) drawdiscI(t) = Shape([psurf2(t, 2*i*pi/100) for i in 1:101]) - # project a line between two points - pline!(p,q; kwargs...) = plot!([proj(p),proj(q)]; - line_style..., kwargs...) α = 1.0 line_style = (; line=(:black, 1)) plot(; empty_style..., aspect_ratio=:equal) - - # by layering, we get x-axis as desired - pline!([-1,0,0], [0,0,0]) + plot!(pline(viewp, [-1,0,0], [0,0,0]); line_style...) plot!(drawdiscF(0); fill =(:lightgray, α)) plot!(drawdiscI(0); fill=(:white, .5)) - pline!([0,0,0], [1,0,0]) + + plot!(pline(viewp, [0,0,0], [1,0,0]); line_style...) plot!(drawdiscF(1); fill =(:black, α)) # black to lightgray gives thickness plot!(drawdiscI(1); fill=(:white, .5)) plot!(drawdiscF(1.1); fill=(:lightgray, α)) plot!(drawdiscI(1.1); fill=(:white, .5)) - pline!([1.1,0,0], [2,0,0]) + + plot!(pline(viewp, [1.1,0,0], [2,0,0]); line_style...) plot!(drawdiscF(2); fill=(:lightgray, α)) plot!(drawdiscI(2); fill=(:white, .5)) - pline!([2,0,0], [3,0,0]) + + plot!(pline(viewp, [2,0,0], [3,0,0]); line_style...) plot!(drawdiscF(3); fill=(:lightgray, α)) plot!(drawdiscI(3); fill=(:white, .5)) - pline!([3,0,0], [4,0,0]; arrow=true, side=:head) - pline!([0,0,0], [0,0,1.25]; arrow=true, side=:head) + plot!(pline(viewp, [3,0,0], [4,0,0]); line_style..., arrow=true, side=:head) + plot!(pline(viewp, [0,0,0], [0,0,1.25]); line_style..., arrow=true, side=:head) - tt = range(0, pi, 30) + ## bounding curves + ### main spine + tt = range(trange..., 30) curve = [psurf(t, pi/2) for t in tt] plot!(curve; line=(:black, 2)) - f1 = [[t, _fold(t, 0, pi)] for t in tt] + ### the folds + f1 = [(t, _fold(psurf, t, 0, pi)) for t in tt] curve = [psurf(f[1], f[2]) for f in f1] plot!(curve; line=(:black,)) - f2 = [[t, _fold(t, pi, 2*pi)] for t in tt] + f2 = [(t, _fold(psurf, t, pi, 2*pi)) for t in tt] curve = [psurf(f[1], f[2]) for f in f2] plot!(curve; line=(:black,)) + ## add shading + ### find bottom edge (t,θ) again + f1 = [[t, _fold(psurf, t, pi, 2*pi)] for t in range(trange..., 120)] + ### shade bottom by adding bigger density of lines near bottom + for (i,f) ∈ enumerate(f1) + λ = iseven(i) ? 6 : 4 # adjust density by have some lines only extend to 6 + isnan(f[1]) || isnan(f[2]) && continue - tt= [0.025*i for i in 1:121] - f1 = [[t, _fold(t, pi, 2*pi)] for t in tt] - for f in f1 - plot!([psurf(f[1], f[2]-k*0.01*(4-f[1])) for k in 1:21]; - line=(:black,1)) - end - tt= [0.05*i for i in 1:61] - f1 = [[t, _fold(t, pi, 2*pi)] for t in tt] - for f in f1 - plot!([psurf( f[1], f[2]-k*0.01*(6-f[1]) ) - for k in 1:21]; line=(:black, 1)) + curve = [psurf(f[1], θ) for θ in range(f[2] - 0.2*(λ - f[1]), f[2], 20)] + + plot!(curve; line=(:black, 1)) end current() @@ -686,44 +653,12 @@ Let $h$ be the distance from the apex to the base. Consider cones with the prope plt = let gr() rad(t) = 3/2 - t + trange = (0, 3/2) + θrange = (0, 2pi) viewp = [2,-1/1.5,1/2+.2] - empty_style = (xaxis=([], false), - yaxis=([], false), - framestyle=:origin, - legend=false) - axis_style = (arrow=true, side=:head, line=(:gray, 1)) - unitize(x) = x / norm(x) - - """Orthogonal projection along the vector viewp""" - function make_Pmat(viewp) - a = unitize( [-viewp[2], viewp[1], 0] ) - b = unitize( [-viewp[3]*viewp[1], - -viewp[3]*viewp[2], - viewp[1]^2 + viewp[2]^2] ) - collect(zip(a,b)) - end - - #linear projection of R^3 onto R^2 - function proj(X, viewp) - Pmat = make_Pmat(viewp) - x=sum([Pmat[i][1]*X[i] for i in 1:3]) - y=sum([Pmat[i][2]*X[i] for i in 1:3]) - (x, y) # a point - end - proj(X) = proj(X, viewp) - - drawdiscF(t) = Shape([psurf(t, 2*i*pi/100) for i in 1:101]) - - # discrete determinant of Jacobian; area multiplier? - function jac(X, u, v) - ϵ = 0.000001 - A = map((p,q) -> (p-q)/ϵ, X(u+ϵ/2, v), X(u-ϵ/2, v)) - B = map((p,q) -> (p-q)/ϵ, X(u, v+ϵ/2), X(u, v-ϵ/2)) - return A[1]*B[2]-A[2]*B[1] - end - - + ## + proj(X) = _proj(X, viewp) # our surface R, r, rho = 1, 1/4, 1/4 f(t) = (R-r) * cos(t) + rho * cos((R-r)/r * t) @@ -731,6 +666,17 @@ plt = let surf(t, θ) = (rad(t)*f(θ), rad(t)*g(θ), t) psurf(t,θ) = proj(surf(t,θ)) + + empty_style = (xaxis=([], false), + yaxis=([], false), + framestyle=:origin, + legend=false) + axis_style = (arrow=true, side=:head, line=(:gray, 1)) + + + drawdiscF(t) = Shape([psurf(t, 2*i*pi/100) for i in 1:101]) + + plot(; empty_style..., aspect_ratio=:equal) for (i,t) in enumerate(range(0, 3/2, 30)) plot!(drawdiscF(t); fill=(:gray,1), line=(:black,1)) @@ -943,38 +889,37 @@ plt = let rad2(t) = 1/2 viewp = [2,-2,1] + ## + function _proj(X, v) + # a is ⟂ to v and b is v × a + vx, vy, vz = v + a = [-vy, vx, 0] + b = cross([vx,vy,vz], a) + a′, b′ = a/norm(a), b/norm(b) + return (a′ ⋅ X, b′ ⋅ X) + end + + # project a curve in R3 onto R2 + pline(viewp, ps...) = [_proj(p, viewp) for p in ps] + + # determinant of Jacobian; area multiplier + # det(J); used to identify folds + function jac(X, u, v) + return det(ForwardDiff.jacobian(xs -> collect(X(xs...)), [u,v])) + end + + function _fold(F, t, θmin, θmax) + λ = θ -> jac(F, t, θ) # F is projected surface, psurf + iszero(λ(θmin)) && return θmin + iszero(λ(θmax)) && return θmax + + return solve(ZeroProblem(λ, (θmin, θmax))) + end + ## - unitize(x) = x / norm(x) - - """Orthogonal projection along the vector viewp""" - function make_Pmat(viewp) - a = unitize( [-viewp[2], viewp[1], 0] ) - b = unitize( [-viewp[3]*viewp[1], - -viewp[3]*viewp[2], - viewp[1]^2 + viewp[2]^2] ) - collect(zip(a,b)) - end - - #linear projection of R^3 onto R^2 - function proj(X, viewp) - Pmat = make_Pmat(viewp) - x=sum([Pmat[i][1]*X[i] for i in 1:3]) - y=sum([Pmat[i][2]*X[i] for i in 1:3]) - (x, y) # a point - end - proj(X) = proj(X, viewp) - - # discrete determinant of Jacobian; area multiplier? - function jac(X, u, v) - ϵ = 0.000001 - A = map((p,q) -> (p-q)/ϵ, X(u+ϵ/2, v), X(u-ϵ/2, v)) - B = map((p,q) -> (p-q)/ϵ, X(u, v+ϵ/2), X(u, v-ϵ/2)) - return A[1]*B[2]-A[2]*B[1] - end - + proj(X) = _proj(X, viewp) # surface of revolution about the z axis - surf(t, z) = (rad(t)*cos(z), rad(t)*sin(z), t) surf2(t, z) = (rad2(t)*cos(z), rad2(t)*sin(z), t) @@ -983,15 +928,12 @@ plt = let psurf2(t, z) = proj(surf2(t,z)) bisect(f, a, b) = find_zero(f, (a,b), Bisection()) - _fold(t, zmin, zmax) = bisect(z -> jac(psurf, t, z), zmin, zmax) - _foldz(z, tmin, tmax) = bisect(t -> jac(psurf, t, z), tmin, tmax) + # create shape holding project disc drawdiscF(t) = Shape([psurf(t, 2*i*pi/100) for i in 1:101]) drawdiscI(t) = Shape([psurf2(t, 2*i*pi/100) for i in 1:101]) - # project a line between two points - pline!(p,q; kwargs...) = plot!([proj(p),proj(q)]; - line_style..., kwargs...) + α = 1.0 line_style = (; line=(:black, 1)) @@ -1014,17 +956,8 @@ plt = let plot!(drawdiscI(x₀); fill=(:white,1.0), line=(:black,1)) z0 = 3pi/2 - δ - pline!(surf(t0, z0), surf(-t0, z0); line=(:black, 1)) - pline!(surf(t0, z0+pi), surf(-t0, z0+pi); line=(:black, 1)) - - - # boundary of sphere - z0 = 3pi/2 - δ - curve = [psurf(t, z0) for t in range(-t0, t0, 100)] - plot!(curve; line=(:black,3)) - z0 = 3pi/2 - δ + pi - curve = [psurf(t, z0) for t in range(-t0, t0, 100)] - plot!(curve; line=(:black,3)) + plot!(pline(viewp, surf(t0, z0), surf(-t0, z0)); line=(:black, 1)) + plot!(pline(viewp, surf(t0, z0+pi), surf(-t0, z0+pi)); line=(:black, 1)) # caps curve = [psurf(t0, θ) for θ in range(0, 2pi, 100)] @@ -1033,6 +966,17 @@ plt = let plot!(curve, line=(:black, 2)) + ## folds + tθs = [(t, _fold(psurf, t, 0,pi)) for t in range(-t0, t0, 50)] + curve = [psurf(t, θ) for (t,θ) ∈ tθs] + plot!(curve, line=(:black, 3)) + + tθs = [(t, _fold(psurf, t, pi, 2pi)) for t in range(-t0, t0, 50)] + curve = [psurf(t, θ) for (t,θ) ∈ tθs] + plot!(curve, line=(:black, 3)) + + + # Shade lines δ = pi/6 Δₜ = (4pi/2 - (3pi/2 - δ))/(2*25) @@ -1046,7 +990,7 @@ plt = let end #= - f1 = [[t, _fold(t, 0, pi/2)] for t in range(-0.5, -0.1, 26)] + f1 = [[t, _fold(psurf, t, 0, pi/2)] for t in range(-0.5, -0.1, 26)] for f in f1 plot!([psurf( f[1], f[2]-k*0.01*(6-f[1]) ) for k in 1:21]; line=(:black, 1)) diff --git a/quarto/limits/continuity.qmd b/quarto/limits/continuity.qmd index 9cc1801..25890b2 100644 --- a/quarto/limits/continuity.qmd +++ b/quarto/limits/continuity.qmd @@ -72,7 +72,7 @@ The definition says three things * The value of the limit is the same as $f(c)$. -The defined speaks to continuity at a point, we can extend it to continuity over an interval $(a,b)$ by saying: +The definition speaks to continuity at a point, we can extend it to continuity over an interval $(a,b)$ by saying: ::: {.callout-note icon=false} ## Definition of continuity over an open interval @@ -130,9 +130,9 @@ There are various reasons why a function may not be continuous. $$ f(x) = \begin{cases} - -1 & x < 0 \\ - 0 & x = 0 \\ - 1 & x > 0 + -1 &~ x < 0 \\ + 0 &~ x = 0 \\ + 1 &~ x > 0 \end{cases} $$ @@ -148,25 +148,57 @@ is implemented by `Julia`'s `sign` function. It has a value at $0$, but no limit plot([-1,-.01], [-1,-.01], legend=false, color=:black) plot!([.01, 1], [.01, 1], color=:black) scatter!([0], [1/2], markersize=5, markershape=:circle) +ts = range(0, 2pi, 100) +C = Shape(0.02 * sin.(ts), 0.03 * cos.(ts)) +plot!(C, fill=(:white,1), line=(:black, 1)) ``` is not continuous at $x=0$. It has a limit of $0$ at $0$, a function value $f(0) =1/2$, but the limit and the function value are not equal. - * The `floor` function, which rounds down to the nearest integer, is also not continuous at the integers, but is right continuous at the integers, as, for example, $\lim_{x \rightarrow 0+} f(x) = f(0)$. This graph emphasizes the right continuity by placing a point for the value of the function when there is a jump: - + * The `floor` function, which rounds down to the nearest integer, is also not continuous at the integers, but is right continuous at the integers, as, for example, $\lim_{x \rightarrow 0+} f(x) = f(0)$. This graph emphasizes the right continuity by placing a filled marker for the value of the function when there is a jump and an open marker where the function is not that value. ```{julia} -#| hold: true #| echo: false -x = [0,1]; y=[0,0] -plt = plot(x.-2, y.-2, color=:black, legend=false) -plot!(plt, x.-1, y.-1, color=:black) -plot!(plt, x.-0, y.-0, color=:black) -plot!(plt, x.+1, y.+1, color=:black) -plot!(plt, x.+2, y.+2, color=:black) -scatter!(plt, [-2,-1,0,1,2], [-2,-1,0,1,2], markersize=5, markershape=:circle) +plt = let + empty_style = (xticks=-4:4, yticks=-4:4, + framestyle=:origin, + legend=false) + axis_style = (arrow=true, side=:head, line=(:gray, 1)) + text_style = (10,) + fn_style = (;line=(:black, 3)) + fn2_style = (;line=(:red, 4)) + mark_style = (;line=(:gray, 1, :dot)) + domain_style = (;fill=(:orange, 0.35), line=nothing) + range_style = (; fill=(:blue, 0.35), line=nothing) + + ts = range(0, 2pi, 100) + xys = sincos.(ts) + xys = [.1 .* xy for xy in xys] + + plot(; empty_style..., aspect_ratio=:equal) + plot!([-4.25,4.25], [0,0]; axis_style...) + plot!([0,0], [-4.25, 4.25]; axis_style...) + + for k in -4:4 + P,Q = (k,k),(k+1,k) + plot!([P,Q], line=(:black,1)) + S = Shape([k .+ xy for xy in xys]) + plot!(S; fill=(:black,)) + S = Shape([(k+1,k) .+ xy for xy in xys]) + plot!(S; fill=(:white,), line=(:black,1)) + end + + + current() +end plt +``` + +```{julia} +#| echo: false +plotly() +nothing ``` * The function $f(x) = 1/x^2$ is not continuous at $x=0$: $f(x)$ is not defined at $x=0$ and $f(x)$ has no limit at $x=0$ (in the usual sense). @@ -176,8 +208,8 @@ plt $$ f(x) = \begin{cases} -0 & \text{if } x \text{ is irrational,}\\ -1 & \text{if } x \text{ is rational.} +0 &~ \text{if } x \text{ is irrational,}\\ +1 &~ \text{if } x \text{ is rational.} \end{cases} $$ @@ -192,8 +224,8 @@ Let a function be defined by cases: $$ f(x) = \begin{cases} -3x^2 + c & x \geq 0,\\ -2x-3 & x < 0. +3x^2 + c &~ x \geq 0,\\ +2x-3 &~ x < 0. \end{cases} $$ @@ -383,8 +415,8 @@ Let $f(x)$ be defined by $$ f(x) = \begin{cases} -c + \sin(2x - \pi/2) & x > 0\\ -3x - 4 & x \leq 0. +c + \sin(2x - \pi/2) &~ x > 0\\ +3x - 4 &~ x \leq 0. \end{cases} $$ @@ -423,12 +455,22 @@ Consider the function $f(x)$ given by the following graph ```{julia} #| hold: true #| echo: false -xs = range(0, stop=2, length=50) -plot(xs, [sqrt(1 - (x-1)^2) for x in xs], legend=false, xlims=(0,4)) -plot!([2,3], [1,0]) -scatter!([3],[0], markersize=5) -plot!([3,4],[1,0]) -scatter!([4],[0], markersize=5) +let + xs = range(0, stop=2, length=50) + + plot(xs, [sqrt(1 - (x-1)^2) for x in xs]; + line=(:black,1), + legend=false, xlims=(-0.1,4.1)) + plot!([2,3], [1,0]; line=(:black,1)) + plot!([3,4],[1,0]; line=(:black,1)) + + scatter!([(0,0)], markersize=5, markercolor=:black) + scatter!([(2,0)], markersize=5, markercolor=:white) + scatter!([(2, 1)], markersize=5; markercolor=:black) + scatter!([(3,0)], markersize=5; markercolor=:black) + scatter!([(3,1)], markersize=5; markercolor=:white) + scatter!([(4,0)], markersize=5; markercolor=:black) +end ``` The function $f(x)$ is continuous at $x=1$? @@ -513,3 +555,29 @@ choices = ["Can't tell", answ = 1 radioq(choices, answ) ``` + + +###### Question + + +A parametric equation is specified by a parameterization $(f(t), g(t)), a \leq t \leq b$. The parameterization will be continuous if and only if each function is continuous. + + +Suppose $k_x$ and $k_y$ are positive integers and $a, b$ are positive numbers, will the [Lissajous](https://en.wikipedia.org/wiki/Parametric_equation#Lissajous_Curve) curve given by $(a\cos(k_x t), b\sin(k_y t))$ be continuous? + + +```{julia} +#| hold: true +#| echo: false +yesnoq(true) +``` + +Here is a sample graph for $a=1, b=2, k_x=3, k_y=4$: + + +```{julia} +#| hold: true +a,b = 1, 2 +k_x, k_y = 3, 4 +plot(t -> a * cos(k_x *t), t-> b * sin(k_y * t), 0, 4pi) +``` diff --git a/quarto/limits/intermediate_value_theorem.qmd b/quarto/limits/intermediate_value_theorem.qmd index 5562eeb..b5b3288 100644 --- a/quarto/limits/intermediate_value_theorem.qmd +++ b/quarto/limits/intermediate_value_theorem.qmd @@ -17,9 +17,9 @@ using SymPy --- -![Between points M and M lies an F](figures/ivt.jpg){width=40%} +![Between points M and M lies an F for a continuous curve. [L'Hospitals](https://ia801601.us.archive.org/26/items/infinimentpetits1716lhos00uoft/infinimentpetits1716lhos00uoft.pdf) figure 55.](figures/ivt.jpg){width=40%} -Continuity for functions is a valued property which carries implications. In this section we discuss two: the intermediate value theorem and the extreme value theorem. These two theorems speak to some fundamental applications of calculus: finding zeros of a function and finding extrema of a function. [L'Hospitals](https://ia801601.us.archive.org/26/items/infinimentpetits1716lhos00uoft/infinimentpetits1716lhos00uoft.pdf) figure 55, above, suggests why. +Continuity for functions is a valued property which carries implications. In this section we discuss two: the intermediate value theorem and the extreme value theorem. These two theorems speak to some fundamental applications of calculus: finding zeros of a function and finding extrema of a function. ## Intermediate Value Theorem @@ -38,7 +38,7 @@ If $f$ is continuous on $[a,b]$ with, say, $f(a) < f(b)$, then for any $y$ with #| echo: false #| cache: true ### {{{IVT}}} -plt = let +let gr() # IVT empty_style = (xaxis=([], false), @@ -68,7 +68,7 @@ plt = let plot!([(a,0),(a,f(a))]; line=(:black, 1, :dash)) plot!([(b,0),(b,f(b))]; line=(:black, 1, :dash)) - m = f(a/2 + b/2) + m = f(a/2 + b/2) + 1.5 plot!([a, b], [m,m]; line=(:black, 1, :dashdot)) δx = 0.03 @@ -76,10 +76,10 @@ plt = let domain_style...) plot!(Shape((a-.1) .+ 2δx * [-1,1,1,-1], [f(a),f(a),f(b), f(b)]); range_style...) - plot!(Shape((a-.1) .+ δx/2 * [-1,1,1,-1], [y0,y0,y1,y1]); + plot!(Shape((a-.1) .+ 2δx/3 * [-1,1,1,-1], [y0,y0,y1,y1]); range_style...) - zs = find_zeros(f, (a,b)) + zs = find_zeros(x -> f(x) - m, (a,b)) c = zs[2] plot!([(c,0), (c,f(c))]; line=(:black, 1, :dashdot)) @@ -89,52 +89,11 @@ plt = let (c, 0, text(L"c", 12, :top)), (a-.1, f(a), text(L"f(a)", 12, :right)), (a-.1, f(b), text(L"f(b)", 12, :right)), - (b, m, text(L"y", 12, :left)), + (a-0.2, m, text(L"y", 12, :right)), ]) end -plt - -#= -function IVT_graph(n) - f(x) = sin(pi*x) + 9x/10 - a,b = [0,3] - - xs = range(a,stop=b, length=50) - - - ## cheat -- pick an x, then find a y - Δ = .2 - x = range(a + Δ, stop=b - Δ, length=6)[n] - y = f(x) - - plt = plot(f, a, b, legend=false, size=fig_size) - plot!(plt, [0,x,x], [f(x),f(x),0], color=:orange, linewidth=3) - - plt - -end - -n = 6 -anim = @animate for i=1:n - IVT_graph(i) -end - -imgfile = tempname() * ".gif" -gif(anim, imgfile, fps = 1) - - -caption = L""" - -Illustration of intermediate value theorem. The theorem implies that any randomly chosen $y$ -value between $f(a)$ and $f(b)$ will have at least one $x$ in $[a,b]$ -with $f(x)=y$. - -""" - -plotly() -ImageFile(imgfile, caption) -=# ``` + ```{julia} #| echo: false plotly() @@ -145,7 +104,7 @@ Illustration of the intermediate value theorem. The theorem implies that any ran ::: -In the early years of calculus, the intermediate value theorem was intricately connected with the definition of continuity, now it is a consequence. +In the early years of calculus, the intermediate value theorem was intricately connected with the definition of continuity, now it is an important consequence. The basic proof starts with a set of points in $[a,b]$: $C = \{x \text{ in } [a,b] \text{ with } f(x) \leq y\}$. The set is not empty (as $a$ is in $C$) so it *must* have a largest value, call it $c$ (this might seem obvious, but it requires the completeness property of the real numbers). By continuity of $f$, it can be shown that $\lim_{x \rightarrow c-} f(x) = f(c) \leq y$ and $\lim_{y \rightarrow c+}f(x) =f(c) \geq y$, which forces $f(c) = y$. @@ -157,18 +116,6 @@ The basic proof starts with a set of points in $[a,b]$: $C = \{x \text{ in } [a, Suppose we have a continuous function $f(x)$ on $[a,b]$ with $f(a) < 0$ and $f(b) > 0$. Then as $f(a) < 0 < f(b)$, the intermediate value theorem guarantees the existence of a $c$ in $[a,b]$ with $f(c) = 0$. This was a special case of the intermediate value theorem proved by Bolzano first. Such $c$ are called *zeros* of the function $f$. -We use this fact when building a "sign chart" of a polynomial function. Between any two consecutive real zeros the polynomial can not change sign. (Why?) So a "test point" can be used to determine the sign of the function over an entire interval. - -The `sign_chart` function from `CalculusWithJulia` uses this to indicate where an *assumed* continuous function changes sign: - -```{julia} -f(x) = sin(x + x^2) + x/2 -sign_chart(f, -3, 3) -``` - - -The intermediate value theorem can find the sign of the function *between* adjacent zeros, but how are the zeros identified? - Here, we use the Bolzano theorem to give an algorithm - the *bisection method* - to locate a value $c$ in $[a,b]$ with $f(c) = 0$ under the assumptions: * $f$ is continuous on $[a,b]$ @@ -178,7 +125,7 @@ Here, we use the Bolzano theorem to give an algorithm - the *bisection method* - ::: {.callout-note} #### Between -The bisection method is used to find a zero, $c$, of $f(x)$ *between* two values, $a$ and $b$. The method is guaranteed to work under assumptions, the most important being the continuous function having different signs at $a$ and $b$. +The bisection method is used to find a zero, $c$, of $f(x)$ *between* two values, $a$ and $b$. The method is guaranteed to work under the assumption of a continuous function having different signs at $a$ and $b$. ::: @@ -305,7 +252,7 @@ sin(c) (Even `1pi` itself is not a "zero" due to floating point issues.) -### The `find_zero` function. +### The `find_zero` function to solve `f(x) = 0` The `Roots` package has a function `find_zero` that implements the bisection method when called as `find_zero(f, (a,b))` where $[a,b]$ is a bracket. Its use is similar to `simple_bisection` above. This package is loaded when `CalculusWithJulia` is. We illlustrate the usage of `find_zero` in the following: @@ -315,8 +262,8 @@ The `Roots` package has a function `find_zero` that implements the bisection met xstar = find_zero(sin, (3, 4)) ``` -:::{.callout-warning} -## Warning +:::{.callout-note} +## Action template Notice, the call `find_zero(sin, (3, 4))` again fits the template `action(function, args...)` that we see repeatedly. The `find_zero` function can also be called through `fzero`. The use of `(3, 4)` to specify the interval is not necessary. For example `[3,4]` would work equally as well. (Anything where `extrema` is defined works.) ::: @@ -396,6 +343,10 @@ It appears (and a plot over $[0,1]$ verifies) that there is one zero between $-2 find_zero(x^3 - x + 1, (-2, -1)) ``` +#### The `find_zero` function to solve `f(x) = c` + +Solving `f(x) = c` is related to solving `h(x) = 0`. The key is to make a new function using the difference of the two sides: `h(x) = f(x) - c`. + ##### Example Solve for a value of $x$ where `erfc(x)` is equal to `0.5`. @@ -415,6 +366,40 @@ find_zero(h, (-Inf, Inf)) # as wide as possible in this case ``` +##### Example: Inverse functions + +If $f(x)$ is *monotonic* and *continuous* over an interval $[a,b]$ then it has an *inverse function*. That is for any $y$ between $f(a)$ and $f(b)$ we can find an $x$ satisfying $y = f(x)$ with $a \leq x \leq b$. This is due, of course, to both the intermediate value theorem (which guarantees an $x$) and monotonicity (which guarantees just one $x$). + +To see how we can *numerically* find an inverse function using `find_zero`, we have this function: + +```{julia} +function inverse_function(f, a, b, args...; kwargs...) + fa, fb = f(a), f(b) + m, M = fa < fb ? (fa, fb) : (fb, fa) + y -> begin + @assert m ≤ y ≤ M + find_zero(x ->f(x) - y, (a,b), args...; kwargs...) + end +end +``` + +The check on `fa < fb` is due to the possibility that $f$ is increasing (in which case `fa < fb`) or decreasing (in which case `fa > fb`). + +To see this used, we consider the monotonic function $f(x) = x - \sin(x)$ over $[0, 5\pi]$. To graph, we have: + +```{julia} +f(x) = x - sin(x) +a, b = 0, 5pi +plot(inverse_function(f, a, b), f(a), f(b); aspect_ratio=:equal) +``` + +(We plot over the range $[f(a), f(b)]$ here, as we can guess $f(x)$ is *increasing*.) + + +#### The `find_zero` function to solve `f(x) = g(x)` + +Solving `f(x) = g(x)` is related to solving `h(x) = 0`. The key is to make a new function using the difference of the two sides: `h(x) = f(x) - g(x)`. + ##### Example @@ -455,36 +440,6 @@ find_zero(cos(x) ~ x, (0, 2)) [![Intersection of two curves as illustrated by Canadian artist Kapwani Kiwanga.](figures/intersection-biennale.jpg)](https://www.gallery.ca/whats-on/touring-exhibitions-and-loans/around-the-world/canada-pavilion-at-the-venice-biennale/kapwani-kiwanga-trinket){width=40%} -##### Example: Inverse functions - -If $f(x)$ is *monotonic* and *continuous* over an interval $[a,b]$ then it has an *inverse function*. That is for any $y$ between $f(a)$ and $f(b)$ we can find an $x$ satisfying $y = f(x)$ with $a \leq x \leq b$. This is due, of course, to both the intermediate value theorem (which guarantees an $x$) and monotonicity (which guarantees just one $x$). - -To see how we can *numerically* find an inverse function using `find_zero`, we have this function: - -```{julia} -function inverse_function(f, a, b, args...; kwargs...) - fa, fb = f(a), f(b) - m, M = fa < fb ? (fa, fb) : (fb, fa) - y -> begin - @assert m ≤ y ≤ M - find_zero(x ->f(x) - y, (a,b), args...; kwargs...) - end -end -``` - -The check on `fa < fb` is due to the possibility that $f$ is increasing (in which case `fa < fb`) or decreasing (in which case `fa > fb`). - -To see this used, we consider the monotonic function $f(x) = x - \sin(x)$ over $[0, 5\pi]$. To graph, we have: - -```{julia} -f(x) = x - sin(x) -a, b = 0, 5pi -plot(inverse_function(f, a, b), f(a), f(b); aspect_ratio=:equal) -``` - -(We plot over the range $[f(a), f(b)]$ here, as we can guess $f(x)$ is *increasing*.) - - ##### Example @@ -679,7 +634,7 @@ Geometry will tell us that $\cos(x) - x/p$ for *one* $x$ in $[0, \pi/2]$ wheneve #| hold: true f(x, p=1) = cos(x) - x/p I = (0, pi/2) -find_zero(f, I), find_zero(f, I, p=2) +find_zero(f, I), find_zero(f, I; p=2) ``` The second number is the solution when `p=2`. @@ -752,7 +707,7 @@ f.(zs) The `find_zero` function in the `Roots` package is an interface to one of several methods. For now we focus on the *bracketing* methods, later we will see others. Bracketing methods, among others, include `Roots.Bisection()`, the basic bisection method though with a different sense of "middle" than $(a+b)/2$ and used by default above; `Roots.A42()`, which will typically converge much faster than simple bisection; `Roots.Brent()` for the classic method of Brent, and `FalsePosition()` for a family of *regula falsi* methods. These can all be used by specifying the method in a call to `find_zero`. -Alternatively, `Roots` implements the `CommonSolve` interface popularized by its use in the `DifferentialEquations.jl` ecosystem, a wildly successful area for `Julia`. The basic setup involves two steps: setup a "problem;" solve the problem. +Alternatively, `Roots` implements the `CommonSolve` interface popularized by its use in the `DifferentialEquations.jl` ecosystem, a wildly successful area for `Julia`. The basic setup involves two steps: setup a "problem"; solve the problem. To set up a problem we call `ZeroProblem` with the function and an initial interval, as in: @@ -822,6 +777,74 @@ nothing [![Elevation profile of the Hardrock 100 ultramarathon. Treating the elevation profile as a function, the absolute maximum is just about 14,000 feet and the absolute minimum about 7600 feet. These are of interest to the runner for different reasons. Also of interest would be each local maxima and local minima - the peaks and valleys of the graph - and the total elevation climbed - the latter so important/unforgettable its value makes it into the chart's title. ](figures/hardrock-100.jpeg)](https://hardrock100.com){width=50%} + +This figure shows the two concepts as well. + +::: {#fig-absolute-relative} + +```{julia} +#| echo: false +plt = let + + gr() + empty_style = (xaxis=([], false), + yaxis=([], false), + framestyle=:origin, + legend=false) + axis_style = (arrow=true, side=:head, line=(:gray, 1)) + + + p(x) = (x-1)*(x-2)*(x-3)*(x-4) + x/2 + 2 + a, b = 0.25, 4.5 + z₁, z₂, z₃ = zs = find_zeros(x -> ForwardDiff.derivative(p,x), (a, b)) + + a′ = -0.0 + plot(; empty_style...) + plot!(p, a, b; line=(:black, 2)) + plot!([a′,b+0.25], [0,0]; axis_style...) + plot!([a′,a′] .+ .1, [-1, p(0)]; axis_style...) + + δ = .5 + ts = range(0, 2pi, 100) + + for z in zs + plot!([z-δ,z+δ],[p(z),p(z)]; line=(:black, 1)) + C = Shape(z .+ 0.03 * sin.(ts), p(z) .+ 0.3 * cos.(ts)) + plot!(C; fill=(:periwinkle, 1), line=(:black, 1)) + end + for z in (a,b) + C = Shape(z .+ 0.03 * sin.(ts), p(z) .+ 0.3 * cos.(ts)) + plot!(C; fill=(:black, 1), line=(:black, 1)) + end + + κ = 0.33 + annotate!([ + (a, 0, text(L"a", :top)), + (b,0, text(L"b", :top)), + (a + κ/5, p(a), text(raw"absolute max", 10, :left)), + (z₁, p(z₁)-κ, text(raw"absolute min", 10, :top)), + (z₂, p(z₂) + κ, text(raw"relative max", 10, :bottom)), + (z₃, p(z₃) - κ, text(raw"relative min", 10, :top)), + (b, p(b) + κ, text(raw"endpoint", 10, :bottom)) + + ]) + + current() +end +plt +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + +Figure illustrating absolute and relative minima for a function $f(x)$ over $I=[a,b]$. The leftmost point has a $y$ value, $f(a)$, which is an absolute maximum of $f(x)$ over $I$. The three points highlighted between $a$ and $b$ are all relative extrema. The first one is *also* the absolute minimum over $I$. The endpoint is not considered a relative maximum for technical reasons --- there is no interval around $b$, it being on the boundary of $I$. + +::: + + The extreme value theorem discusses an assumption that ensures absolute maximum and absolute minimum values exist. ::: {.callout-note icon=false} @@ -849,7 +872,7 @@ The function $f(x) = \sqrt{1-x^2}$ is continuous on the interval $[-1,1]$ (in th ##### Example -The function $f(x) = x \cdot e^{-x}$ on the closed interval $[0, 5]$ is continuous. Hence it has an absolute maximum, which a graph shows to be $0.4$. It has an absolute minimum, clearly the value $0$ occurring at the endpoint. +The function $f(x) = x \cdot e^{-x}$ on the closed interval $[0, 5]$ is continuous. Hence it has an absolute maximum, which a graph shows to be about $0.4$ and occurring near $x=1$. It has an absolute minimum, clearly the value $0$ occurring at the endpoint. ```{julia} @@ -886,7 +909,7 @@ A New York Times [article](https://www.nytimes.com/2016/07/30/world/europe/norwa ## Continuity and closed and open sets -We comment on two implications of continuity that can be generalized to more general settings. +We comment on two implications of continuity that can be generalized. The two intervals $(a,b)$ and $[a,b]$ differ as the latter includes the endpoints. The extreme value theorem shows this distinction can make a big difference in what can be said regarding *images* of such interval. @@ -1212,7 +1235,7 @@ radioq(choices, answ, keep_order=true) ###### Question -The extreme value theorem has two assumptions: a continuous function and a *closed* interval. Which of the following examples fails to satisfy the consequence of the extreme value theorem because the function is not continuous? +The extreme value theorem has two assumptions: a continuous function and a *closed* interval. Which of the following examples fails to satisfy the consequence of the extreme value theorem because the function is defined on $I$ but is not continuous on $I$? ```{julia} @@ -1227,6 +1250,170 @@ answ = 4 radioq(choices, answ, keep_order=true) ``` +###### Question + +The extreme value theorem is true when $f$ is a continuous function on an interval $I$ *and* $I=[a,b]$ is a *closed* interval. Which of these illustrates why it doesn't apply as $f$ is not continuous on $I$ but is defined on $I$? + + +```{julia} +#| hold: true +#| echo: false + +let + gr() + empty_style = (xaxis=([], false), + yaxis=([], false), + framestyle=:origin, + legend=false) + axis_style = (arrow=true, side=:head, line=(:gray, 1)) + + ts = range(0, 2pi, 100) + + # defined on I; not continuous on I + p1 = plot(;empty_style..., aspect_ratio=:equal) + title!(p1, "(a)") + plot!(p1, x -> 1 - abs(2x), -1, 1, color=:black) + plot!(p1, zero; line=(:black, 1), arrow=true, side=:head) + C = Shape(0.03 .* sin.(ts), 1 .+ 0.03 .* cos.(ts)) + plot!(p1, C, fill=(:white, 1), line=(:black,1)) + C = Shape(0.03 .* sin.(ts), - 0.25 .+ 0.03 .* cos.(ts)) + plot!(p1, C, fill=(:black,1)) + annotate!(p1, [ + (-1,0,text(L"a", :top)), + (1,0,text(L"b", :top)) + ]) + + # not defined on I + p2 = plot(;empty_style...) + title!(p2, "(b)") + plot!(p2, x -> 1/(1-x), 0, .95, color=:black) + plot!(p2, x-> -1/(1-x), 1.05, 2, color=:black) + plot!(p2, zero; axis_style...) + annotate!(p2,[ + (0,0,text(L"a", :top)), + (2, 0, text(L"b", :top)) + ]) + + # not continuous on I + p3 = plot(;empty_style...) + title!(p3, "(c)") + plot!(p3, x -> 1/(1-x), 0, .95, color=:black) + ylims!((-0.25, 1/(1 - 0.96))) + plot!(p3, [0,1.05],[0,0]; axis_style...) + vline!(p3, [1]; line=(:black, 1, :dash)) + annotate!(p3,[ + (0,0,text(L"a", :top)), + (1, 0, text(L"b", :top)) + ]) + + # continuous + p4 = plot(;empty_style...) + title!(p4, "(d)") + f(x) = x^x + a, b = 0, 2 + plot!(p4, f, a, b; line=(:black,1)) + ylims!(p4, (-.25, f(b))) + plot!(p4, [a-.1, b+.1], [0,0]; axis_style...) + scatter!([0,2],[ f(0),f(2)]; marker=(:circle,:black)) + annotate!([ + (a, 0, text(L"a", :top)), + (b, 0, text(L"b", :top)) + + ]) + + l = @layout[a b; c d] + p = plot(p1, p2, p3, p4, layout=l) + imgfile = tempname() * ".png" + savefig(p, imgfile) + hotspotq(imgfile, (0,1/2), (1/2,1)) +end +``` + + + +The extreme value theorem is true when $f$ is a continuous function on an interval $I$ and $I=[a,b]$ is a *closed* interval. Which of these illustrates when the theorem's assumptions are true? + + +```{julia} +#| hold: true +#| echo: false +## come on; save this figure... +let + gr() + empty_style = (xaxis=([], false), + yaxis=([], false), + framestyle=:origin, + legend=false) + axis_style = (arrow=true, side=:head, line=(:gray, 1)) + + ts = range(0, 2pi, 100) + + # defined on I; not continuous on I + p1 = plot(;empty_style..., aspect_ratio=:equal) + title!(p1, "(a)") + plot!(p1, x -> 1 - abs(2x), -1, 1, color=:black) + plot!(p1, zero; line=(:black, 1), arrow=true, side=:head) + C = Shape(0.03 .* sin.(ts), 1 .+ 0.03 .* cos.(ts)) + plot!(p1, C, fill=(:white, 1), line=(:black,1)) + C = Shape(0.03 .* sin.(ts), - 0.25 .+ 0.03 .* cos.(ts)) + plot!(p1, C, fill=(:black,1)) + annotate!(p1, [ + (-1,0,text(L"a", :top)), + (1,0,text(L"b", :top)) + ]) + + # not defined on I + p2 = plot(;empty_style...) + title!(p2, "(b)") + plot!(p2, x -> 1/(1-x), 0, .95, color=:black) + plot!(p2, x-> -1/(1-x), 1.05, 2, color=:black) + plot!(p2, zero; axis_style...) + annotate!(p2,[ + (0,0,text(L"a", :top)), + (2, 0, text(L"b", :top)) + ]) + + # not continuous on I + p3 = plot(;empty_style...) + title!(p3, "(c)") + plot!(p3, x -> 1/(1-x), 0, .95, color=:black) + ylims!((-0.1, 1/(1 - 0.96))) + plot!(p3, [0,1.05],[0,0]; axis_style...) + vline!(p3, [1]; line=(:black, 1, :dash)) + annotate!(p3,[ + (0,0,text(L"a", :top)), + (1, 0, text(L"b", :top)) + ]) + + # continuous + p4 = plot(;empty_style...) + title!(p4, "(d)") + f(x) = x^x + a, b = 0, 2 + ylims!(p4, (-.25, f(b))) + plot!(p4, f, a, b; line=(:black,1)) + plot!(p4, [a-.1, b+.1], [0,0]; axis_style...) + scatter!([0,2],[ f(0),f(2)]; marker=(:circle,:black)) + annotate!([ + (a, 0, text(L"a", :top)), + (b, 0, text(L"b", :top)) + + ]) + + l = @layout[a b; c d] + p = plot(p1, p2, p3, p4, layout=l) + imgfile = tempname() * ".png" + savefig(p, imgfile) + hotspotq(imgfile, (1/2,1), (0,1/2)) +end +``` + +```{julia} +#| echo: false +plotly(); +``` + + ###### Question @@ -1318,28 +1505,3 @@ The zeros of the equation $\cos(x) \cdot \cosh(x) = 1$ are related to vibrations val = maximum(find_zeros(x -> cos(x) * cosh(x) - 1, (0, 6pi))) numericq(val) ``` - -###### Question - - -A parametric equation is specified by a parameterization $(f(t), g(t)), a \leq t \leq b$. The parameterization will be continuous if and only if each function is continuous. - - -Suppose $k_x$ and $k_y$ are positive integers and $a, b$ are positive numbers, will the [Lissajous](https://en.wikipedia.org/wiki/Parametric_equation#Lissajous_Curve) curve given by $(a\cos(k_x t), b\sin(k_y t))$ be continuous? - - -```{julia} -#| hold: true -#| echo: false -yesnoq(true) -``` - -Here is a sample graph for $a=1, b=2, k_x=3, k_y=4$: - - -```{julia} -#| hold: true -a,b = 1, 2 -k_x, k_y = 3, 4 -plot(t -> a * cos(k_x *t), t-> b * sin(k_y * t), 0, 4pi) -``` diff --git a/quarto/limits/limits.qmd b/quarto/limits/limits.qmd index 614319d..84a256b 100644 --- a/quarto/limits/limits.qmd +++ b/quarto/limits/limits.qmd @@ -36,27 +36,38 @@ colors = [:black, :blue, :orange, :red, :green, :orange, :purple] ## Area of parabola function make_triangle_graph(n) title = "Area of parabolic cup ..." - n==1 && (title = "\${Area = }1/2\$") - n==2 && (title = "\${Area = previous }+ 1/8\$") - n==3 && (title = "\${Area = previous }+ 2\\cdot(1/8)^2\$") - n==4 && (title = "\${Area = previous }+ 4\\cdot(1/8)^3\$") - n==5 && (title = "\${Area = previous }+ 8\\cdot(1/8)^4\$") - n==6 && (title = "\${Area = previous }+ 16\\cdot(1/8)^5\$") - n==7 && (title = "\${Area = previous }+ 32\\cdot(1/8)^6\$") + n==1 && (title = L"Area $= 1/2$") + n==2 && (title = L"Area $=$ previous $+\; \frac{1}{8}$") + n==3 && (title = L"Area $=$ previous $+\; 2\cdot\frac{1}{8^2}$") + n==4 && (title = L"Area $=$ previous $+\; 4\cdot\frac{1}{8^3}$") + n==5 && (title = L"Area $=$ previous $+\; 8\cdot\frac{1}{8^4}$") + n==6 && (title = L"Area $=$ previous $+\; 16\cdot\frac{1}{8^5}$") + n==7 && (title = L"Area $=$ previous $+\; 32\cdot\frac{1}{8^6}$") - plt = plot(f, 0, 1, legend=false, size = fig_size, linewidth=2) - annotate!(plt, [(0.05, 0.9, text(title,:left))]) # if in title, it grows funny with gr - n >= 1 && plot!(plt, [1,0,0,1, 0], [1,1,0,1,1], color=colors[1], linetype=:polygon, fill=colors[1], alpha=.2) - n == 1 && plot!(plt, [1,0,0,1, 0], [1,1,0,1,1], color=colors[1], linewidth=2) + plt = plot(f, 0, 1; + legend=false, + size = fig_size, + linewidth=2) + annotate!(plt, [ + (0.05, 0.9, text(title,:left)) + ]) # if in title, it grows funny with gr + n >= 1 && plot!(plt, [1,0,0,1, 0], [1,1,0,1,1]; + color=colors[1], linetype=:polygon, + fill=colors[1], alpha=.2) + n == 1 && plot!(plt, [1,0,0,1, 0], [1,1,0,1,1]; + color=colors[1], linewidth=2) for k in 2:n - xs = range(0,stop=1, length=1+2^(k-1)) - ys = map(f, xs) - k < n && plot!(plt, xs, ys, linetype=:polygon, fill=:black, alpha=.2) + xs = range(0, stop=1, length=1+2^(k-1)) + ys = f.(xs) + k < n && plot!(plt, xs, ys; + linetype=:polygon, fill=:black, alpha=.2) if k == n - plot!(plt, xs, ys, color=colors[k], linetype=:polygon, fill=:black, alpha=.2) - plot!(plt, xs, ys, color=:black, linewidth=2) + plot!(plt, xs, ys; + color=colors[k], linetype=:polygon, fill=:black, alpha=.2) + plot!(plt, xs, ys; + color=:black, linewidth=2) end end plt @@ -1041,18 +1052,34 @@ $$ Why? We can express the function $e^{\csc(x)}/e^{\cot(x)}$ as the above function plus the polynomial $1 + x/2 + x^2/8$. The above is then the sum of two functions whose limits exist and are finite, hence, we can conclude that $M = 0 + 1$. -### The [squeeze](http://en.wikipedia.org/wiki/Squeeze_theorem) theorem +### The squeeze theorem + +Sometimes limits can be found by bounding more complicated functions by easier functions. + +::: {.callout-note icon=false} +## The [squeeze theorem](http://en.wikipedia.org/wiki/Squeeze_theorem) + +Fix $c$ in $I=(a,b)$. Suppose for all $x$ in $I$, except possibly $c$, there are two functions $l$ and $u$, satisfying: -We note one more limit law. Suppose we wish to compute $\lim_{x \rightarrow c}f(x)$ and we have two other functions, $l$ and $u$, satisfying: +* $l(x) \leq f(x) \leq u(x)$. + +* These limits exist and are equal: + +$$ +L = \lim_{x \rightarrow c} l(x) = \lim_{x \rightarrow c} u(x). +$$ - * for all $x$ near $c$ (possibly not including $c$) $l(x) \leq f(x) \leq u(x)$. - * These limits exist and are equal: $L = \lim_{x \rightarrow c} l(x) = \lim_{x \rightarrow c} u(x)$. +Then +$$ +\lim_{x\rightarrow c} f(x) = L. +$$ -Then the limit of $f$ must also be $L$. +::: +The figure shows a usage of the squeeze theorem to show $\sin(x)/x \rightarrow 1$ as $\cos(x) \leq \sin(x)x \leq 1$ for $x$ close to $0$. ```{julia} #| hold: true @@ -1088,8 +1115,71 @@ ImageFile(imgfile, caption) The formal definition of a limit involves clarifying what it means for $f(x)$ to be "close to $L$" when $x$ is "close to $c$". These are quantified by the inequalities $0 < \lvert x-c\rvert < \delta$ and the $\lvert f(x) - L\rvert < \epsilon$. The second does not have the restriction that it is greater than $0$, as indeed $f(x)$ can equal $L$. The order is important: it says for any idea of close for $f(x)$ to $L$, an idea of close must be found for $x$ to $c$. + The key is identifying a value for $\delta$ for a given value of $\epsilon$. +::: {#fig-limit-e-d} +```{julia} +#| echo: false +plt = let + gr() + f(x) = (x+1)^2 -1 + a, b = -1/4, 2 + c = 1 + L = f(c) + δ = 0.2 + ϵ = 3sqrt(δ) + + plot(; empty_style...)#, aspect_ratio=:equal) + plot!(f, a, b; line=(:black, 2)) + plot!([a,b],[0,0]; axis_style...) + plot!([0,0], [f(a), f(2)]; axis_style...) + + plot!([c, c, 0], [0, f(c), f(c)]; line=(:black, 1, :dash)) + plot!([c-δ, c-δ, 0], [0, f(c-δ), f(c-δ)]; line=(:black, 1, :dashdot)) + plot!([c+δ, c+δ, 0], [0, f(c+δ), f(c+δ)]; line=(:black, 1, :dashdot)) + + S = Shape([0,b,b,0],[L-ϵ,L-ϵ,L+ϵ,L+ϵ]) + plot!(S; fill=(:lightblue, 0.25), line=(nothing,)) + + domain_color=:red + range_color=:blue + + + S = Shape([c-δ, c+δ, c+δ, c-δ], 0.05*[-1,-1,1,1]) + plot!(S, fill=(domain_color,0.4), line=nothing) + m,M = f(c-δ), f(c+δ) + T = Shape(0.015 * [-1,1,1,-1], [m,m,M,M]) + plot!(T, fill=(range_color, 0.4), line=nothing) + + + C = Plots.scale(Shape(:circle), 0.02, 0.1) + plot!(Plots.translate(C, c, L), fill=(:white,1,0), line=(:black, 1)) + plot!(Plots.translate(C, c, 0), fill=(:white,1,0), line=(domain_color, 1)) + plot!(Plots.translate(C, 0, L), fill=(:white,1,0), line=(range_color, 1)) + + annotate!([ + (c, 0, text(L"c", :top)), + (c-δ, 0, text(L"c - \delta", 10, :top)), + (c+δ, 0, text(L"c + \delta", 10, :top)), + (0, L, text(L"L", :right)), + (0, L+ϵ, text(L"L + \epsilon", 10, :right)), + (0, L-ϵ, text(L"L - \epsilon", 10, :right)), + + ]) + +end +plt +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + +Figure illustrating requirements of $\epsilon-\delta$ definition of the limit. The image (shaded red on $y$ axis) of the $x$ within $\delta$ of $c$ (except for $c$ and shaded blue on the $x$ axis) must stay within the bounds of $L-\epsilon$ and $L+ \epsilon$, where $\delta$ may be chosen based on $\epsilon$ but needs to be chosen for every positive $\epsilon$, not just a fixed one as in this figure. +::: A simple case is the linear case. Consider the function $f(x) = 3x + 2$. Verify that the limit at $c=1$ is $5$. @@ -1117,64 +1207,78 @@ These lines produce a random $\epsilon$, the resulting $\delta$, and then verify (The random numbers are technically in $[0,1)$, so in theory `epsilon` could be `0`. So the above approach would be more solid if some guard, such as `epsilon = max(eps(), rand())`, was used. As the formal definition is the domain of paper-and-pencil, we don't fuss.) -In this case, $\delta$ is easy to guess, as the function is linear and has slope $3$. This basically says the $y$ scale is 3 times the $x$ scale. For non-linear functions, finding $\delta$ for a given $\epsilon$ can be a challenge. For the function $f(x) = x^3$, illustrated below, a value of $\delta=\epsilon^{1/3}$ is used for $c=0$: +In this case, $\delta$ is easy to guess, as the function is linear and has slope $3$. This basically says the $y$ scale is 3 times the $x$ scale. For non-linear functions, finding $\delta$ for a given $\epsilon$ can be more of a challenge. -```{julia} -#| hold: true -#| echo: false -#| cache: true -## {{{ limit_e_d }}} -gr() -function make_limit_e_d(n) - f(x) = x^3 +##### Example - xs = range(-.9, stop=.9, length=50) - ys = map(f, xs) +We show using the definition that for any fixed $a$ and $n$: +$$ +\lim_{x \rightarrow a} x^n = a^n. +$$ + +This proof uses a bound based on properties of the absolute value. - plt = plot(f, -.9, .9, legend=false, size=fig_size) - if n == 0 - nothing - else - k = div(n+1,2) - epsilon = 1/2^k - delta = cbrt(epsilon) - if isodd(n) - plot!(plt, xs, 0*xs .+ epsilon, color=:orange) - plot!(plt, xs, 0*xs .- epsilon, color=:orange) - else - plot!(delta * [-1, 1], epsilon * [ 1, 1], color=:orange) - plot!(delta * [ 1, -1], epsilon * [-1,-1], color=:orange) - plot!(delta * [-1, -1], epsilon * [-1, 1], color=:red) - plot!(delta * [ 1, 1], epsilon * [-1, 1], color=:red) - end - end - plt -end +We look at $f(x) - L = x^n - a^n = (x-a)(x^{n-1} + x^{n-2}a + \cdots + x^1a^{n-1} + a^n)$. +Taking absolute values gives an inequality by the triangle inequality: -n = 11 -anim = @animate for i=1:n - make_limit_e_d(i-1) -end +$$ +\lvert x^n - a^n\rvert \leq \lvert x-a\rvert\cdot +\left( +\lvert x^{n-1}\rvert + +\lvert x^{n-2}\rvert\lvert a\rvert + +\cdots + +\lvert x^1\rvert\lvert a^{n-1}\rvert + +\lvert a^n\rvert +\right). +$$ -imgfile = tempname() * ".gif" -gif(anim, imgfile, fps = 1) +Now, for a given $\epsilon>0$ we seek a $\delta>0$ satisfying the properties of the limit definition for $f(x) = x^n$ and $L=a^n$. For now, assume $\delta < 1$. Then we can assume $\lvert x-a\rvert < \delta$ and +$$ +\lvert x\rvert = \lvert x - a + a\rvert \leq \lvert x-a\rvert + \lvert a\rvert < 1 + \lvert a\rvert +$$ -caption = L""" +This says then -Demonstration of $\epsilon$-$\delta$ proof of $\lim_{x \rightarrow 0} -x^3 = 0$. For any $\epsilon>0$ (the orange lines) there exists a -$\delta>0$ (the red lines of the box) for which the function $f(x)$ -does not leave the top or bottom of the box (except possibly at the -edges). In this example $\delta^3=\epsilon$. +$$ +\begin{align*} +\lvert x^n - a^n\rvert +&\leq +\lvert x-a\rvert\cdot \left( +\lvert x\rvert^{n-1} + +\lvert x\rvert^{n-2}\lvert a\rvert + +\cdots + +\lvert x\rvert^1\lvert a^{n-1}\rvert + +\lvert a^n\rvert +\right)\\ +%% +&\leq \lvert x - a\rvert +\cdot \left( +(\lvert a\rvert+1)^{n-1} + +(\lvert a\rvert+1)^{n-2}\lvert a\rvert ++ \cdots + +(\lvert a\rvert+1)^1 \lvert a^{n-1}\rvert + +\lvert a^n \rvert +\right)\\ +&\leq \lvert x-a\rvert \cdot C, +\end{align*} +$$ + +where $C$ is just some constant not depending on $x$, just $a$ and $n$. + +Now if $\delta < 1$ and $\delta < \epsilon/C$ and if +$0 < \lvert x - a \rvert < \delta$ then + +$$ +\lvert f(x) - L \rvert = +\lvert x^n - a^n\rvert \leq \lvert x-a\rvert \cdot C \leq \delta\cdot C < \frac{\epsilon}{C} \cdot C = \epsilon. +$$ + +With this result, the rules of limits can immediately extend this to any polynomial, $p(x),$ it follows that $\lim_{x \rightarrow c} p(x) = p(a)$. (Because $c_n x^n \rightarrow c_n a^n$ and the sum of two functions with a limit has the limit of the sums.) Based on this, we will say later that any polynomial is *continuous* for all $x$. -""" -plotly() -ImageFile(imgfile, caption) -``` ## Questions @@ -1290,26 +1394,26 @@ let title!(p1, "(a)") plot!(p1, x -> x^2, 0, 2, color=:black) plot!(p1, zero, linestyle=:dash) - annotate!(p1,[(1,0,"a")]) + annotate!(p1,[(1,0,text(L"a",:top))]) p2 = plot(;axis=nothing, legend=false) title!(p2, "(b)") plot!(p2, x -> 1/(1-x), 0, .95, color=:black) plot!(p2, x-> -1/(1-x), 1.05, 2, color=:black) plot!(p2, zero, linestyle=:dash) - annotate!(p2,[(1,0,"a")]) + annotate!(p2,[(1,0,text(L"a",:top))]) p3 = plot(;axis=nothing, legend=false) title!(p3, "(c)") plot!(p3, sinpi, 0, 2, color=:black) plot!(p3, zero, linestyle=:dash) - annotate!(p3,[(1,0,"a")]) + annotate!(p3,[(1,0,text("a",:top))]) p4 = plot(;axis=nothing, legend=false) title!(p4, "(d)") plot!(p4, x -> x^x, 0, 2, color=:black) plot!(p4, zero, linestyle=:dash) - annotate!(p4,[(1,0,"a")]) + annotate!(p4,[(1,0,text(L"a",:top))]) l = @layout[a b; c d] p = plot(p1, p2, p3, p4, layout=l) @@ -1520,8 +1624,8 @@ Take $$ f(x) = \begin{cases} -0 & x \neq 0\\ -1 & x = 0 +0 &~ x \neq 0\\ +1 &~ x = 0 \end{cases} $$ diff --git a/quarto/limits/limits_extensions.qmd b/quarto/limits/limits_extensions.qmd index 1ac9858..0dd4c2a 100644 --- a/quarto/limits/limits_extensions.qmd +++ b/quarto/limits/limits_extensions.qmd @@ -32,22 +32,24 @@ Let's begin with a function that is just problematic. Consider $$ -f(x) = \sin(1/x) +f(x) = \sin(\frac{1}{x}) $$ As this is a composition of nice functions it will have a limit everywhere except possibly when $x=0$, as then $1/x$ may not have a limit. So rather than talk about where it is nice, let's consider the question of whether a limit exists at $c=0$. +@fig-sin-1-over-x shows the issue: -A graph shows the issue: - +:::{#fig-sin-1-over-x} ```{julia} -#| hold: true #| echo: false f(x) = sin(1/x) plot(f, range(-1, stop=1, length=1000)) ``` +Graph of the function $f(x) = \sin(1/x)$ near $0$. It oscillates infinitely many times around $0$. +::: + The graph oscillates between $-1$ and $1$ infinitely many times on this interval - so many times, that no matter how close one zooms in, the graph on the screen will fail to capture them all. Graphically, there is no single value of $L$ that the function gets close to, as it varies between all the values in $[-1,1]$ as $x$ gets close to $0$. A simple proof that there is no limit, is to take any $\epsilon$ less than $1$, then with any $\delta > 0$, there are infinitely many $x$ values where $f(x)=1$ and infinitely many where $f(x) = -1$. That is, there is no $L$ with $|f(x) - L| < \epsilon$ when $\epsilon$ is less than $1$ for all $x$ near $0$. @@ -65,11 +67,10 @@ The following figure illustrates: ```{julia} -#| hold: true f(x) = x * sin(1/x) -plot(f, -1, 1) -plot!(abs) -plot!(x -> -abs(x)) +plot(f, -1, 1; label="f") +plot!(abs; label="|.|") +plot!(x -> -abs(x); label="-|.|") ``` The [squeeze](http://en.wikipedia.org/wiki/Squeeze_theorem) theorem of calculus is the formal reason $f$ has a limit at $0$, as both the upper function, $|x|$, and the lower function, $-|x|$, have a limit of $0$ at $0$. @@ -181,21 +182,38 @@ Consider this funny graph: ```{julia} #| hold: true #| echo: false -xs = range(0,stop=1, length=50) - -plot(x->x^2, -2, -1, legend=false) +let + xs = range(0,stop=1, length=50) +plot(; legend=false, aspect_ratio=true, + xticks = -4:4) + plot!([(-4, -1.5),(-2,4)]; line=(:black,1)) + plot!(x->x^2, -2, -1; line=(:black,1)) plot!(exp, -1,0) plot!(x -> 1-2x, 0, 1) plot!(sqrt, 1, 2) plot!(x -> 1-x, 2,3) +S = Plots.scale(Shape(:circle), 0.05) + + plot!(Plots.translate(S, -4, -1.5); fill=(:black,)) + plot!(Plots.translate(S, -1, (-1)^2); fill=(:white,)) + plot!(Plots.translate(S, -1, exp(-1)); fill=(:black,)) + plot!(Plots.translate(S, 1, 1 - 2(1)); fill=(:black,)) + plot!(Plots.translate(S, 1, sqrt(1)); fill=(:white,)) + plot!(Plots.translate(S, 2, sqrt(2)); fill=(:white,)) + plot!(Plots.translate(S, 2, 1 - (2)); fill=(:black,)) + plot!(Plots.translate(S, 3, 1 - (3)); fill=(:black,)) + +end ``` Describe the limits at $-1$, $0$, and $1$. - * At $-1$ we see a jump, there is no limit but instead a left limit of 1 and a right limit appearing to be $1/2$. - * At $0$ we see a limit of $1$. - * Finally, at $1$ again there is a jump, so no limit. Instead the left limit is about $-1$ and the right limit $1$. +* At $-1$ we see a jump, there is no limit but instead a left limit of 1 and a right limit appearing to be $1/2$. + +* At $0$ we see a limit of $1$. + +* Finally, at $1$ again there is a jump, so no limit. Instead the left limit is about $-1$ and the right limit $1$. ## Limits at infinity diff --git a/quarto/precalc.qmd b/quarto/precalc.qmd index 04b2deb..7e0cfb5 100644 --- a/quarto/precalc.qmd +++ b/quarto/precalc.qmd @@ -1,3 +1,7 @@ +--- +engine: julia +--- + # Precalculus Concepts The mathematical topics in this chapter come from pre-calculus. However, much of the `Julia` usage needed for the rest of the notes are introduced. diff --git a/quarto/precalc/Project.toml b/quarto/precalc/Project.toml index 4c8c9ae..8701513 100644 --- a/quarto/precalc/Project.toml +++ b/quarto/precalc/Project.toml @@ -4,6 +4,7 @@ DataFrames = "a93c6f00-e57d-5684-b7b6-d8193f3e46c0" IJulia = "7073ff75-c697-5162-941a-fcdaad2a7d2a" IntervalArithmetic = "d1acc4aa-44c8-5952-acd4-ba5d80a2a253" LaTeXStrings = "b964fa9f-0449-5b57-a5c2-d3ea65f4040f" +Latexify = "23fbe1c1-3f47-55db-b15f-69d7ec21a316" Markdown = "d6f4376e-aef5-505a-96c1-9c027394607a" Measures = "442fdcdd-2543-5da2-b0f3-8c86c306513e" Mustache = "ffc61752-8dc7-55ee-8c37-f3e9cdd09e70" diff --git a/quarto/precalc/functions.qmd b/quarto/precalc/functions.qmd index 2df9f4b..cb83721 100644 --- a/quarto/precalc/functions.qmd +++ b/quarto/precalc/functions.qmd @@ -42,7 +42,7 @@ For these examples, the domain of both $f(x)$ and $g(x)$ is all real values of $ In general the range is harder to identify than the domain, and this is the case for these functions too. For $f(x)$ we may know the $\cos$ function is trapped in $[-1,1]$ and it is intuitively clear than all values in that set are possible. The function $h(x)$ would have range $[0,\infty)$. The $s(x)$ function is either $-1$ or $1$, so only has two possible values in its range. What about $g(x)$? It is a parabola that opens upward, so any $y$ values below the $y$ value of its vertex will not appear in the range. In this case, the symmetry indicates that the vertex will be at $(1/2, -1/4)$, so the range is $[-1/4, \infty)$. -::: {#fig-domain-range layout-nrow=2} +::: {#fig-domain-range-1} ```{julia} #| echo: false plt = let @@ -70,7 +70,7 @@ plt = let Δy = (y1-y0)/60 Δx = (b - a)/75 - plot(; aspect_ratio=:equal, empty_style...) + plot(; empty_style...) # aspect_ratio=:equal, plot!([-.25,3.25],[0,0]; axis_style...) plot!([0,0], [min(-2Δy, y0 - Δy), y1 + 4Δy]; axis_style... ) @@ -83,9 +83,13 @@ plt = let plot!([a, b], [f(a), f(a)]; mark_style...) plot!([a, b], [f(b), f(b)]; mark_style...) end -plot +plt ``` +Figure of the plot of a function over an interval $[a,b]$ highlighting the domain and the range. +::: + +::: {#fig-domain-range-2} ```{julia} #| echo: false plt = let @@ -124,7 +128,7 @@ plt = let plot!(Shape(Δx*[-1,1,1,-1], [-5, -5,-1,-1]); range_style...) plot!(Shape(Δx*[-1,1,1,-1], [ 5, 5,1,1]); range_style...) end -plot +plt ``` @@ -134,7 +138,7 @@ plotly() nothing ``` -The top figure shows the domain and range of the function as highlighted intervals. The bottom figure shows that the domain may be a collection of intervals. (In this case the $\sec$ function is not defined at $\pi/2 + k \pi$ for integer $k$) and the range may be a collection of intervals. (In this case, the $\sec$ function never have a value in $(-1,1)$. +This figure shows that the domain of a function may be a collection of intervals. (In this case the $\sec$ function is not defined at $\pi/2 + k \pi$ for integer $k$) and the range may be a collection of intervals. (In this case, the $\sec$ function never has a value in $(-1,1)$. ::: @@ -176,7 +180,7 @@ For typical cases like the three above, there isn't really much new to learn. :::{.callout-note} -## Note +## The equals sign is used differently between math and Julia The equals sign in `Julia` always indicates either an assignment or a mutation of the object on the left side. The definition of a function above is an *assignment*, in that a function is added (or modified) in a table holding the methods associated with the function's name. The equals sign restricts the expressions available on the *left*-hand side to a) a variable name, for assignment; b) mutating an object at an index, as in `xs[1]`; c) mutating a property of a struct; or d) a function assignment following this form `function_name(args...)`. @@ -216,7 +220,10 @@ $$ f(x) = 5/9 \cdot (x - 32) $$ -In fact, the graph of a function $f(x)$ is simply defined as the graph of the equation $y=f(x)$. There is a distinction in `Julia` as a command such as +In fact, the graph of a function $f(x)$ is simply defined as the graph of the equation $y=f(x)$. + + +There **is** a distinction in `Julia`. The last command here ```{julia} @@ -236,13 +243,65 @@ f(72) ## room temperature will create a function object with a value of `x` determined at a later time - the time the function is called. So the value of `x` defined when the function is created is not important here (as the value of `x` used by `f` is passed in as an argument). -Within `Julia`, we make note of the distinction between a function object versus a function call. In the definition `f(x)=cos(x)`, the variable `f` refers to a function object, whereas the expression `f(pi)` is a function call. This mirrors the math notation where an $f$ is used when properties of a function are being emphasized (such as $f \circ g$ for composition) and $f(x)$ is used when the values related to the function are being emphasized (such as saying "the plot of the equation $y=f(x)$). +Within `Julia`, we make note of the distinction between a function object versus a function call. In the definition `f(x)=cos(x)`, the variable `f` refers to a function object, whereas the expression `f(pi)` is a function call, resulting in a value. This mirrors the math notation where an $f$ is used when properties of a function are being emphasized (such as $f \circ g$ for composition) and $f(x)$ is used when the values related to the function are being emphasized (such as saying "the plot of the equation $y=f(x)$). -Distinguishing these three related but different concepts -- equations and expressions, function objects, and function calls -- is important when modeling mathematics on the computer. +Distinguishing these related but different concepts --- expressions, equations, values from function calls, and function objects --- is important when modeling mathematics on the computer. + +::: {#fig-kidney} + +```{julia} +#| echo: false +plt = let + gr() + # two kidneys and an arrow + ts = range(0, 2pi, 200) + a = 1 + x(t) = 6a*cos(t) + sin(t) - 4a*cos(t)^5 + y(t) = 4a*sin(t)^3 + S = Shape(x.(ts), y.(ts)) + + y1(t) = 6a*cos(t) -3sin(t) - 4a*cos(t)^5 + x1(t) = 4a*sin(t)^3 + + T = Shape(x1.(ts), y1.(ts)) + T = Plots.translate(T, 10, 0) + + plot(; empty_style...) + plot!(S, fill=(:gray, 0.2), line=(:black, 1)) + plot!(T, fill=(:gray, 0.2), line=(:black, 1)) + + P = (0,0) + Q = (10, 0) + scatter!([P,Q], marker=(:circle,)) + + ts = reverse(range(pi/4+.1, 3pi/4-.1, 100)) + plot!(5 .+ 5*sqrt(2)*cos.(ts), -5 .+ 5*sqrt(2)*sin.(ts); + line=(:black, 1), arrow=true, side=:head) -### Cases + annotate!([ + (P..., text(L"x", :top)), + (Q..., text(L"f(x)", :top)), + (5, 3, text(L"f", :top)), + (0, -6, text("Domain")), + (10, -6, text("Range")) + ]) + +end +plt +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + +Common illustration of an abstract function mapping a value $x$ in the domain to a value $y=f(x)$ in the range. In `Julia`, the values are named, as with `x`, or computed, as with `f(x)`. The function *object* `f` is like the named arrow, which is the name assigned to a particular mapping. +::: + +### Cases, the ternary operator The definition of $s(x)$ above has two cases: @@ -256,7 +315,7 @@ s(x) = \end{cases} $$ -We learn to read this as: when $x$ is less than $0$, then the answer is $-1$. If $x$ is greater than $0$ the answer is $1.$ Often - but not in this example - there is an "otherwise" case to catch those values of $x$ that are not explicitly mentioned. As there is no such "otherwise" case here, we can see that this function has no definition when $x=0$. This function is often called the "sign" function and is also defined by $\lvert x\rvert/x$. (`Julia`'s `sign` function actually defines `sign(0)` to be `0`.) +We learn to read this as: when $x$ is less than $0$, then the answer is $-1$. If $x$ is greater than $0$ the answer is $1.$ Often--but not in this example--there is an "otherwise" case to catch those values of $x$ that are not explicitly mentioned. As there is no such "otherwise" case here, we can see that this function has no definition when $x=0$. This function is often called the "sign" function and is also defined by $\lvert x\rvert/x$. (`Julia`'s `sign` function defines `sign(0)` to be `0`.) How do we create conditional statements in `Julia`? Programming languages generally have "if-then-else" constructs to handle conditional evaluation. In `Julia`, the following code will handle the above condition: @@ -298,7 +357,7 @@ For example, here is one way to define an absolute value function: abs_val(x) = x >= 0 ? x : -x ``` -The condition is `x >= 0` - or is `x` non-negative? If so, the value `x` is used, otherwise `-x` is used. +The condition is `x >= 0`--or is `x` non-negative? If so, the value `x` is used, otherwise `-x` is used. Here is a means to implement a function which takes the larger of `x` or `10`: @@ -311,16 +370,16 @@ bigger_10(x) = x > 10 ? x : 10.0 (This could also utilize the `max` function: `f(x) = max(x, 10.0)`.) -Or similarly, a function to represent a cell phone plan where the first $500$ minutes are $20$ dollars and every additional minute is $5$ cents: +Or similarly, a function to represent a cell phone plan where the first $5$ Gb of data are $11$ dollars and every additional GB is $3.50$: ```{julia} -cellplan(x) = x < 500 ? 20.0 : 20.0 + 0.05 * (x-500) +cellplan(x) = x <= 5 ? 11.0 : 11.0 + 3.50 * (x-5) ``` :::{.callout-warning} ## Warning -Type stability. These last two definitions used `10.0` and `20.0` instead of the integers `10` and `20` for the answer. Why the extra typing? When `Julia` can predict the type of the output from the type of inputs, it can be more efficient. So when possible, we help out and ensure the output is always the same type. +Type stability. These last two definitions used `10.0` and `11.0` instead of the integers `10` and `11` for the answer. Why the extra typing? When `Julia` can predict the type of the output from the type of inputs, it can be more efficient. So when possible, we help out and ensure the output is always the same type. ::: @@ -391,7 +450,7 @@ $$ f(x) = \left(\frac{g}{k v_0\cos(\theta)} + \tan(\theta) \right) x + \frac{g}{k^2}\ln\left(1 - \frac{k}{v_0\cos(\theta)} x \right) $$ -Here $g$ is the gravitational constant $9.8$ and $v_0$, $\theta$ and $k$ parameters, which we take to be $200$, $45$ degrees and $1/2$ respectively. With these values, the above function can be computed when $x=100$ with: +Here $g$ is the gravitational constant $9.8$ and $v_0$, $\theta$, and $k$ parameters, which we take to be $200$, $45$ degrees and $1/2$ respectively. With these values, the above function can be computed when $x=100$ with: ```{julia} @@ -541,7 +600,12 @@ $$ f(x) = m \cdot x + b, \quad g(x) = y_0 + m \cdot (x - x_0). $$ -Both functions use the variable $x$, but there is no confusion, as we learn that this is just a dummy variable to be substituted for and so could have any name. Both also share a variable $m$ for a slope. Where does that value come from? In practice, there is a context that gives an answer. Despite the same name, there is no expectation that the slope will be the same for each function if the context is different. So when parameters are involved, a function involves a rule and a context to give specific values to the parameters. Euler had said initially that functions composed of "the variable quantity and numbers or constant quantities." The term "variable," we still use, but instead of "constant quantities," we use the name "parameters." +Both functions use the variable $x$, but there is no confusion, as we +learn that this is just a dummy variable to be substituted for and so +could have any name. Both also share a variable $m$ for a slope. +Where does the value for $m$ come from? + +In practice, there is a context that gives an answer. Despite the same name, there is no expectation that the slope will be the same for each function if the context is different. So when parameters are involved, a function involves a rule and a context to give specific values to the parameters. Euler had said initially that functions composed of "the variable quantity and numbers or constant quantities." The term "variable," we still use, but instead of "constant quantities," we use the name "parameters." In computer language, instead of context, we use the word *scope*. Something similar is also true with `Julia`. Consider the example of writing a function to model a linear equation with slope $m=2$ and $y$-intercept $3$. A typical means to do this would be to define constants, and then use the familiar formula: @@ -580,7 +644,11 @@ So the `b` is found from the currently stored value. This fact can be exploited. How `Julia` resolves what a variable refers to is described in detail in the manual page [Scope of Variables](https://docs.julialang.org/en/v1/manual/variables-and-scoping/). In this case, the function definition finds variables in the context of where the function was defined, the main workspace, and not where it is called. As seen, this context can be modified after the function definition and prior to the function call. It is only when `b` is needed, that the context is consulted, so the most recent binding is retrieved. Contexts allow the user to repurpose variable names without there being name collision. For example, we typically use `x` as a function argument, and different contexts allow this `x` to refer to different values. -Mostly this works as expected, but at times it can be complicated to reason about. In our example, definitions of the parameters can be forgotten, or the same variable name may have been used for some other purpose. The potential issue is with the parameters, the value for `x` is straightforward, as it is passed into the function. However, we can also pass the parameters, such as $m$ and $b$, as arguments. For parameters, one suggestion is to use [keyword](https://docs.julialang.org/en/v1/manual/functions/#Keyword-Arguments) arguments. These allow the specification of parameters, but also give a default value. This can make usage explicit, yet still convenient. For example, here is an alternate way of defining a line with parameters `m` and `b`: +Mostly this works as expected, but at times it can be complicated to reason about. In our example, definitions of the parameters can be forgotten, or the same variable name may have been used for some other purpose. The potential issue is with the parameters, the value for `x` is straightforward, as it is passed into the function. + +However, we can also pass the parameters, such as $m$ and $b$, as arguments. There are different styles employed. + +For parameters, one suggestion is to use [keyword](https://docs.julialang.org/en/v1/manual/functions/#Keyword-Arguments) arguments. These allow the specification of parameters, but also give a default value. This can make usage explicit, yet still convenient. For example, here is an alternate way of defining a line with parameters `m` and `b`: ```{julia} @@ -604,7 +672,7 @@ During this call, values for `m` and `b` are found from how the function is call mxplusb(0; m=3, b=2) ``` -Keywords are used to mark the parameters whose values are to be changed from the default. Though one can use *positional arguments* for parameters - and there are good reasons to do so - using keyword arguments is a good practice if performance isn't paramount, as their usage is more explicit yet the defaults mean that a minimum amount of typing needs to be done. +Keywords are used to mark the parameters whose values are to be changed from the default. Though one can use *positional arguments* for parameters--and there are good reasons to do so--using keyword arguments is a good practice if performance isn't paramount, as their usage is more explicit yet the defaults mean that a minimum amount of typing needs to be done. Keyword arguments are widely used with plotting commands, as there are numerous options to adjust, but typically only a handful adjusted per call. The `Plots` package whose commands we illustrate throughout these notes starting with the next section has this in its docs: `Plots.jl` follows two simple rules with data and attributes: @@ -640,14 +708,14 @@ For example, here we use a *named tuple* to pass parameters to `f`: ```{julia} #| hold: true -function trajectory(x ,p) +function trajectory(x, p) g, v0, theta, k = p.g, p.v0, p.theta, p.k # unpack parameters a = v0 * cos(theta) (g/(k*a) + tan(theta))* x + (g/k^2) * log(1 - k/a*x) end -p = (g=9.8, v0=200, theta = 45*pi/180, k=1/2) +p = (g=9.8, v0=200, theta=45*pi/180, k=1/2) trajectory(100, p) ``` @@ -687,7 +755,7 @@ The right-hand sides may or may not be familiar, but it should be reasonable to Earlier we saw the `log` function can use a second argument to express the base. This function is basically defined by `log(b,x)=log(x)/log(b)`. The `log(x)` value is the natural log, and this definition just uses the change-of-base formula for logarithms. -But not so fast, on the left side is a function with two arguments and on the right side the functions have one argument - yet they share the same name. How does `Julia` know which to use? `Julia` uses the number, order, and *type* of the positional arguments passed to a function to determine which function definition to use. This is technically known as [multiple dispatch](http://en.wikipedia.org/wiki/Multiple_dispatch) or **polymorphism**. As a feature of the language, it can be used to greatly simplify the number of functions the user must learn. The basic idea is that many functions are "generic" in that they have methods which will work differently in different scenarios. +But not so fast, on the left side is a function with two arguments and on the right side the functions have one argument--yet they share the same name. How does `Julia` know which to use? `Julia` uses the number, order, and *type* of the positional arguments passed to a function to determine which function definition to use. This is technically known as [multiple dispatch](http://en.wikipedia.org/wiki/Multiple_dispatch) or **polymorphism**. As a feature of the language, it can be used to greatly simplify the number of functions the user must learn. The basic idea is that many functions are "generic" in that they have methods which will work differently in different scenarios. :::{.callout-warning} @@ -696,7 +764,7 @@ Multiple dispatch is very common in mathematics. For example, we learn different ::: -`Julia` is similarly structured. `Julia` terminology would be to call the operation "`+`" a *generic function* and the different implementations *methods* of "`+`". This allows the user to just need to know a smaller collection of generic concepts yet still have the power of detail-specific implementations. To see how many different methods are defined in the base `Julia` language for the `+` operator, we can use the command `methods(+)`. As there are so many (well over $200$ when `Julia` is started), we illustrate how many different logarithm methods are implemented for "numbers:" +`Julia` is similarly structured. `Julia` terminology would be to call the operation "`+`" a *generic function* and the different implementations *methods* of "`+`". This allows the user to just need to know a smaller collection of generic concepts yet still have the power of detail-specific implementations. To see how many different methods are defined in the base `Julia` language for the `+` operator, we can use the command `methods(+)`. As there are so many (well over $100$ when `Julia` is started), we illustrate how many different logarithm methods are implemented for "numbers:" ```{julia} @@ -743,7 +811,7 @@ Representing the area of a rectangle in terms of two variables is easy, as the f Area(w, h) = w * h ``` -But the other fact about this problem - that the perimeter is $20$ - means that height depends on width. For this question, we can see that $P=2w + 2h$ so that - as a function - `height` depends on `w` as follows: +But the other fact about this problem--that the perimeter is $20$--means that height depends on width. For this question, we can see that $P=2w + 2h$ so that--as a function--`height` depends on `w` as follows: ```{julia} @@ -813,7 +881,7 @@ function shift_right(f; c=0) end ``` -That takes some parsing. In the body of the `shift_right` is the definition of a function. But this function has no name– it is *anonymous*. But what it does should be clear - it subtracts $c$ from $x$ and evaluates $f$ at this new value. Since the last expression creates a function, this function is returned by `shift_right`. +That takes some parsing. In the body of the `shift_right` is the definition of a function. But this function has no name–-it is *anonymous*. But what it does should be clear--it subtracts $c$ from $x$ and evaluates $f$ at this new value. Since the last expression creates a function, this function is returned by `shift_right`. So we could have done something more complicated like: @@ -833,7 +901,7 @@ The value of `c` used when `l` is called is the one passed to `shift_right`. Fun ::: -Anonymous functions can be created with the `function` keyword, but we will use the "arrow" notation, `arg->body` to create them, The above, could have been defined as: +Anonymous functions can be created with the `function` keyword, but we will use the "arrow" notation, `arg->body`, to create them, The above, could have been defined as: ```{julia} @@ -845,7 +913,11 @@ When the `->` is seen a function is being created. :::{.callout-warning} ## Warning -Generic versus anonymous functions. Julia has two types of functions, generic ones, as defined by `f(x)=x^2` and anonymous ones, as defined by `x -> x^2`. One gotcha is that `Julia` does not like to use the same variable name for the two types. In general, Julia is a dynamic language, meaning variable names can be reused with different types of variables. But generic functions take more care, as when a new method is defined it gets added to a method table. So repurposing the name of a generic function for something else is not allowed. Similarly, repurposing an already defined variable name for a generic function is not allowed. This comes up when we use functions that return functions as we have different styles that can be used: When we defined `l = shift_right(f, c=3)` the value of `l` is assigned an anonymous function. This binding can be reused to define other variables. However, we could have defined the function `l` through `l(x) = shift_right(f, c=3)(x)`, being explicit about what happens to the variable `x`. This would add a method to the generic function `l`. Meaning, we get an error if we tried to assign a variable to `l`, such as an expression like `l=3`. The latter style is inefficient, so is not preferred. +Generic versus anonymous functions. Julia has two types of functions, generic ones, as defined by `f(x)=x^2` and anonymous ones, as defined by `x -> x^2`. One gotcha is that `Julia` does not like to use the same variable name for the two types. In general, Julia is a dynamic language, meaning variable names can be reused with different types of variables. But generic functions take more care, as when a new method is defined it gets added to a method table. So repurposing the name of a generic function for something else is not allowed. Similarly, repurposing an already defined variable name for a generic function is not allowed. + +This comes up when we use functions that return functions as we have different styles that can be used: When we defined `l = shift_right(f, c=3)` the value of `l` is assigned to name an anonymous function for later use. This binding can be reused to define other variables. + +However, we could have defined the function `l` through `l(x) = shift_right(f, c=3)(x)`, being explicit about what happens to the variable `x`. This would add a method to the generic function `l`. Meaning, we get an error if we tried to assign a variable to `l`, such as an expression like `l=3`. The latter style is inefficient, so is not preferred. ::: @@ -918,6 +990,10 @@ which picks off the values of `0` and `1` in a somewhat obscure way but less ver The `Fix2` function is also helpful when using the `f(x, p)` form for passing parameters to a function. The result of `Base.Fix2(f, p)` is a function with its parameters fixed that can be passed along for plotting or other uses. +:::{.callout-note} +## Fix +In Julia v1.12 the `Fix` constructor can fix an arbitrary position of a variadic function. +::: ### The `do` notation @@ -1038,9 +1114,7 @@ radioq(choices, answ) ```{julia} #| echo: false plt = let - gr() - empty_style = (xaxis=([], false), - yaxis=([], false), + empty_style = (xticks=-4:4, yticks=-4:4, framestyle=:origin, legend=false) axis_style = (arrow=true, side=:head, line=(:gray, 1)) @@ -1067,12 +1141,19 @@ plt = let S = Shape([(k+1,k) .+ xy for xy in xys]) plot!(S; fill=(:white,), line=(:black,1)) end + + current() end -plotly() plt ``` +```{julia} +#| echo: false +plotly() +nothing +``` + The `floor` function rounds down. For example, any value in $[k,k+1)$ rounds to $k$ for integer $k$. ::: @@ -1494,7 +1575,7 @@ Where $N(m)$ counts the number of stars brighter than magnitude $m$ *per* square A [square degree](https://en.wikipedia.org/wiki/Square_degree) is a unit of a solid angle. An entire sphere has a solid angle of $4\pi$ and $4\pi \cdot (180/\pi)^2$ square degrees. -With this we can answer agequestions, such as: +With this we can answer questions, such as: > How many stars can we see in the sky? diff --git a/quarto/precalc/inversefunctions.qmd b/quarto/precalc/inversefunctions.qmd index 822b3cf..8e0eb68 100644 --- a/quarto/precalc/inversefunctions.qmd +++ b/quarto/precalc/inversefunctions.qmd @@ -15,10 +15,17 @@ plotly() --- -A (univariate) mathematical function relates or associates values of $x$ to values $y$ using the notation $y=f(x)$. A key point is a given $x$ is associated with just one $y$ value, though a given $y$ value may be associated with several different $x$ values. (Graphically, this is the vertical line test.) +A (univariate) mathematical function relates or associates values of $x$ to values $y$ using the notation $y=f(x)$. A key point is a given $x$ is associated with just one $y$ value, though a given $y$ value may be associated with several different $x$ values. (Graphically, this is the horizontal line test.) -We may conceptualize such a relation in many ways: through an algebraic rule; through the graph of $f;$ through a description of what $f$ does; or through a table of paired values, say. For the moment, let's consider a function as a rule that takes in a value of $x$ and outputs a value $y$. If a rule is given defining the function, the computation of $y$ is straightforward. A different question is not so easy: for a given value $y$ what value - or *values* - of $x$ (if any) produce an output of $y$? That is, what $x$ value(s) satisfy $f(x)=y$? +We may conceptualize such a relation in many ways: + +* through an algebraic rule; +* through the graph of $f$; +* through a description of what $f$ does; +* or through a table of paired values, say. + +For the moment, let's consider a function as a rule that takes in a value of $x$ and outputs a value $y$. If a rule is given defining the function, the computation of $y$ is straightforward. A different question is not so easy: for a given value $y$ what value--or *values*--of $x$ (if any) produce an output of $y$? That is, what $x$ value(s) satisfy $f(x)=y$? *If* for each $y$ in some set of values there is just one $x$ value, then this operation associates to each value $y$ a single value $x$, so it too is a function. When that is the case we call this an *inverse* function. @@ -80,7 +87,7 @@ p The graph of a function is a representation of points $(x,f(x))$, so to *find* $y = f(c)$ from the graph, we begin on the $x$ axis at $c$, move vertically to the graph (the point $(c, f(c))$), and then move horizontally to the $y$ axis, intersecting it at $y = f(c)$. The figure shows this for $c=2$, from which we can read that $f(c)$ is about $4$. This is how an $x$ is associated to a single $y$. -If we were to *reverse* the direction, starting at $y = f(d)$ on the $y$ axis and then moving horizontally to the graph, and then vertically to the $x$-axis we end up at a value $d$ with the correct $f(d)$. This allows solving for $x$ knowing $y$ in $y=f(x)$. +If we were to *reverse* the direction, starting at $y = f(d)$ on the $y$ axis and then moving horizontally to the graph, and then vertically to the $x$-axis we end up at a value $d$ with the correct value of $f(d)$. This allows solving for $x$ knowing $y$ in $y=f(x)$. The operation described will form a function **if** the initial movement horizontally is guaranteed to find *no more than one* value on the graph. That is, to have an inverse function, there can not be two $x$ values corresponding to a given $y$ value. This observation is often visualized through the "horizontal line test" - the graph of a function with an inverse function can only intersect a horizontal line at most in one place. @@ -195,7 +202,7 @@ In the section on the [intermediate value theorem](../limits/intermediate_value_ ## Functions which are not always invertible -Consider the function $f(x) = x^2$. The graph - a parabola - is clearly not *monotonic*. Hence no inverse function exists. Yet, we can solve equations $y=x^2$ quite easily: $y=\sqrt{x}$ *or* $y=-\sqrt{x}$. We know the square root undoes the squaring, but we need to be a little more careful to say the square root is the inverse of the squaring function. +Consider the function $f(x) = x^2$. The graph--a parabola--is clearly not *monotonic*. Hence no inverse function exists. Yet, we can solve equations $y=x^2$ quite easily: $y=\sqrt{x}$ *or* $y=-\sqrt{x}$. We know the square root undoes the squaring, but we need to be a little more careful to say the square root is the inverse of the squaring function. The issue is there are generally *two* possible answers. To avoid this, we might choose to only take the *non-negative* answer. To make this all work as above, we restrict the domain of $f(x)$ and now consider the related function $f(x)=x^2, x \geq 0$. This is now a monotonic function, so will have an inverse function. This is clearly $f^{-1}(x) = \sqrt{x}$. (The $\sqrt{x}$ being defined as the principle square root or the unique *non-negative* answer to $u^2-x=0$.) @@ -231,8 +238,9 @@ Consider again the graph of a monotonic function, in this case $f(x) = x^2 + 2, ```{julia} #| hold: true f(x) = x^2 + 2 -plot(f, 0, 4, legend=false) -plot!([2,2,0], [0,f(2),f(2)]) +plot(f, 0, 4; yticks=[2,4,8,16], + legend=false, framestyle=:origin) +plot!([(2,0), (2, f(2)), (0, f(2))]) ``` The graph is shown over the interval $(0,4)$, but the *domain* of $f(x)$ is all $x \geq 0$. The *range* of $f(x)$ is clearly $2 \leq y \leq \infty$. @@ -241,24 +249,18 @@ The graph is shown over the interval $(0,4)$, but the *domain* of $f(x)$ is all The lines layered on the plot show how to associate an $x$ value to a $y$ value or vice versa (as $f(x)$ is one-to-one). The domain then of the inverse function is all the $y$ values for which a corresponding $x$ value exists: this is clearly all values bigger or equal to $2$. The *range* of the inverse function can be seen to be all the images for the values of $y$, which would be all $x \geq 0$. This gives the relationship: -> the *range* of $f(x)$ is the *domain* of $f^{-1}(x)$; furthermore the *domain* of $f(x)$ is the *range* for $f^{-1}(x)$; - - +> * the *domain* of $f^{-1}(x)$ is the *range* of $f(x)$; +> * the *range* of $f^{-1}(x)$ is the *domain* of $f(x)$; From this we can see if we start at $x$, apply $f$ we get $y$, if we then apply $f^{-1}$ we will get back to $x$ so we have: > For all $x$ in the domain of $f$: $f^{-1}(f(x)) = x$. - - Similarly, were we to start on the $y$ axis, we would see: - > For all $x$ in the domain of $f^{-1}$: $f(f^{-1}(x)) = x$. - - In short $f^{-1} \circ f$ and $f \circ f^{-1}$ are both identity functions, though on possibly different domains. @@ -280,11 +282,12 @@ Let's see this in action. Take the function $2^x$. We can plot it by generating f(x) = 2^x xs = range(0, 2, length=50) ys = f.(xs) -plot(xs, ys, color=:blue, label="f") -plot!(ys, xs, color=:red, label="f⁻¹") # the inverse +plot(xs, ys; color=:blue, label="f", + aspect_ratio=:equal, framestyle=:origin, xlims=(0,4)) +plot!(ys, xs; color=:red, label="f⁻¹") # the inverse ``` -By flipping around the $x$ and $y$ values in the `plot!` command, we produce the graph of the inverse function - when viewed as a function of $x$. We can see that the domain of the inverse function (in red) is clearly different from that of the function (in blue). +By flipping around the $x$ and $y$ values in the `plot!` command, we produce the graph of the inverse function--when viewed as a function of $x$. We can see that the domain of the inverse function (in red) is clearly different from that of the function (in blue). The inverse function graph can be viewed as a symmetry of the graph of the function. Flipping the graph for $f(x)$ around the line $y=x$ will produce the graph of the inverse function: Here we see for the graph of $f(x) = x^{1/3}$ and its inverse function: @@ -295,15 +298,16 @@ The inverse function graph can be viewed as a symmetry of the graph of the funct f(x) = cbrt(x) xs = range(-2, 2, length=150) ys = f.(xs) -plot(xs, ys, color=:blue, aspect_ratio=:equal, legend=false) -plot!(ys, xs, color=:red) -plot!(identity, color=:green, linestyle=:dash) -x, y = 1/2, f(1/2) -plot!([x,y], [y,x], color=:green, linestyle=:dot) +plot(xs, ys; color=:blue, + aspect_ratio=:equal, legend=false) +plot!(ys, xs; line=(:red,)) +plot!(identity; line=(:green, :dash)) +x = 1/4 +y = f(x) +plot!([(x,y), (y,x)]; line=(:green, :dot)) ``` -We drew a line connecting $(1/2, f(1/2))$ to $(f(1/2),1/2)$. We can see that it crosses the line $y=x$ perpendicularly, indicating that points are symmetric about this line. (The plotting argument `aspect_ratio=:equal` ensures that the $x$ and $y$ axes are on the same scale, so that this type of line will look perpendicular.) - +We drew a line connecting $(1/4, f(1/4))$ to $(f(1/4),1/4)$. We can see that it crosses the line $y=x$ perpendicularly, indicating that points are symmetric about this line. (The plotting argument `aspect_ratio=:equal` ensures that the $x$ and $y$ axes are on the same scale, so that this type of line will look perpendicular.) One consequence of this symmetry, is that if $f$ is strictly increasing, then so is its inverse. @@ -472,7 +476,7 @@ plotly() nothing ``` -The key here is that the shape of $f(x)$ near $x=c$ is somewhat related to the shape of $f^{-1}(x)$ at $f(c)$. In this case, if we use the tangent line as a fill in for how steep a function is, we see from the relationship that if $f(x)$ is "steep" at $x=c$, then $f^{-1}(x)$ will be "shallow" at $x=f(c)$. +The key here is that the shape of $f(x)$ near $x=c$ is directly related to the shape of $f^{-1}(x)$ at $f(c)$. In this case, if we use the tangent line as a fill in for how steep a function is, we see from the relationship that if $f(x)$ is "steep" at $x=c$, then $f^{-1}(x)$ will be "shallow" at $x=f(c)$. ## Questions diff --git a/quarto/precalc/numbers_types.qmd b/quarto/precalc/numbers_types.qmd index 454abfc..67223d9 100644 --- a/quarto/precalc/numbers_types.qmd +++ b/quarto/precalc/numbers_types.qmd @@ -165,7 +165,7 @@ Integers are often used casually, as they come about from parsing. As with a cal As per IEEE Standard 754, the `Float64` type gives 52 bits to the precision (with an additional implied one), 11 bits to the exponent and the other bit is used to represent the sign. Positive, finite, floating point numbers have a range approximately between $10^{-308}$ and $10^{308}$, as 308 is about $\log_{10} 2^{1023}$. The numbers are not evenly spread out over this range, but, rather, are much more concentrated closer to $0$. -The use of 32-bit floating point values is common, as some widley used computer chips expect this. These values have a narrower range of possible values. +The use of 32-bit floating point values is common, as some widely used computer chips expect this. These values have a narrower range of possible values. :::{.callout-warning} ## More on floating point numbers @@ -404,6 +404,42 @@ Though complex numbers are stored as pairs of numbers, the imaginary unit, `im`, ::: +### Strings and symbols + +For text, `Julia` has a `String` type. When double quotes are used to specify a string, the parser creates this type: + +```{julia} +x = "The quick brown fox jumped over the lazy dog" +typeof(x) +``` + +Values can be inserted into a string through *interpolation* using a dollar sign. + +```{julia} +animal = "lion" +x = "The quick brown $(animal) jumped over the lazy dog" +``` + +The use of parentheses allows more complicated expressions; it isn't always necessary. + +Longer strings can be produced using *triple* quotes: + +```{julia} +lincoln = """ +Four score and seven years ago our fathers brought forth, upon this continent, a new nation, conceived in Liberty, and dedicated to the proposition that all men are created equal. +""" +``` + +Strings are comprised of *characters* which can be produced directly using *single* quotes: + +```{julia} +'c' +``` + +We won't use these. + +Finally, `Julia` has *symbols* which are *interned* strings which are used as identifiers. Symbols are used for advanced programming techniques; we will only see them as shortcuts to specify plotting arguments. + ## Type stability diff --git a/quarto/precalc/plotting.qmd b/quarto/precalc/plotting.qmd index 4218c76..ae57f5a 100644 --- a/quarto/precalc/plotting.qmd +++ b/quarto/precalc/plotting.qmd @@ -9,7 +9,7 @@ This section will use the following packages: ```{julia} using CalculusWithJulia using Plots -plotly() +plotly(); ``` ```{julia} @@ -18,6 +18,7 @@ plotly() using Roots using SymPy using DataFrames +using Latexify nothing ``` @@ -83,7 +84,7 @@ plotly() (Certain graphics are produced with the `gr()` backend.) -With `Plots` loaded, it is straightforward to graph a function. +With `Plots` loaded and a backend chosen, it is straightforward to graph a function. For example, to graph $f(x) = 1 - x^2/2$ over the interval $[-3,3]$ we have: @@ -97,8 +98,9 @@ plot(f, -3, 3) The `plot` command does the hard work behind the scenes. It needs $2$ pieces of information declared: - * **What** to plot. With this invocation, this detail is expressed by passing a function object to `plot` - * **Where** to plot; the `xmin` and `xmax` values. As with a sketch, it is impossible in this case to render a graph with all possible $x$ values in the domain of $f$, so we need to pick some viewing window. In the example this is $[-3,3]$ which is expressed by passing the two endpoints as the second and third arguments. +* **What** to plot. With this invocation, this detail is expressed by passing a function object to `plot` + +* **Where** to plot; the `xmin` and `xmax` values. As with a sketch, it is impossible in this case to render a graph with all possible $x$ values in the domain of $f$, so we need to pick some viewing window. In the example this has $x$ limits of $[-3,3]$; expressed by passing the two endpoints as the second and third arguments. Plotting a function is then this simple: `plot(f, xmin, xmax)`. @@ -198,9 +200,9 @@ This choices does *not* show the $x-y$ axes. As such, we might layer on the axes To emphasize concepts, we may stylize a function graph, rather than display the basic graphic. For example, in this graphic highlighting the amount the function goes up as it moves from $1$ to $x$: ```{julia} -gr() #| echo: false -let +plt = let + gr() f(x) = x^2 empty_style = (xaxis=([], false), @@ -239,6 +241,7 @@ let ]) current() end +plt ``` ```{julia} @@ -315,6 +318,7 @@ plot(xs, ys) This plots the points as pairs and then connects them in order using straight lines. Basically, it creates a dot-to-dot graph. The above graph looks primitive, as it doesn't utilize enough points. + ##### Example: Reflections @@ -396,17 +400,16 @@ The graph is that of the "inverse function" for $\sin(x), x \text{ in } [-\pi/2, When plotting a univariate function there are three basic patterns that can be employed. We have examples above of: - * `plot(f, xmin, xmax)` uses an adaptive algorithm to identify values for $x$ in the interval `[xmin, xmas]`, - * `plot(xs, f.(xs))` to manually choose the values of $x$ to plot points for, and +* `plot(f, xmin, xmax)` uses a recipe implementing an adaptive algorithm to identify values for $x$ in the interval `[xmin, xmas]`, + +* `plot(xs, f.(xs))` to manually choose the values of $x$ to plot points for, and + +Finally, there is a merging of the first two following the pattern: + +* `plot(xs, f)` -Finally there is a merging of these following either of these patterns: - - - * `plot(f, xs)` *or* `plot(xs, f)` - - -Both require a manual choice of the values of the $x$-values to plot, but the broadcasting is carried out in the `plot` command. This style is convenient, for example, to down sample the $x$ range to see the plotting mechanics, such as: +All require a manual choice of the values of the $x$-values to plot, but the broadcasting is carried out in the `plot` command. This style is convenient, for example, to down sample the $x$ range to see the plotting mechanics, such as: ```{julia} @@ -416,10 +419,10 @@ plot(0:pi/4:2pi, sin) #### NaN values -At times it is not desirable to draw lines between each successive point. For example, if there is a discontinuity in the function or if there were a vertical asymptote, such as what happens at $0$ with $f(x) = 1/x$. +At times it is not desirable to draw lines between each successive point. For example, if there is a discontinuity in the function or if there were a vertical asymptote. -The most straightforward plot is dominated by the vertical asymptote at $x=0$: +For example,what happens at $0$ with $f(x) = 1/x$. The most straightforward plot is dominated by the vertical asymptote at $x=0$: ```{julia} @@ -437,10 +440,10 @@ As we see, even with this adjustment, the spurious line connecting the points wi plot(q, -1, 1, ylims=(-10,10)) ``` -The dot-to-dot algorithm, at some level, assumes the underlying function is continuous; here $q(x)=1/x$ is not. +The dot-to-dot algorithm, at some level, assumes the underlying function is *continuous*; here $q(x)=1/x$ is not. -There is a convention for most plotting programs that **if** the $y$ value for a point is `NaN` that no lines will connect to that point, `(x,NaN)`. `NaN` conveniently appears in many cases where a plot may have an issue, though not with $1/x$ as `1/0` is `Inf` and not `NaN`. (Unlike, say, `0/0` which is NaN.) +There is a convention for most plotting programs that **if** the $y$ value for a point is `NaN` then no lines will connect to that point, `(x,NaN)`. `NaN` conveniently appears in many cases where a plot may have an issue, though not with $1/x$ as `1/0` is `Inf` and not `NaN`. (Unlike, say, `0/0` which is NaN.) Here is one way to plot $q(x) = 1/x$ over $[-1,1]$ taking advantage of this convention: @@ -457,7 +460,7 @@ plot(xs, ys) By using an odd number of points, we should have that $0.0$ is amongst the `xs`. The next to last line replaces the $y$ value that would be infinite with `NaN`. -As a recommended alternative, we might modify the function so that if it is too large, the values are replaced by `NaN`. Here is one such function consuming a function and returning a modified function put to use to make this graph: +The above is fussy. As a recommended alternative, we might modify the function so that if it is too large, the values are replaced by `NaN`. Here is one such function consuming a function and returning a modified function put to use to make this graph: ```{julia} @@ -471,7 +474,7 @@ plot(rangeclamp(x -> 1/x), -1, 1) ## Layers -Graphing more than one function over the same viewing window is often desirable. Though this is easily done in `Plots` by specifying a vector of functions as the first argument to `plot` instead of a single function object, we instead focus on building the graph layer by layer. +Graphing more than one function over the same viewing window is often desirable. Though this is easily done all at once in `Plots` by specifying a vector of functions as the first argument to `plot` instead of a single function object, we instead focus on building the graph layer by layer.^[The style of `Plots` is to combine multiple *series* to plot into one object and let `Plots` sort out which (every column is treated as a separate series). This can be very efficient from a programming perspective, but we leave it for power users. The use of layers, seems much easier to understand.] For example, to see that a polynomial and the cosine function are "close" near $0$, we can plot *both* $\cos(x)$ and the function $f(x) = 1 - x^2/2$ over $[-\pi/2,\pi/2]$: @@ -505,8 +508,8 @@ For another example, suppose we wish to plot the function $f(x)=x\cdot(x-1)$ ove ```{julia} #| hold: true -f(x) = x*(x-1) -plot(f, -1, 2, legend=false) # turn off legend +f(x) = x * (x-1) +plot(f, -1, 2; legend=false) # turn off legend plot!(zero) scatter!([0,1], [0,0]) ``` @@ -514,43 +517,53 @@ scatter!([0,1], [0,0]) The $3$ main functions used in these notes for adding layers are: - * `plot!(f, a, b)` to add the graph of the function `f`; also `plot!(xs, ys)` - * `scatter!(xs, ys)` to add points $(x_1, y_1), (x_2, y_2), \dots$. - * `annotate!((x,y, label))` to add a label at $(x,y)$ +* `plot!(f, a, b)` to add the graph of the function `f`; also `plot!(xs, ys)` + +* `scatter!(xs, ys)` to add points $(x_1, y_1), (x_2, y_2), \dots$. + +* `annotate!((x,y, label))` to add a label at $(x,y)$ :::{.callout-warning} -## Warning +## Trailing ! convention Julia has a convention to use functions named with a `!` suffix to indicate that they mutate some object. In this case, the object is the current graph, though it is implicit. Both `plot!`, `scatter!`, and `annotate!` (others too) do this by adding a layer. ::: ## Additional arguments - -The `Plots` package provides many arguments for adjusting a graphic, here we mention just a few of the [attributes](https://docs.juliaplots.org/latest/attributes/): +The `Plots` package uses positional arguments for input data and keyword arguments for [attributes](https://docs.juliaplots.org/latest/attributes/). +The `Plots` package provides many such arguments for adjusting a graphic, here we mention just a few: - * `plot(..., title="main title", xlab="x axis label", ylab="y axis label")`: add title and label information to a graphic - * `plot(..., color="green")`: this argument can be used to adjust the color of the drawn figure (color can be a string,`"green"`, or a symbol, `:green`, among other specifications) - * `plot(..., linewidth=5)`: this argument can be used to adjust the width of drawn lines - * `plot(..., xlims=(a,b), ylims=(c,d))`: either or both `xlims` and `ylims` can be used to control the viewing window - * `plot(..., linestyle=:dash)`: will change the line style of the plotted lines to dashed lines. Also `:dot`, ... - * `plot(..., aspect_ratio=:equal)`: will keep $x$ and $y$ axis on same scale so that squares look square. - * `plot(..., legend=false)`: by default, different layers will be indicated with a legend, this will turn off this feature - * `plot(..., label="a label")` the `label` attribute will show up when a legend is present. Using an empty string, `""`, will suppress add the layer to the legend. + * `plot(...; title="main title", xlab="x axis label", ylab="y axis label")`: add title and label information to a graphic + * `plot(...; color="green")`: this argument can be used to adjust the color of the drawn figure (color can be a string,`"green"`, or a symbol, `:green`, among other specifications) + * `plot(...; linewidth=5)`: this argument can be used to adjust the width of drawn lines + * `plot(...; linestyle=:dash)`: will change the line style of the plotted lines to dashed lines. Also `:dot`, ... + * `plot(...; label="a label")` the `label` attribute will show up when a legend is present. Using an empty string, `""`, will suppress add the layer to the legend. + * `plot(...; legend=false)`: by default, different layers will be indicated with a legend, this will turn off this feature + * `plot(...; xlims=(a,b), ylims=(c,d))`: either or both `xlims` and `ylims` can be used to control the viewing window + * `plot(...; xticks=[xs..], yticks=[ys...]: either or both `xticks` and `yticks` can be used to specify where the tick marks are to be drawn + * `plot(...; aspect_ratio=:equal)`: will keep $x$ and $y$ axis on same scale so that squares look square. + * `plot(...; framestyle=:origin)`: The default `framestyle` places $x$-$y$ guides on the edges; this specification places them on the $x-y$ plane. For plotting points with `scatter`, or `scatter!` the markers can be adjusted via - * `scatter(..., markersize=5)`: increase marker size - * `scatter(..., marker=:square)`: change the marker (uses a symbol, not a string to specify) + * `scatter(...; markersize=5)`: increase marker size + * `scatter(...; marker=:square)`: change the marker (uses a symbol, not a string to specify) Of course, zero, one, or more of these can be used on any given call to `plot`, `plot!`, `scatter`, or `scatter!`. -There are also several *shorthands* in `Plots` that allows several related attributes to be specified to a single argument that is disambiguated using the type of the value. (Eg. `line=(5, 0.25, "blue")` will specify the line have width `5`, color `blue`, and alpha-transparency `0.25`.) +### Shorthands + +There are also several *shorthands* in `Plots` that allows several related attributes to be specified to a single argument that is disambiguated using the type of the value. A few used herein are: + +* `line`. For example, `line=(5, 0.25, "blue")` will specify `linewidth=5` (integer), `linecolor="blue"` (string or symbol), `linealpha=0.25` (floating point) +* `marker`. For example `marker=(:star, 5)` will specify `markerstyle=:star` (symbol) and `markersize=5` (integer). +* `fill`. For example `fill=(:blue, 0.25)` will specify `fillcolor=:blue` (string or symbol) and `fillalpha=0.25` (floating point). #### Example: Bresenham's algorithm @@ -607,9 +620,9 @@ p = plot(f, x₀, x₁; legend=false, aspect_ratio=:equal, col = RGBA(.64,.64,.64, 0.25) for xy ∈ xs x, y = xy - scatter!([x], [y]; markersize=5) - scatter!([x+1], [y - 1/2], markersize=5, markershape=:star7) - plot!(Shape(x .+ [0, 1, 1, 0], y .+ [0, 0, -1, -1]), color=col) + scatter!([x], [y]; marker=(5,)) + scatter!([x+1], [y - 1/2]; marker=(5,:star)) + plot!(Shape(x .+ [0, 1, 1, 0], y .+ [0, 0, -1, -1]); color=col) end p ``` @@ -618,14 +631,46 @@ We see a number of additional arguments used: different marker sizes and shapes Of course, generalizations for positive slope and slope with magnitude greater than $1$ are needed. As well, this basic algorithm could be optimized, especially if it is part of a lower-level drawing primitive. But this illustrates the considerations involved. +## Points, lines, polygons + +Two basic objects to graph are points and lines. Add to these polygons. + +A point in two-dimensional space has two coordinates, often denoted by $(x,y)$. In `Julia`, the same notation produces a `tuple`. Using square brackets, as in `[x,y]`, produces a vector. Vectors are are more commonly used in these notes, as we have seen there are algebraic operations defined for them. However, tuples have other advantages and are how `Plots` designates a point. + +The plot command `plot(xs, ys)` plots the points $(x_1,y_1), \dots, (x_n, y_n)$ and then connects adjacent points with with lines. The command `scatter(xs, ys)` just plots the points. + +However, the points might be more naturally specified as coordinate pairs. If tuples are used to pair them off, then `Plots` will plot a vector of tuples as a sequence of points through `plot([(x1,y1), (x2, y2), ..., (xn, yn)])`: + +```{julia} +pts = [( 1, 0), ( 1/4, 1/4), (0, 1), (-1/4, 1/4), + (-1, 0), (-1/4, -1/4), (0, -1), ( 1/4, -1/4)] +scatter(pts; legend=false) +``` + +A line segment simply connects two points. While these can be specified as vectors of $x$ and $y$ values, again it may be more convenient to use coordinate pairs to specify the points. Continuing the above, we can connect adjacent points with line segments: + +```{julia} +plot!(pts; line=(:gray, 0.5, :dash)) +``` + +This uses the shorthand notation of `Plots` to specify `linecolor=:gray, linealpha=0.5, linestyle=:dash`. To plot just a line segment, just specifying two points suffices. + +The four-pointed star is not closed off, as there isn't a value from the last point to the first point. A polygon closes itself off. The `Shape` function can take a vector of points or a pair of `xs` and `ys` to specify a polygon. When these are plotted, the arguments to `fill` describe the interior of the polygon, the arguments to `line` the boundary: + + +```{julia} +plot(Shape(pts); fill=(:gray, 0.25), line=(:black, 2), legend=false) +scatter!(pts) +``` + ## Graphs of parametric equations -If we have two functions $f(x)$ and $g(x)$ there are a few ways to investigate their joint behavior. As just mentioned, we can graph both $f$ and $g$ over the same interval using layers. Such a graph allows an easy comparison of the shape of the two functions and can be useful in solving $f(x) = g(x)$. For the latter, the graph of $h(x) = f(x) - g(x)$ is also of value: solutions to $f(x)=g(x)$ appear as crossing points on the graphs of `f` and `g`, whereas they appear as zeros (crossings of the $x$-axis) when `h` is plotted. +If we have two functions $f(x)$ and $g(x)$ there are a few ways to investigate their joint behavior. As mentioned, we can graph both $f$ and $g$ over the same interval using layers. Such a graph allows an easy comparison of the shape of the two functions and can be useful in solving $f(x) = g(x)$, as the $x$ solutions are where the two curves intersect. -A different graph can be made to compare the two functions side-by-side. This is a parametric plot. Rather than plotting points $(x,f(x))$ and $(x,g(x))$ with two separate graphs, the graph consists of points $(f(x), g(x))$. We illustrate with some examples below: +A different graph can be made to compare the two functions side-by-side. This is a parametric plot. Rather than plotting points $(x,f(x))$ and $(x,g(x))$ with two separate graphs, the graph consists of points $(f(x), g(x))$ over a range of $x$ values. We illustrate with some examples below: ##### Example @@ -643,7 +688,7 @@ plot(f.(ts), g.(ts), aspect_ratio=:equal) # make equal axes Any point $(a,b)$ on this graph is represented by $(\cos(t), \sin(t))$ for some value of $t$, and in fact multiple values of $t$, since $t + 2k\pi$ will produce the same $(a,b)$ value as $t$ will. -Making the parametric plot is similar to creating a plot using lower level commands. There a sequence of values is generated to approximate the $x$ values in the graph (`xs`), a set of commands to create the corresponding function values (e.g., `f.(xs)`), and some instruction on how to represent the values, in this case with lines connecting the points (the default for `plot` for two sets of numbers). +Making the parametric plot is similar to creating a plot using lower level commands. There a sequence of values is generated to approximate the $x$ values in the graph (`xs`), a set of commands to create the corresponding function values (e.g., `f.(xs)`), and some instruction on how to represent the values, in this case with lines connecting the points (the default for `plot` for two vectors of numbers). In this next plot, the angle values are chosen to be the familiar ones, so the mechanics of the graph can be emphasized. Only the upper half is plotted: @@ -653,7 +698,7 @@ In this next plot, the angle values are chosen to be the familiar ones, so the m #| hold: true #| echo: false θs =[0, PI/6, PI/4, PI/3, PI/2, 2PI/3, 3PI/4,5PI/6, PI] -DataFrame(θ=θs, x=cos.(θs), y=sin.(θs)) +latexify(DataFrame(θ=θs, x=cos.(θs), y=sin.(θs))) ``` ```{julia} @@ -718,7 +763,7 @@ This graph is *nearly* a straight line. At the point $(0,0)=(f(0), g(0))$, we se ##### Example: Etch A Sketch -[Etch A sketch](http://en.wikipedia.org/wiki/Etch_A_Sketch) is a drawing toy where two knobs control the motion of a pointer, one knob controlling the $x$ motion, the other the $y$ motion. The trace of the movement of the pointer is recorded until the display is cleared by shaking. Shake to clear is now a motion incorporated by some smart-phone apps. +[Etch A Sketch](http://en.wikipedia.org/wiki/Etch_A_Sketch) is a drawing toy where two knobs control the motion of a pointer, one knob controlling the $x$ motion, the other the $y$ motion. The trace of the movement of the pointer is recorded until the display is cleared by shaking. Shake to clear is now a motion incorporated by some smart-phone apps. Playing with the toy makes a few things become clear: @@ -757,38 +802,6 @@ plot(f, g, 0, max((R-r)/r, r/(R-r))*2pi) In the above, one can fix $R=1$. Then different values for `r` and `rho` will produce different graphs. These graphs will be periodic if $(R-r)/r$ is a rational. (Nothing about these equations requires $\rho < r$.) -## Points, lines, polygons - -Two basic objects to graph are points and lines. - -A point in two-dimensional space has two coordinates, often denoted by $(x,y)$. In `Julia`, the same notation produces a `tuple`. Using square brackets, as in `[x,y]`, produces a vector. Vectors are usually used, as we have seen there are algebraic operations defined for them. However, tuples have other advantages and are how `Plots` designates a point. - -The plot command `plot(xs, ys)` plots the points $(x_1,y_1), \dots, (x_n, y_n)$ and then connects adjacent points with with lines. The command `scatter(xs, ys)` just plots the points. - -However, the points might be more naturally specified as coordinate pairs. If tuples are used to pair them off, then `Plots` will plot a vector of tuples as a sequence of points: - -```{julia} -pts = [(1, 0), (1/4, 1/4), (0, 1), (-1/4, 1/4), (-1, 0), - (-1/4, -1/4), (0, -1), (1/4, -1/4)] -scatter(pts; legend=false) -``` - -A line segment simply connects two points. While these can be specified as vectors of $x$ and $y$ values, again it may be more convenient to use coordinate pairs to specify the points. Continuing the above, we can connect adjacent points with line segments: - -```{julia} -plot!(pts; line=(:gray, 0.5, :dash)) -``` - -This uses the shorthand notation of `Plots` to specify `linecolor=:gray, linealpha=0.5, linestyle=:dash`. To plot just a line segment, just specifying two points suffices. - -The four-pointed star is not closed off, as there isn't a value from the last point to the first point. A polygon closes itself off. The `Shape` function can take a vector of points or a pair of `xs` and `ys` to specify a polygon. When these are plotted, the arguments to `fill` describe the interior of the polygon, the arguments to `line` the boundary: - - -```{julia} -plot(Shape(pts); fill=(:gray, 0.25), line=(:black, 2), legend=false) -scatter!(pts) -``` - ## Questions diff --git a/quarto/precalc/polynomial.qmd b/quarto/precalc/polynomial.qmd index e59151b..4efbac6 100644 --- a/quarto/precalc/polynomial.qmd +++ b/quarto/precalc/polynomial.qmd @@ -51,7 +51,12 @@ $$ gr() anim = @animate for m in 2:2:10 fn = x -> x^m - plot(fn, -1, 1, size = fig_size, legend=false, title="graph of x^{$m}", xlims=(-1,1), ylims=(-.1,1)) + title = L"graph of $x^{%$m}$" + plot(fn, -1, 1; + size = fig_size, + legend=false, + title=title, + xlims=(-1,1), ylims=(-.1,1)) end imgfile = tempname() * ".gif" @@ -96,7 +101,12 @@ $$ gr() anim = @animate for m in [-5, -2, -1, 1, 2, 5, 10, 20] fn = x -> m * x - plot(fn, -1, 1, size = fig_size, legend=false, title="m = $m", xlims=(-1,1), ylims=(-20, 20)) + title = L"m = %$m" + plot(fn, -1, 1; + size = fig_size, + legend=false, + title=title, + xlims=(-1,1), ylims=(-20, 20)) end imgfile = tempname() * ".gif" @@ -301,7 +311,9 @@ float(y) -The use of the generic `float` method returns a floating point number. `SymPy` objects have their own internal types. To preserve these on conversion to a related `Julia` value, the `N` function from `SymPy` is useful: +The use of the generic `float` method returns a floating point number. (The `.evalf()` method of `SymPy` objects uses `SymPy` to produce floating point versions of symbolic values. + +`SymPy` objects have their own internal types. To preserve these on conversion to a related `Julia` value, the `N` function from `SymPy` is useful: ```{julia} @@ -310,14 +322,7 @@ p = -16x^2 + 100 N(p(2)) ``` -Where `convert(T, x)` requires a specification of the type to convert `x` to, `N` attempts to match the data type used by SymPy to store the number. As such, the output type of `N` may vary (rational, a BigFloat, a float, etc.) For getting more digits of accuracy, a precision can be passed to `N`. The following command will take the symbolic value for $\pi$, `PI`, and produce about $60$ digits worth as a `BigFloat` value: - - -```{julia} -N(PI, 60) -``` - -Conversion by `N` will fail if the value to be converted contains free symbols, as would be expected. +Where `convert(T, x)` requires a specification of the type to convert `x` to, `N` attempts to match the data type used by SymPy to store the number. As such, the output type of `N` may vary (rational, a BigFloat, a float, etc.) Conversion by `N` will fail if the value to be converted contains free symbols, as would be expected. ### Converting symbolic expressions into Julia functions @@ -362,6 +367,8 @@ pp = lambdify(p, (x,a,b)) pp(1,2,3) # computes 2*1^2 + 3 ``` +(We suggest using the pair notation when there is more than one variable.) + ## Graphical properties of polynomials @@ -394,7 +401,12 @@ To investigate this last point, let's consider the case of the monomial $x^n$. W gr() anim = @animate for m in 0:2:12 fn = x -> x^m - plot(fn, -1.2, 1.2, size = fig_size, legend=false, xlims=(-1.2, 1.2), ylims=(0, 1.2^12), title="x^{$m} over [-1.2, 1.2]") + title = L"$x^{%$m}$ over $[-1.2, 1.2]$" + plot(fn, -1.2, 1.2; + size = fig_size, + legend=false, + xlims=(-1.2, 1.2), ylims=(0, 1.2^12), + title=title) end imgfile = tempname() * ".gif" @@ -433,8 +445,13 @@ anim = @animate for n in 1:6 m = [1, .5, -1, -5, -20, -25] M = [2, 4, 5, 10, 25, 30] fn = x -> (x-1)*(x-2)*(x-3)*(x-5) + title = L"Graph of on $(%$(m[n]), %$(M[n]))$" - plt = plot(fn, m[n], M[n], size=fig_size, legend=false, linewidth=2, title ="Graph of on ($(m[n]), $(M[n]))") + plt = plot(fn, m[n], M[n]; + size=fig_size, + legend=false, + linewidth=2, + title=title) if n > 1 plot!(plt, fn, m[n-1], M[n-1], color=:red, linewidth=4) end @@ -468,11 +485,20 @@ The following graphic illustrates the $4$ basic *overall* shapes that can result ```{julia} #| echo: false -plot(; layout=4) -plot!(x -> x^4, -3,3, legend=false, xticks=false, yticks=false, subplot=1, title="n = even, aₙ > 0") -plot!(x -> x^5, -3,3, legend=false, xticks=false, yticks=false, subplot=2, title="n = odd, aₙ > 0") -plot!(x -> -x^4, -3,3, legend=false, xticks=false, yticks=false, subplot=3, title="n = even, aₙ < 0") -plot!(x -> -x^5, -3,3, legend=false, xticks=false, yticks=false, subplot=4, title="n = odd, aₙ < 0") +let + gr() + plot(; layout=4) + plot!(x -> x^4, -3,3, legend=false, xticks=false, yticks=false, subplot=1, title=L"$n = $ even, $a_n > 0$") + plot!(x -> x^5, -3,3, legend=false, xticks=false, yticks=false, subplot=2, title=L"$n = $ odd, $a_m > 0$") + plot!(x -> -x^4, -3,3, legend=false, xticks=false, yticks=false, subplot=3, title=L"$n = $ even, $a_n < 0$") + plot!(x -> -x^5, -3,3, legend=false, xticks=false, yticks=false, subplot=4, title=L"$n = $ odd, $a_n < 0$") +end +``` + +```{julia} +#| echo: false +plotly() +nothing ``` ##### Example @@ -513,14 +539,15 @@ This observation is the start of Descartes' rule of [signs](http://sepwww.stanfo Among numerous others, there are two common ways of representing a non-zero polynomial: - * expanded form, as in $a_n x^n + a_{n-1}x^{n-1} + \cdots + a_1 x + a_0, a_n \neq 0$; or - * factored form, as in $a\cdot(x-r_1)\cdot(x-r_2)\cdots(x-r_n), a \neq 0$. + * expanded form, as in $a_n x^n + a_{n-1}x^{n-1} + \cdots + a_1 x + a_0,\quad a_n \neq 0$; or + * factored form, as in $a\cdot(x-r_1)\cdot(x-r_2)\cdots(x-r_n), \quad a \neq 0$. +The former uses the *standard basis* to represent the polynomial $p$. The latter writes $p$ as a product of linear factors, though this is only possible in general if we consider complex roots. With real roots only, then the factors are either linear or quadratic, as will be discussed later. -There are values to each representation. One value of the expanded form is that polynomial addition and scalar multiplication is much easier than in factored form. For example, adding polynomials just requires matching up the monomials of similar powers. For the factored form, polynomial multiplication is much easier than expanded form. For the factored form it is easy to read off *roots* of the polynomial (values of $x$ where $p$ is $0$), as a product is $0$ only if a term is $0$, so any zero must be a zero of a factor. Factored form has other technical advantages. For example, the polynomial $(x-1)^{1000}$ can be compactly represented using the factored form, but would require $1001$ coefficients to store in expanded form. (As well, due to floating point differences, the two would evaluate quite differently as one would require over a $1000$ operations to compute, the other just two.) +There are values to each representation. One value of the expanded form is that polynomial addition and scalar multiplication is much easier than in factored form. For example, adding polynomials just requires matching up the monomials of similar powers. (These can be realized easily as vector operations.) For the factored form, polynomial multiplication is much easier than expanded form. For the factored form it is easy to read off *roots* of the polynomial (values of $x$ where $p$ is $0$), as a product is $0$ only if a term is $0$, so any zero must be a zero of a factor. Factored form has other technical advantages. For example, the polynomial $(x-1)^{1000}$ can be compactly represented using the factored form, but would require $1001$ coefficients to store in expanded form. (As well, due to floating point differences, the two would evaluate quite differently as one would require over a $1000$ operations to compute, the other just two.) Translating from factored form to expanded form can be done by carefully following the distributive law of multiplication. For example, with some care it can be shown that: @@ -587,7 +614,7 @@ It is easy to create a symbolic expression from a function - just evaluate the f f(x) ``` -This is easy - but can also be confusing. The function object is `f`, the expression is `f(x)` - the function evaluated on a symbolic object. Moreover, as seen, the symbolic expression can be evaluated using the same syntax as a function call: +This is easy--but can also be confusing. The function object is `f`, the expression is `f(x)`--the function evaluated on a symbolic object. Moreover, as seen, the symbolic expression can be evaluated using the same syntax as a function call: ```{julia} diff --git a/quarto/precalc/polynomial_roots.qmd b/quarto/precalc/polynomial_roots.qmd index 7d760eb..6c00148 100644 --- a/quarto/precalc/polynomial_roots.qmd +++ b/quarto/precalc/polynomial_roots.qmd @@ -453,7 +453,7 @@ N.(solveset(p ~ 0, x)) ## Do numeric methods matter when you can just graph? -It may seem that certain practices related to roots of polynomials are unnecessary as we could just graph the equation and look for the roots. This feeling is perhaps motivated by the examples given in textbooks to be worked by hand, which necessarily focus on smallish solutions. But, in general, without some sense of where the roots are, an informative graph itself can be hard to produce. That is, technology doesn't displace thinking - it only supplements it. +It may seem that certain practices related to roots of polynomials are unnecessary as we could just graph the equation and look for the roots. This feeling is perhaps motivated by the examples given in textbooks to be worked by hand, which necessarily focus on smallish solutions. But, in general, without some sense of where the roots are, an informative graph itself can be hard to produce. That is, technology doesn't displace thinking--it only supplements it. For another example, consider the polynomial $(x-20)^5 - (x-20) + 1$. In this form we might think the roots are near $20$. However, were we presented with this polynomial in expanded form: $x^5 - 100x^4 + 4000x^3 - 80000x^2 + 799999x - 3199979$, we might be tempted to just graph it to find roots. A naive graph might be to plot over $[-10, 10]$: @@ -553,7 +553,7 @@ N.(solve(j ~ 0, x)) ### Cauchy's bound on the magnitude of the real roots. -Descartes' rule gives a bound on how many real roots there may be. Cauchy provided a bound on how large they can be. Assume our polynomial is monic (if not, divide by $a_n$ to make it so, as this won't effect the roots). Then any real root is no larger in absolute value than $|a_0| + |a_1| + |a_2| + \cdots + |a_{n-1}| + 1$, (this is expressed in different ways.) +Descartes' rule gives a bound on how many real roots there may be. Cauchy provided a bound on how large they can be. Assume our polynomial is monic (if not, divide by $a_n$ to make it so, as this won't effect the roots). Then any real root is no larger in absolute value than $h = 1 + |a_0| + |a_1| + |a_2| + \cdots + |a_{n-1}|$, (this is expressed in different ways.) To see precisely [why](https://captainblack.wordpress.com/2009/03/08/cauchys-upper-bound-for-the-roots-of-a-polynomial/) this bound works, suppose $x$ is a root with $|x| > 1$ and let $h$ be the bound. Then since $x$ is a root, we can solve $a_0 + a_1x + \cdots + 1 \cdot x^n = 0$ for $x^n$ as: @@ -563,7 +563,7 @@ $$ x^n = -(a_0 + a_1 x + \cdots a_{n-1}x^{n-1}) $$ -Which after taking absolute values of both sides, yields: +Which after taking absolute values of both sides, yields by the triangle inequality: $$ diff --git a/quarto/precalc/polynomials_package.qmd b/quarto/precalc/polynomials_package.qmd index 2cfae53..ac8844f 100644 --- a/quarto/precalc/polynomials_package.qmd +++ b/quarto/precalc/polynomials_package.qmd @@ -18,7 +18,7 @@ import SymPy # imported only: some functions, e.g. degree, need qualification --- -While `SymPy` can be used to represent polynomials, there are also native `Julia` packages available for this and related tasks. These packages include `Polynomials`, `MultivariatePolynomials`, and `AbstractAlgebra`, among many others. (A search on [juliahub.com](juliahub.com) found over $50$ packages matching "polynomial".) We will look at the `Polynomials` package in the following, as it is straightforward to use and provides the features we are looking at for univariate polynomials. +While `SymPy` can be used to represent polynomials, there are also native `Julia` packages available for this and related tasks. These packages include `Polynomials`, `MultivariatePolynomials`, and `AbstractAlgebra`, among many others. (A search on [juliahub.com](https://juliahub.com) found almost $100$ packages matching "polynomial".) We will look at the `Polynomials` package in the following, as it is straightforward to use and provides the features we are looking at for *univariate* polynomials. ## Construction @@ -76,7 +76,7 @@ Polynomials may be evaluated using function notation, that is: p(1) ``` -This blurs the distinction between a polynomial expression – a formal object consisting of an indeterminate, coefficients, and the operations of addition, subtraction, multiplication, and non-negative integer powers – and a polynomial function. +This blurs the distinction between a polynomial expression--a formal object consisting of an indeterminate, coefficients, and the operations of addition, subtraction, multiplication, and non-negative integer powers--and a polynomial function. The polynomial variable, in this case `1x`, can be returned by `variable`: @@ -86,14 +86,15 @@ The polynomial variable, in this case `1x`, can be returned by `variable`: x = variable(p) ``` -This variable is a `Polynomial` object, so can be manipulated as a polynomial; we can then construct polynomials through expressions like: +This variable is a `Polynomial` object that prints as `x`. +These variables can be manipulated as any polynomial. We can then construct polynomials through expressions like: ```{julia} r = (x-2)^2 * (x-1) * (x+1) ``` -The product is expanded for storage by `Polynomials`, which may not be desirable for some uses. A new variable can be produced by calling `variable()`; so we could have constructed `p` by: +The product is expanded for storage by `Polynomials`, which may not be desirable for some uses. A new variable can also be produced by calling `variable()`; so we could have constructed `p` by: ```{julia} @@ -125,7 +126,7 @@ The `Polynomials` package has different ways to represent polynomials, and a fac fromroots(FactoredPolynomial, [2, 2, 1, -1]) ``` -This form is helpful for some operations, for example polynomial multiplication and positive integer exponentiation, but not others such as addition of polynomials, where such polynomials must first be converted to the standard basis to add and are then converted back into a factored form. +This form is helpful for some operations, for example polynomial multiplication and positive integer exponentiation, but not others such as addition of polynomials, where such polynomials must first be converted to the standard basis to add and are then converted back into a factored form. (A task that may suffer from floating point roundoff.) --- @@ -181,7 +182,7 @@ A consequence of the fundamental theorem of algebra and the factor theorem is th :::{.callout-note} ## Note -`SymPy` also has a `roots` function. If both `Polynomials` and `SymPy` are used together, calling `roots` must be qualified, as with `Polynomials.roots(...)`. Similarly, `degree` is provided in both, so it too must be qualified. +`SymPy` also has a `roots` function. If both `Polynomials` and `SymPy` are used together, calling `roots` must be qualified, as with `Polynomials.roots(...)`. Similarly, `degree` is exported in both, so it too must be qualified. ::: diff --git a/quarto/precalc/ranges.qmd b/quarto/precalc/ranges.qmd index ca23b23..110df17 100644 --- a/quarto/precalc/ranges.qmd +++ b/quarto/precalc/ranges.qmd @@ -103,13 +103,13 @@ h = (b-a)/(n-1) collect(a:h:b) ``` -Pretty neat. If we were doing this many times - such as once per plot - we'd want to encapsulate this into a function, for example: +Pretty neat. If we were doing this many times - such as once per plot - we'd want to encapsulate this into a function, for example using a comprehension: ```{julia} function evenly_spaced(a, b, n) h = (b-a)/(n-1) - collect(a:h:b) + [a + i*h for i in 0:n-1] end ``` @@ -131,10 +131,10 @@ It seems to work as expected. But looking just at the algorithm it isn't quite s ```{julia} -1/5 + 2*1/5 # last value +1/5 + 2*1/5 # last value if h is exactly 1/5 or 0.2 ``` -Floating point roundoff leads to the last value *exceeding* `0.6`, so should it be included? Well, here it is pretty clear it *should* be, but better to have something programmed that hits both `a` and `b` and adjusts `h` accordingly. +Floating point roundoff leads to the last value *exceeding* `0.6`, so should it be included? Well, here it is pretty clear it *should* be, but better to have something programmed that hits both `a` and `b` and adjusts `h` accordingly. Something which isn't subject to the vagaries of `(3/5 - 1/5)/2` not being `0.2`. Enter the base function `range` which solves this seemingly simple - but not really - task. It can use `a`, `b`, and `n`. Like the range operation, this function returns a generator which can be collected to realize the values. @@ -144,14 +144,7 @@ The number of points is specified as a third argument (though keyword arguments ```{julia} -xs = range(-1, 1,9) -``` - -and - - -```{julia} -collect(xs) +xs = range(1/5, 3/5, 3) |> collect ``` :::{.callout-note} @@ -266,7 +259,7 @@ Here are decreasing powers of $2$: [1/2^i for i in 1:10] ``` -Sometimes, the comprehension does not produce the type of output that may be expected. This is related to `Julia`'s more limited abilities to infer types at the command line. If the output type is important, the extra prefix of `T[]` can be used, where `T` is the desired type. We will see that this will be needed at times with symbolic math. +Sometimes, the comprehension does not produce the type of output that may be expected. This is related to `Julia`'s more limited abilities to infer types at the command line. If the output type is important, the extra prefix of `T[]` can be used, where `T` is the desired type. ### Generators diff --git a/quarto/precalc/rational_functions.qmd b/quarto/precalc/rational_functions.qmd index f74b6af..c049075 100644 --- a/quarto/precalc/rational_functions.qmd +++ b/quarto/precalc/rational_functions.qmd @@ -15,7 +15,7 @@ import Polynomials using RealPolynomialRoots ``` -The `Polynomials` package is "imported" to avoid naming collisions with `SymPy`; names will need to be qualified. +The `Polynomials` package is "imported" to avoid naming collisions with `SymPy`;some names may need to be qualified. @@ -410,7 +410,7 @@ The usual recipe for construction follows these steps: * Identify "test points" within each implied interval (these are $(-\infty, -1)$, $(-1,0)$, $(0,1)$, and $(1, \infty)$ in the example) and check for the sign of $f(x)$ at these test points. Write in `-`, `+`, `0`, or `*`, as appropriate. The value comes from the fact that "continuous" functions may only change sign when they cross $0$ or are undefined. -With the computer, where it is convenient to draw a graph, it might be better to emphasize the sign on the graph of the function. The `sign_chart` function from `CalculusWithJulia` does this by numerically identifying points where the function is $0$ or $\infty$ and indicating the sign as $x$ crosses over these points. +With the computer, where it is convenient to draw a graph, it might be better to emphasize the sign on the graph of the function, but at times numeric values are preferred. The `sign_chart` function from `CalculusWithJulia` does this analysis by numerically identifying points where the function is $0$ or $\infty$ and indicating the sign as $x$ crosses over these points. ```{julia} @@ -419,6 +419,8 @@ f(x) = x^3 - x sign_chart(f, -3/2, 3/2) ``` +This format is a bit different from above, but shows to the left of $-1$ a minussign; between $-1$ and $0$ a plus sign; between $0$ and $1$ a minus sign; and between $1$ and $3/2$ a plus sign. + ## Pade approximate diff --git a/quarto/precalc/transformations.qmd b/quarto/precalc/transformations.qmd index 5f7c9f7..c6dfaa1 100644 --- a/quarto/precalc/transformations.qmd +++ b/quarto/precalc/transformations.qmd @@ -220,7 +220,7 @@ Scaling by $2$ shrinks the non-zero domain, scaling by $1/2$ would stretch it. I --- -More exciting is what happens if we combine these operations. +More exciting is what happens if we compose these operations. A shift right by $2$ and up by $1$ is achieved through @@ -267,7 +267,7 @@ We can view this as a composition of "scale" by $1/a$, then "over" by $b$, and #| hold: true a = 2; b = 5 h(x) = stretch(over(scale(f, 1/a), b), 1/a)(x) -plot(f, -1, 8, label="f") +plot(f, -1, 8, label="f"; xticks=-1:8) plot!(h, label="h") ``` diff --git a/quarto/precalc/trig_functions.qmd b/quarto/precalc/trig_functions.qmd index b3f32b8..67a72eb 100644 --- a/quarto/precalc/trig_functions.qmd +++ b/quarto/precalc/trig_functions.qmd @@ -34,12 +34,37 @@ For a right triangle with angles $\theta$, $\pi/2 - \theta$, and $\pi/2$ ($0 < \ ```{julia} #| hide: true #| echo: false -p = plot(legend=false, xlim=(-1/4,5), ylim=(-1/2, 3), - xticks=nothing, yticks=nothing, border=:none) -plot!([0,4,4,0],[0,0,3,0], linewidth=3) -del = .25 -plot!([4-del, 4-del,4], [0, del, del], color=:black, linewidth=3) -annotate!([(.75, .25, "θ"), (4.0, 1.25, "opposite"), (2, -.25, "adjacent"), (1.5, 1.25, "hypotenuse")]) +let + gr() + p = plot(;legend=false, + xticks=nothing, yticks=nothing, + border=:none, + xlim=(-1/4,5), ylim=(-1/2, 3)) + plot!([0,4,4,0],[0,0,3,0], linewidth=3) + θ = atand(3,4) + del = .25 + plot!([4-del, 4-del,4], [0, del, del], color=:black, linewidth=3) + theta = pi/20 + r = sqrt((3/4)^2 + (1/4)^2) + ts = range(0, theta, 20) + plot!(r*cos.(ts), r*sin.(ts); line=(:gray, 1)) + ts = range(atan(3/4) - theta, atan(3,4), 20) + plot!(r*cos.(ts), r*sin.(ts); line=(:gray, 1)) + +annotate!([ + (.75, .25, L"\theta"), + (4.0, 1.5+.1, text("opposite", rotation=90,:top)), + (2, -.25, "adjacent"), + (2, 1.5+.1, text("hypotenuse", rotation=θ,:bottom)) +]) + +end +``` + +```{julia} +#| echo: false +plotly() +nothing ``` With these, the basic definitions for the primary trigonometric functions are @@ -77,9 +102,13 @@ gr() function plot_angle(m) r = m*pi + n,d = numerator(m), denominator(m) - ts = range(0, stop=2pi, length=100) - tit = "$m ⋅ pi -> ($(round(cos(r), digits=2)), $(round(sin(r), digits=2)))" + tit = latexstring("\\frac{$n}{$d}") * + L"\cdot\pi\rightarrow (" * + latexstring("$(round(cos(r), digits=2)),$(round(sin(r), digits=2)))") + + ts = range(0, 2pi, 151) p = plot(cos.(ts), sin.(ts), legend=false, aspect_ratio=:equal,title=tit) plot!(p, [-1,1], [0,0], color=:gray30) plot!(p, [0,0], [-1,1], color=:gray30) @@ -95,13 +124,11 @@ function plot_angle(m) plot!(p, [0,l*cos(r)], [0,l*sin(r)], color=:green, linewidth=4) scatter!(p, [cos(r)], [sin(r)], markersize=5) - annotate!(p, [(1/4+cos(r), sin(r), "(x,y)")]) + annotate!(p, [(1/4+cos(r), sin(r), L"(x,y)")]) p end - - ## different linear graphs anim = @animate for m in -4//3:1//6:10//3 plot_angle(m) @@ -406,7 +433,7 @@ Suppose both $\alpha$ and $\beta$ are positive with $\alpha + \beta \leq \pi/2$. ```{julia} #| echo: false - +gr() using Plots, LaTeXStrings # two angles @@ -425,7 +452,10 @@ color1 = :royalblue color2 = :forestgreen color3 = :brown3 color4 = :mediumorchid2 -canvas() = plot(axis=([],false), legend=false, aspect_ratio=:equal) +canvas() = plot(; + axis=([],false), + legend=false, + aspect_ratio=:equal) p1 = canvas() plot!(Shape([A,B,F]), fill=(color4, 0.15)) @@ -440,29 +470,29 @@ ddf = sqrt(sum((D.-F).^2)) Δ = 0.0 alphabeta = (r*cos(α/2 + β/2), r*sin(α/2 + β/2), - text("α + β",:hcenter; rotation=pi/2)) -cosαβ = (B[1]/2, 0, text("cos(α + β)", :top)) -sinαβ = (B[1], F[2]/2, text("sin(α + β)")) + text(L"\alpha + \beta",:left, rotation=pi/2)) +cosαβ = (B[1]/2, 0, text(L"\cos(\alpha + \beta)", :top)) +sinαβ = (B[1], F[2]/2, text(L"\sin(\alpha + \beta)", rotation=90,:top)) txtpoints = ( - one = (F[1]/2, F[2]/2, "1",:right), + one = (F[1]/2, F[2]/2, text(L"1", :bottom)), beta=(r*cos(α + β/2), r*sin(α + β/2), - text("β", :hcenter)), + text(L"\beta", :hcenter)), alpha = (r*cos(α/2), r*sin(α/2), - text("α",:hcenter)), + text(L"\alpha",:hcenter)), alphaa = (F[1] + r*sin(α/2), F[2] - r*cos(α/2) , - text("α"),:hcenter), + text(L"\alpha"),:hcenter), cosβ = (dae/2*cos(α),dae/2*sin(α) + Δ, - text("cos(β)",:hcenter)), + text(L"\cos(\beta)",:bottom, rotation=rad2deg(α))), sinβ = (B[1] + dbc/2 + Δ/2, D[2] + ddf/2 + Δ/2, - text("sin(β)",:bottom)), - cosαcosβ = (C[1]/2, 0 - Δ, text("cos(α)cos(β)", :top)), - sinαcosβ = (cos(α)*cos(β) - 0.1, dce/2 , - text("sin(α)cos(β)", :hcenter)), + text(L"\sin(\beta)",:bottom, rotation=-(90-rad2deg(α)))), + cosαcosβ = (C[1]/2, 0 - Δ, text(L"\cos(\alpha)\cos(\beta)", :top)), + sinαcosβ = (cos(α)*cos(β), dce/2 , + text(L"\sin(\alpha)\cos(\beta)", :top, rotation=90)), cosαsinβ = (D[1] - Δ, D[2] + ddf/2 , - text("cos(α)sin(β)", :top)), + text(L"\cos(\alpha)\sin(\beta)", :bottom, 10, rotation=90)), sinαsinβ = (D[1] + dde/2, D[2] + Δ , - text("sin(α)sin(β)", :top)), + text(L"\sin(\alpha)\sin(\beta)", 10, :top)), ) # Plot 1 @@ -574,6 +604,12 @@ $$ $$ ::: +```{julia} +#| echo: false +plotly() +nothing +``` + ##### Example @@ -1028,4 +1064,3 @@ Is this identical to the pattern for the regular sine function? #| echo: false yesnoq(false) ``` - diff --git a/quarto/precalc/vectors.qmd b/quarto/precalc/vectors.qmd index bb23e18..d24f5fb 100644 --- a/quarto/precalc/vectors.qmd +++ b/quarto/precalc/vectors.qmd @@ -1,5 +1,4 @@ - -# Vectors +# Vectors and containers {{< include ../_common_code.qmd >}} @@ -143,8 +142,9 @@ We call the values $x$ and $y$ of the vector $\vec{v} = \langle x,~ y \rangle$ t Two operations on vectors are fundamental. - * Vectors can be multiplied by a scalar (a real number): $c\vec{v} = \langle cx,~ cy \rangle$. Geometrically this scales the vector by a factor of $\lvert c \rvert$ and switches the direction of the vector by $180$ degrees (in the $2$-dimensional case) when $c < 0$. A *unit vector* is one with magnitude $1$, and, except for the $\vec{0}$ vector, can be formed from $\vec{v}$ by dividing $\vec{v}$ by its magnitude. A vector's two parts are summarized by its direction given by a unit vector **and** its magnitude given by the norm. - * Vectors can be added: $\vec{v} + \vec{w} = \langle v_x + w_x,~ v_y + w_y \rangle$. That is, each corresponding component adds to form a new vector. Similarly for subtraction. The $\vec{0}$ vector then would be just $\langle 0,~ 0 \rangle$ and would satisfy $\vec{0} + \vec{v} = \vec{v}$ for any vector $\vec{v}$. Vector addition, $\vec{v} + \vec{w}$, is visualized by placing the tail of $\vec{w}$ at the tip of $\vec{v}$ and then considering the new vector with tail coming from $\vec{v}$ and tip coming from the position of the tip of $\vec{w}$. Subtraction is different, place both the tails of $\vec{v}$ and $\vec{w}$ at the same place and the new vector has tail at the tip of $\vec{w}$ and tip at the tip of $\vec{v}$. +* *Scalar multiplication*: Vectors can be multiplied by a scalar (a real number): $c\vec{v} = \langle cx,~ cy \rangle$. Geometrically this scales the vector by a factor of $\lvert c \rvert$ and switches the direction of the vector by $180$ degrees (in the $2$-dimensional case) when $c < 0$. A *unit vector* is one with magnitude $1$, and, except for the $\vec{0}$ vector, can be formed from $\vec{v}$ by dividing $\vec{v}$ by its magnitude. A vector's two parts are summarized by its direction given by a unit vector **and** its magnitude given by the norm. + +* *Vector addition*: Vectors can be added: $\vec{v} + \vec{w} = \langle v_x + w_x,~ v_y + w_y \rangle$. That is, each corresponding component adds to form a new vector. Similarly for subtraction. The $\vec{0}$ vector then would be just $\langle 0,~ 0 \rangle$ and would satisfy $\vec{0} + \vec{v} = \vec{v}$ for any vector $\vec{v}$. Vector addition, $\vec{v} + \vec{w}$, is visualized by placing the tail of $\vec{w}$ at the tip of $\vec{v}$ and then considering the new vector with tail coming from $\vec{v}$ and tip coming from the position of the tip of $\vec{w}$. Subtraction is different, place both the tails of $\vec{v}$ and $\vec{w}$ at the same place and the new vector has tail at the tip of $\vec{w}$ and tip at the tip of $\vec{v}$. ```{julia} @@ -334,7 +334,7 @@ Finally, to find an angle $\theta$ from a vector $\langle x,~ y\rangle$, we can norm(v), atan(y, x) # v = [x, y] ``` -## Higher dimensional vectors +### Higher dimensional vectors Mathematically, vectors can be generalized to more than $2$ dimensions. For example, using $3$-dimensional vectors are common when modeling events happening in space, and $4$-dimensional vectors are common when modeling space and time. @@ -395,334 +395,6 @@ Whereas, in this example where there is no common type to promote the values to, ["one", 2, 3.0, 4//1] ``` -## Indexing - - -Getting the components out of a vector can be done in a manner similar to multiple assignment: - - -```{julia} -vs = [1, 2] -v₁, v₂ = vs -``` - -When the same number of variable names are on the left hand side of the assignment as in the container on the right, each is assigned in order. - - -Though this is convenient for small vectors, it is far from being so if the vector has a large number of components. However, the vector is stored in order with a first, second, third, $\dots$ component. `Julia` allows these values to be referred to by *index*. This too uses the `[]` notation, though differently. Here is how we get the second component of `vs`: - - -```{julia} -vs[2] -``` - -The last value of a vector is usually denoted by $v_n$. In `Julia`, the `length` function will return $n$, the number of items in the container. So `v[length(v)]` will refer to the last component. However, the special keyword `end` will do so as well, when put into the context of indexing. So `v[end]` is more idiomatic. (Similarly, there is a `begin` keyword that is useful when the vector is not $1$-based, as is typical but not mandatory.) - -The functions `first` and `last` refer to the first and last components of a collection. An additional argument can be specified to take the first (or last) $n$ components. The function `only` will return the only component of a vector, if it has length $1$ and error otherwise. - - -:::{.callout-note} -## More on indexing -There is [much more](https://docs.julialang.org/en/v1/manual/arrays/#man-array-indexing) to indexing than just indexing by a single integer value. For example, the following can be used for indexing: - - * a scalar integer (as seen) - * a range - * a vector of integers - * a boolean vector - -::: - -Some add-on packages extend this further. - - -### Assignment and indexing - - -Indexing notation can also be used with assignment, meaning it can appear on the left hand side of an equals sign. The following expression replaces the second component with a new value: - - -```{julia} -vs[2] = 10 -``` - -The value of the right hand side is returned, not the value for `vs`. We can check that `vs` is then $\langle 1,~ 10 \rangle$ by showing it: - - -```{julia} -#| hold: true -vs = [1,2] -vs[2] = 10 -vs -``` - -The assignment `vs[2]` is different than the initial assignment `vs=[1,2]` in that, `vs[2]=10` **modifies** the container that `vs` points to, whereas `vs=[1,2]` **replaces** any binding for `vs`. The indexed assignment is more memory efficient when vectors are large. This point is also of interest when passing vectors to functions, as a function may modify components of the vector passed to it, though can't replace the container itself. - - -## Some useful functions for working with vectors. - - -As mentioned, the `length` function returns the number of components in a vector. It is one of several useful functions for vectors. - - -The `sum` and `prod` function will add and multiply the elements in a vector: - - -```{julia} -v1 = [1,1,2,3,5,8] -sum(v1), prod(v1) -``` - -The `unique` function will throw out any duplicates: - - -```{julia} -unique(v1) # drop a `1` -``` - -The functions `maximum` and `minimum` will return the largest and smallest values of an appropriate vector. - - -```{julia} -maximum(v1) -``` - -(These should not be confused with `max` and `min` which give the largest or smallest value over all their arguments.) - - -The `extrema` function returns both the smallest and largest value of a collection: - - -```{julia} -extrema(v1) -``` - -Consider now - - -```{julia} -𝒗 = [1,4,2,3] -``` - -The `sort` function will rearrange the values in `𝒗`: - - -```{julia} -sort(𝒗) -``` - -The keyword argument, `rev=true` can be given to get values in decreasing order: - - -```{julia} -sort(𝒗, rev=true) -``` - -For adding a new element to a vector the `push!` method can be used, as in - - -```{julia} -push!(𝒗, 5) -``` - -To append more than one value, the `append!` function can be used: - - -```{julia} -append!(v1, [6,8,7]) -``` - -These two functions modify or mutate the values stored within the vector `𝒗` that passed as an argument. In the `push!` example above, the value `5` is added to the vector of $4$ elements. In `Julia`, a convention is to name mutating functions with a trailing exclamation mark. (Again, these do not mutate the binding of `𝒗` to the container, but do mutate the contents of the container.) There are functions with mutating and non-mutating definitions, an example is `sort` and `sort!`. - - -If only a mutating function is available, like `push!`, and this is not desired a copy of the vector can be made. It is not enough to copy by assignment, as with `w = 𝒗`. As both `w` and `𝒗` will be bound to the same memory location. Rather, you call `copy` (or sometimes `deepcopy`) to make a new container with copied contents, as in `w = copy(𝒗)`. - - -Creating new vectors of a given size is common for programming, though not much use will be made here. There are many different functions to do so: `ones` to make a vector of ones, `zeros` to make a vector of zeros, `trues` and `falses` to make Boolean vectors of a given size, and `similar` to make a similar-sized vector (with no particular values assigned). - - -## Applying functions element by element to values in a vector - - -Functions such as `sum` or `length` are known as *reductions* as they reduce the "dimensionality" of the data: a vector is in some sense $1$-dimensional, the sum or length are $0$-dimensional numbers. Applying a reduction is straightforward – it is just a regular function call. - - -```{julia} -#| hold: true -v = [1, 2, 3, 4] -sum(v), length(v) -``` - -Other desired operations with vectors act differently. Rather than reduce a collection of values using some formula, the goal is to apply some formula to *each* of the values, returning a modified vector. A simple example might be to square each element, or subtract the average value from each element. An example comes from statistics. When computing a variance, we start with data $x_1, x_2, \dots, x_n$ and along the way form the values $(x_1-\bar{x})^2, (x_2-\bar{x})^2, \dots, (x_n-\bar{x})^2$. - - -Such things can be done in *many* different ways. Here we describe two, but will primarily utilize the first. - - -### Broadcasting a function call - - -If we have a vector, `xs`, and a function, `f`, to apply to each value, there is a simple means to achieve this task. By adding a "dot" between the function name and the parenthesis that enclose the arguments, instructs `Julia` to "broadcast" the function call. The details allow for more flexibility, but, for this purpose, broadcasting will take each value in `xs` and apply `f` to it, returning a vector of the same size as `xs`. When more than one argument is involved, broadcasting will try to fill out different sized objects. - - -For example, the following will find, using `sqrt`, the square root of each value in a vector: - - -```{julia} -xs = [1, 1, 3, 4, 7] -sqrt.(xs) -``` - -This would find the sine of each number in `xs`: - - -```{julia} -sin.(xs) -``` - -For each function, the `.(` (and not `(`) after the name is the surface syntax for broadcasting. - - -The `^` operator is an *infix* operator. Infix operators can be broadcast, as well, by using the form `.` prior to the operator, as in: - - -```{julia} -xs .^ 2 -``` - -Here is an example involving the logarithm of a set of numbers. In astronomy, a logarithm with base $100^{1/5}$ is used for star [brightness](http://tinyurl.com/ycp7k8ay). We can use broadcasting to find this value for several values at once through: - - -```{julia} -ys = [1/5000, 1/500, 1/50, 1/5, 5, 50] -base = (100)^(1/5) -log.(base, ys) -``` - -Broadcasting with multiple arguments allows for mixing of vectors and scalar values, as above, making it convenient when parameters are used. - - -As a final example, the task from statistics of centering and then squaring can be done with broadcasting. We go a bit further, showing how to compute the [sample variance](http://tinyurl.com/p6wa4r8) of a data set. This has the formula - - -$$ -\frac{1}{n-1}\cdot ((x_1-\bar{x})^2 + \cdots + (x_n - \bar{x})^2). -$$ - -This can be computed, with broadcasting, through: - - -```{julia} -#| hold: true -import Statistics: mean -xs = [1, 1, 2, 3, 5, 8, 13] -n = length(xs) -(1/(n-1)) * sum(abs2.(xs .- mean(xs))) -``` - -This shows many of the manipulations that can be made with vectors. Rather than write `.^2`, we follow the definition of `var` and chose the possibly more performant `abs2` function which, in general, efficiently finds $|x|^2$ for various number types. The `.-` uses broadcasting to subtract a scalar (`mean(xs)`) from a vector (`xs`). Without the `.`, this would error. - - -:::{.callout-note} -## Note -The `map` function is very much related to broadcasting and similarly named functions are found in many different programming languages. (The "dot" broadcast is mostly limited to `Julia` and mirrors a similar usage of a dot in `MATLAB`.) For those familiar with other programming languages, using `map` may seem more natural. Its syntax is `map(f, xs)`. - -::: - -### Comprehensions - - -In mathematics, set notation is often used to describe elements in a set. - - -For example, the first $5$ cubed numbers can be described by: - - -$$ -\{x^3: x \text{ in } 1, 2,\dots, 5\} -$$ - -Comprehension notation is similar. The above could be created in `Julia` with: - - -```{julia} -xs = [1,2,3,4,5] -[x^3 for x in xs] -``` - -Something similar can be done more succinctly: - - -```{julia} -xs .^ 3 -``` - -However, comprehensions have a value when more complicated expressions are desired as they work with an expression of `xs`, and not a pre-defined or user-defined function. - - -Another typical example of set notation might include a condition, such as, the numbers divisible by $7$ between $1$ and $100$. Set notation might be: - - -$$ -\{x: \text{rem}(x, 7) = 0 \text{ for } x \text{ in } 1, 2, \dots, 100\}. -$$ - -This would be read: "the set of $x$ such that the remainder on division by $7$ is $0$ for all x in $1, 2, \dots, 100$." - - -In `Julia`, a comprehension can include an `if` clause to mirror, somewhat, the math notation. For example, the above would become (using `1:100` as a means to create the numbers $1,2,\dots, 100$, as will be described in an upcoming section): - - -```{julia} -[x for x in 1:100 if rem(x,7) == 0] -``` - -Comprehensions can be a convenient means to describe a collection of numbers, especially when no function is defined, but the simplicity of the broadcast notation (just adding a judicious ".") leads to its more common use in these notes. - - -##### Example: creating a "T" table for creating a graph - - -The process of plotting a function is usually first taught by generating a "T" table: values of $x$ and corresponding values of $y$. These pairs are then plotted on a Cartesian grid and the points are connected with lines to form the graph. Generating a "T" table in `Julia` is easy: create the $x$ values, then create the $y$ values for each $x$. - - -To be concrete, let's generate $7$ points to plot $f(x) = x^2$ over $[-1,1]$. - - -The first task is to create the data. We will soon see more convenient ways to generate patterned data, but for now, we do this by hand: - - -```{julia} -a, b, n = -1, 1, 7 -d = (b-a) // (n-1) -xs = [a, a+d, a+2d, a+3d, a+4d, a+5d, a+6d] # 7 points -``` - -To get the corresponding $y$ values, we can use a compression (or define a function and broadcast): - - -```{julia} -ys = [x^2 for x in xs] -``` - -Vectors can be compared together by combining them into a separate container, as follows: - - -```{julia} -[xs ys] -``` - -(If there is a space between objects they are horizontally combined. In our construction of vectors using `[]` we used a comma for vertical combination. More generally we should use a `;` for vertical concatenation.) - - -In the sequel, we will typically use broadcasting for this task using two steps: one to define a function the second to broadcast it. - - -:::{.callout-note} -## Note -The style generally employed here is to use plural variable names for a collection of values, such as the vector of $y$ values and singular names when a single value is being referred to, leading to expressions like "`x in xs`". - -::: - ## Other container types We end this section with some general comments that are for those interested in a bit more, but in general aren't needed to understand most all of what follows later. @@ -761,12 +433,16 @@ Tuples are fixed-length containers where there is no expectation or enforcement While a vector is formed by placing comma-separated values within a `[]` pair (e.g., `[1,2,3]`), a tuple is formed by placing comma-separated values within a `()` pair. A tuple of length $1$ uses a convention of a trailing comma to distinguish it from a parenthesized expression (e.g. `(1,)` is a tuple, `(1)` is just the value `1`). + +Vectors and tuples can appear at the same time: a vector of tuples--each of length $n$--can be used in plotting to specify points. + :::{.callout-note} ## Well, actually... Technically, the tuple is formed just by the use of commas, which separate different expressions. The parentheses are typically used, as they clarify the intent and disambiguate some usage. In a notebook interface, it is useful to just use commas to separate values to output, as typically the only the last command is displayed. This usage just forms a tuple of the values and displays that. ::: +#### Named tuples There are *named tuples* where each component has an associated name. Like a tuple these can be indexed by number and unlike regular tuples also by name. @@ -795,12 +471,14 @@ The values in a named tuple can be accessed using the "dot" notation: nt.x1 ``` -Alternatively, the index notation -- using a *symbol* for the name -- can be used: +Alternatively, the index notation--using a *symbol* for the name--can be used: ```{julia} nt[:x1] ``` +(Indexing is described a bit later, but it is a way to pull elements out of a collection.) + Named tuples are employed to pass parameters to functions. To find the slope, we could do: ```{julia} @@ -820,11 +498,15 @@ x1 - x0 ::: -### Associative arrays +### Pairs, associative arrays Named tuples associate a name (in this case a symbol) to a value. More generally an associative array associates to each key a value, where the keys and values may be of different types. -The `pair` notation, `key => value`, is used to make one association. A *dictionary* is used to have a container of associations. For example, this constructs a simple dictionary associating a spelled out name with a numeric value: +The `pair` notation, `key => value`, is used to make one association between the first and second value. + +A *dictionary* is used to have a container of associations. + +This example constructs a simple dictionary associating a spelled out name with a numeric value: ```{julia} d = Dict("one" => 1, "two" => 2, "three" => 3) @@ -842,7 +524,449 @@ d["two"] Named tuples are associative arrays where the keys are restricted to symbols. There are other types of associative arrays, specialized cases of the `AbstractDict` type with performance benefits for specific use cases. In these notes, dictionaries appear as output in some function calls. -Unlike vectors and tuples, dictionaries are not currently supported by broadcasting. This causes no loss in usefulness, as the values can easily be iterated over, but the convenience of the dot notation is lost. +Unlike vectors and tuples, dictionaries are not currently supported by broadcasting. (To be described in the next section.) This causes no loss in usefulness, as the values can easily be iterated over, but the convenience of the dot notation is lost. + + +## The container interface in Julia + +There are numerous generic functions for working across the many different types of containers. Some are specific to containers which can be modified, some to associative arrays. But it is expected for different container types to implement as many as possible. We list a few here for completeness. Only a few will be used in these notes. + +### Indexing + +Vectors have an implied order: first element, second, last, etc. Tuples do as well. Matrices have two orders: by a row-column pair or by linear order where the first column precedes the second etc. Arrays are similar in that they have a linear order and can be accessed by their individual dimensions. + +To access an element in a vector, say the second, the underlying `getindex` function is used. This is rarely typed, as the `[` notation is used. This notation is used in a style similar to a function call, the indexes go between matching pairs. + +For example, we create a vector, tuple, and matrix: + +```{julia} +v = [1,2,3,4] +t = (1,2,3,4) +m = [1 2; 3 4] +``` + +The second element of each is accessed similarly: + +```{julia} +v[2], t[2], m[2] +``` + +(All of `v`, `t`, and `m` have $1$-based indexing.) + +There is special syntax to reference the last index when used within the square braces: + +```{julia} +v[end], t[end], m[end] +``` + +The last element is also returned by `last`: + +```{julia} +last(v), last(t), last(m) +``` + +These use `lastindex` behind the scenes. There is also a `firstindex` which is associated with the `first` method: + + +```{julia} +first(v), first(t), first(m) +``` + +For indexing by a numeric index, a container of numbers may be used. Containers can be generated different ways, here we just use a vector to get the second and third elements: + +```{julia} +I = [2,3] +v[I], t[I], m[I] +``` + +When indexing by a vector, the value will not be a scalar, even if there is only one element indicated. + +Indexing can also be done by a mask of Boolean values with a matching length. This following mask should do the same as indexing by `I` above: + +```{julia} +J = [false, true, true, false] +v[J], t[J], m[J] +``` + +For the matrix, values can be referenced by row/column values. The following will extract the second row, first column: + +```{julia} +m[2, 1] +``` + +*If* a container has *only* one entry, then the `only` method will return that element (not within the container). Here we use a tuple to illustrate to emphasize the trailing comma in construction: + +```{julia} +s = ("one", ) +``` + +```{julia} +only(s) +``` + +There will be an error with `only` should the container not have just one element. + +### Mutating values + +Vectors and matrices can have their elements changed or mutated; tuples can not. The process is similar to assignment--using an equals sign--but the left hand side has indexing notation to reference which values within the container are to be updated. + +To change the last element of `v` to `0` we have: + +```{julia} +v[end] = 0 +v +``` + +We might read this as assignment, but what happens is the underlying container has an element indicated by the index mutated. The `setindex!` function is called behind the scenes. + + +The `setindex!` function will try to promote the value (`0` above) to the element type of the container. This can throw an error if the promotion isn't possible. For example, to specify an element as `missing` with `v[end] = missing` will error, as missing can't be promoted to an integer. + +If more than one value is referenced in the assignment, then more than one value can be specified on the right-hand side. + +Mutation is different from reassignment. A command like `v=[1,2,3,0]` would have had the same effect as `v[end] = 0`, but would be quite different. The first *replaces* the binding to `v` with a new container, the latter reaches into the container and replaces just a value it holds. + +### Size and type + +The `length` of a container is the number of elements in linear order: + +```{julia} +length(v), length(t), length(m) +``` + +The `isempty` method will indicate if the length is 0, perhaps in a performant way: + +```{julia} +isempty(v), isempty([]), isempty(t), isempty(()) +``` + + +The `size` of a container, when defined, takes into account its shape or the dimensions: + +```{julia} +size(v), size(m) # no size defined for tuples +``` + +Arrays, and hence vectors and matrices have an element type given by `eltype` (the `typeof` method returns the container type: + +```{julia} +eltype(v), eltype(t), eltype(m) +``` + +(The element type of the tuple is `Int64`, but this is only because of this particular tuple. Tuples are typically heterogeneous containers--not homogeneous like vectors--and do not expect to have a common type. The `NTuple` type is for tuples with elements of the same type.) + +### Modifying the length of a container + +Vectors and some other containers allow elements to be added on or elements to be taken off. In computer science a queue is a collection that is ordered and has addition at one or the other end. Vectors can be used as a queue, though for just that task, there are more performant structures available. + +Two key methods for queues are `push!` and `pop!`. We `push!` elements onto the end of the queue: + +```{julia} +push!(v, 5) +``` + +The output is expected -- `5` was added to the end of `v`. What might not be expected is the underlying `v` is changed without assignment. (Actually `mutated`, the underlying container assigned to the symbol `v` is extended, not replaced.) + +:::{.callout-note} +## Trailing exclamation point convention +The function `push!` has a trailing exclamation point which is a `Julia` convention to indicate one of the underlying arguments (traditionally the first) will be *mutated* by the function call. +::: + +The `pop!` function is somewhat of a reverse: it takes the last element and "pops" it off the queue, leaving the queue one element shorter and returning the last element. + +```{julia} +pop!(v) +``` + +```{julia} +v +``` + +There are also `pushfirst!`, `popfirst!`, `insert!` and `deleteat!` methods. + + +### Iteration + +A very fundamental operation is to iterate over the elements of a collection one by one. + +In computer science the `for` loop is the basic construct to iterate over values. This example will iterate over `v` and add each value to `tot` which is initialized to be `0`: + +```{julia} +tot = 0 +for e in v + tot = tot + e +end +tot +``` + +The `for` loop construct is central in many programming languages; in `Julia` for loops are very performant and very flexible, however, they are more verbose than needed. (In the above example we had to initialize an accumulator and then write three lines for the loop, whereas `sum(v)` would do the same--and in this case more flexibly, with just a single call.) Alternatives are usually leveraged--we mention a few. + +Iterating over a vector can be done by *value*, as above, or by *index*. For the latter the `eachindex` method creates an iterable for the indices of the container. For rectangular objects, like matrices, there are also many uses for `eachrow` and `eachcol`, though not in these notes. + + +There are a few basic patterns where alternatives to a `for` loop exist. We discuss two: + +* mapping a function or expression over each element in the collection +* a reduction where a larger dimensional object is summarized by a lower dimensional one. In the example above, the $1$-dimensional vector is reduced to a $0$-dimensional scalar by summing the elements. + +#### Comprehensions + +In mathematics, set notation is often used to describe elements in a set. + +For example, the first $5$ cubed numbers can be described by: + + +$$ +\{x^3: x \text{ in } 1, 2,\dots, 5\} +$$ + +Comprehension notation is similar. The above could be created in `Julia` with: + + +```{julia} +xs = [1, 2, 3, 4, 5] +[x^3 for x in xs] +``` + +Comprehensions are one way of iterating over a collection and evaluating an expression on each element. + +In the above, the value `x` takes on each value in `xs`. The variables may be tuples, as well. + +The `enumerate` method wraps a container (or iterable) and iterates both the index and the value. This is useful, say for polynomials: + +```{julia} +x = 3 +as = [1, 2, 3] # evaluate a₀⋅x⁰, a₁⋅x¹, a₂⋅x² +[a*x^(i-1) for (i, a) in enumerate(as)] +``` + +(These values can then be easily summed to evaluate the polynomial.) + +When iterating over `enumerate` a tuple is returned. The use of `(i, a)` to iterate over these tuples destructures the tuple into parts to be used in the expression. + +The `zip` function also is useful to *pair* off iterators. Redoing the above to have the powers iterated over: + +```{julia} +as = [1, 2, 3] +inds = [0, 1, 2] +[a*x^i for (i, a) in zip(inds, as)] +``` + +Like `enumerate`, the `zip` iterator has elements which are tuples. + + +:::{.callout-note} +## Note +The style generally employed herein is to use plural variable names for a collection of values, such as the vector of $y$ values and singular names when a single value is being referred to, leading to expressions like "`x in xs`". + +::: + + + +#### Broadcasting a function call + +If we have a vector, `xs`, and a function, `f`, to apply to each value, there is a simple means to achieve this task that is shorter than a `for` loop or the comprehension `[f(x) for x in s]`. By adding a "dot" between the function name and the parenthesis that enclose the arguments, instructs `Julia` to "broadcast" the function call. The details allow for more much flexibility, but, for this purpose, broadcasting will take each value in `xs` and apply `f` to it, returning a vector of the same size as `xs`. When more than one argument is involved, broadcasting will try to pad out different sized objects to the same shape. Broadcasting can also *fuse* combined function calls. + + +For example, the following will find, using `sqrt`, the square root of each value in a vector: + + +```{julia} +xs = [1, 1, 3, 4, 7] +sqrt.(xs) +``` + +This call finds the sine of each number in `xs`: + + +```{julia} +sin.(xs) +``` + +For each function call, the `.(` (and not `(`) after the name is the surface syntax for broadcasting. + + +The `^` operator is an *infix* operator. Infix operators can be broadcast, as well, by using the form `.` prior to the operator, as in: + + +```{julia} +xs .^ 2 +``` + +Here is an example involving the logarithm of a set of numbers. In astronomy, a logarithm with base $100^{1/5}$ is used for star [brightness](http://tinyurl.com/ycp7k8ay). We can use broadcasting to find this value for several values at once through: + + +```{julia} +ys = [1/5000, 1/500, 1/50, 1/5, 5, 50] +base = (100)^(1/5) +log.(base, ys) +``` + +Broadcasting with multiple arguments allows for mixing of vectors and scalar values, as above, making it convenient when parameters are used. In broadcasting, there are times where it is desirable to treat a container as a scalar-like argument, a common idiom is to wrap that container in a 1-element tuple. + + +As a final example, the task from statistics of centering and then squaring can be done with broadcasting. We go a bit further, showing how to compute the [sample variance](http://tinyurl.com/p6wa4r8) of a data set. This has the formula + + +$$ +\frac{1}{n-1}\cdot ((x_1-\bar{x})^2 + \cdots + (x_n - \bar{x})^2). +$$ + +This can be computed, with broadcasting, through: + + +```{julia} +#| hold: true +import Statistics: mean +xs = [1, 1, 2, 3, 5, 8, 13] +n = length(xs) +(1/(n-1)) * sum(abs2.(xs .- mean(xs))) +``` + +This shows many of the manipulations that can be made with vectors. Rather than write `.^2`, we follow the definition of `var` and chose the possibly more performant `abs2` function which, in general, efficiently finds $|x|^2$ for various number types. The `.-` uses broadcasting to subtract a scalar (`mean(xs)`) from a vector (`xs`). Without the `.`, this would error. + +Broadcasting is a widely used and powerful surface syntax which we will employ occasionally in the sequel. + + +#### Mapping a function over a collection + +The `map` function is very much related to broadcasting. Similarly named functions are found in many different programming languages. (The "dot" broadcast is mostly limited to `Julia` and mirrors a similar usage of a dot in `MATLAB`.) For those familiar with other programming languages, using `map` may seem more natural. Its syntax is `map(f, xs)`. There may be one or more iterable passed to `map`. + + +For example, this will map `sin` over each value in `xs`, computing the same things as `sin.(xs)`: + +```{julia} +map(sin, xs) +``` + +The `map` function can be used with one or more iterators. + +The `map` function can also be used in combination with `reduce`, a reduction. Reductions take a container with one or more dimensions and reduces the number of dimensions. A example might be: + +```{julia} +sum(map(sin, xs)) +``` + +This has a performance drawback--there are two passes through the container, one to apply `sin` another to add. + +The `mapreduce` function combines the map and reduce operations in one pass. It takes a third argument to reduce by in the second position. This is a *binary* operator. So this combination will map `sin` over `xs` and then add the results up: + +```{julia} +mapreduce(sin, +, xs) +``` + +There are other specialized reduction functions that reverse the order of the mapper and the reducer. For example, we have `sum` (used above) and `prod` for adding and multiplying values in a collection: + +```{julia} +sum(xs), prod(xs) +``` + +These are reductions, which which fall back to a `mapreduce` call. They require a starting value (`init`) of `0` and `1` (which in this case can be determined from `xs`). The `sum` and `prod` function also allow as a first argument an initial function to map over the collection: + +```{julia} +sum(sin, xs) +``` + + +#### Other reductions + +There are other reductions, which summarize a container. We mention those related to the maximum or minimum of a collection. For these examples, we have + +```{julia} +v = [1, 2, 3, 4] +``` + +The largest value in a numeric collection is returned by `maximum`: + +```{julia} +maximum(v) +``` + +Where this maximum occurred is returned by `argmax`: + +```{julia} +argmax(v) +``` + +For `v` these are the same. But if we were to apply `sin` to `v` say, then the result may not be in order. This can be done with, say, a call to `map` and then `maximum`, but the functions allow an initial function to be specified: + +```{julia} +maximum(sin, v), argmax(sin, v) +``` + +This combination is also the duty of `findmax`: + +```{julia} +findmax(sin, v) +``` + +There are also `minimum`, `argmin`, and `findmin`. + +The `extrema` function returns the maximum and minimum of the collection: + +```{julia} +extrema(v) +``` + +:::{.callout-note} +## `maximum` and `max` +In `Julia` there are two related functions: `maximum` and `max`. The `maximum` function generically returns the largest element in a collection. The `max` function returns the maximum of its *arguments*. + +That is, these return identical values: + +```{julia} +xs = [1, 3, 2] +maximum(xs), max(1, 3, 2), max(xs...) +``` + +The latter using *splatting* to iterate over each value in `xs` and pass it to `max` as an argument. +::: + +### Predicate functions + +A few reductions work with *predicate* functions--those that return `true` or `false`. Let's use `iseven` as an example, which tests if a number is even. + +We can check if *all* the alements of a container are even or if *any* of the elements of a container are even with `all` and `even`: + +```{julia} +xs = [1, 1, 2, 3, 5] +all(iseven, xs), any(iseven, xs) +``` + + +Related, we can count the number of `true` responses of the predicate function: + +```{julia} +count(iseven, xs) +``` + + +#### methods for associative arrays + +For dictionaries, the collection is unordered (by default), but iteration can still be done over "key-value" pairs. + +In `Julia` a `Pair` matches a key and a value into one entity. Pairs are made with the `=>` notation with the `key` on the left and the value on the right. + + +Dictionaries are a collection of pairs. The `Dict` constructor can be passed pairs directly: + +```{julia} +ascii = Dict("a"=>97, "b"=>98, "c"=>99) # etc. +``` + +To iterate over these, the `pairs` iterator is useful: + +```{julia} +collect(pairs(ascii)) +``` + +(We used `collect` to iterate over values and return them as a vector.) + +The keys are returned by `keys`, the values by `values`: + +```{julia} +keys(ascii) +``` @@ -993,6 +1117,7 @@ From [transum.org](http://www.transum.org/Maths/Exam/Online_Exercise.asp?Topic=V #| hold: true #| echo: false let + gr() p = plot(xlim=(0,10), ylim=(0,5), legend=false, framestyle=:none) for j in (-3):10 plot!(p, [j, j + 5], [0, 5*sqrt(3)], color=:blue, alpha=0.5) @@ -1058,6 +1183,12 @@ answ = 4 radioq(choices, answ) ``` +```{julia} +#| echo: false +plotly() +nothing +``` + ###### Question From 33c6e62d68732f826fa0519b8772fbb7eaef6e8e Mon Sep 17 00:00:00 2001 From: jverzani Date: Sun, 27 Jul 2025 15:26:00 -0400 Subject: [PATCH 18/22] em dash; sentence case --- .gitignore | 4 +- quarto/ODEs/euler.qmd | 8 +- quarto/ODEs/odes.qmd | 13 +- quarto/_quarto.yml | 14 +- quarto/basics.qmd | 3 + quarto/basics/Project.toml | 15 ++ quarto/basics/_pdf_index.qmd | 1 + quarto/{precalc => basics}/calculator.qmd | 2 +- .../figures/calculator.png | Bin .../figures/order_operations_pop_mech.png | Bin quarto/{precalc => basics}/figures/swale.png | Bin .../logical_expressions.qmd | 0 quarto/basics/make_pdf.jl | 16 ++ quarto/{precalc => basics}/numbers_types.qmd | 0 quarto/{precalc => basics}/ranges.qmd | 2 +- quarto/{precalc => basics}/variables.qmd | 2 +- quarto/{precalc => basics}/vectors.qmd | 16 +- quarto/derivatives/curve_sketching.qmd | 2 +- .../derivatives/implicit_differentiation.qmd | 2 +- quarto/derivatives/lhospitals_rule.qmd | 2 +- quarto/derivatives/mean_value_theorem.qmd | 2 +- quarto/derivatives/more_zeros.qmd | 2 +- quarto/derivatives/symbolic_derivatives.qmd | 9 -- .../derivatives/taylor_series_polynomials.qmd | 12 +- .../matrix_calculus_notes.qmd | 47 +++--- .../polar_coordinates.qmd | 4 +- .../scalar_functions.qmd | 8 +- .../scalar_functions_applications.qmd | 14 +- .../vector_fields.qmd | 10 +- .../vector_valued_functions.qmd | 6 +- .../vectors.qmd | 30 ++-- quarto/index.qmd | 4 +- .../div_grad_curl.qmd | 2 +- .../line_integrals.qmd | 6 +- quarto/integral_vector_calculus/review.qmd | 4 +- .../stokes_theorem.qmd | 8 +- quarto/integrals/arc_length.qmd | 2 +- quarto/integrals/area.qmd | 4 +- quarto/integrals/area_between_curves.qmd | 2 +- quarto/integrals/center_of_mass.qmd | 2 +- quarto/integrals/improper_integrals.qmd | 2 +- quarto/integrals/integration_by_parts.qmd | 6 +- quarto/integrals/partial_fractions.qmd | 4 +- quarto/integrals/surface_area.qmd | 2 +- quarto/integrals/twelve-qs.qmd | 34 ++--- quarto/limits/intermediate_value_theorem.qmd | 4 +- quarto/limits/limits.qmd | 2 +- quarto/precalc.qmd | 4 - quarto/precalc/exp_log_functions.qmd | 8 +- quarto/precalc/functions.qmd | 24 +-- quarto/precalc/inversefunctions.qmd | 8 +- quarto/precalc/julia_overview.qmd | 50 +++---- quarto/precalc/make_pdf.jl | 8 +- quarto/precalc/plotting.qmd | 34 ++--- quarto/precalc/polynomial.qmd | 2 +- quarto/precalc/polynomial_roots.qmd | 4 +- quarto/precalc/polynomials_package.qmd | 2 +- quarto/precalc/transformations.qmd | 10 +- quarto/precalc/trig_functions.qmd | 140 +++++++++++++++++- 59 files changed, 385 insertions(+), 243 deletions(-) create mode 100644 quarto/basics.qmd create mode 100644 quarto/basics/Project.toml create mode 120000 quarto/basics/_pdf_index.qmd rename quarto/{precalc => basics}/calculator.qmd (99%) rename quarto/{precalc => basics}/figures/calculator.png (100%) rename quarto/{precalc => basics}/figures/order_operations_pop_mech.png (100%) rename quarto/{precalc => basics}/figures/swale.png (100%) rename quarto/{precalc => basics}/logical_expressions.qmd (100%) create mode 100644 quarto/basics/make_pdf.jl rename quarto/{precalc => basics}/numbers_types.qmd (100%) rename quarto/{precalc => basics}/ranges.qmd (99%) rename quarto/{precalc => basics}/variables.qmd (99%) rename quarto/{precalc => basics}/vectors.qmd (97%) diff --git a/.gitignore b/.gitignore index 66f3cb9..958614b 100644 --- a/.gitignore +++ b/.gitignore @@ -5,6 +5,7 @@ docs/site test/benchmarks.json Manifest.toml TODO.md +Changelog.md /*/_pdf_index.pdf /*/*/_pdf_index.pdf /*/_pdf_index.typ @@ -12,4 +13,5 @@ TODO.md /*/CalculusWithJulia.pdf default.profraw /quarto/default.profraw -/*/*/default.profraw \ No newline at end of file +/*/*/default.profraw +/*/*_files diff --git a/quarto/ODEs/euler.qmd b/quarto/ODEs/euler.qmd index ceff1cb..493ca31 100644 --- a/quarto/ODEs/euler.qmd +++ b/quarto/ODEs/euler.qmd @@ -184,7 +184,7 @@ plot(exp(-1/2)*exp(x^2/2), x0, 2) plot!(xs, ys) ``` -Not bad. We wouldn't expect this to be exact - due to the concavity of the solution, each step is an underestimate. However, we see it is an okay approximation and would likely be better with a smaller $h$. A topic we pursue in just a bit. +Not bad. We wouldn't expect this to be exact---due to the concavity of the solution, each step is an underestimate. However, we see it is an okay approximation and would likely be better with a smaller $h$. A topic we pursue in just a bit. Rather than type in the above command each time, we wrap it all up in a function. The inputs are $n$, $a=x_0$, $b=x_n$, $y_0$, and, most importantly, $F$. The output is massaged into a function through a call to `linterp`, rather than two vectors. The `linterp` function[^Interpolations] we define below just finds a function that linearly interpolates between the points and is `NaN` outside of the range of the $x$ values: @@ -263,7 +263,7 @@ Each step introduces an error. The error in one step is known as the *local trun The total error, or more commonly, *global truncation error*, is the error between the actual answer and the approximate answer at the end of the process. It reflects an accumulation of these local errors. This error is *bounded* by a constant times $h$. Since it gets smaller as $h$ gets smaller in direct proportion, the Euler method is called *first order*. -Other, somewhat more complicated, methods have global truncation errors that involve higher powers of $h$ - that is for the same size $h$, the error is smaller. In analogy is the fact that Riemann sums have error that depends on $h$, whereas other methods of approximating the integral have smaller errors. For example, Simpson's rule had error related to $h^4$. So, the Euler method may not be employed if there is concern about total resources (time, computer, ...), it is important for theoretical purposes in a manner similar to the role of the Riemann integral. +Other, somewhat more complicated, methods have global truncation errors that involve higher powers of $h$---that is for the same size $h$, the error is smaller. In analogy is the fact that Riemann sums have error that depends on $h$, whereas other methods of approximating the integral have smaller errors. For example, Simpson's rule had error related to $h^4$. So, the Euler method may not be employed if there is concern about total resources (time, computer, ...), it is important for theoretical purposes in a manner similar to the role of the Riemann integral. In the examples, we will see that for many problems the simple Euler method is satisfactory, but not always so. The task of numerically solving differential equations is not a one-size-fits-all one. In the following, a few different modifications are presented to the basic Euler method, but this just scratches the surface of the topic. @@ -648,7 +648,7 @@ plot(euler2(x0, xn, y0, yp0, 360), 0, 4T) plot!(x -> pi/4*cos(sqrt(g/l)*x), 0, 4T) ``` -Even now, we still see that something seems amiss, though the issue is not as dramatic as before. The oscillatory nature of the pendulum is seen, but in the Euler solution, the amplitude grows, which would necessarily mean energy is being put into the system. A familiar instance of a pendulum would be a child on a swing. Without pumping the legs - putting energy in the system - the height of the swing's arc will not grow. Though we now have oscillatory motion, this growth indicates the solution is still not quite right. The issue is likely due to each step mildly overcorrecting and resulting in an overall growth. One of the questions pursues this a bit further. +Even now, we still see that something seems amiss, though the issue is not as dramatic as before. The oscillatory nature of the pendulum is seen, but in the Euler solution, the amplitude grows, which would necessarily mean energy is being put into the system. A familiar instance of a pendulum would be a child on a swing. Without pumping the legs---putting energy in the system---the height of the swing's arc will not grow. Though we now have oscillatory motion, this growth indicates the solution is still not quite right. The issue is likely due to each step mildly overcorrecting and resulting in an overall growth. One of the questions pursues this a bit further. ## Questions @@ -794,7 +794,7 @@ Modify the `euler2` function to implement the Euler-Cromer method. What do you s #| hold: true #| echo: false choices = [ -"The same as before - the amplitude grows", +"The same as before---the amplitude grows", "The solution is identical to that of the approximation found by linearization of the sine term", "The solution has a constant amplitude, but its period is slightly *shorter* than that of the approximate solution found by linearization", "The solution has a constant amplitude, but its period is slightly *longer* than that of the approximate solution found by linearization"] diff --git a/quarto/ODEs/odes.qmd b/quarto/ODEs/odes.qmd index 46b2050..d9e4652 100644 --- a/quarto/ODEs/odes.qmd +++ b/quarto/ODEs/odes.qmd @@ -149,7 +149,7 @@ $$ U'(t) = -r U(t), \quad U(0) = U_0. $$ -This shows that the rate of change of $U$ depends on $U$. Large positive values indicate a negative rate of change - a push back towards the origin, and large negative values of $U$ indicate a positive rate of change - again, a push back towards the origin. We shouldn't be surprised to either see a steady decay towards the origin, or oscillations about the origin. +This shows that the rate of change of $U$ depends on $U$. Large positive values indicate a negative rate of change---a push back towards the origin, and large negative values of $U$ indicate a positive rate of change---again, a push back towards the origin. We shouldn't be surprised to either see a steady decay towards the origin, or oscillations about the origin. What will we find? This equation is different from the previous two equations, as the function $U$ appears on both sides. However, we can rearrange to get: @@ -177,7 +177,7 @@ $$ In words, the initial difference in temperature of the object and the environment exponentially decays to $0$. -That is, as $t > 0$ goes to $\infty$, the right hand will go to $0$ for $r > 0$, so $T(t) \rightarrow T_a$ - the temperature of the object will reach the ambient temperature. The rate of this is largest when the difference between $T(t)$ and $T_a$ is largest, so when objects are cooling the statement "hotter things cool faster" is appropriate. +That is, as $t > 0$ goes to $\infty$, the right hand will go to $0$ for $r > 0$, so $T(t) \rightarrow T_a$---the temperature of the object will reach the ambient temperature. The rate of this is largest when the difference between $T(t)$ and $T_a$ is largest, so when objects are cooling the statement "hotter things cool faster" is appropriate. A graph of the solution for $T_0=200$ and $T_a=72$ and $r=1/2$ is made as follows. We've added a few line segments from the defining formula, and see that they are indeed tangent to the solution found for the differential equation. @@ -403,7 +403,7 @@ To finish, we call `dsolve` to find a solution (if possible): out = dsolve(eqn) ``` -This answer - to a first-order equation - has one free constant, `C₁`, which can be solved for from an initial condition. We can see that when $a > 0$, as $x$ goes to positive infinity the solution goes to $1$, and when $x$ goes to negative infinity, the solution goes to $0$ and otherwise is trapped in between, as expected. +This answer---to a first-order equation---has one free constant, `C₁`, which can be solved for from an initial condition. We can see that when $a > 0$, as $x$ goes to positive infinity the solution goes to $1$, and when $x$ goes to negative infinity, the solution goes to $0$ and otherwise is trapped in between, as expected. The limits are confirmed by investigating the limits of the right-hand: @@ -618,6 +618,7 @@ nothing ``` ![The cables of an unloaded suspension bridge have a different shape than a loaded suspension bridge. As seen, the cables in this [figure](https://www.brownstoner.com/brooklyn-life/verrazano-narrows-bridge-anniversary-historic-photos/) would be modeled by a catenary.](./figures/verrazano-narrows-bridge-anniversary-historic-photos-2.jpeg) + --- @@ -641,7 +642,7 @@ $$ x''(t) = 0, \quad y''(t) = -g. $$ -That is, the $x$ position - where no forces act - has $0$ acceleration, and the $y$ position - where the force of gravity acts - has constant acceleration, $-g$, where $g=9.8m/s^2$ is the gravitational constant. These equations can be solved to give: +That is, the $x$ position---where no forces act---has $0$ acceleration, and the $y$ position---where the force of gravity acts---has constant acceleration, $-g$, where $g=9.8m/s^2$ is the gravitational constant. These equations can be solved to give: $$ @@ -957,7 +958,7 @@ radioq(choices, answ) ##### Question -The example with projectile motion in a medium has a parameter $\gamma$ modeling the effect of air resistance. If `y` is the answer - as would be the case if the example were copy-and-pasted in - what can be said about `limit(y, gamma=>0)`? +The example with projectile motion in a medium has a parameter $\gamma$ modeling the effect of air resistance. If `y` is the answer---as would be the case if the example were copy-and-pasted in---what can be said about `limit(y, gamma=>0)`? ```{julia} @@ -966,7 +967,7 @@ The example with projectile motion in a medium has a parameter $\gamma$ modeling choices = [ "The limit is a quadratic polynomial in `x`, mirroring the first part of that example.", "The limit does not exist, but the limit to `oo` gives a quadratic polynomial in `x`, mirroring the first part of that example.", -"The limit does not exist -- there is a singularity -- as seen by setting `gamma=0`." +"The limit does not exist---there is a singularity---as seen by setting `gamma=0`." ] answ = 1 radioq(choices, answ) diff --git a/quarto/_quarto.yml b/quarto/_quarto.yml index 8426908..3e2df4c 100644 --- a/quarto/_quarto.yml +++ b/quarto/_quarto.yml @@ -25,14 +25,16 @@ book: page-footer: "Copyright 2022-25, John Verzani" chapters: - index.qmd + - part: basics.qmd + chapters: + - basics/calculator.qmd + - basics/variables.qmd + - basics/numbers_types.qmd + - basics/logical_expressions.qmd + - basics/vectors.qmd + - basics/ranges.qmd - part: precalc.qmd chapters: - - precalc/calculator.qmd - - precalc/variables.qmd - - precalc/numbers_types.qmd - - precalc/logical_expressions.qmd - - precalc/vectors.qmd - - precalc/ranges.qmd - precalc/functions.qmd - precalc/plotting.qmd - precalc/transformations.qmd diff --git a/quarto/basics.qmd b/quarto/basics.qmd new file mode 100644 index 0000000..baa62cb --- /dev/null +++ b/quarto/basics.qmd @@ -0,0 +1,3 @@ +# Mathematical basics + +This chapter introduces some mathematical basics and their counterparts within the `Julia` programming language. diff --git a/quarto/basics/Project.toml b/quarto/basics/Project.toml new file mode 100644 index 0000000..1c8e033 --- /dev/null +++ b/quarto/basics/Project.toml @@ -0,0 +1,15 @@ +[deps] +CalculusWithJulia = "a2e0e22d-7d4c-5312-9169-8b992201a882" +DataFrames = "a93c6f00-e57d-5684-b7b6-d8193f3e46c0" +LaTeXStrings = "b964fa9f-0449-5b57-a5c2-d3ea65f4040f" +Logging = "56ddb016-857b-54e1-b83d-db4d58db5568" +Measures = "442fdcdd-2543-5da2-b0f3-8c86c306513e" +Mustache = "ffc61752-8dc7-55ee-8c37-f3e9cdd09e70" +PlotlyBase = "a03496cd-edff-5a9b-9e67-9cda94a718b5" +PlotlyKaleido = "f2990250-8cf9-495f-b13a-cce12b45703c" +Plots = "91a5bcdd-55d7-5caf-9e0b-520d859cae80" +Primes = "27ebfcd6-29c5-5fa9-bf4b-fb8fc14df3ae" +QuizQuestions = "612c44de-1021-4a21-84fb-7261cf5eb2d4" +SymPy = "24249f21-da20-56a4-8eb1-6a02cf4ae2e6" +Tables = "bd369af6-aec1-5ad0-b16a-f7cc5008161c" +TextWrap = "b718987f-49a8-5099-9789-dcd902bef87d" diff --git a/quarto/basics/_pdf_index.qmd b/quarto/basics/_pdf_index.qmd new file mode 120000 index 0000000..98da163 --- /dev/null +++ b/quarto/basics/_pdf_index.qmd @@ -0,0 +1 @@ +../basics.qmd \ No newline at end of file diff --git a/quarto/precalc/calculator.qmd b/quarto/basics/calculator.qmd similarity index 99% rename from quarto/precalc/calculator.qmd rename to quarto/basics/calculator.qmd index c91485a..75443dc 100644 --- a/quarto/precalc/calculator.qmd +++ b/quarto/basics/calculator.qmd @@ -244,7 +244,7 @@ With the Google Calculator, typing `1 + 2 x 3 =` will give the value $7$, but *i In `Julia`, the entire expression is typed in before being evaluated, so the usual conventions of mathematics related to the order of operations may be used. These are colloquially summarized by the acronym [PEMDAS](http://en.wikipedia.org/wiki/Order_of_operations). -> **PEMDAS**. This acronym stands for Parentheses, Exponents, Multiplication, Division, Addition, Subtraction. The order indicates which operation has higher precedence, or should happen first. This isn't exactly the case, as "M" and "D" have the same precedence, as do "A" and "S". In the case of two operations with equal precedence, *associativity* is used to decide which to do. For the operations `-`, `/` the associativity is left to right, as in the left one is done first, then the right. However, `^` has right associativity, so `4^3^2` is `4^(3^2)` and not `(4^3)^2` (Be warned that some calculators - and spread sheets, such as Excel - will treat this expression with left associativity). But, `+` and `*` don't have associativity, so `1+2+3` can be `(1+2)+3` or `1+(2+3)`. +> **PEMDAS**. This acronym stands for Parentheses, Exponents, Multiplication, Division, Addition, Subtraction. The order indicates which operation has higher precedence, or should happen first. This isn't exactly the case, as "M" and "D" have the same precedence, as do "A" and "S". In the case of two operations with equal precedence, *associativity* is used to decide which to do. For the operations `-`, `/` the associativity is left to right, as in the left one is done first, then the right. However, `^` has right associativity, so `4^3^2` is `4^(3^2)` and not `(4^3)^2` (Be warned that some calculators - and spread sheets, such as Excel - will treat this expression with left associativity). But, `+` and `*` don't have associativity, so `1+2+3` can be `(1+2)+3` or `1+(2+3)`. diff --git a/quarto/precalc/figures/calculator.png b/quarto/basics/figures/calculator.png similarity index 100% rename from quarto/precalc/figures/calculator.png rename to quarto/basics/figures/calculator.png diff --git a/quarto/precalc/figures/order_operations_pop_mech.png b/quarto/basics/figures/order_operations_pop_mech.png similarity index 100% rename from quarto/precalc/figures/order_operations_pop_mech.png rename to quarto/basics/figures/order_operations_pop_mech.png diff --git a/quarto/precalc/figures/swale.png b/quarto/basics/figures/swale.png similarity index 100% rename from quarto/precalc/figures/swale.png rename to quarto/basics/figures/swale.png diff --git a/quarto/precalc/logical_expressions.qmd b/quarto/basics/logical_expressions.qmd similarity index 100% rename from quarto/precalc/logical_expressions.qmd rename to quarto/basics/logical_expressions.qmd diff --git a/quarto/basics/make_pdf.jl b/quarto/basics/make_pdf.jl new file mode 100644 index 0000000..9a3e640 --- /dev/null +++ b/quarto/basics/make_pdf.jl @@ -0,0 +1,16 @@ +module Make +# makefile for generating typst pdfs +# per directory usage +dir = "basics" +files = ("calculator", + "variables", + "numbers_types", + "logical_expressions", + "vectors", + "ranges", + ) + +include("../_make_pdf.jl") +main() + +end diff --git a/quarto/precalc/numbers_types.qmd b/quarto/basics/numbers_types.qmd similarity index 100% rename from quarto/precalc/numbers_types.qmd rename to quarto/basics/numbers_types.qmd diff --git a/quarto/precalc/ranges.qmd b/quarto/basics/ranges.qmd similarity index 99% rename from quarto/precalc/ranges.qmd rename to quarto/basics/ranges.qmd index 110df17..a50cf77 100644 --- a/quarto/precalc/ranges.qmd +++ b/quarto/basics/ranges.qmd @@ -1,4 +1,4 @@ -# Ranges and Sets +# Ranges and sets {{< include ../_common_code.qmd >}} diff --git a/quarto/precalc/variables.qmd b/quarto/basics/variables.qmd similarity index 99% rename from quarto/precalc/variables.qmd rename to quarto/basics/variables.qmd index 54edc83..ad1a8b8 100644 --- a/quarto/precalc/variables.qmd +++ b/quarto/basics/variables.qmd @@ -231,7 +231,7 @@ The distinction between ``=`` versus `=` is important and one area where common ## Context -The binding of a value to a variable name happens within some context. When a variable is assigned or referenced, the scope of the variable -- the region of code where it is accessible -- is taken into consideration. +The binding of a value to a variable name happens within some context. When a variable is assigned or referenced, the scope of the variable---the region of code where it is accessible---is taken into consideration. For our simple illustrations, we are assigning values, as though they were typed at the command line. This stores the binding in the `Main` module. `Julia` looks for variables in this module when it encounters an expression and the value is substituted. Other uses, such as when variables are defined within a function, involve different contexts which may not be visible within the `Main` module. diff --git a/quarto/precalc/vectors.qmd b/quarto/basics/vectors.qmd similarity index 97% rename from quarto/precalc/vectors.qmd rename to quarto/basics/vectors.qmd index d24f5fb..9a5442e 100644 --- a/quarto/precalc/vectors.qmd +++ b/quarto/basics/vectors.qmd @@ -434,7 +434,7 @@ Tuples are fixed-length containers where there is no expectation or enforcement While a vector is formed by placing comma-separated values within a `[]` pair (e.g., `[1,2,3]`), a tuple is formed by placing comma-separated values within a `()` pair. A tuple of length $1$ uses a convention of a trailing comma to distinguish it from a parenthesized expression (e.g. `(1,)` is a tuple, `(1)` is just the value `1`). -Vectors and tuples can appear at the same time: a vector of tuples--each of length $n$--can be used in plotting to specify points. +Vectors and tuples can appear at the same time: a vector of tuples---each of length $n$---can be used in plotting to specify points. :::{.callout-note} ## Well, actually... @@ -471,7 +471,7 @@ The values in a named tuple can be accessed using the "dot" notation: nt.x1 ``` -Alternatively, the index notation--using a *symbol* for the name--can be used: +Alternatively, the index notation---using a *symbol* for the name---can be used: ```{julia} nt[:x1] @@ -608,7 +608,7 @@ There will be an error with `only` should the container not have just one elemen ### Mutating values -Vectors and matrices can have their elements changed or mutated; tuples can not. The process is similar to assignment--using an equals sign--but the left hand side has indexing notation to reference which values within the container are to be updated. +Vectors and matrices can have their elements changed or mutated; tuples can not. The process is similar to assignment---using an equals sign---but the left hand side has indexing notation to reference which values within the container are to be updated. To change the last element of `v` to `0` we have: @@ -653,7 +653,7 @@ Arrays, and hence vectors and matrices have an element type given by `eltype` (t eltype(v), eltype(t), eltype(m) ``` -(The element type of the tuple is `Int64`, but this is only because of this particular tuple. Tuples are typically heterogeneous containers--not homogeneous like vectors--and do not expect to have a common type. The `NTuple` type is for tuples with elements of the same type.) +(The element type of the tuple is `Int64`, but this is only because of this particular tuple. Tuples are typically heterogeneous containers---not homogeneous like vectors---and do not expect to have a common type. The `NTuple` type is for tuples with elements of the same type.) ### Modifying the length of a container @@ -665,7 +665,7 @@ Two key methods for queues are `push!` and `pop!`. We `push!` elements onto the push!(v, 5) ``` -The output is expected -- `5` was added to the end of `v`. What might not be expected is the underlying `v` is changed without assignment. (Actually `mutated`, the underlying container assigned to the symbol `v` is extended, not replaced.) +The output is expected---`5` was added to the end of `v`. What might not be expected is the underlying `v` is changed without assignment. (Actually `mutated`, the underlying container assigned to the symbol `v` is extended, not replaced.) :::{.callout-note} ## Trailing exclamation point convention @@ -699,7 +699,7 @@ end tot ``` -The `for` loop construct is central in many programming languages; in `Julia` for loops are very performant and very flexible, however, they are more verbose than needed. (In the above example we had to initialize an accumulator and then write three lines for the loop, whereas `sum(v)` would do the same--and in this case more flexibly, with just a single call.) Alternatives are usually leveraged--we mention a few. +The `for` loop construct is central in many programming languages; in `Julia` for loops are very performant and very flexible, however, they are more verbose than needed. (In the above example we had to initialize an accumulator and then write three lines for the loop, whereas `sum(v)` would do the same---and in this case more flexibly, with just a single call.) Alternatives are usually leveraged---we mention a few. Iterating over a vector can be done by *value*, as above, or by *index*. For the latter the `eachindex` method creates an iterable for the indices of the container. For rectangular objects, like matrices, there are also many uses for `eachrow` and `eachcol`, though not in these notes. @@ -847,7 +847,7 @@ The `map` function can also be used in combination with `reduce`, a reduction. R sum(map(sin, xs)) ``` -This has a performance drawback--there are two passes through the container, one to apply `sin` another to add. +This has a performance drawback---there are two passes through the container, one to apply `sin` another to add. The `mapreduce` function combines the map and reduce operations in one pass. It takes a third argument to reduce by in the second position. This is a *binary* operator. So this combination will map `sin` over `xs` and then add the results up: @@ -924,7 +924,7 @@ The latter using *splatting* to iterate over each value in `xs` and pass it to ` ### Predicate functions -A few reductions work with *predicate* functions--those that return `true` or `false`. Let's use `iseven` as an example, which tests if a number is even. +A few reductions work with *predicate* functions---those that return `true` or `false`. Let's use `iseven` as an example, which tests if a number is even. We can check if *all* the alements of a container are even or if *any* of the elements of a container are even with `all` and `even`: diff --git a/quarto/derivatives/curve_sketching.qmd b/quarto/derivatives/curve_sketching.qmd index 3c99eed..34a93a2 100644 --- a/quarto/derivatives/curve_sketching.qmd +++ b/quarto/derivatives/curve_sketching.qmd @@ -1,4 +1,4 @@ -# Curve Sketching +# Curve sketching {{< include ../_common_code.qmd >}} diff --git a/quarto/derivatives/implicit_differentiation.qmd b/quarto/derivatives/implicit_differentiation.qmd index 52ab46e..8d14a50 100644 --- a/quarto/derivatives/implicit_differentiation.qmd +++ b/quarto/derivatives/implicit_differentiation.qmd @@ -1,4 +1,4 @@ -# Implicit Differentiation +# Implicit differentiation {{< include ../_common_code.qmd >}} diff --git a/quarto/derivatives/lhospitals_rule.qmd b/quarto/derivatives/lhospitals_rule.qmd index 13b7867..5e9160b 100644 --- a/quarto/derivatives/lhospitals_rule.qmd +++ b/quarto/derivatives/lhospitals_rule.qmd @@ -1,4 +1,4 @@ -# L'Hospital's Rule +# L'Hospital's rule {{< include ../_common_code.qmd >}} diff --git a/quarto/derivatives/mean_value_theorem.qmd b/quarto/derivatives/mean_value_theorem.qmd index c2f2d06..554fe1e 100644 --- a/quarto/derivatives/mean_value_theorem.qmd +++ b/quarto/derivatives/mean_value_theorem.qmd @@ -1,4 +1,4 @@ -# The mean value theorem for differentiable functions. +# The mean value theorem for differentiable functions {{< include ../_common_code.qmd >}} diff --git a/quarto/derivatives/more_zeros.qmd b/quarto/derivatives/more_zeros.qmd index 4d05d14..14dec55 100644 --- a/quarto/derivatives/more_zeros.qmd +++ b/quarto/derivatives/more_zeros.qmd @@ -368,7 +368,7 @@ $$ So we should consider `f(x_n)` an *approximate zero* when it is on the scale of $f'(\alpha) \cdot \alpha \delta$. That $\alpha$ factor means we consider a *relative* tolerance for `f`. -> For checking if $f(x_n) \approx 0$ both a relative and absolute error should be used--the relative error involving the size of $x_n$. +> For checking if $f(x_n) \approx 0$ both a relative and absolute error should be used---the relative error involving the size of $x_n$. A good condition to check if `f(x_n)` is small is diff --git a/quarto/derivatives/symbolic_derivatives.qmd b/quarto/derivatives/symbolic_derivatives.qmd index 1308da3..3388032 100644 --- a/quarto/derivatives/symbolic_derivatives.qmd +++ b/quarto/derivatives/symbolic_derivatives.qmd @@ -10,15 +10,6 @@ This section uses the `TermInterface` add-on package. using TermInterface ``` -```{julia} -#| echo: false -const frontmatter = ( - title = "Symbolic derivatives", - description = "Calculus with Julia: Symbolic derivatives", - tags = ["CalculusWithJulia", "derivatives", "symbolic derivatives"], -); -``` - --- diff --git a/quarto/derivatives/taylor_series_polynomials.qmd b/quarto/derivatives/taylor_series_polynomials.qmd index 2b95e42..4fed196 100644 --- a/quarto/derivatives/taylor_series_polynomials.qmd +++ b/quarto/derivatives/taylor_series_polynomials.qmd @@ -1,4 +1,4 @@ -# Taylor Polynomials and other Approximating Polynomials +# Taylor polynomials and other approximating polynomials {{< include ../_common_code.qmd >}} @@ -104,7 +104,7 @@ $$ tl(x) = f(c) + f'(c) \cdot(x - c). $$ -The key is the term multiplying $(x-c)$ -- for the secant line this is an approximation to the related term for the tangent line. That is, the secant line approximates the tangent line, which is the linear function that best approximates the function at the point $(c, f(c))$. +The key is the term multiplying $(x-c)$---for the secant line this is an approximation to the related term for the tangent line. That is, the secant line approximates the tangent line, which is the linear function that best approximates the function at the point $(c, f(c))$. This is quantified by the *mean value theorem* which states under our assumptions on $f(x)$ that there exists some $\xi$ between $x$ and $c$ for which: @@ -194,7 +194,7 @@ function divided_differences(f, x, xs...) end ``` -In the following--even though it is *type piracy*--by adding a `getindex` method, we enable the `[]` notation of Newton to work with symbolic functions, like `u()` defined below, which is used in place of $f$: +In the following---even though it is *type piracy*---by adding a `getindex` method, we enable the `[]` notation of Newton to work with symbolic functions, like `u()` defined below, which is used in place of $f$: ```{julia} @@ -215,7 +215,7 @@ Now, let's look at: ex₂ = u[c, c+h, c+2h] ``` -If multiply by $2$ and simplify, a discrete approximation for the second derivative--the second order forward [difference equation](http://tinyurl.com/n4235xy)--is seen: +If multiply by $2$ and simplify, a discrete approximation for the second derivative---the second order forward [difference equation](http://tinyurl.com/n4235xy)---is seen: ```{julia} simplify(2ex₂) @@ -794,7 +794,7 @@ This is re-expressed as $2s + s \cdot p$ with $p$ given by: ```{julia} -cancel((a_b - 2s)/s) +p = cancel((a_b - 2s)/s) ``` Now, $2s = m - s\cdot m$, so the above can be reworked to be $\log(1+m) = m - s\cdot(m-p)$. @@ -807,7 +807,7 @@ How big can the error be between this *approximations* and $\log(1+m)$? The expr ```{julia} -Max = (v/(2+v))(v => sqrt(2) - 1) +Max = (x/(2+x))(x => sqrt(2) - 1) ``` The error term is like $2/19 \cdot \xi^{19}$ which is largest at this value of $M$. Large is relative - it is really small: diff --git a/quarto/differentiable_vector_calculus/matrix_calculus_notes.qmd b/quarto/differentiable_vector_calculus/matrix_calculus_notes.qmd index 545c972..cb65b27 100644 --- a/quarto/differentiable_vector_calculus/matrix_calculus_notes.qmd +++ b/quarto/differentiable_vector_calculus/matrix_calculus_notes.qmd @@ -1,4 +1,4 @@ -# Matrix Calculus +# Matrix calculus This section illustrates a more general setting for taking derivatives, that unifies the different expositions taken prior. @@ -74,7 +74,7 @@ Additionally, many other set of objects form vector spaces. Certain families of Let's take differentiable functions as an example. These form a vector space as the derivative of a linear combination of differentiable functions is defined through the simplest derivative rule: $[af(x) + bg(x)]' = a[f(x)]' + b[g(x)]'$. If $f$ and $g$ are differentiable, then so is $af(x)+bg(x)$. -A finite vector space is described by a *basis* -- a minimal set of vectors needed to describe the space, after consideration of linear combinations. For some typical vector spaces, this is the set of special vectors with $1$ as one of the entries, and $0$ otherwise. +A finite vector space is described by a *basis*---a minimal set of vectors needed to describe the space, after consideration of linear combinations. For some typical vector spaces, this is the set of special vectors with $1$ as one of the entries, and $0$ otherwise. A key fact about a basis for a finite vector space is every vector in the vector space can be expressed *uniquely* as a linear combination of the basis vectors. The set of numbers used in the linear combination, along with an order to the basis, means an element in a finite vector space can be associated with a unique coordinate vector. @@ -88,7 +88,7 @@ Vectors and matrices have properties that are generalizations of the real number * Viewing a vector as a matrix is possible. The association chosen here is common and is through a *column* vector. -* The *transpose* of a matrix comes by permuting the rows and columns. The transpose of a column vector is a row vector, so $v\cdot w = v^T w$, where we use a superscript $T$ for the transpose. The transpose of a product, is the product of the transposes -- reversed: $(AB)^T = B^T A^T$; the tranpose of a transpose is an identity operation: $(A^T)^T = A$; the inverse of a transpose is the tranpose of the inverse: $(A^{-1})^T = (A^T)^{-1}$. +* The *transpose* of a matrix comes by permuting the rows and columns. The transpose of a column vector is a row vector, so $v\cdot w = v^T w$, where we use a superscript $T$ for the transpose. The transpose of a product, is the product of the transposes---reversed: $(AB)^T = B^T A^T$; the tranpose of a transpose is an identity operation: $(A^T)^T = A$; the inverse of a transpose is the tranpose of the inverse: $(A^{-1})^T = (A^T)^{-1}$. * Matrices for which $A = A^T$ are called symmetric. @@ -231,7 +231,7 @@ Various differentiation rules are still available such as the sum, product, and ### Sum and product rules for the derivative -Using the differential notation -- which implicitly ignores higher order terms as they vanish in a limit -- the sum and product rules can be derived. +Using the differential notation---which implicitly ignores higher order terms as they vanish in a limit---the sum and product rules can be derived. For the sum rule, let $f(x) = g(x) + h(x)$. Then @@ -377,7 +377,7 @@ Multiplying left to right (the first) is called reverse mode; multiplying right The reason comes down to the shape of the matrices. To see, we need to know that matrix multiplication of an $m \times q$ matrix times a $q \times n$ matrix takes an order of $mqn$ operations. -When $m=1$, the derviative is a product of matrices of size $n\times j$, $j\times k$, and $k \times 1$ yielding a matrix of size $n \times 1$ matching the function dimension. +When $m=1$, the derivative is a product of matrices of size $n\times j$, $j\times k$, and $k \times 1$ yielding a matrix of size $n \times 1$ matching the function dimension. The operations involved in multiplication from left to right can be quantified. The first operation takes $njk$ operation leaving an $n\times k$ matrix, the next multiplication then takes another $nk1$ operations or $njk + nk$ together. @@ -435,7 +435,7 @@ That is $f'(A)$ is the operator $f'(A)[\delta A] = A \delta A + \delta A A$. (Th Alternatively, we can identify $A$ through its components, as a vector in $R^{n^2}$ and then leverage the Jacobian. -One such identification is vectorization -- consecutively stacking the +One such identification is vectorization---consecutively stacking the column vectors into a single vector. In `Julia` the `vec` function does this operation: @@ -444,7 +444,7 @@ operation: vec(A) ``` -The stacking by column follows how `Julia` stores matrices and how `Julia` references a matrices entries by linear index: +The stacking by column follows how `Julia` stores matrices and how `Julia` references entries in a matrix by linear index: ```{julia} vec(A) == [A[i] for i in eachindex(A)] @@ -562,7 +562,7 @@ all(l == r for (l, r) ∈ zip(L, R)) ---- -Now to use this relationship to recognize $df = A dA + dA A$ with the Jacobian computed from $\text{vec}{f(a)}$. +Now to use this relationship to recognize $df = A dA + dA A$ with the Jacobian computed from $\text{vec}(f(a))$. We have $\text{vec}(A dA + dA A) = \text{vec}(A dA) + \text{vec}(dA A)$, by obvious linearity of $\text{vec}$. Now inserting an identity matrix, $I$, which is symmteric, in a useful spot we have: @@ -683,7 +683,7 @@ det(I + dA) - det(I) ## The adjoint method -The chain rule brings about a series of products. The adjoint method illustrated below, shows how to approach the computation of the series in a direction that minimizes the computational cost, illustrating why reverse mode is preferred to forward mode when a scalar function of several variables is considered. +The chain rule brings about a series of products. The adjoint method illustrated by @BrightEdelmanJohnson and summarize below, shows how to approach the computation of the series in a direction that minimizes the computational cost, illustrating why reverse mode is preferred to forward mode when a scalar function of several variables is considered. @BrightEdelmanJohnson consider the derivative of @@ -778,9 +778,9 @@ Here $v$ can be solved for by taking adjoints (as before). Let $A = \partial h/\ ## Second derivatives, Hessian -@CarlssonNikitinTroedssonWendt -We reference a theorem presented by [Carlsson, Nikitin, Troedsson, and Wendt](https://arxiv.org/pdf/2502.03070v1) for exposition with some modification + +We reference a theorem presented by @CarlssonNikitinTroedssonWendt for exposition with some modification ::: {.callout-note appearance="minimal"} Theorem 1. Let $f:X \rightarrow Y$, where $X,Y$ are finite dimensional *inner product* spaces with elements in $R$. Suppose $f$ is smooth (a certain number of derivatives). Then for each $x$ in $X$ there exists a unique linear operator, $f'(x)$, and a unique *bilinear* *symmetric* operator $f'': X \oplus X \rightarrow Y$ such that @@ -804,7 +804,7 @@ $$ \begin{align*} f(x + dx) &= f(x) + \frac{\partial f}{\partial x_1} dx_1 + \frac{\partial f}{\partial x_2} dx_2\\ -&+ \frac{1}{2}\left( +&{+} \frac{1}{2}\left( \frac{\partial^2 f}{\partial x_1^2}dx_1^2 + \frac{\partial^2 f}{\partial x_1 \partial x_2}dx_1dx_2 + \frac{\partial^2 f}{\partial x_2^2}dx_2^2 @@ -832,7 +832,7 @@ $$ $H$ being the *Hessian* with entries $H_{ij} = \frac{\partial f}{\partial x_i \partial x_j}$. -This formula -- $f(x+dx)-f(x) \approx f'(x)dx + dx^T H dx$ -- is valid for any $n$, showing $n=2$ was just for ease of notation when expressing in the coordinates and not as matrices. +This formula---$f(x+dx)-f(x) \approx f'(x)dx + dx^T H dx$---is valid for any $n$, showing $n=2$ was just for ease of notation when expressing in the coordinates and not as matrices. By uniqueness, we have under these assumptions that the Hessian is *symmetric* and the expression $dx^T H dx$ is a *bilinear* form, which we can identify as $f''(x)[dx,dx]$. @@ -909,24 +909,23 @@ $$ &= \left( \text{det}(A) + \text{det}(A)\text{tr}(A^{-1}dA') \right) -\text{tr}((A^{-1} - A^{-1}dA' A^{-1})dA) - \text{det}(A) \text{tr}(A^{-1}dA) \\ +\text{tr}((A^{-1} - A^{-1}dA' A^{-1})dA)\\ +&\quad{-} \text{det}(A) \text{tr}(A^{-1}dA) \\ &= -\text{det}(A) \text{tr}(A^{-1}dA)\\ -&+ \text{det}(A)\text{tr}(A^{-1}dA')\text{tr}(A^{-1}dA) \\ -&- \text{det}(A)\text{tr}(A^{-1}dA' A^{-1}dA)\\ -&- \text{det}(A)\text{tr}(A^{-1}dA')\text{tr}(A^{-1}dA' A^{-1}dA)\\ -&- \text{det}(A) \text{tr}(A^{-1}dA) \\ +\textcolor{blue}{\text{det}(A) \text{tr}(A^{-1}dA)}\\ +&\quad{+} \text{det}(A)\text{tr}(A^{-1}dA')\text{tr}(A^{-1}dA) \\ +&\quad{-} \text{det}(A)\text{tr}(A^{-1}dA' A^{-1}dA)\\ +&\quad{-} \textcolor{red}{\text{det}(A)\text{tr}(A^{-1}dA')\text{tr}(A^{-1}dA' A^{-1}dA)}\\ +&\quad{-} \textcolor{blue}{\text{det}(A) \text{tr}(A^{-1}dA)} \\ &= \text{det}(A)\text{tr}(A^{-1}dA')\text{tr}(A^{-1}dA) - \text{det}(A)\text{tr}(A^{-1}dA' A^{-1}dA)\\ -&+ \text{third order term} +&\quad{+} \textcolor{red}{\text{third order term}} \end{align*} $$ So, after dropping the third-order term, we see: $$ -\begin{align*} f''(A)[dA,dA'] -&= \text{det}(A)\text{tr}(A^{-1}dA')\text{tr}(A^{-1}dA)\\ -&\quad - \text{det}(A)\text{tr}(A^{-1}dA' A^{-1}dA). -\end{align*} += \text{det}(A)\text{tr}(A^{-1}dA')\text{tr}(A^{-1}dA) - +\text{det}(A)\text{tr}(A^{-1}dA' A^{-1}dA). $$ diff --git a/quarto/differentiable_vector_calculus/polar_coordinates.qmd b/quarto/differentiable_vector_calculus/polar_coordinates.qmd index b2a1975..1679a21 100644 --- a/quarto/differentiable_vector_calculus/polar_coordinates.qmd +++ b/quarto/differentiable_vector_calculus/polar_coordinates.qmd @@ -1,4 +1,4 @@ -# Polar Coordinates and Curves +# Polar coordinates and curves {{< include ../_common_code.qmd >}} @@ -226,7 +226,7 @@ The folium has radial part $0$ when $\cos(\theta) = 0$ or $\sin(2\theta) = b/4a$ plot_polar(𝒂0..(pi/2-𝒂0), 𝒓) ``` -The second - which is too small to appear in the initial plot without zooming in - with +The second---which is too small to appear in the initial plot without zooming in---with ```{julia} diff --git a/quarto/differentiable_vector_calculus/scalar_functions.qmd b/quarto/differentiable_vector_calculus/scalar_functions.qmd index af75fec..efcd38f 100644 --- a/quarto/differentiable_vector_calculus/scalar_functions.qmd +++ b/quarto/differentiable_vector_calculus/scalar_functions.qmd @@ -388,7 +388,7 @@ For a scalar function, Define a *level curve* as the solutions to the equations contour(xsₛ, ysₛ, zzsₛ) ``` -Were one to walk along one of the contour lines, then there would be no change in elevation. The areas of greatest change in elevation - basically the hills - occur where the different contour lines are closest. In this particular area, there is a river that runs from the upper right through to the lower left and this is flanked by hills. +Were one to walk along one of the contour lines, then there would be no change in elevation. The areas of greatest change in elevation---basically the hills--- occur where the different contour lines are closest. In this particular area, there is a river that runs from the upper right through to the lower left and this is flanked by hills. The $c$ values for the levels drawn may be specified through the `levels` argument: @@ -636,7 +636,7 @@ This says, informally, for any scale about $L$ there is a "ball" about $C$ (not In the univariate case, it can be useful to characterize a limit at $x=c$ existing if *both* the left and right limits exist and the two are equal. Generalizing to getting close in $R^m$ leads to the intuitive idea of a limit existing in terms of any continuous "path" that approaches $C$ in the $x$-$y$ plane has a limit and all are equal. Let $\gamma$ describe the path, and $\lim_{s \rightarrow t}\gamma(s) = C$. Then $f \circ \gamma$ will be a univariate function. If there is a limit, $L$, then this composition will also have the same limit as $s \rightarrow t$. Conversely, if for *every* path this composition has the *same* limit, then $f$ will have a limit. -The "two path corollary" is a trick to show a limit does not exist - just find two paths where there is a limit, but they differ, then a limit does not exist in general. +The "two path corollary" is a trick to show a limit does not exist---just find two paths where there is a limit, but they differ, then a limit does not exist in general. ### Continuity of scalar functions @@ -997,7 +997,7 @@ The figure suggests a potential geometric relationship between the gradient and We see here how the gradient of $f$, $\nabla{f} = \langle f_{x_1}, f_{x_2}, \dots, f_{x_n} \rangle$, plays a similar role as the derivative does for univariate functions. -First, we consider the role of the derivative for univariate functions. The main characterization - the derivative is the slope of the line that best approximates the function at a point - is quantified by Taylor's theorem. For a function $f$ with a continuous second derivative: +First, we consider the role of the derivative for univariate functions. The main characterization---the derivative is the slope of the line that best approximates the function at a point---is quantified by Taylor's theorem. For a function $f$ with a continuous second derivative: $$ @@ -1174,7 +1174,7 @@ atand(mean(slopes)) Which seems about right for a generally uphill trail section, as this is. -In the above example, the data is given in terms of a sample, not a functional representation. Suppose instead, the surface was generated by `f` and the path - in the $x$-$y$ plane - by $\gamma$. Then we could estimate the maximum and average steepness by a process like this: +In the above example, the data is given in terms of a sample, not a functional representation. Suppose instead, the surface was generated by `f` and the path---in the $x$-$y$ plane---by $\gamma$. Then we could estimate the maximum and average steepness by a process like this: ```{julia} diff --git a/quarto/differentiable_vector_calculus/scalar_functions_applications.qmd b/quarto/differentiable_vector_calculus/scalar_functions_applications.qmd index f3a7428..d48728e 100644 --- a/quarto/differentiable_vector_calculus/scalar_functions_applications.qmd +++ b/quarto/differentiable_vector_calculus/scalar_functions_applications.qmd @@ -918,7 +918,7 @@ zs = fₗ.(xs, ys) scatter3d!(xs, ys, zs) ``` -A contour plot also shows that some - and only one - extrema happens on the interior: +A contour plot also shows that some---and only one---extrema happens on the interior: ```{julia} @@ -967,10 +967,10 @@ We confirm this by looking at the Hessian and noting $H_{11} > 0$: Hₛ = subs.(hessian(exₛ, [x,y]), x=>xstarₛ[x], y=>xstarₛ[y]) ``` -As it occurs at $(\bar{x}, \bar{y})$ where $\bar{x} = (x_1 + x_2 + x_3)/3$ and $\bar{y} = (y_1+y_2+y_3)/3$ - the averages of the three values - the critical point is an interior point of the triangle. +As it occurs at $(\bar{x}, \bar{y})$ where $\bar{x} = (x_1 + x_2 + x_3)/3$ and $\bar{y} = (y_1+y_2+y_3)/3$---the averages of the three values---the critical point is an interior point of the triangle. -As mentioned by Strang, the real problem is to minimize $d_1 + d_2 + d_3$. A direct approach with `SymPy` - just replacing `d2` above with the square root fails. Consider instead the gradient of $d_1$, say. To avoid square roots, this is taken implicitly from $d_1^2$: +As mentioned by Strang, the real problem is to minimize $d_1 + d_2 + d_3$. A direct approach with `SymPy`---just replacing `d2` above with the square root fails. Consider instead the gradient of $d_1$, say. To avoid square roots, this is taken implicitly from $d_1^2$: $$ @@ -1016,7 +1016,7 @@ psₛₗ = [a*u for (a,u) in zip(asₛ₁, usₛ)] plot!(polygon(psₛₗ)...) ``` -Let's see where the minimum distance point is by constructing a plot. The minimum must be on the boundary, as the only point where the gradient vanishes is the origin, not in the triangle. The plot of the triangle has a contour plot of the distance function, so we see clearly that the minimum happens at the point `[0.5, -0.866025]`. On this plot, we drew the gradient at some points along the boundary. The gradient points in the direction of greatest increase - away from the minimum. That the gradient vectors have a non-zero projection onto the edges of the triangle in a direction pointing away from the point indicates that the function `d` would increase if moved along the boundary in that direction, as indeed it does. +Let's see where the minimum distance point is by constructing a plot. The minimum must be on the boundary, as the only point where the gradient vanishes is the origin, not in the triangle. The plot of the triangle has a contour plot of the distance function, so we see clearly that the minimum happens at the point `[0.5, -0.866025]`. On this plot, we drew the gradient at some points along the boundary. The gradient points in the direction of greatest increase---away from the minimum. That the gradient vectors have a non-zero projection onto the edges of the triangle in a direction pointing away from the point indicates that the function `d` would increase if moved along the boundary in that direction, as indeed it does. ```{julia} @@ -1064,7 +1064,7 @@ The smallest value is when $t=0$ or $t=1$, so at one of the points, as `li` is d ##### Example: least squares -We know that two points determine a line. What happens when there are more than two points? This is common in statistics where a bivariate data set (pairs of points $(x,y)$) are summarized through a linear model $\mu_{y|x} = \alpha + \beta x$, That is the average value for $y$ given a particular $x$ value is given through the equation of a line. The data is used to identify what the slope and intercept are for this line. We consider a simple case - $3$ points. The case of $n \geq 3$ being similar. +We know that two points determine a line. What happens when there are more than two points? This is common in statistics where a bivariate data set (pairs of points $(x,y)$) are summarized through a linear model $\mu_{y|x} = \alpha + \beta x$, That is the average value for $y$ given a particular $x$ value is given through the equation of a line. The data is used to identify what the slope and intercept are for this line. We consider a simple case---$3$ points. The case of $n \geq 3$ being similar. We have a line $l(x) = \alpha + \beta(x)$ and three points $(x_1, y_1)$, $(x_2, y_2)$, and $(x_3, y_3)$. Unless these three points *happen* to be collinear, they can't possibly all lie on the same line. So to *approximate* a relationship by a line requires some inexactness. One measure of inexactness is the *vertical* distance to the line: @@ -1118,7 +1118,7 @@ As found, the formulas aren't pretty. If $x_1 + x_2 + x_3 = 0$ they simplify. Fo subs(outₗₛ[β], sum(xₗₛ) => 0) ``` -Let $\vec{x} = \langle x_1, x_2, x_3 \rangle$ and $\vec{y} = \langle y_1, y_2, y_3 \rangle$ this is simply $(\vec{x} \cdot \vec{y})/(\vec{x}\cdot \vec{x})$, a formula that will generalize to $n > 3$. The assumption is not a restriction - it comes about by subtracting the mean, $\bar{x} = (x_1 + x_2 + x_3)/3$, from each $x$ term (and similarly subtract $\bar{y}$ from each $y$ term). A process called "centering." +Let $\vec{x} = \langle x_1, x_2, x_3 \rangle$ and $\vec{y} = \langle y_1, y_2, y_3 \rangle$ this is simply $(\vec{x} \cdot \vec{y})/(\vec{x}\cdot \vec{x})$, a formula that will generalize to $n > 3$. The assumption is not a restriction---it comes about by subtracting the mean, $\bar{x} = (x_1 + x_2 + x_3)/3$, from each $x$ term (and similarly subtract $\bar{y}$ from each $y$ term). A process called "centering." With this observation, the formulas can be re-expressed through: @@ -1587,7 +1587,7 @@ $$ G(\epsilon_1, \epsilon_2) = L. $$ -Now, Lagrange's method can be employed. This will be fruitful - even though we know the answer - it being $\epsilon_1 = \epsilon_2 = 0$! +Now, Lagrange's method can be employed. This will be fruitful---even though we know the answer---it being $\epsilon_1 = \epsilon_2 = 0$! Forging ahead, we compute $\nabla{F}$ and $\lambda \nabla{G}$ and set $\epsilon_1 = \epsilon_2 = 0$ where the two are equal. This will lead to a description of $y$ in terms of $y'$. diff --git a/quarto/differentiable_vector_calculus/vector_fields.qmd b/quarto/differentiable_vector_calculus/vector_fields.qmd index 8f612e7..f18e2b1 100644 --- a/quarto/differentiable_vector_calculus/vector_fields.qmd +++ b/quarto/differentiable_vector_calculus/vector_fields.qmd @@ -111,7 +111,7 @@ Plot of a vector field from $R^2 \rightarrow R^2$ illustrated by drawing curves To the plot, we added the partial derivatives with respect to $r$ (in red) and with respect to $\theta$ (in blue). These are found with the soon-to-be discussed Jacobian. From the graph, you can see that these vectors are tangent vectors to the drawn curves. -The curves form a non-rectangular grid. Were the cells exactly parallelograms, the area would be computed taking into account the length of the vectors and the angle between them -- the same values that come out of a cross product. +The curves form a non-rectangular grid. Were the cells exactly parallelograms, the area would be computed taking into account the length of the vectors and the angle between them---the same values that come out of a cross product. ## Parametrically defined surfaces @@ -323,7 +323,7 @@ plt = plot_axes() We are using the vector of tuples interface (representing points) to specify the curve to draw. -Now we add on some curves for fixed $t$ and then fixed $\theta$ utilizing the fact that `project` returns a tuple of $x$--$y$ values to display. +Now we add on some curves for fixed $t$ and then fixed $\theta$ utilizing the fact that `project` returns a tuple of $x$---$y$ values to display. ```{julia} for t in range(t₀, tₙ, 20) @@ -1225,9 +1225,9 @@ q = interpolate(vcat(basic_conditions, new_conds)) plot_q_level_curve(q;layout=(1,2)) ``` -For this shape, if $b$ increases away from $b_0$, the secant line connecting $(a_0,0)$ and $(b, f(b)$ will have a negative slope, but there are no points nearby $x=c_0$ where the derivative has a tangent line with negative slope, so the continuous function is only on the left side of $b_0$. Mathematically, as $f$ is increasing $c_0$ -- as $f'''(c_0) = 3 > 0$ -- and $f$ is decreasing at $f(b_0)$ -- as $f'(b_0) = -1 < 0$, the signs alone suggest the scenario. The contour plot reveals, not one, but two one-sided functions of $b$ giving $c$. +For this shape, if $b$ increases away from $b_0$, the secant line connecting $(a_0,0)$ and $(b, f(b)$ will have a negative slope, but there are no points nearby $x=c_0$ where the derivative has a tangent line with negative slope, so the continuous function is only on the left side of $b_0$. Mathematically, as $f$ is increasing $c_0$---as $f'''(c_0) = 3 > 0$---and $f$ is decreasing at $f(b_0)$---as $f'(b_0) = -1 < 0$, the signs alone suggest the scenario. The contour plot reveals, not one, but two one-sided functions of $b$ giving $c$. ----- +--- Now to characterize all possibilities. @@ -1291,7 +1291,7 @@ $$ Then $F(c, b) = g_1(b) - g_2(c)$. By construction, $g_2(c_0) = 0$ and $g_2^{(k)}(c_0) = f^{(k+1)}(c_0)$, -Adjusting $f$ to have a vanishing second -- but not third -- derivative at $c_0$ means $g_2$ will satisfy the assumptions of the lemma assuming $f$ has at least four continuous derivatives (as all our example polynomials do). +Adjusting $f$ to have a vanishing second---but not third---derivative at $c_0$ means $g_2$ will satisfy the assumptions of the lemma assuming $f$ has at least four continuous derivatives (as all our example polynomials do). As for $g_1$, we have by construction $g_1(b_0) = 0$. By differentiation we get a pattern for some constants $c_j = (j+1)\cdot(j+2)\cdots \cdot k$ with $c_k = 1$. diff --git a/quarto/differentiable_vector_calculus/vector_valued_functions.qmd b/quarto/differentiable_vector_calculus/vector_valued_functions.qmd index b6267b6..ba9650d 100644 --- a/quarto/differentiable_vector_calculus/vector_valued_functions.qmd +++ b/quarto/differentiable_vector_calculus/vector_valued_functions.qmd @@ -981,7 +981,7 @@ $$ \vec{v} \times \vec{c} = GM \hat{x} + \vec{d}. $$ -As $\vec{x}$ and $\vec{v}\times\vec{c}$ lie in the same plane - orthogonal to $\vec{c}$ - so does $\vec{d}$. With a suitable re-orientation, so that $\vec{d}$ is along the $x$ axis, $\vec{c}$ is along the $z$-axis, then we have $\vec{c} = \langle 0,0,c\rangle$ and $\vec{d} = \langle d ,0,0 \rangle$, and $\vec{x} = \langle x, y, 0 \rangle$. Set $\theta$ to be the angle, then $\hat{x} = \langle \cos(\theta), \sin(\theta), 0\rangle$. +As $\vec{x}$ and $\vec{v}\times\vec{c}$ lie in the same plane---orthogonal to $\vec{c}$---so does $\vec{d}$. With a suitable re-orientation, so that $\vec{d}$ is along the $x$ axis, $\vec{c}$ is along the $z$-axis, then we have $\vec{c} = \langle 0,0,c\rangle$ and $\vec{d} = \langle d ,0,0 \rangle$, and $\vec{x} = \langle x, y, 0 \rangle$. Set $\theta$ to be the angle, then $\hat{x} = \langle \cos(\theta), \sin(\theta), 0\rangle$. Now @@ -1662,7 +1662,7 @@ $$ The first equation relates the steering angle with the curvature. If the steering angle is not changed ($d\alpha/du=0$) then the curvature is constant and the motion is circular. It will be greater for larger angles (up to $\pi/2$). As the curvature is the reciprocal of the radius, this means the radius of the circular trajectory will be smaller. For the same constant steering angle, the curvature will be smaller for longer wheelbases, meaning the circular trajectory will have a larger radius. For cars, which have similar dynamics, this means longer wheelbase cars will take more room to make a U-turn. -The second equation may be interpreted in ratio of arc lengths. The infinitesimal arc length of the rear wheel is proportional to that of the front wheel only scaled down by $\cos(\alpha)$. When $\alpha=0$ - the bike is moving in a straight line - and the two are the same. At the other extreme - when $\alpha=\pi/2$ - the bike must be pivoting on its rear wheel and the rear wheel has no arc length. This cosine, is related to the speed of the back wheel relative to the speed of the front wheel, which was used in the initial differential equation. +The second equation may be interpreted in ratio of arc lengths. The infinitesimal arc length of the rear wheel is proportional to that of the front wheel only scaled down by $\cos(\alpha)$. When $\alpha=0$---the bike is moving in a straight line---and the two are the same. At the other extreme---when $\alpha=\pi/2$---the bike must be pivoting on its rear wheel and the rear wheel has no arc length. This cosine, is related to the speed of the back wheel relative to the speed of the front wheel, which was used in the initial differential equation. The last equation, relates the curvature of the back wheel track to the steering angle of the front wheel. When $\alpha=\pm\pi/2$, the rear-wheel curvature, $k$, is infinite, resulting in a cusp (no circle with non-zero radius will approximate the trajectory). This occurs when the front wheel is steered orthogonal to the direction of motion. As was seen in previous graphs of the trajectories, a cusp can happen for quite regular front wheel trajectories. @@ -1875,7 +1875,7 @@ $$ $$ -We see $\vec\beta'$ is zero (the curve is non-regular) when $\kappa'(s) = 0$. The curvature changes from increasing to decreasing, or vice versa at each of the $4$ crossings of the major and minor axes - there are $4$ non-regular points, and we see $4$ cusps in the evolute. +We see $\vec\beta'$ is zero (the curve is non-regular) when $\kappa'(s) = 0$. The curvature changes from increasing to decreasing, or vice versa at each of the $4$ crossings of the major and minor axes--there are $4$ non-regular points, and we see $4$ cusps in the evolute. The curve parameterized by $\vec{r}(t) = 2(1 - \cos(t)) \langle \cos(t), \sin(t)\rangle$ over $[0,2\pi]$ is cardiod. It is formed by rolling a circle of radius $r$ around another similar sized circle. The following graphically shows the evolute is a smaller cardiod (one-third the size). For fun, the evolute of the evolute is drawn: diff --git a/quarto/differentiable_vector_calculus/vectors.qmd b/quarto/differentiable_vector_calculus/vectors.qmd index 31d648f..2dad038 100644 --- a/quarto/differentiable_vector_calculus/vectors.qmd +++ b/quarto/differentiable_vector_calculus/vectors.qmd @@ -81,7 +81,7 @@ $$ \| \vec{v} \| = \sqrt{ v_1^2 + v_2^2 + \cdots + v_n^2}. $$ -The definition of a norm leads to a few properties. First, if $c$ is a scalar, $\| c\vec{v} \| = |c| \| \vec{v} \|$ - which says scalar multiplication by $c$ changes the length by $|c|$. (Sometimes, scalar multiplication is described as "scaling by....") The other property is an analog of the triangle inequality, in which for any two vectors $\| \vec{v} + \vec{w} \| \leq \| \vec{v} \| + \| \vec{w} \|$. The right hand side is equal only when the two vectors are parallel. +The definition of a norm leads to a few properties. First, if $c$ is a scalar, $\| c\vec{v} \| = |c| \| \vec{v} \|$---which says scalar multiplication by $c$ changes the length by $|c|$. (Sometimes, scalar multiplication is described as "scaling by....") The other property is an analog of the triangle inequality, in which for any two vectors $\| \vec{v} + \vec{w} \| \leq \| \vec{v} \| + \| \vec{w} \|$. The right hand side is equal only when the two vectors are parallel. A vector with length $1$ is called a *unit* vector. Dividing a non-zero vector by its norm will yield a unit vector, a consequence of the first property above. Unit vectors are often written with a "hat:" $\hat{v}$. @@ -234,7 +234,7 @@ A simple example might be to add up a sequence of numbers. A direct way might be x1, x2, x3, x4, x5, x6 = 1, 2, 3, 4, 5, 6 x1 + x2 + x3 + x4 + x5 + x6 ``` - + Someone doesn't need to know `Julia`'s syntax to guess what this computes, save for the idiosyncratic tuple assignment used, which could have been bypassed at the cost of even more typing. A more efficient means to do, as each component isn't named, this would be to store the data in a container: @@ -267,7 +267,7 @@ These two functions are *reductions*. There are others, such as `maximum` and `m reduce(+, xs; init=0) # sum(xs) ``` -or +or ```{julia} reduce(*, xs; init=1) # prod(xs) @@ -289,9 +289,9 @@ and foldr(=>, xs) ``` -Next, we do a slightly more complicated problem. +Next, we do a slightly more complicated problem. -Recall the distance formula between two points, also called the *norm*. It is written here with the square root on the other side: $d^2 = (x_1-y_1)^2 + (x_0 - y_0)^2$. This computation can be usefully generalized to higher dimensional points (with $n$ components each). +Recall the distance formula between two points, also called the *norm*. It is written here with the square root on the other side: $d^2 = (x_1-y_1)^2 + (x_0 - y_0)^2$. This computation can be usefully generalized to higher dimensional points (with $n$ components each). This first example shows how the value for $d^2$ can be found using broadcasting and `sum`: @@ -309,10 +309,10 @@ This formula is a sum after applying an operation to the paired off values. Usin sum((xi - yi)^2 for (xi, yi) in zip(xs, ys)) ``` -The `zip` function, used above, produces an iterator over tuples of the paired off values in the two (or more) containers passed to it. +The `zip` function, used above, produces an iterator over tuples of the paired off values in the two (or more) containers passed to it. -This pattern -- where a reduction follows a function's application to the components -- is implemented in `mapreduce`. +This pattern---where a reduction follows a function's application to the components---is implemented in `mapreduce`. ```{julia} @@ -337,7 +337,7 @@ mapreduce((xi,yi) -> (xi-yi)^2, +, xs, ys) At times, extracting all but the first or last value can be of interest. For example, a polygon comprised of $n$ points (the vertices), might be stored using a vector for the $x$ and $y$ values with an additional point that mirrors the first. Here are the points: ```{julia} -xs = [1, 3, 4, 2] +xs = [1, 3, 4, 2] ys = [1, 1, 2, 3] pts = zip(xs, ys) # recipe for [(x1,y1), (x2,y2), (x3,y3), (x4,y4)] ``` @@ -392,7 +392,7 @@ The `take` method could be used to remove the padded value from the `xs` and `ys ##### Example: Riemann sums -In the computation of a Riemann sum, the interval $[a,b]$ is partitioned using $n+1$ points $a=x_0 < x_1 < \cdots < x_{n-1} < x_n = b$. +In the computation of a Riemann sum, the interval $[a,b]$ is partitioned using $n+1$ points $a=x_0 < x_1 < \cdots < x_{n-1} < x_n = b$. ```{julia} a, b, n = 0, 1, 4 @@ -414,7 +414,7 @@ sum(f ∘ first, partitions) ``` This uses a few things: like `mapreduce`, `sum` allows a function to -be applied to each element in the `partitions` collection. (Indeed, the default method to compute `sum(xs)` for an arbitrary container resolves to `mapreduce(identity, add_sum, xs)` where `add_sum` is basically `+`.) +be applied to each element in the `partitions` collection. (Indeed, the default method to compute `sum(xs)` for an arbitrary container resolves to `mapreduce(identity, add_sum, xs)` where `add_sum` is basically `+`.) In this case, the values come as tuples to the function to apply to each component. @@ -636,7 +636,7 @@ But the associative property does not make sense, as $(\vec{u} \cdot \vec{v}) \c ## Matrices -Algebraically, the dot product of two vectors - pair off by components, multiply these, then add - is a common operation. Take for example, the general equation of a line, or a plane: +Algebraically, the dot product of two vectors---pair off by components, multiply these, then add---is a common operation. Take for example, the general equation of a line, or a plane: $$ @@ -764,7 +764,7 @@ Vectors are defined similarly. As they are identified with *column* vectors, we ```{julia} -𝒷 = [10, 11, 12] # not 𝒷 = [10 11 12], which would be a row vector. +a = [10, 11, 12] # not a = [10 11 12], which would be a row vector. ``` In `Julia`, entries in a matrix (or a vector) are stored in a container with a type wide enough accommodate each entry. In this example, the type is SymPy's `Sym` type: @@ -822,7 +822,7 @@ We can then see how the system of equations is represented with matrices: ```{julia} -M * xs - 𝒷 +M * xs - a ``` Here we use `SymPy` to verify the above: @@ -899,7 +899,7 @@ and ``` :::{.callout-note} ## Note -The adjoint is defined *recursively* in `Julia`. In the `CalculusWithJulia` package, we overload the `'` notation for *functions* to yield a univariate derivative found with automatic differentiation. This can lead to problems: if we have a matrix of functions, `M`, and took the transpose with `M'`, then the entries of `M'` would be the derivatives of the functions in `M` - not the original functions. This is very much likely to not be what is desired. The `CalculusWithJulia` package commits **type piracy** here *and* abuses the generic idea for `'` in Julia. In general type piracy is very much frowned upon, as it can change expected behaviour. It is defined in `CalculusWithJulia`, as that package is intended only to act as a means to ease users into the wider package ecosystem of `Julia`. +The adjoint is defined *recursively* in `Julia`. In the `CalculusWithJulia` package, we overload the `'` notation for *functions* to yield a univariate derivative found with automatic differentiation. This can lead to problems: if we have a matrix of functions, `M`, and took the transpose with `M'`, then the entries of `M'` would be the derivatives of the functions in `M`---not the original functions. This is very much likely to not be what is desired. The `CalculusWithJulia` package commits **type piracy** here *and* abuses the generic idea for `'` in Julia. In general type piracy is very much frowned upon, as it can change expected behaviour. It is defined in `CalculusWithJulia`, as that package is intended only to act as a means to ease users into the wider package ecosystem of `Julia`. ::: --- @@ -1081,7 +1081,7 @@ norm(u₂ × v₂) --- -This analysis can be extended to the case of 3 vectors, which - when not co-planar - will form a *parallelepiped*. +This analysis can be extended to the case of 3 vectors, which---when not co-planar---will form a *parallelepiped*. ```{julia} diff --git a/quarto/index.qmd b/quarto/index.qmd index 0529f15..581cae5 100644 --- a/quarto/index.qmd +++ b/quarto/index.qmd @@ -77,7 +77,9 @@ These notes may be compiled into a `pdf` file through Quarto. As the result is r --> To *contribute* -- say by suggesting additional topics, correcting a -mistake, or fixing a typo -- click the "Edit this page" link and join the list of [contributors](https://github.com/jverzani/CalculusWithJuliaNotes.jl/graphs/contributors). Thanks to all contributors and a *very* special thanks to `@fangliu-tju` for their careful and most-appreciated proofreading. +mistake, or fixing a typo -- click the "Edit this page" link and join the list of [contributors](https://github.com/jverzani/CalculusWithJuliaNotes.jl/graphs/contributors). Thanks to all contributors. + +A *very* special thanks goes out to `@fangliu-tju` for their careful and most-appreciated proofreading and error spotting spread over a series of PRs. ## Running Julia diff --git a/quarto/integral_vector_calculus/div_grad_curl.qmd b/quarto/integral_vector_calculus/div_grad_curl.qmd index 5180b12..877eadc 100644 --- a/quarto/integral_vector_calculus/div_grad_curl.qmd +++ b/quarto/integral_vector_calculus/div_grad_curl.qmd @@ -1,4 +1,4 @@ -# The Gradient, Divergence, and Curl +# The gradient, divergence, and curl {{< include ../_common_code.qmd >}} diff --git a/quarto/integral_vector_calculus/line_integrals.qmd b/quarto/integral_vector_calculus/line_integrals.qmd index 94b0b8d..50bd29b 100644 --- a/quarto/integral_vector_calculus/line_integrals.qmd +++ b/quarto/integral_vector_calculus/line_integrals.qmd @@ -1,4 +1,4 @@ -# Line and Surface Integrals +# Line and surface integrals {{< include ../_common_code.qmd >}} @@ -340,7 +340,7 @@ W = integrate(F(r(t)) ⋅ T(r(t)), (t, 0, 2PI)) There are technical assumptions about curves and regions that are necessary for some statements to be made: - * Let $C$ be a [Jordan](https://en.wikipedia.org/wiki/Jordan_curve_theorem) curve - a non-self-intersecting continuous loop in the plane. Such a curve divides the plane into two regions, one bounded and one unbounded. The normal to a Jordan curve is assumed to be in the direction of the unbounded part. + * Let $C$ be a [Jordan](https://en.wikipedia.org/wiki/Jordan_curve_theorem) curve---a non-self-intersecting continuous loop in the plane. Such a curve divides the plane into two regions, one bounded and one unbounded. The normal to a Jordan curve is assumed to be in the direction of the unbounded part. * Further, we will assume that our curves are *piecewise smooth*. That is comprised of finitely many smooth pieces, continuously connected. * The region enclosed by a closed curve has an *interior*, $D$, which we assume is an *open* set (one for which every point in $D$ has some "ball" about it entirely within $D$ as well.) * The region $D$ is *connected* meaning between any two points there is a continuous path in $D$ between the two points. @@ -471,7 +471,7 @@ The flow integral is typically computed for a closed (Jordan) curve, measuring t :::{.callout-note} ## Note -For a Jordan curve, the positive orientation of the curve is such that the normal direction (proportional to $\hat{T}'$) points away from the bounded interior. For a non-closed path, the choice of parameterization will determine the normal and the integral for flow across a curve is dependent - up to its sign - on this choice. +For a Jordan curve, the positive orientation of the curve is such that the normal direction (proportional to $\hat{T}'$) points away from the bounded interior. For a non-closed path, the choice of parameterization will determine the normal and the integral for flow across a curve is dependent---up to its sign---on this choice. ::: diff --git a/quarto/integral_vector_calculus/review.qmd b/quarto/integral_vector_calculus/review.qmd index 7e56b1e..b7a538d 100644 --- a/quarto/integral_vector_calculus/review.qmd +++ b/quarto/integral_vector_calculus/review.qmd @@ -1,4 +1,4 @@ -# Quick Review of Vector Calculus +# Quick review of vector calculus {{< include ../_common_code.qmd >}} @@ -133,7 +133,7 @@ $$ $$ -The generalization to $n>2$ is clear - the partial derivative in $x_i$ is the derivative of $f$ when the *other* $x_j$ are held constant. +The generalization to $n>2$ is clear---the partial derivative in $x_i$ is the derivative of $f$ when the *other* $x_j$ are held constant. This may be viewed as the derivative of the univariate function $(f\circ\vec{r})(t)$ where $\vec{r}(t) = p + t \hat{e}_i$, $\hat{e}_i$ being the unit vector of all $0$s except a $1$ in the $i$th component. diff --git a/quarto/integral_vector_calculus/stokes_theorem.qmd b/quarto/integral_vector_calculus/stokes_theorem.qmd index 0901e9f..0ed531e 100644 --- a/quarto/integral_vector_calculus/stokes_theorem.qmd +++ b/quarto/integral_vector_calculus/stokes_theorem.qmd @@ -1,4 +1,4 @@ -# Green's Theorem, Stokes' Theorem, and the Divergence Theorem +# Green's theorem, Stokes' theorem, and the divergence theorem {{< include ../_common_code.qmd >}} @@ -721,13 +721,13 @@ The fluid would flow along the blue (stream) lines. The red lines have equal pot # https://en.wikipedia.org/wiki/Jiffy_Pop#/media/File:JiffyPop.jpg imgfile ="figures/jiffy-pop.png" caption =""" -The Jiffy Pop popcorn design has a top surface that is designed to expand to accommodate the popped popcorn. Viewed as a surface, the surface area grows, but the boundary - where the surface meets the pan - stays the same. This is an example that many different surfaces can have the same bounding curve. Stokes' theorem will relate a surface integral over the surface to a line integral about the bounding curve. +The Jiffy Pop popcorn design has a top surface that is designed to expand to accommodate the popped popcorn. Viewed as a surface, the surface area grows, but the boundary---where the surface meets the pan---stays the same. This is an example that many different surfaces can have the same bounding curve. Stokes' theorem will relate a surface integral over the surface to a line integral about the bounding curve. """ # ImageFile(:integral_vector_calculus, imgfile, caption) nothing ``` -![The Jiffy Pop popcorn design has a top surface that is designed to expand to accommodate the popped popcorn. Viewed as a surface, the surface area grows, but the boundary - where the surface meets the pan - stays the same. This is an example that many different surfaces can have the same bounding curve. Stokes' theorem will relate a surface integral over the surface to a line integral about the bounding curve. +![The Jiffy Pop popcorn design has a top surface that is designed to expand to accommodate the popped popcorn. Viewed as a surface, the surface area grows, but the boundary---where the surface meets the pan---stays the same. This is an example that many different surfaces can have the same bounding curve. Stokes' theorem will relate a surface integral over the surface to a line integral about the bounding curve. ](./figures/jiffy-pop.png) Were the figure of Jiffy Pop popcorn animated, the surface of foil would slowly expand due to pressure of popping popcorn until the popcorn was ready. However, the boundary would remain the same. Many different surfaces can have the same boundary. Take for instance the upper half unit sphere in $R^3$ it having the curve $x^2 + y^2 = 1$ as a boundary curve. This is the same curve as the surface of the cone $z = 1 - (x^2 + y^2)$ that lies above the $x-y$ plane. This would also be the same curve as the surface formed by a Mickey Mouse glove if the collar were scaled and positioned onto the unit circle. @@ -761,7 +761,7 @@ $$ $$ -In terms of our expanding popcorn, the boundary integral - after accounting for cancellations, as in Green's theorem - can be seen as a microscopic sum of boundary integrals each of which is approximated by a term $\nabla\times{F}\cdot\hat{N} \Delta{S}$ which is viewed as a Riemann sum approximation for the the integral of the curl over the surface. The cancellation depends on a proper choice of orientation, but with that we have: +In terms of our expanding popcorn, the boundary integral---after accounting for cancellations, as in Green's theorem---can be seen as a microscopic sum of boundary integrals each of which is approximated by a term $\nabla\times{F}\cdot\hat{N} \Delta{S}$ which is viewed as a Riemann sum approximation for the the integral of the curl over the surface. The cancellation depends on a proper choice of orientation, but with that we have: ::: {.callout-note icon=false} ## Stokes' theorem diff --git a/quarto/integrals/arc_length.qmd b/quarto/integrals/arc_length.qmd index 0e9d159..0753932 100644 --- a/quarto/integrals/arc_length.qmd +++ b/quarto/integrals/arc_length.qmd @@ -557,7 +557,7 @@ Following (faithfully) [Kantorwitz and Neumann](https://www.researchgate.net/pub @fig-kantorwitz-neumann is clearly of a concave down function. The asymmetry about the critical point will be seen to be a result of the derivative also being concave down. This asymmetry will be characterized in several different ways in the following including showing that the arc length from $(a,0)$ to $(c,f(c))$ is longer than from $(c,f(c))$ to $(b,0)$. -::: {#@fig-kantorwitz-neumann} +::: {#fig-kantorwitz-neumann} ```{julia} diff --git a/quarto/integrals/area.qmd b/quarto/integrals/area.qmd index 7882d29..de22c4b 100644 --- a/quarto/integrals/area.qmd +++ b/quarto/integrals/area.qmd @@ -16,7 +16,7 @@ using Roots --- -![A jigsaw puzzle needs a certain amount of area to complete. For a traditional rectangular puzzle, this area is comprised of the sum of the areas for each piece. Decomposing a total area into the sum of smaller, known, ones--even if only approximate--is the basis of definite integration.](figures/jigsaw.png) +![A jigsaw puzzle needs a certain amount of area to complete. For a traditional rectangular puzzle, this area is comprised of the sum of the areas for each piece. Decomposing a total area into the sum of smaller, known, ones---even if only approximate---is the basis of definite integration.](figures/jigsaw.png) The question of area has long fascinated human culture. As children, we learn early on the formulas for the areas of some geometric figures: a square is $b^2$, a rectangle $b\cdot h$, a triangle $1/2 \cdot b \cdot h$ and for a circle, $\pi r^2$. The area of a rectangle is often the intuitive basis for illustrating multiplication. The area of a triangle has been known for ages. Even complicated expressions, such as [Heron's](http://tinyurl.com/mqm9z) formula which relates the area of a triangle with measurements from its perimeter have been around for 2000 years. The formula for the area of a circle is also quite old. Wikipedia dates it as far back as the [Rhind](http://en.wikipedia.org/wiki/Rhind_Mathematical_Papyrus) papyrus for 1700 BC, with the approximation of $256/81$ for $\pi$. @@ -1067,7 +1067,7 @@ plot!(zero) We could add the signed area over $[0,1]$ to the above, but instead see a square of area $1$, a triangle with area $1/2$ and a triangle with signed area $-1$. The total is then $1/2$. -This figure--using equal sized axes--may make the above decomposition more clear: +This figure---using equal sized axes---may make the above decomposition more clear: ```{julia} #| echo: false diff --git a/quarto/integrals/area_between_curves.qmd b/quarto/integrals/area_between_curves.qmd index 471322f..4088acc 100644 --- a/quarto/integrals/area_between_curves.qmd +++ b/quarto/integrals/area_between_curves.qmd @@ -448,7 +448,7 @@ When doing problems by hand this latter style can often reduce the complications Consider two overlapping circles, one with smaller radius. How much area is in the larger circle that is not in the smaller? The question came up on the `Julia` [discourse](https://discourse.julialang.org/t/is-there-package-or-method-to-calculate-certain-area-in-julia-symbolically-with-sympy/99751) discussion board. A solution, modified from an answer of `@rocco_sprmnt21`, follows. Without losing too-much generality, we can consider the smaller circle to have radius $a$, the larger circle to have radius $b$ and centered at $(0,c)$. -We assume some overlap -- $a \ge c-b$, but not too much -- $c-b \ge 0$ or $0 \le c-b \le a$. +We assume some overlap---$a \ge c-b$, but not too much---$c-b \ge 0$ or $0 \le c-b \le a$. ```{julia} @syms x::real y::real a::positive b::positive c::positive diff --git a/quarto/integrals/center_of_mass.qmd b/quarto/integrals/center_of_mass.qmd index e148a5b..b19f409 100644 --- a/quarto/integrals/center_of_mass.qmd +++ b/quarto/integrals/center_of_mass.qmd @@ -1,4 +1,4 @@ -# Center of Mass +# Center of mass {{< include ../_common_code.qmd >}} diff --git a/quarto/integrals/improper_integrals.qmd b/quarto/integrals/improper_integrals.qmd index 2534f08..887be19 100644 --- a/quarto/integrals/improper_integrals.qmd +++ b/quarto/integrals/improper_integrals.qmd @@ -1,4 +1,4 @@ -# Improper Integrals +# Improper integrals {{< include ../_common_code.qmd >}} diff --git a/quarto/integrals/integration_by_parts.qmd b/quarto/integrals/integration_by_parts.qmd index 9271f72..a7e0a5b 100644 --- a/quarto/integrals/integration_by_parts.qmd +++ b/quarto/integrals/integration_by_parts.qmd @@ -1,4 +1,4 @@ -# Integration By Parts +# Integration by parts {{< include ../_common_code.qmd >}} @@ -116,7 +116,7 @@ $$ $B$ is similar with the roles of $u$ and $v$ reversed. ----- +--- Informally, the integration by parts formula is sometimes seen as $\int udv = uv - \int v du$, as well can be somewhat confusingly written as: @@ -382,7 +382,7 @@ Recall, just using *either* $x_i$ or $x_{i-1}$ for $c_i$ gives an error that is This [proof](http://www.math.ucsd.edu/~ebender/20B/77_Trap.pdf) for the error estimate is involved, but is reproduced here, as it nicely integrates many of the theoretical concepts of integration discussed so far. -First, for convenience, we consider the interval $x_i$ to $x_i+h$. The actual answer over this is just $\int_{x_i}^{x_i+h}f(x) dx$. By a $u$-substitution with $u=x-x_i$ this becomes $\int_0^h f(t + x_i) dt$. For analyzing this we integrate once by parts using $u=f(t+x_i)$ and $dv=dt$. But instead of letting $v=t$, we choose to add--as is our prerogative--a constant of integration $A$, so $v=t+A$: +First, for convenience, we consider the interval $x_i$ to $x_i+h$. The actual answer over this is just $\int_{x_i}^{x_i+h}f(x) dx$. By a $u$-substitution with $u=x-x_i$ this becomes $\int_0^h f(t + x_i) dt$. For analyzing this we integrate once by parts using $u=f(t+x_i)$ and $dv=dt$. But instead of letting $v=t$, we choose to add---as is our prerogative---a constant of integration $A$, so $v=t+A$: $$ diff --git a/quarto/integrals/partial_fractions.qmd b/quarto/integrals/partial_fractions.qmd index 0af818f..5a871c5 100644 --- a/quarto/integrals/partial_fractions.qmd +++ b/quarto/integrals/partial_fractions.qmd @@ -1,4 +1,4 @@ -# Partial Fractions +# Partial fractions {{< include ../_common_code.qmd >}} @@ -14,7 +14,7 @@ using SymPy Integration is facilitated when an antiderivative for $f$ can be found, as then definite integrals can be evaluated through the fundamental theorem of calculus. -However, despite differentiation being an algorithmic procedure, integration is not. There are "tricks" to try, such as substitution and integration by parts. These work in some cases--but not all! +However, despite differentiation being an algorithmic procedure, integration is not. There are "tricks" to try, such as substitution and integration by parts. These work in some cases---but not all! However, there are classes of functions for which algorithms exist. For example, the `SymPy` `integrate` function mostly implements an algorithm that decides if an elementary function has an antiderivative. The [elementary](http://en.wikipedia.org/wiki/Elementary_function) functions include exponentials, their inverses (logarithms), trigonometric functions, their inverses, and powers, including $n$th roots. Not every elementary function will have an antiderivative comprised of (finite) combinations of elementary functions. The typical example is $e^{x^2}$, which has no simple antiderivative, despite its ubiquitousness. diff --git a/quarto/integrals/surface_area.qmd b/quarto/integrals/surface_area.qmd index 088dd89..eacde05 100644 --- a/quarto/integrals/surface_area.qmd +++ b/quarto/integrals/surface_area.qmd @@ -1,4 +1,4 @@ -# Surface Area +# Surface area {{< include ../_common_code.qmd >}} diff --git a/quarto/integrals/twelve-qs.qmd b/quarto/integrals/twelve-qs.qmd index f8a9485..947e794 100644 --- a/quarto/integrals/twelve-qs.qmd +++ b/quarto/integrals/twelve-qs.qmd @@ -14,7 +14,7 @@ using LaTeXStrings gr(); ``` ----- +--- In the March 2003 issue of the College Mathematics Journal, Leon M Hall posed 12 questions related to the following figure: @@ -80,7 +80,7 @@ zs = solve(f(x) ~ nl, x) q = only(filter(!=(a), zs)) ``` ----- +--- The first question is simply: @@ -115,7 +115,7 @@ In the remaining examples we don't show the code by default. ::: ----- +--- > 1b. The length of the line segment $PQ$ @@ -133,7 +133,7 @@ lseg = sqrt((f(a) - f(q))^2 + (a - q)^2); ``` ----- +--- > 2a. The horizontal distance between $P$ and $Q$ @@ -151,7 +151,7 @@ plot!([q₀, a₀], [f(a₀), f(a₀)], linewidth=5) hd = a - q; ``` ----- +--- > 2b. The area of the parabolic segment @@ -172,7 +172,7 @@ plot!(xs, ys, fill=(:green, 0.25, 0)) A = simplify(integrate(nl - f(x), (x, q, a))); ``` ----- +--- > 2c. The volume of the rotated solid formed by revolving the parabolic segment around the vertical line $k$ units to the right of $P$ or to the left of $Q$ where $k > 0$. @@ -185,7 +185,7 @@ A = simplify(integrate(nl - f(x), (x, q, a))); V = simplify(integrate(2PI*(nl-f(x))*(a - x + k),(x, q, a))); ``` ----- +--- > 3. The $y$ coordinate of the centroid of the parabolic segment @@ -214,7 +214,7 @@ yₘ = integrate( (1//2) * (nl^2 - f(x)^2), (x, q, a)) / A yₘ = simplify(yₘ); ``` ----- +--- > 4. The length of the arc of the parabola between $P$ and $Q$ @@ -233,7 +233,7 @@ p L = integrate(sqrt(1 + fp(x)^2), (x, q, a)); ``` ----- +--- > 5. The $y$ coordinate of the midpoint of the line segment $PQ$ @@ -254,7 +254,7 @@ p mp = nl(x => (a + q)/2); ``` ----- +--- > 6. The area of the trapezoid bound by the normal line, the $x$-axis, and the vertical lines through $P$ and $Q$. @@ -273,7 +273,7 @@ p trap = 1//2 * (f(q) + f(a)) * (a - q); ``` ----- +--- > 7. The area bounded by the parabola and the $x$ axis and the vertical lines through $P$ and $Q$ @@ -295,7 +295,7 @@ p pa = integrate(x^2, (x, q, a)); ``` ----- +--- > 8. The area of the surface formed by revolving the arc of the parabola between $P$ and $Q$ around the vertical line through $P$ @@ -321,7 +321,7 @@ vv(x) = f(a - uu(x)) SA = 2PI * integrate(uu(x) * sqrt(diff(uu(x),x)^2 + diff(vv(x),x)^2), (x, q, a)); ``` ----- +--- > 9. The height of the parabolic segment (i.e. the distance between the normal line and the tangent line to the parabola that is parallel to the normal line) @@ -350,7 +350,7 @@ segment_height = sqrt((b-b′)^2 + (f(b) - nl(x=>b′))^2); ``` ----- +--- > 10. The volume of the solid formed by revolving the parabolic segment around the $x$-axis @@ -371,7 +371,7 @@ end Vₓ = integrate(pi * (nl^2 - f(x)^2), (x, q, a)); ``` ----- +--- > 11. The area of the triangle bound by the normal line, the vertical line through $Q$ and the $x$-axis @@ -392,7 +392,7 @@ plot!([p₀,q₀,q₀,p₀], [0,f(q₀),0,0]; triangle = 1/2 * f(q) * (a - f(a)/(-1/fp(a)) - q); ``` ----- +--- > 12. The area of the quadrilateral bound by the normal line, the tangent line, the vertical line through $Q$ and the $x$-axis @@ -417,7 +417,7 @@ x₁,x₂,x₃,x₄ = (a,q,q,tl₀) y₁, y₂, y₃, y₄ = (f(a), f(q), 0, 0) quadrilateral = (x₁ - x₂)*(y₁ - y₃)/2 - (x₁ - x₃)*(y₁ - y₂)/2 + (x₁ - x₃)*(y₁ - y₄)/2 - (x₁ - x₄)*(y₁ - y₃)/2; ``` ----- +--- The answers appear here in sorted order, some given as approximate floating point values: diff --git a/quarto/limits/intermediate_value_theorem.qmd b/quarto/limits/intermediate_value_theorem.qmd index b5b3288..8aa9ab9 100644 --- a/quarto/limits/intermediate_value_theorem.qmd +++ b/quarto/limits/intermediate_value_theorem.qmd @@ -315,7 +315,7 @@ find_zero(q, (5, 10)) ::: {.callout-note} ### Between need not be near -Later, we will see more efficient algorithms to find a zero *near* a given guess. The bisection method finds a zero *between* two values of a bracketing interval. This interval need not be small. Indeed in many cases it can be infinite. For this particular problem, any interval like `(2,N)` will work as long as `N` is bigger than the zero and small enough that `q(N)` is finite *or* infinite *but* not `NaN`. (Basically, `q` must evaluate to a number with a sign. Here, the value of `q(Inf)` is `NaN` as it evaluates to the indeterminate `Inf - Inf`. But `q` is still not `NaN` for quite large numbers, such as `1e77`, as `x^4` can as big as `1e308` -- technically `floatmax(Float64)` -- and be finite.) +Later, we will see more efficient algorithms to find a zero *near* a given guess. The bisection method finds a zero *between* two values of a bracketing interval. This interval need not be small. Indeed in many cases it can be infinite. For this particular problem, any interval like `(2,N)` will work as long as `N` is bigger than the zero and small enough that `q(N)` is finite *or* infinite *but* not `NaN`. (Basically, `q` must evaluate to a number with a sign. Here, the value of `q(Inf)` is `NaN` as it evaluates to the indeterminate `Inf - Inf`. But `q` is still not `NaN` for quite large numbers, such as `1e77`, as `x^4` can as big as `1e308`---technically `floatmax(Float64)`---and be finite.) ::: @@ -840,7 +840,7 @@ plotly() nothing ``` -Figure illustrating absolute and relative minima for a function $f(x)$ over $I=[a,b]$. The leftmost point has a $y$ value, $f(a)$, which is an absolute maximum of $f(x)$ over $I$. The three points highlighted between $a$ and $b$ are all relative extrema. The first one is *also* the absolute minimum over $I$. The endpoint is not considered a relative maximum for technical reasons --- there is no interval around $b$, it being on the boundary of $I$. +Figure illustrating absolute and relative minima for a function $f(x)$ over $I=[a,b]$. The leftmost point has a $y$ value, $f(a)$, which is an absolute maximum of $f(x)$ over $I$. The three points highlighted between $a$ and $b$ are all relative extrema. The first one is *also* the absolute minimum over $I$. The endpoint is not considered a relative maximum for technical reasons---there is no interval around $b$, it being on the boundary of $I$. ::: diff --git a/quarto/limits/limits.qmd b/quarto/limits/limits.qmd index 84a256b..7b170e9 100644 --- a/quarto/limits/limits.qmd +++ b/quarto/limits/limits.qmd @@ -689,7 +689,7 @@ c = 15/11 lim(h, c; n = 16) ``` -(Though the graph and table do hint at something a bit odd -- the graph shows a blip, the table doesn't show values in the second column going towards a specific value.) +(Though the graph and table do hint at something a bit odd---the graph shows a blip, the table doesn't show values in the second column going towards a specific value.) However the limit in this case is $-\infty$ (or DNE), as there is an aysmptote at $c=15/11$. The problem is the asymptote due to the logarithm is extremely narrow and happens between floating point values to the left and right of $15/11$. diff --git a/quarto/precalc.qmd b/quarto/precalc.qmd index 7e0cfb5..04b2deb 100644 --- a/quarto/precalc.qmd +++ b/quarto/precalc.qmd @@ -1,7 +1,3 @@ ---- -engine: julia ---- - # Precalculus Concepts The mathematical topics in this chapter come from pre-calculus. However, much of the `Julia` usage needed for the rest of the notes are introduced. diff --git a/quarto/precalc/exp_log_functions.qmd b/quarto/precalc/exp_log_functions.qmd index d3e4694..eb5750b 100644 --- a/quarto/precalc/exp_log_functions.qmd +++ b/quarto/precalc/exp_log_functions.qmd @@ -22,7 +22,7 @@ The family of exponential functions is used to model growth and decay. The famil ## Exponential functions -The family of exponential functions is defined by $f(x) = a^x, -\infty< x < \infty$ and $a > 0$. For $0 < a < 1$ these functions decay or decrease, for $a > 1$ the functions grow or increase, and if $a=1$ the function is constantly $1$. +The family of exponential functions is defined by $f(x) = a^x, -\infty< x < \infty$ and $a > 0$. For $0 < a < 1$ these functions decay or decrease, for $a > 1$ these functions grow or increase, and if $a=1$ the function is constantly $1$. For a given $a$, defining $a^n$ for positive integers is straightforward, as it means multiplying $n$ copies of $a.$ From this, for *integer powers*, the key properties of exponents: $a^x \cdot a^y = a^{x+y}$, and $(a^x)^y = a^{x \cdot y}$ are immediate consequences. For example with $x=3$ and $y=2$: @@ -114,7 +114,7 @@ t2, t8 = 72/2, 72/8 exp(r2*t2), exp(r8*t8) ``` -So fairly close - after $72/r$ years the amount is $2.05...$ times more than the initial amount. +So fairly close---after $72/r$ years the amount is $2.05...$ times more than the initial amount. ##### Example @@ -259,7 +259,7 @@ The inverse function will solve for $x$ in the equation $a^x = y$. The answer, f That is $a^{\log_a(x)} = x$ for $x > 0$ and $\log_a(a^x) = x$ for all $x$. -To see how a logarithm is mathematically defined will have to wait, though the family of functions - one for each $a>0$ - are implemented in `Julia` through the function `log(a,x)`. There are special cases requiring just one argument: `log(x)` will compute the natural log, base $e$ - the inverse of $f(x) = e^x$; `log2(x)` will compute the log base $2$ - the inverse of $f(x) = 2^x$; and `log10(x)` will compute the log base $10$ - the inverse of $f(x)=10^x$. (Also `log1p` computes an accurate value of $\log(1 + p)$ when $p \approx 0$.) +To see how a logarithm is mathematically defined will have to wait, though the family of functions---one for each $a>0$---are implemented in `Julia` through the function `log(a,x)`. There are special cases requiring just one argument: `log(x)` will compute the natural log, base $e$---the inverse of $f(x) = e^x$; `log2(x)` will compute the log base $2$---the inverse of $f(x) = 2^x$; and `log10(x)` will compute the log base $10$- the inverse of $f(x)=10^x$. (Also `log1p` computes an accurate value of $\log(1 + p)$ when $p \approx 0$.) To see this in an example, we plot for base $2$ the exponential function $f(x)=2^x$, its inverse, and the logarithm function with base $2$: @@ -398,7 +398,7 @@ $$ ##### Example -Before the ubiquity of electronic calculating devices, the need to compute was still present. Ancient civilizations had abacuses to make addition easier. For multiplication and powers a [slide rule](https://en.wikipedia.org/wiki/Slide_rule) could be used. It is easy to represent addition physically with two straight pieces of wood - just represent a number with a distance and align the two pieces so that the distances are sequentially arranged. To multiply then was as easy: represent the logarithm of a number with a distance then add the logarithms. The sum of the logarithms is the logarithm of the *product* of the original two values. Converting back to a number answers the question. The conversion back and forth is done by simply labeling the wood using a logartithmic scale. The slide rule was [invented](http://tinyurl.com/qytxo3e) soon after Napier's initial publication on the logarithm in 1614. +Before the ubiquity of electronic calculating devices, the need to compute was still present. Ancient civilizations had abacuses to make addition easier. For multiplication and powers a [slide rule](https://en.wikipedia.org/wiki/Slide_rule) could be used. It is easy to represent addition physically with two straight pieces of wood---just represent a number with a distance and align the two pieces so that the distances are sequentially arranged. To multiply then was as easy: represent the logarithm of a number with a distance then add the logarithms. The sum of the logarithms is the logarithm of the *product* of the original two values. Converting back to a number answers the question. The conversion back and forth is done by simply labeling the wood using a logartithmic scale. The slide rule was [invented](http://tinyurl.com/qytxo3e) soon after Napier's initial publication on the logarithm in 1614. ##### Example diff --git a/quarto/precalc/functions.qmd b/quarto/precalc/functions.qmd index cb83721..c647dbd 100644 --- a/quarto/precalc/functions.qmd +++ b/quarto/precalc/functions.qmd @@ -240,13 +240,13 @@ f(x) = 5/9 * (x - 32) f(72) ## room temperature ``` -will create a function object with a value of `x` determined at a later time - the time the function is called. So the value of `x` defined when the function is created is not important here (as the value of `x` used by `f` is passed in as an argument). +will create a function object with a value of `x` determined at a later time---the time the function is called. So the value of `x` defined when the function is created is not important here (as the value of `x` used by `f` is passed in as an argument). Within `Julia`, we make note of the distinction between a function object versus a function call. In the definition `f(x)=cos(x)`, the variable `f` refers to a function object, whereas the expression `f(pi)` is a function call, resulting in a value. This mirrors the math notation where an $f$ is used when properties of a function are being emphasized (such as $f \circ g$ for composition) and $f(x)$ is used when the values related to the function are being emphasized (such as saying "the plot of the equation $y=f(x)$). -Distinguishing these related but different concepts --- expressions, equations, values from function calls, and function objects --- is important when modeling mathematics on the computer. +Distinguishing these related but different concepts---expressions, equations, values from function calls, and function objects---is important when modeling mathematics on the computer. ::: {#fig-kidney} @@ -315,7 +315,7 @@ s(x) = \end{cases} $$ -We learn to read this as: when $x$ is less than $0$, then the answer is $-1$. If $x$ is greater than $0$ the answer is $1.$ Often--but not in this example--there is an "otherwise" case to catch those values of $x$ that are not explicitly mentioned. As there is no such "otherwise" case here, we can see that this function has no definition when $x=0$. This function is often called the "sign" function and is also defined by $\lvert x\rvert/x$. (`Julia`'s `sign` function defines `sign(0)` to be `0`.) +We learn to read this as: when $x$ is less than $0$, then the answer is $-1$. If $x$ is greater than $0$ the answer is $1.$ Often---but not in this example---there is an "otherwise" case to catch those values of $x$ that are not explicitly mentioned. As there is no such "otherwise" case here, we can see that this function has no definition when $x=0$. This function is often called the "sign" function and is also defined by $\lvert x\rvert/x$. (`Julia`'s `sign` function defines `sign(0)` to be `0`.) How do we create conditional statements in `Julia`? Programming languages generally have "if-then-else" constructs to handle conditional evaluation. In `Julia`, the following code will handle the above condition: @@ -357,7 +357,7 @@ For example, here is one way to define an absolute value function: abs_val(x) = x >= 0 ? x : -x ``` -The condition is `x >= 0`--or is `x` non-negative? If so, the value `x` is used, otherwise `-x` is used. +The condition is `x >= 0`---or is `x` non-negative? If so, the value `x` is used, otherwise `-x` is used. Here is a means to implement a function which takes the larger of `x` or `10`: @@ -672,7 +672,7 @@ During this call, values for `m` and `b` are found from how the function is call mxplusb(0; m=3, b=2) ``` -Keywords are used to mark the parameters whose values are to be changed from the default. Though one can use *positional arguments* for parameters--and there are good reasons to do so--using keyword arguments is a good practice if performance isn't paramount, as their usage is more explicit yet the defaults mean that a minimum amount of typing needs to be done. +Keywords are used to mark the parameters whose values are to be changed from the default. Though one can use *positional arguments* for parameters---and there are good reasons to do so---using keyword arguments is a good practice if performance isn't paramount, as their usage is more explicit yet the defaults mean that a minimum amount of typing needs to be done. Keyword arguments are widely used with plotting commands, as there are numerous options to adjust, but typically only a handful adjusted per call. The `Plots` package whose commands we illustrate throughout these notes starting with the next section has this in its docs: `Plots.jl` follows two simple rules with data and attributes: @@ -726,7 +726,7 @@ The style isn't so different from using keyword arguments, save the extra step o v0, theta ``` -The *big* advantage of bundling parameters into a container is consistency – the function is always called in an identical manner regardless of the number of parameters (or variables). +The *big* advantage of bundling parameters into a container is consistency-–-the function is always called in an identical manner regardless of the number of parameters (or variables). ::: {.callout-note} @@ -749,13 +749,13 @@ Volume(r, h) = pi * r^2 * h # of a cylinder SurfaceArea(r, h) = pi * r * (r + sqrt(h^2 + r^2)) # of a right circular cone, including the base ``` -The right-hand sides may or may not be familiar, but it should be reasonable to believe that if push came to shove, the formulas could be looked up. However, the left-hand sides are subtly different - they have two arguments, not one. In `Julia` it is trivial to define functions with multiple arguments - we just did. +The right-hand sides may or may not be familiar, but it should be reasonable to believe that if push came to shove, the formulas could be looked up. However, the left-hand sides are subtly different---they have two arguments, not one. In `Julia` it is trivial to define functions with multiple arguments---we just did. Earlier we saw the `log` function can use a second argument to express the base. This function is basically defined by `log(b,x)=log(x)/log(b)`. The `log(x)` value is the natural log, and this definition just uses the change-of-base formula for logarithms. -But not so fast, on the left side is a function with two arguments and on the right side the functions have one argument--yet they share the same name. How does `Julia` know which to use? `Julia` uses the number, order, and *type* of the positional arguments passed to a function to determine which function definition to use. This is technically known as [multiple dispatch](http://en.wikipedia.org/wiki/Multiple_dispatch) or **polymorphism**. As a feature of the language, it can be used to greatly simplify the number of functions the user must learn. The basic idea is that many functions are "generic" in that they have methods which will work differently in different scenarios. +But not so fast, on the left side is a function with two arguments and on the right side the functions have one argument---yet they share the same name. How does `Julia` know which to use? `Julia` uses the number, order, and *type* of the positional arguments passed to a function to determine which function definition to use. This is technically known as [multiple dispatch](http://en.wikipedia.org/wiki/Multiple_dispatch) or **polymorphism**. As a feature of the language, it can be used to greatly simplify the number of functions the user must learn. The basic idea is that many functions are "generic" in that they have methods which will work differently in different scenarios. :::{.callout-warning} @@ -785,7 +785,7 @@ twotox(x::Real) = (2.0)^x twotox(x::Complex) = (2.0 + 0.0im)^x ``` -This is for illustration purposes -- the latter two are actually already done through `Julia`'s *promotion* mechanism -- but we see that `twotox` will return a rational number when `x` is an integer unlike `Julia` which, when `x` is non-negative will return an integer and will otherwise will error or return a float (when `x` is a numeric literal, like `2^(-3)`). +This is for illustration purposes---the latter two are actually already done through `Julia`'s *promotion* mechanism---but we see that `twotox` will return a rational number when `x` is an integer unlike `Julia` which, when `x` is non-negative will return an integer and will otherwise will error or return a float (when `x` is a numeric literal, like `2^(-3)`). The key to reading the above is the type annotation acts like a gatekeeper allowing in only variables of that type or a subtype of that type. @@ -811,7 +811,7 @@ Representing the area of a rectangle in terms of two variables is easy, as the f Area(w, h) = w * h ``` -But the other fact about this problem--that the perimeter is $20$--means that height depends on width. For this question, we can see that $P=2w + 2h$ so that--as a function--`height` depends on `w` as follows: +But the other fact about this problem---that the perimeter is $20$---means that height depends on width. For this question, we can see that $P=2w + 2h$ so that---as a function---`height` depends on `w` as follows: ```{julia} @@ -858,7 +858,7 @@ $$ g(x) = f(x-c) $$ -has an interpretation - the graph of $g$ will be the same as the graph of $f$ shifted to the right by $c$ units. That is $g$ is a transformation of $f$. From one perspective, the act of replacing $x$ with $x-c$ transforms a function into a new function. Mathematically, when we focus on transforming functions, the word [operator](http://en.wikipedia.org/wiki/Operator_%28mathematics%29) is sometimes used. This concept of transforming a function can be viewed as a certain type of function, in an abstract enough way. The relation would be to just pair off the functions $(f,g)$ where $g(x) = f(x-c)$. +has an interpretation---the graph of $g$ will be the same as the graph of $f$ shifted to the right by $c$ units. That is $g$ is a transformation of $f$. From one perspective, the act of replacing $x$ with $x-c$ transforms a function into a new function. Mathematically, when we focus on transforming functions, the word [operator](http://en.wikipedia.org/wiki/Operator_%28mathematics%29) is sometimes used. This concept of transforming a function can be viewed as a certain type of function, in an abstract enough way. The relation would be to just pair off the functions $(f,g)$ where $g(x) = f(x-c)$. With `Julia` we can represent such operations. The simplest thing would be to do something like: @@ -881,7 +881,7 @@ function shift_right(f; c=0) end ``` -That takes some parsing. In the body of the `shift_right` is the definition of a function. But this function has no name–-it is *anonymous*. But what it does should be clear--it subtracts $c$ from $x$ and evaluates $f$ at this new value. Since the last expression creates a function, this function is returned by `shift_right`. +That takes some parsing. In the body of the `shift_right` is the definition of a function. But this function has no name–-it is *anonymous*. But what it does should be clear---it subtracts $c$ from $x$ and evaluates $f$ at this new value. Since the last expression creates a function, this function is returned by `shift_right`. So we could have done something more complicated like: diff --git a/quarto/precalc/inversefunctions.qmd b/quarto/precalc/inversefunctions.qmd index 8e0eb68..51dca2d 100644 --- a/quarto/precalc/inversefunctions.qmd +++ b/quarto/precalc/inversefunctions.qmd @@ -1,4 +1,4 @@ -# The Inverse of a Function +# The inverse of a function {{< include ../_common_code.qmd >}} @@ -25,7 +25,7 @@ We may conceptualize such a relation in many ways: * through a description of what $f$ does; * or through a table of paired values, say. -For the moment, let's consider a function as a rule that takes in a value of $x$ and outputs a value $y$. If a rule is given defining the function, the computation of $y$ is straightforward. A different question is not so easy: for a given value $y$ what value--or *values*--of $x$ (if any) produce an output of $y$? That is, what $x$ value(s) satisfy $f(x)=y$? +For the moment, let's consider a function as a rule that takes in a value of $x$ and outputs a value $y$. If a rule is given defining the function, the computation of $y$ is straightforward. A different question is not so easy: for a given value $y$ what value---or *values*---of $x$ (if any) produce an output of $y$? That is, what $x$ value(s) satisfy $f(x)=y$? *If* for each $y$ in some set of values there is just one $x$ value, then this operation associates to each value $y$ a single value $x$, so it too is a function. When that is the case we call this an *inverse* function. @@ -202,7 +202,7 @@ In the section on the [intermediate value theorem](../limits/intermediate_value_ ## Functions which are not always invertible -Consider the function $f(x) = x^2$. The graph--a parabola--is clearly not *monotonic*. Hence no inverse function exists. Yet, we can solve equations $y=x^2$ quite easily: $y=\sqrt{x}$ *or* $y=-\sqrt{x}$. We know the square root undoes the squaring, but we need to be a little more careful to say the square root is the inverse of the squaring function. +Consider the function $f(x) = x^2$. The graph---a parabola---is clearly not *monotonic*. Hence no inverse function exists. Yet, we can solve equations $y=x^2$ quite easily: $y=\sqrt{x}$ *or* $y=-\sqrt{x}$. We know the square root undoes the squaring, but we need to be a little more careful to say the square root is the inverse of the squaring function. The issue is there are generally *two* possible answers. To avoid this, we might choose to only take the *non-negative* answer. To make this all work as above, we restrict the domain of $f(x)$ and now consider the related function $f(x)=x^2, x \geq 0$. This is now a monotonic function, so will have an inverse function. This is clearly $f^{-1}(x) = \sqrt{x}$. (The $\sqrt{x}$ being defined as the principle square root or the unique *non-negative* answer to $u^2-x=0$.) @@ -287,7 +287,7 @@ plot(xs, ys; color=:blue, label="f", plot!(ys, xs; color=:red, label="f⁻¹") # the inverse ``` -By flipping around the $x$ and $y$ values in the `plot!` command, we produce the graph of the inverse function--when viewed as a function of $x$. We can see that the domain of the inverse function (in red) is clearly different from that of the function (in blue). +By flipping around the $x$ and $y$ values in the `plot!` command, we produce the graph of the inverse function---when viewed as a function of $x$. We can see that the domain of the inverse function (in red) is clearly different from that of the function (in blue). The inverse function graph can be viewed as a symmetry of the graph of the function. Flipping the graph for $f(x)$ around the line $y=x$ will produce the graph of the inverse function: Here we see for the graph of $f(x) = x^{1/3}$ and its inverse function: diff --git a/quarto/precalc/julia_overview.qmd b/quarto/precalc/julia_overview.qmd index 86adc26..880206a 100644 --- a/quarto/precalc/julia_overview.qmd +++ b/quarto/precalc/julia_overview.qmd @@ -11,6 +11,8 @@ using CalculusWithJulia nothing ``` + + The [`Julia`](http://www.julialang.org) programming language is well suited as a computer accompaniment while learning the concepts of calculus. The following overview covers the language-specific aspects of the pre-calculus part of the [Calculus with Julia](calculuswithjulia.github.io) notes. @@ -34,35 +36,29 @@ The [https://mybinder.org/](https://mybinder.org/) service in particular allows [Google colab](https://colab.research.google.com/) offers a free service with more computing power than `binder`, though setup is a bit more fussy. To use `colab` along with these notes, you need to execute a command that downloads `Julia` and installs the `CalculusWithJulia` package and a plotting package. (Modify the `pkg"add ..."` command to add other desired packages; update the julia version as necessary): -``` -# Installation cell -%%capture -%%shell -if ! command -v julia 3>&1 > /dev/null -then - wget -q 'https://julialang-s3.julialang.org/bin/linux/x64/1.10/julia-1.10.2-linux-x86_64.tar.gz' \ - -O /tmp/julia.tar.gz - tar -x -f /tmp/julia.tar.gz -C /usr/local --strip-components 1 - rm /tmp/julia.tar.gz -fi -julia -e 'using Pkg; pkg"add IJulia CalculusWithJulia; precompile;"' -julia -e 'using Pkg; Pkg.add(url="https://github.com/mth229/BinderPlots.jl")' -echo 'Now change the runtime type' -``` - (The `BinderPlots` is a light-weight, barebones, plotting package that uses `PlotlyLight` to render graphics with commands mostly following those of the `Plots` package. Though suitable for most examples herein, the `Plots` package could instead be installed) +> Go to google colab: + +[https://colab.research.google.com/](https://colab.research.google.com/) -After this executes (which can take quite some time, as in a few minutes) under the `Runtime` menu select `Change runtime type` and then select `Julia`. +> Click on "Runtime" menu and then "Change Runtime Type" -After that, in a cell execute these commands to load the two installed packages: +> Select Julia as the "Runtime Type" then save + + +> Copy and paste then run this set of commands ``` +using Pkg +Pkg.add("Plots") +Pkg.add("CalculusWithJulia") using CalculusWithJulia -using BinderPlots +using Plots ``` -As mentioned, other packages can be chosen for installation. + +This may take 2-3 minutes to load. The `plotly()` backend doesn't work out of the box. Use `gr()` to recover if that command is issued. @@ -85,7 +81,7 @@ $ julia (_) | (_) (_) | _ _ _| |_ __ _ | Type "?" for help, "]?" for Pkg help. | | | | | | |/ _` | | - | | |_| | | | (_| | | Version 1.11.1 (2024-10-16) + | | |_| | | | (_| | | Version 1.11.6 (2025-07-09) _/ |\__'_|_|_|\__'_| | Official https://julialang.org/ release |__/ | @@ -452,7 +448,7 @@ a = 4 f(3) # now 2 * 4 + 3 ``` -User-defined functions can have $0$, $1$ or more positional arguments: +User-defined functions can have $0$, $1$, or more positional arguments: ```{julia} @@ -573,7 +569,7 @@ With `Plots` loaded, we can plot a function by passing the function object by na ```{julia} -plot(sin, 0, 2pi) # plot a function - by name - over an interval [a,b] +plot(sin, 0, 2pi) ``` ::: {.callout-note} @@ -629,7 +625,7 @@ ys = f.(xs) plot(f, a, b) # recipe for a function plot(xs, f) # alternate recipe plot(xs, ys) # plot coordinates as two vectors -plot([(x,f(x)) for x in xs]) # plot a vector o points +plot([(x,f(x)) for x in xs]) # plot a vector of points ``` The choice should depend on convenience. @@ -637,7 +633,7 @@ The choice should depend on convenience. ## Equations -Notation for `Julia` and math is *similar* for functions - but not for equations. In math, an equation might look like: +Notation for `Julia` and math is *similar* for functions---but not for equations. In math, an equation might look like: $$ @@ -664,13 +660,15 @@ using SymPy (A macro rewrites values into other commands before they are interpreted. Macros are prefixed with the `@` sign. In this use, the "macro" `@syms` translates `x a b c` into a command involving `SymPy`s `symbols` function.) -Symbolic expressions - unlike numeric expressions - are not immediately evaluated, though they may be simplified: +Symbolic expressions---unlike numeric expressions---are not immediately evaluated, though they may be simplified: ```{julia} p = a*x^2 + b*x + c ``` +The above command illustrates that the mathematical operations of `*`, `^`, and `+` work with symbolic objects. This is the case for most mathematical functions as well. + To substitute a value, we can use `Julia`'s `pair` notation (`variable=>value`): diff --git a/quarto/precalc/make_pdf.jl b/quarto/precalc/make_pdf.jl index e9bcc30..8282316 100644 --- a/quarto/precalc/make_pdf.jl +++ b/quarto/precalc/make_pdf.jl @@ -2,13 +2,7 @@ module Make # makefile for generating typst pdfs # per directory usage dir = "precalc" -files = ("calculator", - "variables", - "numbers_types", - "logical_expressions", - "vectors", - "ranges", - "functions", +files = ("functions", "plotting", "transformations", "inversefunctions", diff --git a/quarto/precalc/plotting.qmd b/quarto/precalc/plotting.qmd index ae57f5a..cbf56bc 100644 --- a/quarto/precalc/plotting.qmd +++ b/quarto/precalc/plotting.qmd @@ -1,4 +1,4 @@ -# The Graph of a Function +# The graph of a function {{< include ../_common_code.qmd >}} @@ -254,10 +254,10 @@ nothing ----- +--- -Making a graph with `Plots` is easy, but producing a graph that is informative can be a challenge, as the choice of a viewing window can make a big difference in what is seen. For example, trying to make a graph of $f(x) = \tan(x)$, as below, will result in a bit of a mess - the chosen viewing window crosses several places where the function blows up: +Making a graph with `Plots` is easy, but producing a graph that is informative can be a challenge, as the choice of a viewing window can make a big difference in what is seen. For example, trying to make a graph of $f(x) = \tan(x)$, as below, will result in a bit of a mess---the chosen viewing window crosses several places where the function blows up: ```{julia} @@ -536,16 +536,16 @@ The `Plots` package uses positional arguments for input data and keyword argumen The `Plots` package provides many such arguments for adjusting a graphic, here we mention just a few: - * `plot(...; title="main title", xlab="x axis label", ylab="y axis label")`: add title and label information to a graphic - * `plot(...; color="green")`: this argument can be used to adjust the color of the drawn figure (color can be a string,`"green"`, or a symbol, `:green`, among other specifications) - * `plot(...; linewidth=5)`: this argument can be used to adjust the width of drawn lines - * `plot(...; linestyle=:dash)`: will change the line style of the plotted lines to dashed lines. Also `:dot`, ... + * `plot(...; title="main title", xlabel="x axis label", ylabel="y axis label")`: add title and label information to a graphic * `plot(...; label="a label")` the `label` attribute will show up when a legend is present. Using an empty string, `""`, will suppress add the layer to the legend. * `plot(...; legend=false)`: by default, different layers will be indicated with a legend, this will turn off this feature * `plot(...; xlims=(a,b), ylims=(c,d))`: either or both `xlims` and `ylims` can be used to control the viewing window * `plot(...; xticks=[xs..], yticks=[ys...]: either or both `xticks` and `yticks` can be used to specify where the tick marks are to be drawn * `plot(...; aspect_ratio=:equal)`: will keep $x$ and $y$ axis on same scale so that squares look square. * `plot(...; framestyle=:origin)`: The default `framestyle` places $x$-$y$ guides on the edges; this specification places them on the $x-y$ plane. + * `plot(...; color="green")`: this argument can be used to adjust the color of the drawn figure (color can be a string,`"green"`, or a symbol, `:green`, among other specifications) + * `plot(...; linewidth=5)`: this argument can be used to adjust the width of drawn lines + * `plot(...; linestyle=:dash)`: will change the line style of the plotted lines to dashed lines. Also `:dot`, ... For plotting points with `scatter`, or `scatter!` the markers can be adjusted via @@ -583,7 +583,7 @@ With these assumptions, we have an initial decision to make: We re-express our equation $y=f(x)= mx+b$ in general form $f(x,y) = 0 = Ax + By + C$. Using the other point on the line $A=-(y_1-y_0)$, $B=(x_1-x_0)$, and $C = -x_1y_0 + x_0 y_1$. In particular, by assumption both $A$ and $B$ are positive. -With this, we have $f(x_0,y_0) = 0$. But moreover, any point with $y>y_0$ will have $f(x_0,y)>0$ and if $y < y_0$ the opposite. That is this equation divides the plane into two pieces depending on whether $f$ is positive, the line is the dividing boundary. +With this, we have $f(x_0,y_0) = 0$. But moreover, any point $(x_0,y)$ with $y>y_0$ will have $f(x_0,y)>0$ and if $y < y_0$ the opposite. That is this equation divides the plane into two pieces depending on whether $f$ is positive---the line is the dividing boundary. For the algorithm, we start at $(x_0, y_0)$ and ask if the pixel $(x_0 + 1, y_0)$ or $(x_0 + 1, y_0 - 1)$ will be lit, then we continue to the right. @@ -637,7 +637,7 @@ Two basic objects to graph are points and lines. Add to these polygons. A point in two-dimensional space has two coordinates, often denoted by $(x,y)$. In `Julia`, the same notation produces a `tuple`. Using square brackets, as in `[x,y]`, produces a vector. Vectors are are more commonly used in these notes, as we have seen there are algebraic operations defined for them. However, tuples have other advantages and are how `Plots` designates a point. -The plot command `plot(xs, ys)` plots the points $(x_1,y_1), \dots, (x_n, y_n)$ and then connects adjacent points with with lines. The command `scatter(xs, ys)` just plots the points. +The plot command `plot(xs, ys)` plots the points $(x_1,y_1), \dots, (x_n, y_n)$ and then connects adjacent points with lines. The command `scatter(xs, ys)` just plots the points. However, the points might be more naturally specified as coordinate pairs. If tuples are used to pair them off, then `Plots` will plot a vector of tuples as a sequence of points through `plot([(x1,y1), (x2, y2), ..., (xn, yn)])`: @@ -711,7 +711,7 @@ scatter!(f.(θs), g.(θs)) --- -As with the plot of a univariate function, there is a convenience interface for these plots - just pass the two functions in: +As with the plot of a univariate function, there is a convenience interface for these plots---just pass the two functions in: ```{julia} @@ -777,8 +777,9 @@ Playing with the toy makes a few things become clear: These all apply to parametric plots, as the Etch A Sketch trace is no more than a plot of $(f(t), g(t))$ over some range of values for $t$, where $f$ describes the movement in time of the left knob and $g$ the movement in time of the right. +--- -Now, we revisit the last problem in the context of this. We saw in the last problem that the parametric graph was nearly a line - so close the eye can't really tell otherwise. That means that the growth in both $f(t) = t^3$ and $g(t)=t - \sin(t)$ for $t$ around $0$ are in a nearly fixed ratio, as otherwise the graph would have more curve in it. +Now, we revisit the last problem in the context of this. We saw in the last problem that the parametric graph was nearly a line---so close the eye can't really tell otherwise. That means that the growth in both $f(t) = t^3$ and $g(t)=t - \sin(t)$ for $t$ around $0$ are in a nearly fixed ratio, as otherwise the graph would have more curve in it. ##### Example: Spirograph @@ -1136,14 +1137,3 @@ choices = [ answ = 5 radioq(choices, answ, keep_order=true) ``` - ---- - - -## Technical note - - -The slow "time to first plot" in `Julia` is a well-known hiccup that is related to how `Julia` can be so fast. Loading Plots and the making the first plot are both somewhat time consuming, though the second and subsequent plots are speedy. Why? - - -`Julia` is an interactive language that attains its speed by compiling functions on the fly using the [llvm](llvm.org) compiler. When `Julia` encounters a new combination of a function method and argument types it will compile and cache a function for subsequent speedy execution. The first plot is slow, as there are many internal functions that get compiled. This has sped up of late, as excessive recompilations have been trimmed down, but still has a way to go. This is different from "precompilation" which also helps trim down time for initial executions. There are also some more technically challenging means to create `Julia` images for faster start up that can be pursued if needed. diff --git a/quarto/precalc/polynomial.qmd b/quarto/precalc/polynomial.qmd index 4efbac6..8e0b27b 100644 --- a/quarto/precalc/polynomial.qmd +++ b/quarto/precalc/polynomial.qmd @@ -614,7 +614,7 @@ It is easy to create a symbolic expression from a function - just evaluate the f f(x) ``` -This is easy--but can also be confusing. The function object is `f`, the expression is `f(x)`--the function evaluated on a symbolic object. Moreover, as seen, the symbolic expression can be evaluated using the same syntax as a function call: +This is easy---but can also be confusing. The function object is `f`, the expression is `f(x)`---the function evaluated on a symbolic object. Moreover, as seen, the symbolic expression can be evaluated using the same syntax as a function call: ```{julia} diff --git a/quarto/precalc/polynomial_roots.qmd b/quarto/precalc/polynomial_roots.qmd index 6c00148..3c84918 100644 --- a/quarto/precalc/polynomial_roots.qmd +++ b/quarto/precalc/polynomial_roots.qmd @@ -448,12 +448,12 @@ To get the numeric approximation, we can broadcast: N.(solveset(p ~ 0, x)) ``` -(There is no need to call `collect` -- though you can -- as broadcasting over a set falls back to broadcasting over the iteration of the set and in this case returns a vector.) +(There is no need to call `collect`---though you can---as broadcasting over a set falls back to broadcasting over the iteration of the set and in this case returns a vector.) ## Do numeric methods matter when you can just graph? -It may seem that certain practices related to roots of polynomials are unnecessary as we could just graph the equation and look for the roots. This feeling is perhaps motivated by the examples given in textbooks to be worked by hand, which necessarily focus on smallish solutions. But, in general, without some sense of where the roots are, an informative graph itself can be hard to produce. That is, technology doesn't displace thinking--it only supplements it. +It may seem that certain practices related to roots of polynomials are unnecessary as we could just graph the equation and look for the roots. This feeling is perhaps motivated by the examples given in textbooks to be worked by hand, which necessarily focus on smallish solutions. But, in general, without some sense of where the roots are, an informative graph itself can be hard to produce. That is, technology doesn't displace thinking---it only supplements it. For another example, consider the polynomial $(x-20)^5 - (x-20) + 1$. In this form we might think the roots are near $20$. However, were we presented with this polynomial in expanded form: $x^5 - 100x^4 + 4000x^3 - 80000x^2 + 799999x - 3199979$, we might be tempted to just graph it to find roots. A naive graph might be to plot over $[-10, 10]$: diff --git a/quarto/precalc/polynomials_package.qmd b/quarto/precalc/polynomials_package.qmd index ac8844f..ec878ea 100644 --- a/quarto/precalc/polynomials_package.qmd +++ b/quarto/precalc/polynomials_package.qmd @@ -76,7 +76,7 @@ Polynomials may be evaluated using function notation, that is: p(1) ``` -This blurs the distinction between a polynomial expression--a formal object consisting of an indeterminate, coefficients, and the operations of addition, subtraction, multiplication, and non-negative integer powers--and a polynomial function. +This blurs the distinction between a polynomial expression---a formal object consisting of an indeterminate, coefficients, and the operations of addition, subtraction, multiplication, and non-negative integer powers---and a polynomial function. The polynomial variable, in this case `1x`, can be returned by `variable`: diff --git a/quarto/precalc/transformations.qmd b/quarto/precalc/transformations.qmd index c6dfaa1..313f89e 100644 --- a/quarto/precalc/transformations.qmd +++ b/quarto/precalc/transformations.qmd @@ -22,7 +22,7 @@ nothing --- -Thinking of functions as objects themselves that can be manipulated - rather than just blackboxes for evaluation - is a major abstraction of calculus. The main operations to come: the limit *of a function*, the derivative *of a function*, and the integral *of a function* all operate on functions. Hence the idea of an [operator](http://tinyurl.com/n5gp6mf). Here we discuss manipulations of functions from pre-calculus that have proven to be useful abstractions. +Thinking of functions as objects themselves that can be manipulated---rather than just blackboxes for evaluation---is a major abstraction of calculus. The main operations to come: the limit *of a function*, the derivative *of a function*, and the integral *of a function* all operate on functions. Hence the idea of an [operator](http://tinyurl.com/n5gp6mf). Here we discuss manipulations of functions from pre-calculus that have proven to be useful abstractions. ## The algebra of functions @@ -141,7 +141,7 @@ The real value of composition is to break down more complicated things into a se ### Shifting and scaling graphs -It is very useful to mentally categorize functions within families. The difference between $f(x) = \cos(x)$ and $g(x) = 12\cos(2(x - \pi/4))$ is not that much - both are cosine functions, one is just a simple enough transformation of the other. As such, we expect bounded, oscillatory behaviour with the details of how large and how fast the oscillations are to depend on the specifics of the function. Similarly, both these functions $f(x) = 2^x$ and $g(x)=e^x$ behave like exponential growth, the difference being only in the rate of growth. There are families of functions that are qualitatively similar, but quantitatively different, linked together by a few basic transformations. +It is very useful to mentally categorize functions within families. The difference between $f(x) = \cos(x)$ and $g(x) = 12\cos(2(x - \pi/4))$ is not that much---both are cosine functions, one is just a simple enough transformation of the other. As such, we expect bounded, oscillatory behaviour with the details of how large and how fast the oscillations are to depend on the specifics of the function. Similarly, both these functions $f(x) = 2^x$ and $g(x)=e^x$ behave like exponential growth, the difference being only in the rate of growth. There are families of functions that are qualitatively similar, but quantitatively different, linked together by a few basic transformations. There is a set of operations of functions, which does not really change the type of function. Rather, it basically moves and stretches how the functions are graphed. We discuss these four main transformations of $f$: @@ -322,7 +322,7 @@ datetime = 12 + 10/60 + 38/60/60 delta = (newyork(266) - datetime) * 60 ``` -This is off by a fair amount - almost $8$ minutes. Clearly a trigonometric model, based on the assumption of circular motion of the earth around the sun, is not accurate enough for precise work, but it does help one understand how summer days are longer than winter days and how the length of a day changes fastest at the spring and fall equinoxes. +This is off by a fair amount---almost $8$ minutes. Clearly a trigonometric model, based on the assumption of circular motion of the earth around the sun, is not accurate enough for precise work, but it does help one understand how summer days are longer than winter days and how the length of a day changes fastest at the spring and fall equinoxes. ##### Example: the pipeline operator @@ -358,7 +358,7 @@ Suppose we have a data set like the following:^[Which comes from the "Palmer Pen | 48.8 | 18.4 | 3733 | male | Chinstrap | | 47.5 | 15.0 | 5076 | male | Gentoo | -We might want to plot on an $x-y$ axis flipper length versus bill length but also indicate body size with a large size marker for bigger sizes. +We might want to plot on an $x$-$y$ axis flipper length versus bill length but also indicate body size with a large size marker for bigger sizes. We could do so by transforming a marker: scaling by size, then shifting it to an `x-y` position; then plotting. Something like this: @@ -473,7 +473,7 @@ S(D(f))(15), f(15) - f(0) That is the accumulation of differences is just the difference of the end values. -These two operations are discrete versions of the two main operations of calculus - the derivative and the integral. This relationship will be known as the "fundamental theorem of calculus." +These two operations are discrete versions of the two main operations of calculus---the derivative and the integral. This relationship will be known as the "fundamental theorem of calculus." ## Questions diff --git a/quarto/precalc/trig_functions.qmd b/quarto/precalc/trig_functions.qmd index 67a72eb..c6f3f08 100644 --- a/quarto/precalc/trig_functions.qmd +++ b/quarto/precalc/trig_functions.qmd @@ -269,7 +269,7 @@ plot(sin, 0, 4pi) The graph shows two periods. The wavy aspect of the graph is why this function is used to model periodic motions, such as the amount of sunlight in a day, or the alternating current powering a computer. -From this graph - or considering when the $y$ coordinate is $0$ - we see that the sine function has zeros at any integer multiple of $\pi$, or $k\pi$, $k$ in $\dots,-2,-1, 0, 1, 2, \dots$. +From this graph---or considering when the $y$ coordinate is $0$---we see that the sine function has zeros at any integer multiple of $\pi$, or $k\pi$, $k$ in $\dots,-2,-1, 0, 1, 2, \dots$. The cosine function is similar, in that it has the same domain and range, but is "out of phase" with the sine curve. A graph of both shows the two are related: @@ -693,14 +693,144 @@ atan(y, x) ##### Example -A (white) light shining through a [prism](http://tinyurl.com/y8sczg4t) will be deflected depending on the material of the prism and the angles involved (refer to the link for a figure). The relationship can be analyzed by tracing a ray through the figure and utilizing Snell's law. If the prism has index of refraction $n$ then the ray will deflect by an amount $\delta$ that depends on the angle, $\alpha$ of the prism and the initial angle ($\theta_0$) according to: +A (white) light shining through a [dispersive prism](https://en.wikipedia.org/wiki/Dispersive_prism) will be deflected depending on the material of the prism and the angles involved. The relationship can be analyzed by tracing a ray through the figure and utilizing Snell's law which relates the angle of incidence with the angle of refraction through two different media through: $$ -\delta = \theta_0 - \alpha + \arcsin(n \sin(\alpha - \arcsin(\frac{1}{n}\sin(\theta_0)))). +n_0 \sin(\theta_0) = n_1 \sin(\theta_1) $$ -If $n=1.5$ (glass), $\alpha = \pi/3$ and $\theta_0=\pi/6$, find the deflection (in radians). + +:::{#fig-snells-law-prism} +```{julia} +#| echo: false +p1 = let + gr() + + plot(; empty_style..., aspect_ratio=:equal) + n₀,n₁,n₂ = 1,3, 1 + θ₀ = pi/7 + α = pi/8 + θ₁ = asin(n₀/n₁* θ₀) + θ₁′ = α - θ₁ + θ₂′ = asin(n₁/n₂ * θ₁′) + θ₂ = θ₂′ - α + + plot!([(-1,0), (1,0)]; line=(:black, 1)) + plot!([(0,-1),(0,1)]; line=(:black, 1)) + plot!([(0,1), (2tan(α),-1)]; line=(:black, 1)) + S = Shape([(0,-1),(2tan(α),-1),(0,1)]) + plot!(S, fill=(:gray80, 0.25), line=nothing) + + + xx = tan(α)/ (1 + tan(α)*tan(θ₁)) + sl(x) = 1 - x/tan(α) + yy = sl(xx) + plot!(sl, xx-1/8, xx+1/9; line=(:red,3)) + plot!(x -> yy + tan(α)*(x-xx), xx-1/4, xx+1/4; line=(:red, 3)) + + + plot!([(-1,-sin(θ₀)), (0,0)]; line=(:black, 2)) + plot!([(0,0), (xx, xx*tan(θ₁))]; line=(:black, 2)) + plot!([(xx,yy), (xx + 5/8, yy - 5/8*tan(θ₂))]; line=(:black, 2)) + + + + + annotate!([ + (-1/2,1/2*sin(pi + θ₀/2), text(L"\theta_0")), + (1/5, 1/5*sin(θ₁/2), text(L"\theta_1")), + (2tan(α), -0.075, text(L"\theta_2")), + (-1/2, -3/4, text(L"n_0")), + (2tan(α/2), -3/4, text(L"n_1")), + (1 - (1-2tan(α))/2, -3/4, text(L"n_2")) + + ]) + current() +end +p2 = let + plot(; empty_style..., aspect_ratio=:equal) + n₀,n₁,n₂ = 1,3, 1 + θ₀ = pi/7 + α = pi/8 + θ₁ = asin(n₀/n₁* θ₀) + θ₁′ = α - θ₁ + θ₂′ = asin(n₁/n₂ * θ₁′) + θ₂ = θ₂′ - α + + + + xx = tan(α)/ (1 + tan(α)*tan(θ₁)) + sl(x) = 1 - x/tan(α) + yy = sl(xx) + plot!(sl, xx-1/8, xx+1/9; line=(:red,3)) + plot!(x -> yy + tan(α)*(x-xx), xx-1/4, xx+1/4; line=(:red, 3)) + + S = Shape([(0, sl(xx-1/8)), (xx-1/8,sl(xx-1/8)), + (xx+1/9, sl(xx+1/9)), (0, sl(xx+1/9))]) + plot!(S, fill=(:gray80, 0.25), line=nothing) + + #plot!([(-1,-sin(θ₀)), (0,0)]; line=(:black, 2)) + plot!([(0,0), (xx, xx*tan(θ₁))]; line=(:black, 2)) + plot!([(xx,yy), (xx + 2/8, yy - 2/8*tan(θ₂))]; line=(:black, 2)) + + annotate!([ + (1/5, .1*sin(θ₁/2), text(L"\theta_1\prime")), + (xx + .1, 0.06, text(L"\theta_2\prime")), + (2tan(α/2), -1/8, text(L"n_1")), + (17/32, -1/8, text(L"n_2")) + ]) + current() +end +plot(p1, p2; layout=(1,2)) +``` + +Light bending through a prism. The right graphic shows the second bending. +::: + +```{julia} +#| echo: false +plotly() +nothing +``` + +Following Wikipedia, we have + + + +$$ +\theta_1 = \sin^{-1}\left( \frac{n_0}{n_1} \sin(\theta_0) \right) +$$ + +Both $\theta_0$ and $\theta_1$ are measured with respect to the coordinate system that looks like the $x-y$ plane. The red coordinate system is used to identify the angle of incidence for the second bending. Some right-triangle geometry relates the new angle $\theta'_1$ with $\theta_1$ through $\theta'_1 = \alpha - \theta_1$. With this new angle of incidence, the angle of refraction, $\theta'_2$, satisfies: + +$$ +n1 \sin(\theta'_1) = n2 \sin(\theta'_2) +$$ + +Or + +$$ +\theta'_2 = \sin^{-1}\left(\frac{n_1}{n_2}\sin(\theta'_1) \right) +$$ + +Finally, using right-triangle geometry, the angle $\theta_2 = \theta'_2 - \alpha$ can be identified. + +For a prism, in air, we would have $n_0 = n_2 = 1$. Letting $n_1 = n$, and combining we get + +$$ +\begin{align*} +\delta &= \theta_0 + \theta_2\\ +&=\theta_0 + \sin^{-1}\left(\frac{n_1}{n_2}\sin(\theta'_1) \right)- \alpha\\ +&= \theta_0 - \alpha + \sin^{-1}\left(\frac{n}{1}\sin(\alpha -\theta_1) \right)\\ +&= \theta_0 - \alpha + \sin^{-1}\left(n\sin\left(\alpha - \sin^{-1}\left( \frac{n_0}{n_1} \sin(\theta_0) \right)\right)\right)\\ +&= \theta_0 - \alpha + \sin^{-1}\left(n\sin\left(\alpha -\sin^{-1}\left( \frac{1}{n} \sin(\theta_0) \right)\right) \right) +\end{align*} +$$ + +If the prism has index of refraction $n$ then the ray will deviate by this amount $\delta$ that depends on the initial incidence angle, $\alpha$ of the prism and $n$. + +When $n=1.5$ (glass), $\alpha = \pi/3$ and $\theta_0=\pi/6$, find the deflection (in radians). We have: @@ -795,6 +925,8 @@ plot(abs ∘ T4, -1,1, label="|T₄|") plot!(abs ∘ q, -1,1, label="|q|") ``` +We will return to this family of polynomials in the section on Orthogonal Polynomials. + ## Hyperbolic trigonometric functions From e9efe6985b7248a35db8995a2f4463db5c90a1ad Mon Sep 17 00:00:00 2001 From: jverzani Date: Sun, 27 Jul 2025 15:27:22 -0400 Subject: [PATCH 19/22] ignore more --- .gitignore | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/.gitignore b/.gitignore index 66f3cb9..761b201 100644 --- a/.gitignore +++ b/.gitignore @@ -5,6 +5,7 @@ docs/site test/benchmarks.json Manifest.toml TODO.md +Changelog.md /*/_pdf_index.pdf /*/*/_pdf_index.pdf /*/_pdf_index.typ @@ -12,4 +13,7 @@ TODO.md /*/CalculusWithJulia.pdf default.profraw /quarto/default.profraw -/*/*/default.profraw \ No newline at end of file +/*/*/default.profraw +/*/bonepile.qmd +/*/*/bonepile.qmd +/*/*_files From 50cb6454527e6c071ef1636c476d8ff9350d1b57 Mon Sep 17 00:00:00 2001 From: jverzani Date: Tue, 29 Jul 2025 17:05:59 -0400 Subject: [PATCH 20/22] add some examples, figures --- .../vectors.qmd | 1744 +++++++++++------ 1 file changed, 1141 insertions(+), 603 deletions(-) diff --git a/quarto/differentiable_vector_calculus/vectors.qmd b/quarto/differentiable_vector_calculus/vectors.qmd index 2dad038..cf16f5c 100644 --- a/quarto/differentiable_vector_calculus/vectors.qmd +++ b/quarto/differentiable_vector_calculus/vectors.qmd @@ -13,38 +13,179 @@ using LinearAlgebra using SymPy ``` +```{julia} +#| echo: false +## Some plotting functions +begin + function projection_plane(v) + vx, vy, vz = v + a = [-vy, vx, 0] # v ⋅ a = 0 + b = v × a # so v ⋅ b = 0 + return (a/norm(a), b/norm(b)) +end + +function project(x, v) + â, b̂ = projection_plane(v) + (x ⋅ â, x ⋅ b̂) # (x ⋅ â) â + (x ⋅ b̂) b̂ +end + + empty_style = (xaxis = ([], false), + yaxis = ([], false), + legend=false) + axis_style = (arrow=true, side=:head, line=(:gray, 1)) + + project(x) = project(x, [2, -2, 1]) +end +nothing +``` --- -In [vectors](../precalc/vectors.html) we introduced the concept of a vector. For `Julia`, vectors are a useful storage container and are used to hold, for example, zeros of functions or the coefficients of a polynomial. This section is about their mathematical properties. A [vector](https://en.wikipedia.org/wiki/Euclidean_vector) mathematically is a geometric object with two attributes a magnitude and a direction. (The direction is undefined in the case the magnitude is $0$.) Vectors are typically visualized with an arrow, where the anchoring of the arrow is context dependent and is not particular to a given vector. +In [vectors](../precalc/vectors.html) we introduced the concept of a +vector. For `Julia`, vectors are a useful storage container and are +used to hold, for example, zeros of functions or the coefficients of a +polynomial. Matrices are a two-dimensional version of vector. This +section is about the mathematical properties or vectors and matrices. -Vectors and points are related, but distinct. They are identified when the tail of the vector is taken to be the origin. Let's focus on $3$ dimensions. Mathematically, the notation for a point is $p=(x,y,z)$ while the notation for a vector is $\vec{v} = \langle x, y, z \rangle$. The $i$th component in a vector is referenced by a subscript: $v_i$. With this, we may write a typical vector as $\vec{v} = \langle v_1, v_2, \dots, v_n \rangle$ and a vector in $n=3$ as $\vec{v} =\langle v_1, v_2, v_3 \rangle$. The different grouping notation distinguishes the two objects. As another example, the notation $\{x, y, z\}$ indicates a set. Vectors and points may be *identified* by anchoring the vector at the origin. Sets are quite different from both, as the order of their entries is not unique. +## Vectors + + A [vector](https://en.wikipedia.org/wiki/Euclidean_vector) mathematically is a geometric object with two attributes a magnitude and a direction. (The direction is undefined in the case the magnitude is $0$.) Vectors are typically visualized with an arrow, where the anchoring of the arrow is context dependent and is not particular to a given vector. -In `Julia`, the notation to define a point and a vector would be identical, using square brackets to group like-type values: `[x, y, z]`. The notation `(x,y,z)` would form a [tuple](https://en.wikipedia.org/wiki/Euclidean_vector) which though similar in many respects, are different, as tuples do not have the operations associated with a point or a vector defined for them. +A point is a position in $R^n$, denoted by $n$ components. Vectors and points are related, but distinct concepts. The two are identified when the tail of the vector is taken to be the origin. Let's focus on $3$ dimensions. Mathematically, the notation for a point is $p=(x,y,z)$ while the notation for a vector is $\vec{v} = \langle x, y, z \rangle$. The $i$th component in a vector is referenced by a subscript: $v_i$. With this, we may write a typical vector as $\vec{v} = \langle v_1, v_2, \dots, v_n \rangle$ and a vector in $n=3$ as $\vec{v} =\langle v_1, v_2, v_3 \rangle$. The different grouping notation distinguishes the two objects. As another example, the notation $\{x, y, z\}$ indicates a set. While vectors and points may be *identified* by anchoring the vector at the origin. Sets are quite different from both, as the order of their entries is not unique. +`Julia` uses square brackets to group together the components of a vector. The construction `[v1, v2, ..., vn]` creates a `Vector` object: + +```{julia} +[1, 2, 3] +``` + +(The type of the elements is also indicated.) + +We have seen that `Plots` interprets tuples as points. We can use tuples as points, though it can be a bit cumbersome as certain arithmetic operations aren't defined for tuples. As such, we often use a vector to indicate a point or a vector, leaving the interpretation to the reader. + The square bracket constructor has some subtleties: - * `[x,y,z]` calls `vect` and creates a 1-dimensional array - * `[x; y; z]` calls `vcat` to **v**ertically con**cat**enate values together. With simple (scalar) values `[x,y,z]` and `[x; y; z]` are identical, but not in other cases. (For example, if `A` is a matrix then `[A, A]` is a vector of matrices, `[A; A]` is a matrix combined from the two pieces. - * `[x y z]` calls `hcat` to **h**orizontally con**cat**enate values together. If `x`, `y` are numbers then `[x y]` is *not* a vector, but rather a $2$D array with a single row and two columns. - * finally `[w x; y z]` calls `hvcat` to horizontally and vertically concatenate values together to create a container in two dimensions, like a matrix. +* `[x, y, z]` calls `Base.vect` and creates a 1-dimensional array or vector. + +* `[x; y; z]` calls `vcat` to **v**ertically con**cat**enate values together. With simple (scalar) values `[x, y, z]` and `[x; y; z]` are identical, but not in other cases. (For example, if `A` is a matrix then `[A, A]` is a vector of matrices, `[A; A]` is a matrix combined from the two pieces. + +* `[x y z]` calls `hcat` to **h**orizontally con**cat**enate values together. If `x`, `y` are numbers then `[x y]` is *not* a vector, but rather a $2$D array with a single row and two columns. + +* finally `[w x; y z]` calls `hvcat` to horizontally and vertically concatenate values together to create a container in two dimensions, like a matrix. -(A vector, mathematically, is a one-dimensional collection of numbers, a matrix a two-dimensional *rectangular* collection of numbers, and an array an $n$-dimensional rectangular-like collection of numbers. In `Julia`, a vector can hold a collection of objects of arbitrary type, though each will be promoted to a common type.) +(A vector, mathematically, is a one-dimensional collection of numbers, a matrix a two-dimensional *rectangular* collection of numbers, and an array an $n$-dimensional rectangular-like collection of numbers. In `Julia` the `Array` type is a general type to hold *homogeneous* data along different dimensions. The `Vector` type is an array specialized to one dimension; the `Matrix` type is a specialization to two dimensions.) + +### Vector addition -## Vector addition, scalar multiplication +As seen earlier, vectors have some arithmetic operations defined for them. As a typical use of vectors, mathematically, is to collect the $x$, $y$, and $z$ (in $3$D) components together, operations like addition and subtraction operate component wise. -As seen earlier, vectors have some arithmetic operations defined for them. As a typical use of vectors, mathematically, is to collect the $x$, $y$, and $z$ (in $3$D) components together, operations like addition and subtraction operate component wise. With this, addition can be visualized geometrically: put the tail of $\vec{v}$ at the tip of $\vec{u}$ and draw a vector from the tail of $\vec{u}$ to the tip of $\vec{v}$ and you have $\vec{u}+\vec{v}$. This is identical by $\vec{v} + \vec{u}$ as vector addition is commutative. Unless $\vec{u}$ and $\vec{v}$ are parallel or one has $0$ length, the addition will create a vector with a different direction from the two. +:::{.callout-note appearance="minimal"} +# Vector addition +$$ +\langle a_1, a_2, \dots, a_n \rangle + +\langle b_1, b_2, \dots, b_n \rangle = +\langle (a_1 + b_1), (a_2 + b_2), \dots, (a_n + b_n) \rangle +$$ +::: + +With this, addition can be visualized geometrically: put the tail of $\vec{v}$ at the tip of $\vec{u}$ and draw a vector from the tail of $\vec{u}$ to the tip of $\vec{v}$ and you have $\vec{u}+\vec{v}$. This is identical by $\vec{v} + \vec{u}$ as vector addition is commutative. Unless $\vec{u}$ and $\vec{v}$ are parallel or one has $0$ length, the addition will create a vector with a different direction from the two. + +:::{#fig-vector-addition-2-3d} + +```{julia} +#| echo: false +gr() +p1 = let + # plot two vectors + P,Q = (1,2), (2, 1) + O = (0,0) + plot(; empty_style..., xlim=(0, 3.5)) + plot!([O, (3, 0)]; axis_style...) + plot!([O, (0,3)]; axis_style...) + + plot!([O, O .+ P]; arrow=true, line=(:black, 2)) + plot!([O, O .+ Q]; arrow=true, line=(:black, 2)) + plot!([Q, Q .+ P]; arrow=true, line=(:black, 1, :dash)) + plot!([O, Q .+ P]; arrow=true, line=(:black, 1, :dash)) + + annotate!([ + (3, 0, text(L"x", :top)), + (0, 3, text(L"y", :right)), + (P..., text(L"\vec{v}", :top, :left)), + (Q..., text(L"\vec{u}", :top, :left)), + ((P .+Q)..., text(L"\vec{u}+\vec{v}", :top, :left)) + + ]) +end + +p2 = let + # plot two vectors + P,Q = (1,1,3), (2, 1/3,7/4) + O = (0,0,0) + plot(; empty_style..., xlim=(0, 4)) + plot!(project.([O, (4, 0, 0)]); axis_style..., linealpha=0.5) + plot!(project.([O, (0, 4, 0)]); axis_style..., linealpha=0.5) + plot!(project.([O, (0, 0, 4)]); axis_style..., linealpha=0.5) + annotate!([ + (project((4, 0, 0))..., text(L"x",:top)), + (project((0, 4, 0))..., text(L"y",:top)), + (project((0, 0, 4))..., text(L"z",:left)), + ]) + + plot!(project.([O, P]); arrow=true, line=(:black, 2)) + plot!(project.([O, Q]); arrow=true, line=(:black, 2)) + plot!(project.([Q, P .+ Q]); arrow=true, line=(:black, 1, :dash)) + plot!(project.([O, P .+ Q]); arrow=true, line=(:black, 1, :dash)) -Another operation for vectors is *scalar* multiplication. Geometrically this changes the magnitude, but not the direction of a vector, when the *scalar* is positive. Scalar multiplication is defined component wise, like addition so the $i$th component of $c \vec{v}$ is $c$ times the $i$th component of $\vec{v}$. When the scalar is negative, the direction is "reversed." + for pt in (P, Q, P .+ Q) + plot!(project.([(pt[1],0,0), (pt[1], pt[2],0), pt]); line=(:black, 1, 0.25, :dot)) + end + + + annotate!([ + (project(P)..., text(L"\vec{v}", :top, :left)), + (project(Q)..., text(L"\vec{u}", :top, :left)), + (project(P .+ Q)..., text(L"\vec{u}+\vec{v}", :top, :left)) + + ]) +end + +plot(p1, p2; layout=[2, 1]) +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + +Illustration of vector addition in $2$ and $3$ dimensions +::: + +### Scalar multiplication + +Another operation for vectors is *scalar* multiplication. Geometrically this changes the magnitude, but not the direction of a vector, when the *scalar* is positive. Scalar multiplication is defined component wise, like addition so the $i$th component of $c \vec{v}$ is $c$ times the $i$th component of $\vec{v}$. + + +:::{.callout-note appearance="minimal"} +# Scalar multiplication +$$ +c \cdot \langle a_1, a_2, \dots, a_n \rangle = +\langle c\cdot a_1, c \cdot a_2 , \dots, c \cdot a_n \rangle +$$ +::: + + +When the scalar is negative, the direction is "reversed." To illustrate we define two $3$-dimensional vectors: @@ -71,7 +212,7 @@ Scalar multiplication by `2`, say, multiplies each entry by `2`: 2 * u ``` -## The length and direction of a vector +### The length and direction of a vector If a vector $\vec{v} = \langle v_1, v_2, \dots, v_n\rangle$ then the *norm* (also Euclidean norm or length) of $\vec{v}$ is defined by: @@ -90,7 +231,7 @@ A vector with length $1$ is called a *unit* vector. Dividing a non-zero vector b The direction indicated by $\vec{v}$ can be visualized as an angle in $2$- or $3$-dimensions, but in higher dimensions, visualization is harder. For $2$-dimensions, we might associate with a vector, it's unit vector. This in turn may be identified with a point on the unit circle, which from basic trigonometry can be associated with an angle. Something similar, can be done in $3$ dimensions, using two angles. However, the "direction" of a vector is best thought of in terms of its associated unit vector. With this, we have a decomposition of a non-zero vector $\vec{v}$ into a magnitude and a direction when we write $\vec{v} = \|\vec{v}\| \cdot (\vec{v} / \|\vec{v}\|)=\|\vec{v}\| \hat{v}$. -## Visualization of vectors +### Visualization of vectors Vectors may be visualized in $2$ or $3$ dimensions using `Plots`. In $2$ dimensions, the `quiver` function may be used. To graph a vector, it must have its tail placed at a point, so two values are needed. @@ -103,7 +244,10 @@ To plot `u=[1,2]` from `p=[0,0]` we have the following usage: quiver([0],[0], quiver=([1],[2])) ``` -The cumbersome syntax is typical here. We naturally describe vectors and points using `[a,b,c]` to combine them, but the plotting functions want to plot many such at a time and expect vectors containing just the `x` values, just the `y` values, etc. The above usage looks a bit odd, as these vectors of `x` and `y` values have only one entry. Converting from the one representation to the other requires reshaping the data. We will use the `unzip` function from `CalculusWithJulia` which in turn just uses the the `invert` function of the `SplitApplyCombine` package ("return a new nested container by reversing the order of the nested container") for the bulk of its work. +The cumbersome syntax, `quiver(x, y, quiver=(u, v))`, is typical here. We naturally describe vectors and points using `[a,b,c]` to combine them, but the plotting functions want to plot many such at a time and expect vectors containing just the `x` values, just the `y` values, etc. The above usage looks a bit odd, as these vectors of `x` and `y` values have only one entry. + + +Converting from the one representation to the other requires reshaping the data. We will use the `unzip` function from `CalculusWithJulia` which in turn just uses the the `invert` function of the `SplitApplyCombine` package ("return a new nested container by reversing the order of the nested container") for the bulk of its work. This function takes a vector of vectors, and returns a tuple containing the `x` values, the `y` values, etc. So if `u=[1,2,3]` and `v=[4,5,6]`, then `unzip([u,v])` becomes `([1,4],[2,5],[3,6])`, etc. (The `zip` function in base does essentially the reverse operation, hence the name.) Notationally, `A = [u,v]` can have the third element of the first vector (`u`) accessed by `A[1][3]`, where as `unzip(A)[3][1]` will do the same. We use `unzip([u])` in the following, which for this `u` returns `([1],[2],[3])`. (Note the `[u]` to make a vector of a vector.) @@ -126,6 +270,10 @@ quiver!(unzip([u])..., quiver=unzip([v])) quiver!(unzip([p])..., quiver=unzip([w])) ``` +For some plotting backends---though not `plotly`--- there is an `arrow` argument that can be passed to the line-plotting command. If `p` and `u` are *tuples*, then the command `plot([p, p .+ u]; arrow=true, side=:head)` will draw the vector `u` anchored at `p`. (There can be many more points in the curve used to create curved arrows.) + + + Plotting a $3$-d vector is not supported in all toolkits with `quiver`. A line segment may be substituted and can be produced with `plot(unzip([p,p+v])...)`. To avoid all these details, the `CalculusWithJulia` provides the `arrow!` function to *add* a vector to an existing plot. The function requires a point, `p`, and the vector, `v`: @@ -144,7 +292,7 @@ arrow!(u, v) arrow!(p, w) ``` -The distinction between a point and a vector within `Julia` is only mental. We use the same storage type. Mathematically, we can **identify** a point and a vector, by considering the vector with its tail placed at the origin. In this case, the tip of the arrow is located at the point. But this is only an identification, though a useful one. It allows us to "add" a point and a vector (e.g., writing $P + \vec{v}$) by imagining the point as a vector anchored at the origin. +The distinction between a point and a vector within `Julia` is only mental. We often use the same storage type (`Vector`). Mathematically, we can **identify** a point and a vector, by considering the vector with its tail placed at the origin. In this case, the tip of the arrow is located at the point. But this is only an identification, though a useful one. It allows us to "add" a point and a vector (e.g., writing $P + \vec{v}$) by imagining the point as a vector anchored at the origin. (If `P` is represented with a tuple, we can still add to a vector through broadcasting: `P .+ v`.) To see that a unit vector has the same "direction" as the vector, we might draw them with different widths: @@ -163,13 +311,14 @@ arrow!(p, u, linewidth=5) The `norm` function is in the standard library, `LinearAlgebra`, which must be loaded first through the command `using LinearAlgebra`. (Though here it is redundant, as that package is loaded and reexported when the `CalculusWithJulia` package is loaded.) -## Aside: review of `Julia`'s use of dots to work with containers +#### Aside: review of `Julia`'s use of dots to work with containers `Julia` makes use of the dot, "`.`", in a few ways to simplify usage when containers, such as vectors, are involved: +##### Splatting - * **Splatting**. The use of three dots, "`...`", to "splat" the values from a container like a vector (or tuple) into *arguments* of a function can be very convenient. It was used above in the definition for the `arrow!` function: essentially `quiver!(unzip([p])..., quiver=unzip([v]))`. The `quiver` function expects $2$ (or $3$) arguments describing the `xs` and `ys` (and sometimes `zs`). The `unzip` function returns these in a container, so splatting is used to turn the values in the container into distinct arguments of the function. Whereas the `quiver` argument expects a tuple of vectors, so no splatting is used for that part of the definition. Another use of splatting we will see is with functions of vectors. These can be defined in terms of the vector's components or the vector as a whole, as below: +The use of three dots, "`...`", to "splat" the values from a container like a vector (or tuple) into *arguments* of a function can be very convenient. It was used above in the definition for the `arrow!` function: essentially `quiver!(unzip([p])..., quiver=unzip([v]))`. The `quiver` function expects $2$ (or $3$) arguments describing the `xs` and `ys` (and sometimes `zs`). The `unzip` function returns these in a container, so splatting is used to turn the values in the container into distinct arguments of the function. Whereas the `quiver` argument expects a tuple of vectors, so no splatting is used for that part of the definition. Another use of splatting we will see is with functions of vectors. These can be defined in terms of the vector's components or the vector as a whole, as below: ```{julia} @@ -194,7 +343,9 @@ The splatting will mean `g(v)` eventually calls `g(x, y, z)` through `Julia`'s m (The three dots can also appear in the definition of the arguments to a function, but there the usage is not splatting but rather a specification of a variable number of arguments.) - * **Broadcasting**. For a univariate function, `f`, and vector, `xs`, the call `f.(xs)` *broadcasts* `f` over each value of `xs` and returns a container holding all the values. This is a compact alternative to a comprehension when a function is defined. When `f` depends on more than one value, broadcasting can still be used: `f.(xs, ys)` will broadcast `f` over values formed from *both* `xs` and `ys`. Broadcasting has the extra feature (over `map`) of attempting to match up the shapes of `xs` and `ys` when they are not identical. (See the help page for `broadcast` for more details.) +##### Broadcasting + +For a univariate function, `f`, and vector, `xs`, the call `f.(xs)` *broadcasts* `f` over each value of `xs` and returns a container holding all the values. This is a compact alternative to a comprehension when a function is defined. When `f` depends on more than one value, broadcasting can still be used: `f.(xs, ys)` will broadcast `f` over values formed from *both* `xs` and `ys`. Broadcasting has the extra feature (over `map`) of attempting to match up the shapes of `xs` and `ys` when they are not identical. (See the help page for `broadcast` for more details.) For example, if `xs` is a vector and `ys` a scalar, then the value in `ys` is repeated many times to match up with the values of `xs`. Or if `xs` and `ys` have different dimensions, the values of one will be repeated. Consider this: @@ -224,210 +375,8 @@ The two dimensions are different so for each value of `xs` the vector of `ys` is At times using the "apply" notation: `x |> f`, in place of using `f(x)` is useful, as it can move the wrapping function to the right of the expression. To broadcast, `.|>` is available. -## Aside: simplifying calculations using containers and higher-order functions -Storing data in a container, be it a vector or a tuple, can seem at first more complicated, but in fact can lead to much simpler computations. - -A simple example might be to add up a sequence of numbers. A direct way might be: - -```{julia} -x1, x2, x3, x4, x5, x6 = 1, 2, 3, 4, 5, 6 -x1 + x2 + x3 + x4 + x5 + x6 -``` - -Someone doesn't need to know `Julia`'s syntax to guess what this computes, save for the idiosyncratic tuple assignment used, which could have been bypassed at the cost of even more typing. - -A more efficient means to do, as each component isn't named, this would be to store the data in a container: - -```{julia} -xs = [1, 2, 3, 4, 5, 6] # as a vector -sum(xs) -``` - -Sometimes tuples are used for containers. The difference to `sum` is not noticeable (though a different code path for the computation is taken behind the scenes): - -```{julia} -xs = (1, 2, 3, 4, 5, 6) -sum(xs) -``` - -(Tuples and vectors are related, but tuples don't have built-in arithmetic defined. Several popular packages, such as `Plots`, draw a distinction between the two basic containers, but most generic functions just need to be able to iterate over the values.) - -The `sum` function has a parallel `prod` function for finding the product of the entries: - -```{julia} -prod(xs) -``` - -Both `sum` and `prod` will error if the container is empty. - -These two functions are *reductions*. There are others, such as `maximum` and `minimum`. They reduce the dimensionality of the container, in this case from a vector to a scalar. When applied to higher-dimensional containers, dimenensions to reduce over are specified. The higher-order `reduce` function can be used as a near alternate to `sum`: - -```{julia} -reduce(+, xs; init=0) # sum(xs) -``` - -or - -```{julia} -reduce(*, xs; init=1) # prod(xs) -``` - -The functions (`+` and `*`) are binary operators and are serially passed the running value (or `init`) and the new term from the iterator. - -The initial value above is the unit for the operation (which could be found programmatically by `zero(eltype(xs))` or `one(eltype(xs))` where the type is useful for better performance). - -The `foldl` and `foldr` functions are similar to `reduce` only left (and right) associativity is guaranteed. This example uses the binary, infix `Pair` operator, `=>`, to illustrate the difference: - -```{julia} -foldl(=>, xs) -``` - -and - -```{julia} -foldr(=>, xs) -``` - -Next, we do a slightly more complicated problem. - -Recall the distance formula between two points, also called the *norm*. It is written here with the square root on the other side: $d^2 = (x_1-y_1)^2 + (x_0 - y_0)^2$. This computation can be usefully generalized to higher dimensional points (with $n$ components each). - -This first example shows how the value for $d^2$ can be found using broadcasting and `sum`: - -```{julia} -xs = [1, 2, 3, 4, 5] -ys = [1, 3, 5, 7, 3] - -sum((xs - ys).^2) -``` - - -This formula is a sum after applying an operation to the paired off values. Using a generator that sum would look like: - -```{julia} -sum((xi - yi)^2 for (xi, yi) in zip(xs, ys)) -``` - -The `zip` function, used above, produces an iterator over tuples of the paired off values in the two (or more) containers passed to it. - - -This pattern---where a reduction follows a function's application to the components---is implemented in `mapreduce`. - - -```{julia} -f(xy) = (xy[1] - xy[2])^2 -mapreduce(f, +, zip(xs, ys)) -``` - -In the generator example above, the components of the tuple are destructured into `(xi, yi)`; in the function `f` above $1$-based indexing is used to access the first and second components. - -The `mapreduce` function can take more than one iterator to reduce over, When used this way, the function takes multiple arguments. Unlike the above example, where `f` was first defined and then used, we use an anonymous function below, to make the example a one-liner: - -```{julia} -mapreduce((xi,yi) -> (xi-yi)^2, +, xs, ys) -``` - - -(The `mapreduce` form is more performant than broadcasting where the vectors are traversed more times.) - - -### Extracting pieces of a container - -At times, extracting all but the first or last value can be of interest. For example, a polygon comprised of $n$ points (the vertices), might be stored using a vector for the $x$ and $y$ values with an additional point that mirrors the first. Here are the points: - -```{julia} -xs = [1, 3, 4, 2] -ys = [1, 1, 2, 3] -pts = zip(xs, ys) # recipe for [(x1,y1), (x2,y2), (x3,y3), (x4,y4)] -``` - -To *add* the additional point we might just use `push!`: - -```{julia} -push!(xs, first(xs)) -push!(ys, first(ys)) -``` - -(Though this approach won't work with `pts`, only with mutable containers.) - -The `first` and `last` methods refer to the specific elements in the indexed collection. To get the rest of the values can be done a few ways. For example, this pattern peels off the first and leaves a new container to hold the rest: - -```{julia} -a, bs... = xs -a, bs -``` - -The splatting operation for `bs....` is usually seen inside a function, so this is a bit unusual. The `Iterators.peel` method also can also do this task (the `Iterators` module and its methods are not exported, so `peel` is necessarily qualified if not imported): - -```{julia} -a, bs = Iterators.peel(xs) -bs -``` - - -The `bs` are represented with an iterator and can be collected to yield the values, though often this is unnecessary and possibly a costly step: - -```{julia} -collect(bs), sum(bs) -``` - -The iterators shown here are *lazy* and only construct a recipe to produce the points one after another. Reductions like `sum` and `prod` can use this recipe to produce an answer without needing to realize in memory at one time the entire collection of values being represented. - -The `Iterators.rest` method can be used to take the rest of the container starting a given index. This command also finds the `bs` above: - -```{julia} -bs = Iterators.rest(xs, 2) -collect(bs) -``` - -The `Iterators.take` method can be used to take values at the beginning of a container. This command takes all but the last value of `xs`: - -```{julia} -as = Iterators.take(xs, length(xs) - 1) -collect(as) -``` - -The `take` method could be used to remove the padded value from the `xs` and `ys`. Between `Iterators.take` and `Iterators.rest` the iterable object can be split into a head and tail. - -##### Example: Riemann sums - -In the computation of a Riemann sum, the interval $[a,b]$ is partitioned using $n+1$ points $a=x_0 < x_1 < \cdots < x_{n-1} < x_n = b$. - -```{julia} -a, b, n = 0, 1, 4 -xs = range(a, b, n+1) # n + 1 points gives n subintervals [xᵢ, xᵢ₊₁] -``` - -To grab these points as adjacent pairs can be done by combining the first $n$ points and the last $n$ points, as follows: - -```{julia} -partitions = zip(Iterators.take(xs, n), Iterators.rest(xs, 1)) -collect(partitions) -``` - -A left-hand Riemann sum for `f` could then be done with: - -```{julia} -f(x) = x^2 -sum(f ∘ first, partitions) -``` - -This uses a few things: like `mapreduce`, `sum` allows a function to -be applied to each element in the `partitions` collection. (Indeed, the default method to compute `sum(xs)` for an arbitrary container resolves to `mapreduce(identity, add_sum, xs)` where `add_sum` is basically `+`.) - -In this case, the -values come as tuples to the function to apply to each component. -The function above uses `first` to find the left-endpoint value and then calls `f`. The composition (through `∘`) implements this. - -Alternatively, `zip` can be avoided with: - -```{julia} -mapreduce((l, r) -> l^2, +, Iterators.take(xs, n), Iterators.rest(xs, 1)) -``` - - -## The dot product +### The dot product There is no concept of multiplying two vectors, or for that matter dividing two vectors. However, there are two operations between vectors that are somewhat similar to multiplication, these being the dot product and the cross product. Each has an algebraic definition, but their geometric properties are what motivate their usage. We begin by discussing the dot product. @@ -450,7 +399,59 @@ $$ \vec{v} \cdot \vec{w} = \|\vec{v}\| \|\vec{w}\| \cos(\theta). $$ -If we denoted $\hat{v} = \vec{v} / \| \vec{v} \|$, the unit vector in the direction of $\vec{v}$, then by dividing, we see that $\cos(\theta) = \hat{v} \cdot \hat{w}$. That is the angle does not depend on the magnitude of the vectors involved. +If we denoted $\hat{v} = \vec{v} / \| \vec{v} \|$, the unit vector in the direction of $\vec{v}$, then by dividing, we see that + +$$ +\cos(\theta) = \hat{v} \cdot \hat{w}. +$$ + +That is, the angle does not depend on the magnitude of the vectors involved. + +:::{#fig-angle-between-vectors} +```{julia} +#| echo: false +gr() +let + # dot product + θ₀, θ₁ = pi/8, 7pi/16 + r₀, r₁ = 3/2, 2 + O = (0,0) + x′, y′ = (2,2) + plot(; empty_style..., aspect_ratio=:equal, xlim=(0, x′ + 0.25)) + plot!([O, (x′, 0)]; axis_style...) + plot!([O, (0,y′)]; axis_style...) + annotate!([ + (x′, 0, text(L"x", :top, :left)), + (0, y′, text(L"y", :left)) + ]) + + for (r, θ, l, l̂) ∈ ((r₀,θ₀, L"\vec{u}", L"\hat{u}"), (r₁, θ₁,L"\vec{v}", L"\hat{v}")) + P = (r*cos(θ), r * sin(θ)) + plot!([O, P]; arrow=true, line=(:black, 2)) + plot!([O, (cos(θ), sin(θ))]; arrow=true, line=(:red, 5, 0.5)) + annotate!([(P..., text(l, :top, :left)), + (1/2*cos(θ), 1/2*sin(θ), text(l̂, :top, :left)) + ]) + + end + + rᵢ = 1/5 + pts = [(rᵢ*cos(t), rᵢ*sin(t)) for t in range(θ₀, θ₁, 100)] + plot!(pts; arrow=true, line=(:black, 1) ) + tᵢ = (θ₀ + θ₁)/2 + annotate!([(2rᵢ*cos(tᵢ), 2rᵢ*sin(tᵢ), text(L"\theta"))]) + current() +end +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + +The angle between two vectors is the same as that between their unit vectors. It can be defined to be between $0$ and $\pi$. Its cosine is realized by the dot product of the unit vectors, $\hat{u} \cdot \hat{v}$. +::: The dot product is computed in `Julia` by the `dot` function, which is in the `LinearAlgebra` package of the standard library. This must be loaded (as above) before its use either directly or through the `CalculusWithJulia` package: @@ -505,41 +506,82 @@ u ⋅ v #### Projection - -From right triangle trigonometry, we learn that $\cos(\theta) = \text{adjacent}/\text{hypotenuse}$. If we use a vector, $\vec{h}$ for the hypotenuse, and $\vec{a} = \langle 1, 0 \rangle$, we have this picture: +A key fact about *non* parallel vectors, $\vec{u}$ and $\vec{v}$, is that $\vec{v}$ can be expressed as a scalar multiple of $\vec{u}$ plus a second vector which is orthogonal to $\vec{u}$. +::: {#fig-projection-v-on-u} ```{julia} -#| hold: true -h = [2, 3] -a = [1, 0] # unit vector -h_hat = h / norm(h) -theta = acos(h_hat ⋅ a) +#| echo: false +gr() +let + ## projection + O = (0,0) + A = (2,1) + Â = A ./ norm(A) + B = (1, 2) -plot(legend=false) -arrow!([0,0], h) -arrow!([0,0], norm(h) * cos(theta) * a) -arrow!([0,0], a, linewidth=3) + plot(; empty_style..., aspect_ratio=:equal) + plot!([O,A]; arrow=true, line=(:black, 1)) + plot!([O, B]; arrow=true, line=(:black, 1)) + + p_ab = (B ⋅ Â) .* Â + plot!([O, p_ab]; arrow=true, line=(:red, 3)) + + n = B .- p_ab + plot!([p_ab, B]; arrow=true, line=(:red, 3)) + + λ = 1/8 + plot!([p_ab .- λ .* Â, + p_ab .- λ .* Â .+ λ .* n./norm(n), + p_ab .+ λ .* n./norm(n), + ]; line=(:black, 1)) + + θ₀, θ₁ = atan(A[2],A[1]), atan(B[2],B[1]) + θ₂ = (θ₀ + θ₁)/2 + pts = [(λ*cos(t), λ*sin(t)) for t in range(θ₀, θ₁, 25)] + plot!(pts, line=(:black, 1) ) + + annotate!([ + (2λ*cos(θ₂), 2λ*sin(θ₂), text(L"\theta")), + ((A./1.1)..., text(L"\vec{u}", :top)), + ((B./2)..., text(L"\vec{v}", :bottom,:right)), + ((p_ab./2)..., text(L"proj_{\vec{u}}(\vec{v})", :bottom; + rotation=atand(A[2],A[1]), color=:red)), + ((p_ab .+ n ./ 2)..., text(L"\vec{n}=\vec{v} - proj_{\vec{u}}(\vec{v})", :bottom, + color=:red, rotation=atand(-n[2],-n[1]))) + + ]) +end ``` -We used vectors to find the angle made by `h`, and from there, using the length of the hypotenuse is `norm(h)`, we can identify the length of the adjacent side, it being the length of the hypotenuse times the cosine of $\theta$. Geometrically, we call the vector `norm(h) * cos(theta) * a` the *projection* of $\vec{h}$ onto $\vec{a}$, the word coming from the shadow $\vec{h}$ would cast on the direction of $\vec{a}$ were there light coming perpendicular to $\vec{a}$. +```{julia} +#| echo: false +plotly() +nothing +``` +The vector $\vec{v}$ *projected* onto $\vec{u}$ decomposes $\vec{v}$ into a sum of vectors---one is parallel to $\vec{u}$ the other orthogonal. +::: -The projection can be made for any pair of vectors, and in any dimension $n > 1$. The projection of $\vec{u}$ on $\vec{v}$ would be a vector of length $\vec{u}$ (the hypotenuse) times the cosine of the angle in the direction of $\vec{v}$. In dot-product notation: - +In @fig-projection-v-on-u we see graphically the decomposition. The length of the vector parallel to $\vec{u}$ can be seen from right-triangle geometry to be +$\lVert \vec{v} \rVert \cos(\theta)$, so the vector parallel to $\vec{u}$ would be this length times the unit vector in the direction of $\vec{u}$. +:::{.callout-note appearance="minimal"} +## Projection of one vector onto another $$ -proj_{\vec{v}}(\vec{u}) = \| \vec{u} \| \frac{\vec{u}\cdot\vec{v}}{\|\vec{u}\|\|\vec{v}\|} \frac{\vec{v}}{\|\vec{v}\|}. +\begin{align*} +\text{proj}_{\vec{u}}(\vec{v}) +&= \lVert \vec{v} \rVert \cos(\theta) \hat{u} \\ +&= \frac{\vec{v} \cdot \vec{u}}{\lVert\vec{u}\rVert} \frac{\vec{u}}{\lVert\vec{u}\rVert} \\ +&= (\vec{v} \cdot \hat{u}) \hat{u}. +\end{align*} $$ -This can simplify. After cancelling, and expressing norms in terms of dot products, we have: +::: +With this projection function, the orthogonal piece can be recovered by subtraction: $\vec{v} - (\vec{v} \cdot \hat{u}) \hat{u}$. -$$ -proj_{\vec{v}}(\vec{u}) = \frac{\vec{u} \cdot \vec{v}}{\vec{v} \cdot \vec{v}} \vec{v} = (\vec{u} \cdot \hat{v}) \hat{v}, -$$ - -where $\hat{v}$ is the unit vector in the direction of $\vec{v}$. +This is readily seen to be orthogonal to $\vec{u}$ just by taking the dot product. ##### Example @@ -630,9 +672,678 @@ The dot product is similar to multiplication, but different, as it is an operati The last two can be combined: $\vec{u}\cdot(s \vec{v} + t \vec{w}) = s(\vec{u}\cdot\vec{v}) + t (\vec{u}\cdot\vec{w})$. -But the associative property does not make sense, as $(\vec{u} \cdot \vec{v}) \cdot \vec{w}$ does not make sense as two dot products: the result of the first is not a vector, but a scalar. +But the associative property does not, as $(\vec{u} \cdot \vec{v}) \cdot \vec{w}$ does not make sense as two dot products: the result of the first is not a vector, but a scalar. + +### Cross product + + +In three dimensions, there is a another operation between vectors that is similar to multiplication, though we will see with many differences. + +Let $\vec{u}$ and $\vec{v}$ be two $3$-dimensional vectors, then the *cross* product, $\vec{u} \times \vec{v}$, is defined as a vector with length: + + +$$ +\| \vec{u} \times \vec{v} \| = \| \vec{u} \| \| \vec{v} \| \sin(\theta), +$$ + +with $\theta$ being the angle in $[0, \pi]$ between $\vec{u}$ and $\vec{v}$. Consequently, $\sin(\theta) \geq 0$. + +::: {#fig-cross-product} +```{julia} +#| echo: false +gr() +let + O = (0,0,0) + u = (5/2, 1, 1/8) + v = (2, 1, 1/2) + kross(u,v)= tuple(cross(collect(u), collect(v))...) + w = kross(u,v) + û, v̂, ŵ = u ./ norm(u), v ./ norm(v), w ./ norm(w) + + plot(; empty_style..., aspect_ratio=:equal) + plot!([O,u]; arrow=true, line=(:black, 1)) + plot!([O,v]; arrow=true, line=(:black, 1)) + plot!([O,w]; arrow=true, line=(:black, 1)) + r = 1/8 + plot!([r.*û, r.*û .+ r.*ŵ, r.*ŵ], line=(:black, 1)) + plot!([r.*v̂, r.*v̂ .+ r.*ŵ, r.*ŵ], line=(:black, 1)) + + θ = asin(û⋅v̂) + pts = [((cos(t) .* û) .+ (sin(t) .* v̂)) ./ 4 for t in range(0, θ, 20)] + plot!(pts) + + annotate!([ + (u..., text(L"\vec{u}", :top)), + (v..., text(L"\vec{v}", :top)), + (w..., text(L"\vec{w}", :bottom, :left)), + (((û .+ v̂)./4)..., text(L"\theta")) + + + + ]) + +end +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + +Cross product of $\vec{u}$ and $\vec{v}$ will be orthogonal to both with length given by the lengths of both vectors and the sine of the angle between them. +::: + + +The direction of the cross product is such that it is *orthogonal* to *both* $\vec{u}$ and $\vec{v}$. There are two such directions, to identify which is correct, the [right-hand rule](https://en.wikipedia.org/wiki/Cross_product#Definition) is used. This rule points the right hand fingers in the direction of $\vec{u}$ and curls them towards $\vec{v}$ (so that the angle between the two vectors is in $[0, \pi]$). The thumb will point in the direction. Call this direction $\hat{n}$, a normal unit vector. Then the cross product can be defined by: + + +$$ +\vec{u} \times \vec{v} = \| \vec{u} \| \| \vec{v} \| \sin(\theta) \hat{n}. +$$ + +:::{.callout-note} +## The right-hand rule +The right-hand rule is also useful to understand how standard household screws will behave when twisted with a screwdriver. If the right hand fingers curl in the direction of the twisting screwdriver, then the screw will go in or out following the direction pointed to by the thumb. + +::: + +The right-hand rule depends on the order of consideration of the vectors. If they are reversed, the opposite direction is determined. A consequence is that the cross product is **anti**-commutative, unlike multiplication: + + +$$ +\vec{u} \times \vec{v} = - \vec{v} \times \vec{u}. +$$ + +Mathematically, the definition in terms of its components is a bit involved. + + +$$ +\vec{u} \times \vec{v} = \langle u_2 v_3 - u_3 v_2, u_3 v_1 - u_1 v_3, u_1 v_2 - u_2 v_1 \rangle. +$$ + + +From the $\sin(\theta)$ term in the definition, we see that $\vec{u}\times\vec{u}=0$. In fact, the cross product is $0$ only if the two vectors involved are parallel or there is a zero vector. + + +In `Julia`, the `cross` function from the `LinearAlgebra` package implements the cross product. For example: + + +```{julia} +a = [1, 2, 3] +b = [4, 2, 1] +cross(a, b) +``` + +There is also the *infix* unicode operator `\times[tab]` that can be used for similarity to traditional mathematical syntax. + + +```{julia} +a × b +``` + +We can see the cross product is anti-commutative by comparing the last answer with: + + +```{julia} +b × a +``` + +Using vectors of size different than $n=3$ produces a dimension mismatch error: + + +```{julia} +#| error: true +[1, 2] × [3, 4] +``` + +(It can prove useful to pad $2$-dimensional vectors into $3$-dimensional vectors by adding a $0$ third component. We will see this in the discussion on curvature in the plane.) + + +Consider this extended picture involving two vectors $\vec{u}$ and $\vec{v}$ drawn in two dimensions: + + +```{julia} +u₁ = [1, 2] +v₁ = [2, 1] +p₁ = [0,0] + +plot(aspect_ratio=:equal) +arrow!(p₁, u₁) +arrow!(p₁, v₁) +arrow!(u₁, v₁) +arrow!(v₁, u₁) + +puv₁ = (u₁ ⋅ v₁) / (v₁ ⋅ v₁) * v₁ +porth₁ = u₁ - puv₁ +arrow!(puv₁, porth₁) +``` + +The enclosed shape is a parallelogram. To this we added the projection of $\vec{u}$ onto $\vec{v}$ (`puv`) and then the *orthogonal* part (`porth`). + + +The *area* of a parallelogram is the length of one side times the perpendicular height. The perpendicular height could be found from `norm(porth)`, so the area is: + + +```{julia} +norm(v₁) * norm(porth₁) +``` + +However, from trigonometry we have the height would also be the norm of $\vec{u}$ times $\sin(\theta)$, a value that is given through the length of the cross product of $\vec{u}$ and $\hat{v}$, the unit vector, were these vectors viewed as $3$ dimensional by adding a $0$ third component. In formulas, this is also the case: + + +$$ +\text{area of the parallelogram} = \| \vec{u} \times \hat{v} \| \| \vec{v} \| = \| \vec{u} \times \vec{v} \|. +$$ + +We have, for our figure, after extending `u` and `v` to be three dimensional the area of the parallelogram: + + +```{julia} +u₂ = [1, 2, 0] +v₂ = [2, 1, 0] +norm(u₂ × v₂) +``` + +--- + + +This analysis can be extended to the case of 3 vectors, which---when not co-planar---will form a *parallelepiped*. + + +```{julia} +u₃, v₃, w₃ = [1,2,3], [2,1,0], [1,1,2] +plot() +p₃ = [0,0,0] + +plot(legend=false) +arrow!(p₃, u₃); arrow!(p₃, v₃); arrow!(p₃, w₃) +arrow!(u₃, v₃); arrow!(u₃, w₃) +arrow!(v₃, u₃); arrow!(v₃, w₃) +arrow!(w₃, u₃); arrow!(w₃, v₃) +arrow!(u₃ + v₃, w₃); arrow!(u₃ + w₃, v₃); arrow!(v₃ + w₃, u₃) +``` + +The volume of a parallelepiped is the area of a base parallelogram times the height of a perpendicular. If $\vec{u}$ and $\vec{v}$ form the base parallelogram, then the perpendicular will have height $\|\vec{w}\| \cos(\theta)$ where the angle is the one made by $\vec{w}$ with the normal, $\vec{n}$. Since $\vec{u} \times \vec{v} = \| \vec{u} \times \vec{v}\| \hat{n} = \hat{n}$ times the area of the base parallelogram, we have if we dot this answer with $\vec{w}$: + + +$$ +(\vec{u} \times \vec{v}) \cdot \vec{w} = +\|\vec{u} \times \vec{v}\| (\vec{n} \cdot \vec{w}) = +\|\vec{u} \times \vec{v}\| \| \vec{w}\| \cos(\theta), +$$ + +that is, the area of the parallelepiped. Wait, what about $(\vec{v}\times\vec{u})\cdot\vec{w}$? That will have an opposite sign. Yes, in the above, there is an assumption that $\vec{n}$ and $\vec{w}$ have a an angle between them within $[0, \pi/2]$, otherwise an absolute value must be used, as volume is non-negative. + + +:::{.callout-note} +## Orientation +The triple-scalar product, $\vec{u}\cdot(\vec{v}\times\vec{w})$, gives the volume of the parallelepiped up to sign. If the sign of this is positive, the $3$ vectors are said to have a *positive* orientation, if the triple-scalar product is negative, the vectors have a *negative* orientation. + +::: + +#### Algebraic properties + + +The cross product has many properties, some different from regular multiplication: + + + * scalar multiplication: $(c\vec{u})\times\vec{v} = c(\vec{u}\times\vec{v})$ + * distributive over addition: $\vec{u} \times (\vec{v} + \vec{w}) = \vec{u}\times\vec{v} + \vec{u}\times\vec{w}$. + * *anti*-commutative: $\vec{u} \times \vec{v} = - \vec{v} \times \vec{u}$ + * *not* associative: that is there is no guarantee that $(\vec{u}\times\vec{v})\times\vec{w}$ will be equivalent to $\vec{u}\times(\vec{v}\times\vec{w})$. + * The triple cross product $(\vec{u}\times\vec{v}) \times \vec{w}$ must be orthogonal to $\vec{u}\times\vec{v}$ so lies in a plane with this as a normal vector. But, $\vec{u}$ and $\vec{v}$ will generate this plane, so it should be possible to express this triple product in terms of a sum involving $\vec{u}$ and $\vec{v}$ and indeed: + + +$$ +(\vec{u}\times\vec{v})\times\vec{w} = (\vec{u}\cdot\vec{w})\vec{v} - (\vec{v}\cdot\vec{w})\vec{u}. +$$ + +--- + + +The following shows the algebraic properties stated above hold for symbolic vectors. First the linearity of the dot product: + + +```{julia} +@syms s₄ t₄ u₄[1:3]::real v₄[1:3]::real w₄[1:3]::real + +u₄ ⋅ (s₄ * v₄ + t₄ * w₄) - (s₄ * (u₄ ⋅ v₄) + t₄ * (u₄ ⋅ w₄)) |> simplify +``` + +This shows the dot product is commutative: + + +```{julia} +(u₄ ⋅ v₄) - (v₄ ⋅ u₄) |> simplify +``` + +This shows the linearity of the cross product over scalar multiplication and vector addition: + + +```{julia} +u₄ × (s₄* v₄ + t₄ * w₄) - (s₄ * (u₄ × v₄) + t₄ * (u₄ × w₄)) .|> simplify +``` + +(We use `.|>` to broadcast `simplify` over each component.) + + +The cross product is anti-commutative: + + +```{julia} +u₄ × v₄ + v₄ × u₄ .|> simplify +``` + +but not associative: + + +```{julia} +u₄ × (v₄ × w₄) - (u₄ × v₄) × w₄ .|> simplify +``` + +Finally we verify the decomposition of the triple cross product: + + +```{julia} +(u₄ × v₄) × w₄ - ( (u₄ ⋅ w₄) * v₄ - (v₄ ⋅ w₄) * u₄) .|> simplify +``` + +--- + + +This table shows common usages of the symbols for various multiplication types: `*`, $\cdot$, and $\times$: + + +| Symbol | inputs | output | type | +|:--------:|:-------------- |:----------- |:---------------------- | +| `*` | scalar, scalar | scalar | regular multiplication | +| `*` | scalar, vector | vector | scalar multiplication | +| `*` | vector, vector | *undefined* | | +| $\cdot$ | scalar, scalar | scalar | regular multiplication | +| $\cdot$ | scalar, vector | vector | scalar multiplication | +| $\cdot$ | vector, vector | scalar | dot product | +| $\times$ | scalar, scalar | scalar | regular multiplication | +| $\times$ | scalar, vector | undefined | | +| $\times$ | vector, vector | vector | cross product ($3$D) | + + +### Parameterized lines and planes + + +A line in two dimensions satisfies the equation $ax + by = c$. Suppose $a$ and $b$ are non-zero. This can be represented in vector form, as the collection of all points associated to the vectors: $p + t \vec{v}$ where $p$ is a point on the line, say $(0,c/b)$, and v is the vector $\langle b, -a \rangle$. We can verify, this for values of `t` as follows: + + +```{julia} +#| hold: true +@syms a b c x y t + +eq = c - (a*x + b*y) + +p = [0, c/b] +v = [-b, a] +li = p + t * v + +eq(x=>li[1], y=>li[2]) |> simplify +``` + +Let $\vec{n} = \langle a , b \rangle$, taken from the coefficients in the equation. We can see directly that $\vec{n}$ is orthogonal to $\vec{v}$. The line may then be seen as the collection of all vectors that are orthogonal to $\vec{n}$ that have their tail at the point $p$. + +::: {#fig-parameterized-line} +```{julia} +#| echo: false +gr() +let + O = (0,0) + P = (1,2) + V = (2,-1/2) + V′ = (1/2, 2) + n̂ = V′ ./norm(V′) + V̂ = V ./ norm(V) + + l(t) = P .+ t.* V + + x′, y′ = 4,4 + plot(; empty_style..., aspect_ratio=:equal) + plot!([O, (x′, 0)]; axis_style...) + plot!([O, (0,y′)]; axis_style...) + annotate!([ + (0.95*x′, 0, text(L"x", :top, :left)), + (0, 0.95*y′, text(L"y", :right)) + ]) + + scatter!([P]; marker=(5,:black)) + plot!([P, P.+V]; arrow=true, line=(:black,3)) + plot!([P, P .+ n̂]; arrow=true, line=(:black, 2)) + r = 1/7 + plot!([(P.+ r .* V̂), (P .+ r .* V̂ .+ r .* n̂), (P .+ r .* n̂)]; line=(:black, 1)) + + plot!([l(t) for t in range(-3/4, 1.5, 10)]; line=(:black, 1)) + + plot!([O, l(0)]; arrow=true, line=(:gray, 1, :dash)) + plot!([O, l(1.25)]; arrow=true, line=(:gray, 1, :dash)) + + annotate!([ + (P..., text(L"p", :top, :left)), + ((P .+ V./2)..., text(L"\vec{v}", :bottom)), + ((P .+ n̂ ./ 2)..., text(L"\hat{n}", :top, :left)), + ((l(0)./2)..., text(L"t=0", :top, :left)), + ((l(1.25)./2)..., text(L"t=5/4", :top, :left)), + ((l(1.25))..., text(L"p + t \vec{v}", :bottom, :left)) + + ]) + +end +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + +Parameterized line through the point $p$ along vector $\vec{v}$. The line can be described by $l(t) = p + t\cdot \vec{v}$ or as all vector orthogonal to $\hat{n}$ anchored at $p$. +::: + + +In three dimensions, the equation of a plane is $ax + by + cz = d$. Suppose, $a$, $b$, and $c$ are non-zero, for simplicity. Setting $\vec{n} = \langle a,b,c\rangle$ by comparison, it can be seen that plane is identified with the set of all vectors orthogonal to $\vec{n}$ that are anchored at $p$. + + +::: {#fig-plane-parameterized} +```{julia} +#| echo: false +let + gr() + # cross product + O = (0, 0, 0) + P = (0, 0, 1) + u = (5/2, 2.05, -1.75) + v = (2, 1/3, 1/2) + kross(u,v)= tuple(cross(collect(u), collect(v))...) + + n = kross(v,u) + n̂ = 3 .* n ./ norm(n) + + plot(; empty_style...)#, aspect_ratio=:equal) + plot!(project.([O, (5, 0, 0)]); axis_style..., linealpha=0.5) + plot!(project.([O, (0, 5, 0)]); axis_style..., linealpha=0.5) + plot!(project.([O, (0, 0, 5)]); axis_style..., linealpha=0.5) + annotate!([ + (project((5, 0, 0))..., text(L"x",:top)), + (project((0, 5, 0))..., text(L"y",:top)), + (project((0, 0, 5))..., text(L"z",:left)), + ]) + + + + plot!(project.([P, P .+ u]); arrow=true, line=(:black, 1)) + plot!(project.([P, P .+ v]); arrow=true, line=(:black, 1)) + plot!(project.([P, P .+ n̂]); arrow=true, line=(:black, 1)) + + λ = 1.25 + pts = @. [P - λ*u - λ*v, P - λ*u + λ*v, P + λ*u + λ*v, P + λ*u - λ*v] + S = Shape(project.(pts)) + plot!(S; fill=(:gray90, 0.5), line=(:black, 1)) + + annotate!([ + (project(P .+ u./2)..., text(L"\vec{u}", :top)), + (project(P .+ v./2)..., text(L"\vec{v}", :top)), + (project(P .+ n̂./2)..., text(L"\hat{n}", :bottom, :right)), + (project(P)..., text(L"p", :top)) + ]) + + current() +end +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + +A plane can be described as all vectors orthogonal to $\hat{n}$ which are anchored at a point $p$ or through a parameterization $p + s\vec{u} + t\vec{v}$. +::: + +First, let $p = (0, 0, d/c)$ be a point on the plane. We find two vectors $u = \langle -b, a, 0 \rangle$ and $v = \langle 0, c, -b \rangle$. Then any point on the plane may be identified with the vector $p + s\vec{u} + t\vec{v}$. We can verify this algebraically through: + + +```{julia} +#| hold: true +@syms a b c d x y z s t + +eq = d - (a*x + b*y + c * z) + +p = [0, 0, d/c] +u, v = [-b, a, 0], [0, c, -b] +pl = p + t * u + s * v + +subs(eq, x=>pl[1], y=>pl[2], z=>pl[3]) |> simplify +``` + +The above viewpoint can be reversed: + + +> a plane is determined by two (non-parallel) vectors and a point. + + + +The parameterized version of the plane would be $p + s \vec{u} + t \vec{v}$, as used above. + + +The equation of the plane can be given from $\vec{u}$ and $\vec{v}$. Let $\vec{n} = \vec{u} \times \vec{v}$. Then $\vec{n} \cdot \vec{u} = \vec{n} \cdot \vec{v} = 0$, from the properties of the cross product. As such, $\vec{n} \cdot (s \vec{u} + t \vec{v}) = 0$. That is, the cross product is orthogonal to any *linear* combination of the two vectors. This figure shows one such linear combination: + +```{julia} +#| echo: false +gr(); nothing +``` + +```{julia} +#| hold: true +u = [1,2,3] +v = [2,3,1] +n = u × v +p = [0,0,1] + +plot(legend=false) + +arrow!(p, u) +arrow!(p, v) +arrow!(p + u, v) +arrow!(p + v, u) +arrow!(p, n) + +s, t = 1/2, 1/4 +arrow!(p, s*u + t*v) +``` + + +```{julia} +#| echo: false +plotly(); nothing +``` + +So if $\vec{n} \cdot p = d$ (identifying the point $p$ with a vector so the dot product is defined), we will have for any vector $\vec{v} = \langle x, y, z \rangle = s \vec{u} + t \vec{v}$ that + + +$$ +\vec{n} \cdot (p + s\vec{u} + t \vec{v}) = \vec{n} \cdot p + \vec{n} \cdot (s \vec{u} + t \vec{v}) = d + 0 = d, +$$ + +But if $\vec{n} = \langle a, b, c \rangle$, then this says $d = ax + by + cz$, so from $\vec{n}$ and $p$ the equation of the plane is given. + + +In summary: + + +| Object | Equation | vector equation | +|:------ |:------------------:|:-------------------------------- | +| Line | $ax + by = c$ | line: $p + t\vec{u}$ | +| Plane | $ax + by + cz = d$ | plane: $p + s\vec{u} + t\vec{v}$ | + + +--- + + +##### Example + + +You are given that the vectors $\vec{u} =\langle 6, 3, 1 \rangle$ and $\vec{v} = \langle 3, 2, 1 \rangle$ describe a plane through the point $p=[1,1,2]$. Find the equation of the plane. + + +The key is to find the normal vector to the plane, $\vec{n} = \vec{u} \times \vec{v}$: + + +```{julia} +#| hold: true +u, v, p = [6,3,1], [3,2,1], [1,1,2] +n = u × v +a, b, c = n +d = n ⋅ p +"equation of plane: $a x + $b y + $c z = $d" +``` + +##### Example: orthogonal + +A line is described by a point $P=(a,b)$ and a vector $\vec{v} = \langle v_1, v_2 \rangle$ through $r(t) = P + t \vec{v}$. + +A *normal* to the line is given by $\vec{n} = \langle -v_2, v_1 \rangle$. By construction, the value of $\vec{v} \cdot \vec{n}$ is $0$. + +The line can also be described in terms of orthogonality by the equation + +$$ +\vec{n} \cdot (\vec{x} - P) = 0, +$$ + +where we treat $P$ as a vector for purposes of subtraction. That is, the line is comprised of all vectors anchored at $P$ orthogonal to $\vec{n}$. + +Clearly, as $r(t) - P = \vec{v}$ we have $\vec{n} \cdot r(t) = 0$. Further, if $\vec{x}$ satisfies $\vec{n} \cdot (\vec{x} - P) = 0$ then again by flipping coordinates and taking a minus sign, we must have $\vec{x} - P$ is a multiple of $\langle v_1, v_2 \rangle = \vec{v}$. + + +Furthermore, suppose $P$ is a point in $R^3$ with two vectors $\vec{v}$ and $\vec{w}$ which are not parallel. Then the plane $P + s\vec{v} + t\vec{w}$ formed has a normal given by $\vec{n} = \vec{v} \times \vec{w}$. But +$\vec{n} \cdot ((P + s\vec{v} + t\vec{w}) - P) = s\vec{n} \cdot \vec{v} + t \vec{n} \cdot \vec{w} = 0$ so again, we have the plane described as well by" + +$$ +\vec{n} \cdot (\vec{x} - P) = 0. +$$ + +If $\vec{n} = \langle a,b,d \rangle$ then from $\vec{n}\cdot x = \vec{n} \cdot P$ this becomes $ax + by + cz = d = \vec{n}\cdot P$, as we used in the last example. + +##### Example: distance of a point + +Consider a line in $R^2$ and a point, $Q$, not on the line. How far is the nearest point on the line to $Q$? + +::: {#fig-distance-pt-line} +```{julia} +#| echo: false +gr() +let + # plot two vectors + P,Q = (3/2,1/2), (2, 2) + v = (1,1) + n = (-1, 1) ./ sqrt(2) + r(t) = P .+ t .* v + O = (0, 0) + plot(; empty_style..., aspect_ratio=:equal) + plot!([O, (3, 0)]; axis_style...) + plot!([O, (0,3)]; axis_style...) + annotate!([ + (3, 0, text(L"x", :top)), + (0, 3, text(L"y", :right)), + ]) + plot!([P, P .+ n]; arrow=true, line=(:black, 2)) + plot!([P, P .+ v]; arrow=true, line=(:black, 2)) + plot!([r(t) for t in range(-1, 1.4, 3)]; line=(:black, 1)) + scatter!([P, Q]; marker=(5,)) + + a = n ⋅ (Q .- P) + plot!([Q, Q .- a .* n ]; line=(:black, 1, :dash)) + plot!([P, P .+ a .* n]; arrow=true, line=(:black, 1, :dash)) + plot!([P .+ a .* n, Q]; arrow=true, line=(:black, 1, :dash)) + plot!([P, Q]; line=(:black, 1, :dash)) + + annotate!([ + (P..., text(L"P", :top, :left)), + (Q..., text(L"Q", :bottom, :right)), + ((P .+ v ./2)..., text(L"\vec{v}", :top, :left)), + ((P .+ n ./ 2)..., text(L"\hat{n}", :top, :right)), + ((P .+ (Q .- P) ./ 2 )..., text(L"\vec{w}", :top, :left)), + ((P .+ (a.*n) .+ (v ./2))..., + text(L"\|\vec{w}\cdot\hat{v}\|", :bottom; rotation=atand(1,1))), + ((Q .- (a/2).* n)..., + text(L"\|\vec{w}\cdot\hat{n}\|", :bottom; rotation=atand(-1,1))), + ]) + +end +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + +How far is point $Q$ to the line going through $P$ in direction $\vec{v}$? +::: + + +Suppose the line goes through the point $P$ and is in direction $\vec{v}$. Let $\hat{n}$ be a unit normal vector to $\vec{v}$ and $\hat{v}$ be the unit vector in direction $\vec{v}$. + +Then projection of $\vec{w}$ on $\vec{v}$ decomposes $\vec{w}$: + +$$ +\vec{w} = (\vec{w}\cdot\hat{v}) \hat{v} + a \hat{n} +$$ + +The length of $\text{proj}_{\vec{w}}(\vec{v})$ has been seen to be $\|\vec{w}\cdot\hat{v}\|$. The length of $a$ by a *similar* reason is $\|\vec{w} \cdot \hat{n}\|$. This is the length sought, so the distance from a point to a line is given by + +$$ +\| \hat{n} \cdot \vec{PQ} \|. +$$ + + +:::{.callout-note appearance="minimal"} +## Distance of point to a line and a plane +Let $\hat{n}$ be normal to a line (in $R^2$) or plane (in $R^3$) which goes through a point $P$. Then the distance from a point $Q$ to the line or plane is given by +$$ +\text{dist}(Q) = \| \hat{n} \cdot \vec{PQ} \|. +$$ +::: + + +##### Example: intersection + +In $R^3$ consider a plane through a point $P=(1,2,3)$ with normal vector $\vec{n} = \langle 1, 1/2, 1/4\rangle$. Suppose a line goes through points $A=(2,3,1)$ and $B=(3,1,2)$. Where does this line intersect the plane? + +We have the line is parameterized by $\vec{r}(t) = B + t \vec{AB}$ and the plane is described by $\vec{n} \cdot (\vec{x} - P) = 0$. The intersection point must be a $t$ that satisfies $\vec{n} \cdot (\vec{r}(t) - P) = 0$ + +We have: + +```{julia} +P, A, B = [1,2,3], [2,3,1], [3,1,2] +n = [1, 1//2, 1//4] +@syms t +𝑟(t) = B + t * (B - A) +t₀ = only(solve(n ⋅ (𝑟(t) -P) ~ 0, t)) +𝑟(t₀) +``` + +##### Example: intersection + +In $R^3$ two non-parallel planes will intersect in a line. How to describe that line? + +Let the planes be described by points $P_1$ and $P_2$ with normal vectors $\vec{n_1}$ and $\vec{n_2}$. The line will lie in each of the two planes, so any point on the line will satisfy $\vec{n_i} \cdot (\vec{x} - P)= 0$ for $i=1,2$. The vector $\vec{v} = \vec{n_1} \times \vec{n_2}$ is perpendicular to both $\vec{n_1}$ and $\vec{n_2}$, so the line is in the direction of $\vec{v}$. To describe the line we need a point $Q$ on the line, then the description is $Q + t \vec{v}$. + +Fix the $z$ coordinate of $Q$ to be $0$. Then we can solve for $Q$ in general through two linear equations in $x$ and $y$: + +```{julia} +@syms n₁[1:3]::real, n₂[1:3]::real, p₁[1:3]::real, p₂[1:3]::real, x::real, y::real +q = [x, y, 0] +solve((n₁ ⋅ (q - p₁) ~ 0, n₂ ⋅ (q - p₂) ~ 0), (x, y)) +``` + ## Matrices @@ -647,8 +1358,8 @@ The left hand sides are in the form of a dot product, in this case $\langle a,b $$ \begin{align*} -3x &+ 4y &- 5z &= 10\\ -3x &- 5y &+ 7z &= 11\\ + 3x &+ 4y &- 5z &= 10\\ + 3x &- 5y &+ 7z &= 11\\ -3x &+ 6y &+ 9z &= 12, \end{align*} $$ @@ -923,50 +1634,13 @@ u, v = [1,1,2], [3,5,8] reshape(u,(1,3)) * v ``` -## Cross product +##### Example: The cross product as a matrix operation -In three dimensions, there is a another operation between vectors that is similar to multiplication, though we will see with many differences. -Let $\vec{u}$ and $\vec{v}$ be two $3$-dimensional vectors, then the *cross* product, $\vec{u} \times \vec{v}$, is defined as a vector with length: - -$$ -\| \vec{u} \times \vec{v} \| = \| \vec{u} \| \| \vec{v} \| \sin(\theta), -$$ - -with $\theta$ being the angle in $[0, \pi]$ between $\vec{u}$ and $\vec{v}$. Consequently, $\sin(\theta) \geq 0$. - - -The direction of the cross product is such that it is *orthogonal* to *both* $\vec{u}$ and $\vec{v}$. There are two such directions, to identify which is correct, the [right-hand rule](https://en.wikipedia.org/wiki/Cross_product#Definition) is used. This rule points the right hand fingers in the direction of $\vec{u}$ and curls them towards $\vec{v}$ (so that the angle between the two vectors is in $[0, \pi]$). The thumb will point in the direction. Call this direction $\hat{n}$, a normal unit vector. Then the cross product can be defined by: - - -$$ -\vec{u} \times \vec{v} = \| \vec{u} \| \| \vec{v} \| \sin(\theta) \hat{n}. -$$ - -:::{callout-note} -## Note -The right-hand rule is also useful to understand how standard household screws will behave when twisted with a screwdriver. If the right hand fingers curl in the direction of the twisting screwdriver, then the screw will go in or out following the direction pointed to by the thumb. - -::: - -The right-hand rule depends on the order of consideration of the vectors. If they are reversed, the opposite direction is determined. A consequence is that the cross product is **anti**-commutative, unlike multiplication: - - -$$ -\vec{u} \times \vec{v} = - \vec{v} \times \vec{u}. -$$ - -Mathematically, the definition in terms of its components is a bit involved: - - -$$ -\vec{u} \times \vec{v} = \langle u_2 v_3 - u_3 v_2, u_3 v_1 - u_1 v_3, u_1 v_2 - u_2 v_1 \rangle. -$$ - -There is a matrix notation that can simplify this computation. If we *formally* define $\hat{i}$, $\hat{j}$, and $\hat{k}$ to represent unit vectors in the $x$, $y$, and $z$ direction, then a vector $\langle u_1, u_2, u_3 \rangle$ could be written $u_1\hat{i} + u_2\hat{j} + u_3\hat{k}$. With this the cross product of $\vec{u}$ and $\vec{v}$ is the vector associated with the *determinant* of the matrix +There is a matrix notation that can simplify the cross-product computation. If we *formally* define $\hat{i}$, $\hat{j}$, and $\hat{k}$ to represent unit vectors in the $x$, $y$, and $z$ direction, then a vector $\langle u_1, u_2, u_3 \rangle$ could be written $u_1\hat{i} + u_2\hat{j} + u_3\hat{k}$. With this the cross product of $\vec{u}$ and $\vec{v}$ is the vector associated with the *determinant* of the matrix $$ @@ -977,42 +1651,6 @@ v_1 & v_2 & v_3 \end{bmatrix} $$ -From the $\sin(\theta)$ term in the definition, we see that $\vec{u}\times\vec{u}=0$. In fact, the cross product is $0$ only if the two vectors involved are parallel or there is a zero vector. - - -In `Julia`, the `cross` function from the `LinearAlgebra` package implements the cross product. For example: - - -```{julia} -a = [1, 2, 3] -b = [4, 2, 1] -cross(a, b) -``` - -There is also the *infix* unicode operator `\times[tab]` that can be used for similarity to traditional mathematical syntax. - - -```{julia} -a × b -``` - -We can see the cross product is anti-commutative by comparing the last answer with: - - -```{julia} -b × a -``` - -Using vectors of size different than $n=3$ produces a dimension mismatch error: - - -```{julia} -#| error: true -[1, 2] × [3, 4] -``` - -(It can prove useful to pad $2$-dimensional vectors into $3$-dimensional vectors by adding a $0$ third component. We will see this in the discussion on curvature in the plane.) - Let's see that the matrix definition will be identical (after identifications) to `cross`: @@ -1033,286 +1671,6 @@ M[2,:] × M[3,:] --- -Consider this extended picture involving two vectors $\vec{u}$ and $\vec{v}$ drawn in two dimensions: - - -```{julia} -u₁ = [1, 2] -v₁ = [2, 1] -p₁ = [0,0] - -plot(aspect_ratio=:equal) -arrow!(p₁, u₁) -arrow!(p₁, v₁) -arrow!(u₁, v₁) -arrow!(v₁, u₁) - -puv₁ = (u₁ ⋅ v₁) / (v₁ ⋅ v₁) * v₁ -porth₁ = u₁ - puv₁ -arrow!(puv₁, porth₁) -``` - -The enclosed shape is a parallelogram. To this we added the projection of $\vec{u}$ onto $\vec{v}$ (`puv`) and then the *orthogonal* part (`porth`). - - -The *area* of a parallelogram is the length of one side times the perpendicular height. The perpendicular height could be found from `norm(porth)`, so the area is: - - -```{julia} -norm(v₁) * norm(porth₁) -``` - -However, from trigonometry we have the height would also be the norm of $\vec{u}$ times $\sin(\theta)$, a value that is given through the length of the cross product of $\vec{u}$ and $\hat{v}$, the unit vector, were these vectors viewed as $3$ dimensional by adding a $0$ third component. In formulas, this is also the case: - - -$$ -\text{area of the parallelogram} = \| \vec{u} \times \hat{v} \| \| \vec{v} \| = \| \vec{u} \times \vec{v} \|. -$$ - -We have, for our figure, after extending `u` and `v` to be three dimensional the area of the parallelogram: - - -```{julia} -u₂ = [1, 2, 0] -v₂ = [2, 1, 0] -norm(u₂ × v₂) -``` - ---- - - -This analysis can be extended to the case of 3 vectors, which---when not co-planar---will form a *parallelepiped*. - - -```{julia} -u₃, v₃, w₃ = [1,2,3], [2,1,0], [1,1,2] -plot() -p₃ = [0,0,0] - -plot(legend=false) -arrow!(p₃, u₃); arrow!(p₃, v₃); arrow!(p₃, w₃) -arrow!(u₃, v₃); arrow!(u₃, w₃) -arrow!(v₃, u₃); arrow!(v₃, w₃) -arrow!(w₃, u₃); arrow!(w₃, v₃) -arrow!(u₃ + v₃, w₃); arrow!(u₃ + w₃, v₃); arrow!(v₃ + w₃, u₃) -``` - -The volume of a parallelepiped is the area of a base parallelogram times the height of a perpendicular. If $\vec{u}$ and $\vec{v}$ form the base parallelogram, then the perpendicular will have height $\|\vec{w}\| \cos(\theta)$ where the angle is the one made by $\vec{w}$ with the normal, $\vec{n}$. Since $\vec{u} \times \vec{v} = \| \vec{u} \times \vec{v}\| \hat{n} = \hat{n}$ times the area of the base parallelogram, we have if we dot this answer with $\vec{w}$: - - -$$ -(\vec{u} \times \vec{v}) \cdot \vec{w} = -\|\vec{u} \times \vec{v}\| (\vec{n} \cdot \vec{w}) = -\|\vec{u} \times \vec{v}\| \| \vec{w}\| \cos(\theta), -$$ - -that is, the area of the parallelepiped. Wait, what about $(\vec{v}\times\vec{u})\cdot\vec{w}$? That will have an opposite sign. Yes, in the above, there is an assumption that $\vec{n}$ and $\vec{w}$ have a an angle between them within $[0, \pi/2]$, otherwise an absolute value must be used, as volume is non-negative. - - -:::{.callout-note} -## Orientation -The triple-scalar product, $\vec{u}\cdot(\vec{v}\times\vec{w})$, gives the volume of the parallelepiped up to sign. If the sign of this is positive, the $3$ vectors are said to have a *positive* orientation, if the triple-scalar product is negative, the vectors have a *negative* orientation. - -::: - -#### Algebraic properties - - -The cross product has many properties, some different from regular multiplication: - - - * scalar multiplication: $(c\vec{u})\times\vec{v} = c(\vec{u}\times\vec{v})$ - * distributive over addition: $\vec{u} \times (\vec{v} + \vec{w}) = \vec{u}\times\vec{v} + \vec{u}\times\vec{w}$. - * *anti*-commutative: $\vec{u} \times \vec{v} = - \vec{v} \times \vec{u}$ - * *not* associative: that is there is no guarantee that $(\vec{u}\times\vec{v})\times\vec{w}$ will be equivalent to $\vec{u}\times(\vec{v}\times\vec{w})$. - * The triple cross product $(\vec{u}\times\vec{v}) \times \vec{w}$ must be orthogonal to $\vec{u}\times\vec{v}$ so lies in a plane with this as a normal vector. But, $\vec{u}$ and $\vec{v}$ will generate this plane, so it should be possible to express this triple product in terms of a sum involving $\vec{u}$ and $\vec{v}$ and indeed: - - -$$ -(\vec{u}\times\vec{v})\times\vec{w} = (\vec{u}\cdot\vec{w})\vec{v} - (\vec{v}\cdot\vec{w})\vec{u}. -$$ - ---- - - -The following shows the algebraic properties stated above hold for symbolic vectors. First the linearity of the dot product: - - -```{julia} -@syms s₄ t₄ u₄[1:3]::real v₄[1:3]::real w₄[1:3]::real - -u₄ ⋅ (s₄ * v₄ + t₄ * w₄) - (s₄ * (u₄ ⋅ v₄) + t₄ * (u₄ ⋅ w₄)) |> simplify -``` - -This shows the dot product is commutative: - - -```{julia} -(u₄ ⋅ v₄) - (v₄ ⋅ u₄) |> simplify -``` - -This shows the linearity of the cross product over scalar multiplication and vector addition: - - -```{julia} -u₄ × (s₄* v₄ + t₄ * w₄) - (s₄ * (u₄ × v₄) + t₄ * (u₄ × w₄)) .|> simplify -``` - -(We use `.|>` to broadcast `simplify` over each component.) - - -The cross product is anti-commutative: - - -```{julia} -u₄ × v₄ + v₄ × u₄ .|> simplify -``` - -but not associative: - - -```{julia} -u₄ × (v₄ × w₄) - (u₄ × v₄) × w₄ .|> simplify -``` - -Finally we verify the decomposition of the triple cross product: - - -```{julia} -(u₄ × v₄) × w₄ - ( (u₄ ⋅ w₄) * v₄ - (v₄ ⋅ w₄) * u₄) .|> simplify -``` - ---- - - -This table shows common usages of the symbols for various multiplication types: `*`, $\cdot$, and $\times$: - - -| Symbol | inputs | output | type | -|:--------:|:-------------- |:----------- |:---------------------- | -| `*` | scalar, scalar | scalar | regular multiplication | -| `*` | scalar, vector | vector | scalar multiplication | -| `*` | vector, vector | *undefined* | | -| $\cdot$ | scalar, scalar | scalar | regular multiplication | -| $\cdot$ | scalar, vector | vector | scalar multiplication | -| $\cdot$ | vector, vector | scalar | dot product | -| $\times$ | scalar, scalar | scalar | regular multiplication | -| $\times$ | scalar, vector | undefined | | -| $\times$ | vector, vector | vector | cross product ($3$D) | - - -##### Example: lines and planes - - -A line in two dimensions satisfies the equation $ax + by = c$. Suppose $a$ and $b$ are non-zero. This can be represented in vector form, as the collection of all points associated to the vectors: $p + t \vec{v}$ where $p$ is a point on the line, say $(0,c/b)$, and v is the vector $\langle b, -a \rangle$. We can verify, this for values of `t` as follows: - - -```{julia} -#| hold: true -@syms a b c x y t - -eq = c - (a*x + b*y) - -p = [0, c/b] -v = [-b, a] -li = p + t * v - -eq(x=>li[1], y=>li[2]) |> simplify -``` - -Let $\vec{n} = \langle a , b \rangle$, taken from the coefficients in the equation. We can see directly that $\vec{n}$ is orthogonal to $\vec{v}$. The line may then be seen as the collection of all vectors that are orthogonal to $\vec{n}$ that have their tail at the point $p$. - - -In three dimensions, the equation of a plane is $ax + by + cz = d$. Suppose, $a$, $b$, and $c$ are non-zero, for simplicity. Setting $\vec{n} = \langle a,b,c\rangle$ by comparison, it can be seen that plane is identified with the set of all vectors orthogonal to $\vec{n}$ that are anchored at $p$. - - -First, let $p = (0, 0, d/c)$ be a point on the plane. We find two vectors $u = \langle -b, a, 0 \rangle$ and $v = \langle 0, c, -b \rangle$. Then any point on the plane may be identified with the vector $p + s\vec{u} + t\vec{v}$. We can verify this algebraically through: - - -```{julia} -#| hold: true -@syms a b c d x y z s t - -eq = d - (a*x + b*y + c * z) - -p = [0, 0, d/c] -u, v = [-b, a, 0], [0, c, -b] -pl = p + t * u + s * v - -subs(eq, x=>pl[1], y=>pl[2], z=>pl[3]) |> simplify -``` - -The above viewpoint can be reversed: - - -> a plane is determined by two (non-parallel) vectors and a point. - - - -The parameterized version of the plane would be $p + t \vec{u} + s \vec{v}$, as used above. - - -The equation of the plane can be given from $\vec{u}$ and $\vec{v}$. Let $\vec{n} = \vec{u} \times \vec{v}$. Then $\vec{n} \cdot \vec{u} = \vec{n} \cdot \vec{v} = 0$, from the properties of the cross product. As such, $\vec{n} \cdot (s \vec{u} + t \vec{v}) = 0$. That is, the cross product is orthogonal to any *linear* combination of the two vectors. This figure shows one such linear combination: - - -```{julia} -#| hold: true -u = [1,2,3] -v = [2,3,1] -n = u × v -p = [0,0,1] - -plot(legend=false) - -arrow!(p, u) -arrow!(p, v) -arrow!(p + u, v) -arrow!(p + v, u) -arrow!(p, n) - -s, t = 1/2, 1/4 -arrow!(p, s*u + t*v) -``` - -So if $\vec{n} \cdot p = d$ (identifying the point $p$ with a vector so the dot product is defined), we will have for any vector $\vec{v} = \langle x, y, z \rangle = s \vec{u} + t \vec{v}$ that - - -$$ -\vec{n} \cdot (p + s\vec{u} + t \vec{v}) = \vec{n} \cdot p + \vec{n} \cdot (s \vec{u} + t \vec{v}) = d + 0 = d, -$$ - -But if $\vec{n} = \langle a, b, c \rangle$, then this says $d = ax + by + cz$, so from $\vec{n}$ and $p$ the equation of the plane is given. - - -In summary: - - -| Object | Equation | vector equation | -|:------ |:------------------:|:-------------------------------- | -| Line | $ax + by = c$ | line: $p + t\vec{u}$ | -| Plane | $ax + by + cz = d$ | plane: $p + s\vec{u} + t\vec{v}$ | - - ---- - - -##### Example - - -You are given that the vectors $\vec{u} =\langle 6, 3, 1 \rangle$ and $\vec{v} = \langle 3, 2, 1 \rangle$ describe a plane through the point $p=[1,1,2]$. Find the equation of the plane. - - -The key is to find the normal vector to the plane, $\vec{n} = \vec{u} \times \vec{v}$: - - -```{julia} -#| hold: true -u, v, p = [6,3,1], [3,2,1], [1,1,2] -n = u × v -a, b, c = n -d = n ⋅ p -"equation of plane: $a x + $b y + $c z = $d" -``` ## Questions @@ -1393,6 +1751,76 @@ val = abs((u × v) ⋅ w) numericq(val) ``` +###### Question + +A parallelogram is formed by vectors $\vec{v}$ and $\vec{w}$. + +:::{#fig-parallelogram-question} +```{julia} +#| echo: false +gr() +let + # plot two vectors + P,Q = (1,2), (2, 1) + O = (0,0) + plot(; empty_style..., xlim=(0, 3.5)) + plot!([O, (3, 0)]; axis_style...) + plot!([O, (0,3)]; axis_style...) + + plot!([O, O .+ P]; arrow=true, line=(:black, 2)) + plot!([O, O .+ Q]; arrow=true, line=(:black, 2)) + plot!([Q, Q .+ P]; arrow=true, line=(:black, 1, :dash)) + plot!([P, Q .+ P]; arrow=true, line=(:black, 1, :dash)) + plot!([O, Q .+ P]; arrow=true, line=(:black, 1, :dash)) + plot!([P,Q]; arrow=true, line=(:red, 2)) + annotate!([ + (3, 0, text(L"x", :top)), + (0, 3, text(L"y", :right)), + (P..., text(L"\vec{w}", :top, :left)), + (Q..., text(L"\vec{v}", :top, :left)), + ((P .+Q)..., text(L"\vec{v}+\vec{w}", :top, :left)) + + ]) +end +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + +Parallelogram formed by two vectors $\vec{v}$ and $\vec{w}$. What is desription of red diagonal? +::: + +One diagonal is given by $\vec{v} + \vec{w}$. What is the other (as in the figure)? + +```{julia} +#| echo: false +choices = [ +L"\vec{v} \cdot \vec{w}", +L"\vec{v} \times \vec{w}", +L"\vec{v} - \vec{w}", +L"\vec{v} + \vec{w}", +] +radioq(choices, 3) +``` + +What is the area of the parallelogram? + + +```{julia} +#| echo: false +choices = [ +L"\vec{v} \cdot \vec{w}", +L"\vec{v} \times \vec{w}", +L"\vec{v} - \vec{w}", +L"\vec{v} + \vec{w}", +] +radioq(choices, 2) +``` + + ###### Question @@ -1446,11 +1874,11 @@ radioq(choices, answ) ###### Question -Let $\vec{u}$ and $\vec{v}$ be 3-dimensional **unit** vectors. What is the value of +Let $\hat{u}$ and $\hat{v}$ be 3-dimensional **unit** vectors. What is the value of $$ -(\vec{u} \times \vec{v}) \cdot (\vec{u} \times \vec{v}) + (\vec{u} \cdot \vec{v})^2? +(\hat{u} \times \hat{v}) \cdot (\hat{u} \times \hat{v}) + (\hat{u} \cdot \hat{v})^2? $$ ```{julia} @@ -1603,6 +2031,116 @@ radioq(choices, answ) ###### Question +In the figure, the tangent line to the unit circle for a given $\theta$ is given. + +```{julia} +#| echo: false +gr() +let + empty_style = (xaxis = ([], false), + yaxis = ([], false), + legend=false) + + @syms s θ + xx = (cos(θ), sin(θ)) + xx′ = diff.(xx, θ) + yy(s) = xx .+ s .* xx′ + + # x₀ + sx = only(solve(yy(s)[2] ~ 0, s)) + x₀ = first(yy(sx)) + + # y₀ + sy = only(solve(yy(s)[1] ~ 0, s)) + y₀ = last(yy(sy)) + + # plot with x-y axis + p = plot(; empty_style..., aspect_ratio=:equal) + plot!([(0, -2), (0, 2)]; arrow=true, line=(:gray, 1)) + plot!([(0, 0), (2, 0)]; arrow=true, line=(:gray, 1)) + + + θs = range(0, 2pi, 200) + plot!(sincos.(θs); line=(:black, 1)) + + θ₀ = 50*pi/180 + plot!([(0,0), reverse(sincos(θ₀))]; line=(:black, 1)) + + tl(x) = float(sin(θ₀) + last(xx′)(θ => θ₀) / first(xx′)(θ => θ₀) * (x - cos(θ₀))) + + x′ = -1/4 + plot!(tl, x′, 2) + + r′ = 1/6 + θs = range(0, θ₀, 100) + xys = [(r′*cos(t), r′*sin(t)) for t ∈ θs] + plot!(xys; line=(:gray, 1)) + + annotate!([ + (2r′*cos(θ₀/2), 2r′*sin(θ₀/2), text(L"θ")), + (float(x₀(θ=>θ₀)), 0, text(L"x_0", :top)), + (0, float(y₀(θ=>θ₀)), text(L"y_0", :right)), + (1/2*cos(θ₀), 1/2 * sin(θ₀), text(L"1", :bottom)) + ]) + current() +end +``` + +```{julia} +#| echo: false +plotly() +nothing +``` + +Using `SymPy`, find the values for $x_0$ and $y_0$. + +For $x_0$, this value might be of use: + +```{julia} +@syms s θ +r = [cos(θ), sin(θ)] +v = diff.(r, θ) +tl(s) = r .+ (s * v) # parametric description of the tangent line + +sₓ = only(solve(last(tl(s)) ~ 0, s)) +``` + +(Simplify `tl(sₓ)`.) + + +What is the value of $x_0$? + +```{julia} +#| echo: false +choices = [ + L"\cos(\theta)", + L"\sin(\theta)", + L"\frac{1}{\cos(\theta)}", + L"\frac{1}{\sin(\theta)}", + L"-\tan(θ)", + L"\frac{1}{\tan(\theta)}" +] +radioq(choices, 3) +``` + +What is the value of $y_0$? + +```{julia} +#| echo: false +choices = [ + L"\cos(\theta)", + L"\sin(\theta)", + L"\frac{1}{\cos(\theta)}", + L"\frac{1}{\sin(\theta)}", + L"-\tan(θ)", + L"\frac{1}{\tan(\theta)}" + +] +radioq(choices, 4) +``` + +###### Question + Give a geometric reason for this identity: From 8398f21b87e3f985c52f7bbd0cda86bf4251c26f Mon Sep 17 00:00:00 2001 From: jverzani Date: Tue, 29 Jul 2025 17:11:34 -0400 Subject: [PATCH 21/22] typos --- quarto/basics/vectors.qmd | 2 +- .../matrix_calculus_notes.qmd | 4 ++-- quarto/differentiable_vector_calculus/vectors.qmd | 2 +- quarto/integrals/area_between_curves.qmd | 2 +- quarto/integrals/ftc.qmd | 2 +- quarto/integrals/volumes_slice.qmd | 6 +++--- 6 files changed, 9 insertions(+), 9 deletions(-) diff --git a/quarto/basics/vectors.qmd b/quarto/basics/vectors.qmd index 9a5442e..dafa9b9 100644 --- a/quarto/basics/vectors.qmd +++ b/quarto/basics/vectors.qmd @@ -926,7 +926,7 @@ The latter using *splatting* to iterate over each value in `xs` and pass it to ` A few reductions work with *predicate* functions---those that return `true` or `false`. Let's use `iseven` as an example, which tests if a number is even. -We can check if *all* the alements of a container are even or if *any* of the elements of a container are even with `all` and `even`: +We can check if *all* the elements of a container are even or if *any* of the elements of a container are even with `all` and `even`: ```{julia} xs = [1, 1, 2, 3, 5] diff --git a/quarto/differentiable_vector_calculus/matrix_calculus_notes.qmd b/quarto/differentiable_vector_calculus/matrix_calculus_notes.qmd index cb65b27..bee6d93 100644 --- a/quarto/differentiable_vector_calculus/matrix_calculus_notes.qmd +++ b/quarto/differentiable_vector_calculus/matrix_calculus_notes.qmd @@ -88,7 +88,7 @@ Vectors and matrices have properties that are generalizations of the real number * Viewing a vector as a matrix is possible. The association chosen here is common and is through a *column* vector. -* The *transpose* of a matrix comes by permuting the rows and columns. The transpose of a column vector is a row vector, so $v\cdot w = v^T w$, where we use a superscript $T$ for the transpose. The transpose of a product, is the product of the transposes---reversed: $(AB)^T = B^T A^T$; the tranpose of a transpose is an identity operation: $(A^T)^T = A$; the inverse of a transpose is the tranpose of the inverse: $(A^{-1})^T = (A^T)^{-1}$. +* The *transpose* of a matrix comes by permuting the rows and columns. The transpose of a column vector is a row vector, so $v\cdot w = v^T w$, where we use a superscript $T$ for the transpose. The transpose of a product, is the product of the transposes---reversed: $(AB)^T = B^T A^T$; the transpose of a transpose is an identity operation: $(A^T)^T = A$; the inverse of a transpose is the transpose of the inverse: $(A^{-1})^T = (A^T)^{-1}$. * Matrices for which $A = A^T$ are called symmetric. @@ -133,7 +133,7 @@ The referenced notes identify $f'(x) dx$ as $f'(x)[dx]$, the latter emphasizing We take the view that a derivative is a linear operator where $df = f(x+dx) + f(x) = f'(x)[dx]$. -In writing $df = f(x + dx) - f(x) = f'(x)[dx]$ generically, some underlying facts are left implicit: $dx$ has the same shape as $x$ (so can be added) and there is an underlying concept of distance and size that allows the above to be made rigorous. This may be an abolute value or a norm. +In writing $df = f(x + dx) - f(x) = f'(x)[dx]$ generically, some underlying facts are left implicit: $dx$ has the same shape as $x$ (so can be added) and there is an underlying concept of distance and size that allows the above to be made rigorous. This may be an absolute value or a norm. ##### Example: directional derivatives diff --git a/quarto/differentiable_vector_calculus/vectors.qmd b/quarto/differentiable_vector_calculus/vectors.qmd index cf16f5c..8e625c1 100644 --- a/quarto/differentiable_vector_calculus/vectors.qmd +++ b/quarto/differentiable_vector_calculus/vectors.qmd @@ -1790,7 +1790,7 @@ plotly() nothing ``` -Parallelogram formed by two vectors $\vec{v}$ and $\vec{w}$. What is desription of red diagonal? +Parallelogram formed by two vectors $\vec{v}$ and $\vec{w}$. What is a description of red diagonal? ::: One diagonal is given by $\vec{v} + \vec{w}$. What is the other (as in the figure)? diff --git a/quarto/integrals/area_between_curves.qmd b/quarto/integrals/area_between_curves.qmd index 4088acc..ac7db6a 100644 --- a/quarto/integrals/area_between_curves.qmd +++ b/quarto/integrals/area_between_curves.qmd @@ -118,7 +118,7 @@ p = let # axis plot!([(A,0),(B,0)]; axis_style...) - # hightlight + # highlight x0, x1 = xp[marked], xp[marked+1] _style = (;line=(:gray, 1, :dash)) plot!([(a,0), (a, f(a))]; _style...) diff --git a/quarto/integrals/ftc.qmd b/quarto/integrals/ftc.qmd index 16bb31f..48c104f 100644 --- a/quarto/integrals/ftc.qmd +++ b/quarto/integrals/ftc.qmd @@ -1247,7 +1247,7 @@ Why is $F'(x) = \text{erf}'(x)$? ```{julia} #| echo: false -choices = ["The integrand is an *even* function so the itegral from ``0`` to ``x`` is the same as the integral from ``-x`` to ``0``", +choices = ["The integrand is an *even* function so the integral from ``0`` to ``x`` is the same as the integral from ``-x`` to ``0``", "This isn't true"] radioq(choices, 1; keep_order=true) ``` diff --git a/quarto/integrals/volumes_slice.qmd b/quarto/integrals/volumes_slice.qmd index 72a5f5b..be4af39 100644 --- a/quarto/integrals/volumes_slice.qmd +++ b/quarto/integrals/volumes_slice.qmd @@ -493,7 +493,7 @@ plt = let gr() # Follow lead of # https://github.com/SigurdAngenent/WisconsinCalculus/blob/master/figures/221/09surf_of_rotation2.py # plot surface of revolution around x axis between [0, 3] - # best if r(t) descreases + # best if r(t) decreases rad(x) = 2/(1 + exp(x)) trange = (0, 3) @@ -585,7 +585,7 @@ plotly() nothing ``` -Modification of earlier figure to show washer method. The interior volumn would be given by $\int_a^b \pi r(x)^2 dx$, the entire volume by $\int_a^b \pi R(x)^2 dx$. The difference then is the volume computed by the washer method. +Modification of earlier figure to show washer method. The interior volume would be given by $\int_a^b \pi r(x)^2 dx$, the entire volume by $\int_a^b \pi R(x)^2 dx$. The difference then is the volume computed by the washer method. ::: @@ -883,7 +883,7 @@ Consider a sphere with an interior cylinder bored out of it. (The [Napkin](http plt = let # Follow lead of # https://github.com/SigurdAngenent/WisconsinCalculus/blob/master/figures/221/09surf_of_rotation2.py # plot surface of revolution around x axis between [0, 3] - # best if r(t) descreases + # best if r(t) decreases rad(t) = (t = clamp(t, -1, 1); sqrt(1 - t^2)) rad2(t) = 1/2 From d7c0d08ddee842d2e0d4988d23f0d749694fc1fb Mon Sep 17 00:00:00 2001 From: jverzani Date: Tue, 29 Jul 2025 17:12:23 -0400 Subject: [PATCH 22/22] typos --- quarto/integrals/volumes_slice.qmd | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/quarto/integrals/volumes_slice.qmd b/quarto/integrals/volumes_slice.qmd index be4af39..28eb2a4 100644 --- a/quarto/integrals/volumes_slice.qmd +++ b/quarto/integrals/volumes_slice.qmd @@ -167,7 +167,7 @@ plt = let gr() # Follow lead of # https://github.com/SigurdAngenent/WisconsinCalculus/blob/master/figures/221/09surf_of_rotation2.py # plot surface of revolution around x axis between [0, 3] - # best if r(t) descreases + # best if r(t) decreases rad(x) = 2/(1 + exp(x)) trange = (0,3)