Spaces:
Sleeping
Sleeping
relevant assets and workflow
Browse filesThis view is limited to 50 files because it contains too many changes.
See raw diff
- .github/workflows/deploy.yml +56 -0
- _site/courses.json +216 -0
- _site/index.html +883 -0
- _site/optimization/.nojekyll +0 -0
- _site/optimization/01_least_squares.html +83 -0
- _site/optimization/02_linear_program.html +83 -0
- _site/optimization/03_minimum_fuel_optimal_control.html +83 -0
- _site/optimization/04_quadratic_program.html +83 -0
- _site/optimization/05_portfolio_optimization.html +83 -0
- _site/optimization/06_convex_optimization.html +83 -0
- _site/optimization/07_sdp.html +83 -0
- _site/optimization/android-chrome-192x192.png +0 -0
- _site/optimization/android-chrome-512x512.png +0 -0
- _site/optimization/apple-touch-icon.png +0 -0
- _site/optimization/assets/ConnectedDataExplorerComponent-D39SA74B.js +0 -0
- _site/optimization/assets/FiraMono-Bold-CLVRCuM9.ttf +0 -0
- _site/optimization/assets/FiraMono-Medium-DU3aDxX5.ttf +0 -0
- _site/optimization/assets/FiraMono-Regular-BTCkDNvf.ttf +0 -0
- _site/optimization/assets/KaTeX_AMS-Regular-BQhdFMY1.woff2 +0 -0
- _site/optimization/assets/KaTeX_AMS-Regular-DMm9YOAa.woff +0 -0
- _site/optimization/assets/KaTeX_AMS-Regular-DRggAlZN.ttf +0 -0
- _site/optimization/assets/KaTeX_Caligraphic-Bold-ATXxdsX0.ttf +0 -0
- _site/optimization/assets/KaTeX_Caligraphic-Bold-BEiXGLvX.woff +0 -0
- _site/optimization/assets/KaTeX_Caligraphic-Bold-Dq_IR9rO.woff2 +0 -0
- _site/optimization/assets/KaTeX_Caligraphic-Regular-CTRA-rTL.woff +0 -0
- _site/optimization/assets/KaTeX_Caligraphic-Regular-Di6jR-x-.woff2 +0 -0
- _site/optimization/assets/KaTeX_Caligraphic-Regular-wX97UBjC.ttf +0 -0
- _site/optimization/assets/KaTeX_Fraktur-Bold-BdnERNNW.ttf +0 -0
- _site/optimization/assets/KaTeX_Fraktur-Bold-BsDP51OF.woff +0 -0
- _site/optimization/assets/KaTeX_Fraktur-Bold-CL6g_b3V.woff2 +0 -0
- _site/optimization/assets/KaTeX_Fraktur-Regular-CB_wures.ttf +0 -0
- _site/optimization/assets/KaTeX_Fraktur-Regular-CTYiF6lA.woff2 +0 -0
- _site/optimization/assets/KaTeX_Fraktur-Regular-Dxdc4cR9.woff +0 -0
- _site/optimization/assets/KaTeX_Main-Bold-Cx986IdX.woff2 +0 -0
- _site/optimization/assets/KaTeX_Main-Bold-Jm3AIy58.woff +0 -0
- _site/optimization/assets/KaTeX_Main-Bold-waoOVXN0.ttf +0 -0
- _site/optimization/assets/KaTeX_Main-BoldItalic-DxDJ3AOS.woff2 +0 -0
- _site/optimization/assets/KaTeX_Main-BoldItalic-DzxPMmG6.ttf +0 -0
- _site/optimization/assets/KaTeX_Main-BoldItalic-SpSLRI95.woff +0 -0
- _site/optimization/assets/KaTeX_Main-Italic-3WenGoN9.ttf +0 -0
- _site/optimization/assets/KaTeX_Main-Italic-BMLOBm91.woff +0 -0
- _site/optimization/assets/KaTeX_Main-Italic-NWA7e6Wa.woff2 +0 -0
- _site/optimization/assets/KaTeX_Main-Regular-B22Nviop.woff2 +0 -0
- _site/optimization/assets/KaTeX_Main-Regular-Dr94JaBh.woff +0 -0
- _site/optimization/assets/KaTeX_Main-Regular-ypZvNtVU.ttf +0 -0
- _site/optimization/assets/KaTeX_Math-BoldItalic-B3XSjfu4.ttf +0 -0
- _site/optimization/assets/KaTeX_Math-BoldItalic-CZnvNsCZ.woff2 +0 -0
- _site/optimization/assets/KaTeX_Math-BoldItalic-iY-2wyZ7.woff +0 -0
- _site/optimization/assets/KaTeX_Math-Italic-DA0__PXp.woff +0 -0
- _site/optimization/assets/KaTeX_Math-Italic-flOr_0UB.ttf +0 -0
.github/workflows/deploy.yml
ADDED
@@ -0,0 +1,56 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
name: Deploy to GitHub Pages
|
2 |
+
|
3 |
+
on:
|
4 |
+
push:
|
5 |
+
branches: ['main']
|
6 |
+
workflow_dispatch:
|
7 |
+
|
8 |
+
concurrency:
|
9 |
+
group: 'pages'
|
10 |
+
cancel-in-progress: false
|
11 |
+
|
12 |
+
env:
|
13 |
+
UV_SYSTEM_PYTHON: 1
|
14 |
+
|
15 |
+
jobs:
|
16 |
+
build:
|
17 |
+
runs-on: ubuntu-latest
|
18 |
+
steps:
|
19 |
+
- uses: actions/checkout@v4
|
20 |
+
|
21 |
+
- name: 🚀 Install uv
|
22 |
+
uses: astral-sh/setup-uv@v4
|
23 |
+
|
24 |
+
- name: 🐍 Set up Python
|
25 |
+
uses: actions/setup-python@v5
|
26 |
+
with:
|
27 |
+
python-version: 3.12
|
28 |
+
|
29 |
+
- name: 📦 Install dependencies
|
30 |
+
run: |
|
31 |
+
uv pip install marimo
|
32 |
+
|
33 |
+
- name: 🛠️ Export notebooks
|
34 |
+
run: |
|
35 |
+
python scripts/build.py
|
36 |
+
|
37 |
+
- name: 📤 Upload artifact
|
38 |
+
uses: actions/upload-pages-artifact@v3
|
39 |
+
with:
|
40 |
+
path: _site
|
41 |
+
|
42 |
+
deploy:
|
43 |
+
needs: build
|
44 |
+
|
45 |
+
permissions:
|
46 |
+
pages: write
|
47 |
+
id-token: write
|
48 |
+
|
49 |
+
environment:
|
50 |
+
name: github-pages
|
51 |
+
url: ${{ steps.deployment.outputs.page_url }}
|
52 |
+
runs-on: ubuntu-latest
|
53 |
+
steps:
|
54 |
+
- name: 🚀 Deploy to GitHub Pages
|
55 |
+
id: deployment
|
56 |
+
uses: actions/deploy-pages@v4
|
_site/courses.json
ADDED
@@ -0,0 +1,216 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
{
|
2 |
+
"optimization": {
|
3 |
+
"id": "optimization",
|
4 |
+
"title": "Optimization",
|
5 |
+
"description": "This collection of marimo notebooks teaches you the basics of convex optimization.",
|
6 |
+
"notebooks": [
|
7 |
+
{
|
8 |
+
"id": "01_least_squares",
|
9 |
+
"path": "optimization\\01_least_squares.py",
|
10 |
+
"display_name": "Least Squares",
|
11 |
+
"order": 1
|
12 |
+
},
|
13 |
+
{
|
14 |
+
"id": "02_linear_program",
|
15 |
+
"path": "optimization\\02_linear_program.py",
|
16 |
+
"display_name": "Linear Program",
|
17 |
+
"order": 2
|
18 |
+
},
|
19 |
+
{
|
20 |
+
"id": "03_minimum_fuel_optimal_control",
|
21 |
+
"path": "optimization\\03_minimum_fuel_optimal_control.py",
|
22 |
+
"display_name": "Minimum Fuel Optimal Control",
|
23 |
+
"order": 3
|
24 |
+
},
|
25 |
+
{
|
26 |
+
"id": "04_quadratic_program",
|
27 |
+
"path": "optimization\\04_quadratic_program.py",
|
28 |
+
"display_name": "Quadratic Program",
|
29 |
+
"order": 4
|
30 |
+
},
|
31 |
+
{
|
32 |
+
"id": "05_portfolio_optimization",
|
33 |
+
"path": "optimization\\05_portfolio_optimization.py",
|
34 |
+
"display_name": "Portfolio Optimization",
|
35 |
+
"order": 5
|
36 |
+
},
|
37 |
+
{
|
38 |
+
"id": "06_convex_optimization",
|
39 |
+
"path": "optimization\\06_convex_optimization.py",
|
40 |
+
"display_name": "Convex Optimization",
|
41 |
+
"order": 6
|
42 |
+
},
|
43 |
+
{
|
44 |
+
"id": "07_sdp",
|
45 |
+
"path": "optimization\\07_sdp.py",
|
46 |
+
"display_name": "Sdp",
|
47 |
+
"order": 7
|
48 |
+
}
|
49 |
+
]
|
50 |
+
},
|
51 |
+
"polars": {
|
52 |
+
"id": "polars",
|
53 |
+
"title": "Polars",
|
54 |
+
"description": "_\ud83d\udea7 This collection is a work in progress. Please help us add notebooks!_",
|
55 |
+
"notebooks": [
|
56 |
+
{
|
57 |
+
"id": "01_why_polars",
|
58 |
+
"path": "polars\\01_why_polars.py",
|
59 |
+
"display_name": "Why Polars",
|
60 |
+
"order": 1
|
61 |
+
},
|
62 |
+
{
|
63 |
+
"id": "04_basic_operations",
|
64 |
+
"path": "polars\\04_basic_operations.py",
|
65 |
+
"display_name": "Basic Operations",
|
66 |
+
"order": 4
|
67 |
+
},
|
68 |
+
{
|
69 |
+
"id": "12_aggregations",
|
70 |
+
"path": "polars\\12_aggregations.py",
|
71 |
+
"display_name": "Aggregations",
|
72 |
+
"order": 12
|
73 |
+
},
|
74 |
+
{
|
75 |
+
"id": "14_user_defined_functions",
|
76 |
+
"path": "polars\\14_user_defined_functions.py",
|
77 |
+
"display_name": "User Defined Functions",
|
78 |
+
"order": 14
|
79 |
+
}
|
80 |
+
]
|
81 |
+
},
|
82 |
+
"probability": {
|
83 |
+
"id": "probability",
|
84 |
+
"title": "Probability",
|
85 |
+
"description": "\ud83d\udea7 _This collection is a work in progress. Check back later for new noteboks._",
|
86 |
+
"notebooks": [
|
87 |
+
{
|
88 |
+
"id": "01_sets",
|
89 |
+
"path": "probability\\01_sets.py",
|
90 |
+
"display_name": "Sets",
|
91 |
+
"order": 1
|
92 |
+
},
|
93 |
+
{
|
94 |
+
"id": "02_axioms",
|
95 |
+
"path": "probability\\02_axioms.py",
|
96 |
+
"display_name": "Axioms",
|
97 |
+
"order": 2
|
98 |
+
},
|
99 |
+
{
|
100 |
+
"id": "03_probability_of_or",
|
101 |
+
"path": "probability\\03_probability_of_or.py",
|
102 |
+
"display_name": "Probability Of Or",
|
103 |
+
"order": 3
|
104 |
+
},
|
105 |
+
{
|
106 |
+
"id": "04_conditional_probability",
|
107 |
+
"path": "probability\\04_conditional_probability.py",
|
108 |
+
"display_name": "Conditional Probability",
|
109 |
+
"order": 4
|
110 |
+
},
|
111 |
+
{
|
112 |
+
"id": "05_independence",
|
113 |
+
"path": "probability\\05_independence.py",
|
114 |
+
"display_name": "Independence",
|
115 |
+
"order": 5
|
116 |
+
},
|
117 |
+
{
|
118 |
+
"id": "06_probability_of_and",
|
119 |
+
"path": "probability\\06_probability_of_and.py",
|
120 |
+
"display_name": "Probability Of And",
|
121 |
+
"order": 6
|
122 |
+
},
|
123 |
+
{
|
124 |
+
"id": "07_law_of_total_probability",
|
125 |
+
"path": "probability\\07_law_of_total_probability.py",
|
126 |
+
"display_name": "Law Of Total Probability",
|
127 |
+
"order": 7
|
128 |
+
},
|
129 |
+
{
|
130 |
+
"id": "08_bayes_theorem",
|
131 |
+
"path": "probability\\08_bayes_theorem.py",
|
132 |
+
"display_name": "Bayes Theorem",
|
133 |
+
"order": 8
|
134 |
+
},
|
135 |
+
{
|
136 |
+
"id": "09_random_variables",
|
137 |
+
"path": "probability\\09_random_variables.py",
|
138 |
+
"display_name": "Random Variables",
|
139 |
+
"order": 9
|
140 |
+
},
|
141 |
+
{
|
142 |
+
"id": "10_probability_mass_function",
|
143 |
+
"path": "probability\\10_probability_mass_function.py",
|
144 |
+
"display_name": "Probability Mass Function",
|
145 |
+
"order": 10
|
146 |
+
},
|
147 |
+
{
|
148 |
+
"id": "11_expectation",
|
149 |
+
"path": "probability\\11_expectation.py",
|
150 |
+
"display_name": "Expectation",
|
151 |
+
"order": 11
|
152 |
+
}
|
153 |
+
]
|
154 |
+
},
|
155 |
+
"python": {
|
156 |
+
"id": "python",
|
157 |
+
"title": "Python",
|
158 |
+
"description": "This collection of marimo notebooks is designed to teach you the basics of the Python programming language.",
|
159 |
+
"notebooks": [
|
160 |
+
{
|
161 |
+
"id": "001_numbers",
|
162 |
+
"path": "python\\001_numbers.py",
|
163 |
+
"display_name": "Numbers",
|
164 |
+
"order": 1
|
165 |
+
},
|
166 |
+
{
|
167 |
+
"id": "002_strings",
|
168 |
+
"path": "python\\002_strings.py",
|
169 |
+
"display_name": "Strings",
|
170 |
+
"order": 2
|
171 |
+
},
|
172 |
+
{
|
173 |
+
"id": "003_collections",
|
174 |
+
"path": "python\\003_collections.py",
|
175 |
+
"display_name": "Collections",
|
176 |
+
"order": 3
|
177 |
+
},
|
178 |
+
{
|
179 |
+
"id": "004_conditional_logic",
|
180 |
+
"path": "python\\004_conditional_logic.py",
|
181 |
+
"display_name": "Conditional Logic",
|
182 |
+
"order": 4
|
183 |
+
},
|
184 |
+
{
|
185 |
+
"id": "005_loops",
|
186 |
+
"path": "python\\005_loops.py",
|
187 |
+
"display_name": "Loops",
|
188 |
+
"order": 5
|
189 |
+
},
|
190 |
+
{
|
191 |
+
"id": "007_advanced_collections",
|
192 |
+
"path": "python\\007_advanced_collections.py",
|
193 |
+
"display_name": "Advanced Collections",
|
194 |
+
"order": 7
|
195 |
+
},
|
196 |
+
{
|
197 |
+
"id": "008_functions",
|
198 |
+
"path": "python\\008_functions.py",
|
199 |
+
"display_name": "Functions",
|
200 |
+
"order": 8
|
201 |
+
},
|
202 |
+
{
|
203 |
+
"id": "009_modules",
|
204 |
+
"path": "python\\009_modules.py",
|
205 |
+
"display_name": "Modules",
|
206 |
+
"order": 9
|
207 |
+
},
|
208 |
+
{
|
209 |
+
"id": "010_exceptions",
|
210 |
+
"path": "python\\010_exceptions.py",
|
211 |
+
"display_name": "Exceptions",
|
212 |
+
"order": 10
|
213 |
+
}
|
214 |
+
]
|
215 |
+
}
|
216 |
+
}
|
_site/index.html
ADDED
@@ -0,0 +1,883 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<!DOCTYPE html>
|
2 |
+
<html lang="en">
|
3 |
+
<head>
|
4 |
+
<meta charset="UTF-8">
|
5 |
+
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
6 |
+
<title>Marimo Learn - Interactive Educational Notebooks</title>
|
7 |
+
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.4.0/css/all.min.css">
|
8 |
+
<style>
|
9 |
+
|
10 |
+
:root {
|
11 |
+
--eva-purple: #9a1eb3;
|
12 |
+
--eva-green: #00ff00;
|
13 |
+
--eva-orange: #ff6600;
|
14 |
+
--eva-blue: #0066ff;
|
15 |
+
--eva-red: #ff0000;
|
16 |
+
--eva-black: #111111;
|
17 |
+
--eva-dark: #222222;
|
18 |
+
--eva-terminal-bg: rgba(0, 0, 0, 0.85);
|
19 |
+
--eva-text: #e0e0e0;
|
20 |
+
--eva-border-radius: 4px;
|
21 |
+
--eva-transition: all 0.3s ease;
|
22 |
+
}
|
23 |
+
|
24 |
+
body {
|
25 |
+
background-color: var(--eva-black);
|
26 |
+
color: var(--eva-text);
|
27 |
+
font-family: 'Courier New', monospace;
|
28 |
+
margin: 0;
|
29 |
+
padding: 0;
|
30 |
+
line-height: 1.6;
|
31 |
+
}
|
32 |
+
|
33 |
+
.eva-container {
|
34 |
+
max-width: 1200px;
|
35 |
+
margin: 0 auto;
|
36 |
+
padding: 2rem;
|
37 |
+
}
|
38 |
+
|
39 |
+
.eva-header {
|
40 |
+
border-bottom: 2px solid var(--eva-green);
|
41 |
+
padding-bottom: 1rem;
|
42 |
+
margin-bottom: 2rem;
|
43 |
+
display: flex;
|
44 |
+
justify-content: space-between;
|
45 |
+
align-items: center;
|
46 |
+
position: sticky;
|
47 |
+
top: 0;
|
48 |
+
background-color: rgba(17, 17, 17, 0.95);
|
49 |
+
z-index: 100;
|
50 |
+
backdrop-filter: blur(5px);
|
51 |
+
padding-top: 1rem;
|
52 |
+
}
|
53 |
+
|
54 |
+
.eva-logo {
|
55 |
+
font-size: 2.5rem;
|
56 |
+
font-weight: bold;
|
57 |
+
color: var(--eva-green);
|
58 |
+
text-transform: uppercase;
|
59 |
+
letter-spacing: 2px;
|
60 |
+
text-shadow: 0 0 10px rgba(0, 255, 0, 0.5);
|
61 |
+
}
|
62 |
+
|
63 |
+
.eva-nav {
|
64 |
+
display: flex;
|
65 |
+
gap: 1.5rem;
|
66 |
+
}
|
67 |
+
|
68 |
+
.eva-nav a {
|
69 |
+
color: white;
|
70 |
+
text-decoration: none;
|
71 |
+
text-transform: uppercase;
|
72 |
+
font-size: 0.9rem;
|
73 |
+
letter-spacing: 1px;
|
74 |
+
transition: color 0.3s;
|
75 |
+
position: relative;
|
76 |
+
padding: 0.5rem 0;
|
77 |
+
}
|
78 |
+
|
79 |
+
.eva-nav a:hover {
|
80 |
+
color: var(--eva-green);
|
81 |
+
}
|
82 |
+
|
83 |
+
.eva-nav a:hover::after {
|
84 |
+
content: '';
|
85 |
+
position: absolute;
|
86 |
+
bottom: -5px;
|
87 |
+
left: 0;
|
88 |
+
width: 100%;
|
89 |
+
height: 2px;
|
90 |
+
background-color: var(--eva-green);
|
91 |
+
animation: scanline 1.5s linear infinite;
|
92 |
+
}
|
93 |
+
|
94 |
+
.eva-hero {
|
95 |
+
background-color: var(--eva-terminal-bg);
|
96 |
+
border: 1px solid var(--eva-green);
|
97 |
+
padding: 3rem 2rem;
|
98 |
+
margin-bottom: 3rem;
|
99 |
+
position: relative;
|
100 |
+
overflow: hidden;
|
101 |
+
border-radius: var(--eva-border-radius);
|
102 |
+
display: flex;
|
103 |
+
flex-direction: column;
|
104 |
+
align-items: flex-start;
|
105 |
+
background-image: linear-gradient(45deg, rgba(0, 0, 0, 0.9), rgba(0, 0, 0, 0.7)), url('https://raw.githubusercontent.com/marimo-team/marimo/main/docs/_static/marimo-logotype-thick.svg');
|
106 |
+
background-size: cover;
|
107 |
+
background-position: center;
|
108 |
+
background-blend-mode: overlay;
|
109 |
+
}
|
110 |
+
|
111 |
+
.eva-hero::before {
|
112 |
+
content: '';
|
113 |
+
position: absolute;
|
114 |
+
top: 0;
|
115 |
+
left: 0;
|
116 |
+
width: 100%;
|
117 |
+
height: 2px;
|
118 |
+
background-color: var(--eva-green);
|
119 |
+
animation: scanline 3s linear infinite;
|
120 |
+
}
|
121 |
+
|
122 |
+
.eva-hero h1 {
|
123 |
+
font-size: 2.5rem;
|
124 |
+
margin-bottom: 1rem;
|
125 |
+
color: var(--eva-green);
|
126 |
+
text-transform: uppercase;
|
127 |
+
letter-spacing: 2px;
|
128 |
+
text-shadow: 0 0 10px rgba(0, 255, 0, 0.5);
|
129 |
+
}
|
130 |
+
|
131 |
+
.eva-hero p {
|
132 |
+
font-size: 1.1rem;
|
133 |
+
max-width: 800px;
|
134 |
+
margin-bottom: 2rem;
|
135 |
+
line-height: 1.8;
|
136 |
+
}
|
137 |
+
|
138 |
+
.eva-features {
|
139 |
+
display: grid;
|
140 |
+
grid-template-columns: repeat(auto-fit, minmax(300px, 1fr));
|
141 |
+
gap: 2rem;
|
142 |
+
margin-bottom: 3rem;
|
143 |
+
}
|
144 |
+
|
145 |
+
.eva-feature {
|
146 |
+
background-color: var(--eva-terminal-bg);
|
147 |
+
border: 1px solid var(--eva-blue);
|
148 |
+
padding: 1.5rem;
|
149 |
+
border-radius: var(--eva-border-radius);
|
150 |
+
transition: var(--eva-transition);
|
151 |
+
position: relative;
|
152 |
+
overflow: hidden;
|
153 |
+
}
|
154 |
+
|
155 |
+
.eva-feature:hover {
|
156 |
+
transform: translateY(-5px);
|
157 |
+
box-shadow: 0 10px 20px rgba(0, 102, 255, 0.2);
|
158 |
+
}
|
159 |
+
|
160 |
+
.eva-feature-icon {
|
161 |
+
font-size: 2rem;
|
162 |
+
margin-bottom: 1rem;
|
163 |
+
color: var(--eva-blue);
|
164 |
+
}
|
165 |
+
|
166 |
+
.eva-feature h3 {
|
167 |
+
font-size: 1.3rem;
|
168 |
+
margin-bottom: 1rem;
|
169 |
+
color: var(--eva-blue);
|
170 |
+
}
|
171 |
+
|
172 |
+
.eva-section-title {
|
173 |
+
font-size: 2rem;
|
174 |
+
color: var(--eva-green);
|
175 |
+
margin-bottom: 2rem;
|
176 |
+
text-transform: uppercase;
|
177 |
+
letter-spacing: 2px;
|
178 |
+
text-align: center;
|
179 |
+
position: relative;
|
180 |
+
padding-bottom: 1rem;
|
181 |
+
}
|
182 |
+
|
183 |
+
.eva-section-title::after {
|
184 |
+
content: '';
|
185 |
+
position: absolute;
|
186 |
+
bottom: 0;
|
187 |
+
left: 50%;
|
188 |
+
transform: translateX(-50%);
|
189 |
+
width: 100px;
|
190 |
+
height: 2px;
|
191 |
+
background-color: var(--eva-green);
|
192 |
+
}
|
193 |
+
|
194 |
+
.eva-courses {
|
195 |
+
display: grid;
|
196 |
+
grid-template-columns: repeat(auto-fill, minmax(350px, 1fr));
|
197 |
+
gap: 2rem;
|
198 |
+
}
|
199 |
+
|
200 |
+
.eva-course {
|
201 |
+
background-color: var(--eva-terminal-bg);
|
202 |
+
border: 1px solid var(--eva-purple);
|
203 |
+
border-radius: var(--eva-border-radius);
|
204 |
+
transition: var(--eva-transition);
|
205 |
+
position: relative;
|
206 |
+
overflow: hidden;
|
207 |
+
}
|
208 |
+
|
209 |
+
.eva-course:hover {
|
210 |
+
transform: translateY(-5px);
|
211 |
+
box-shadow: 0 10px 20px rgba(154, 30, 179, 0.3);
|
212 |
+
}
|
213 |
+
|
214 |
+
.eva-course::after {
|
215 |
+
content: '';
|
216 |
+
position: absolute;
|
217 |
+
bottom: 0;
|
218 |
+
left: 0;
|
219 |
+
width: 100%;
|
220 |
+
height: 2px;
|
221 |
+
background-color: var(--eva-purple);
|
222 |
+
animation: scanline 2s linear infinite;
|
223 |
+
}
|
224 |
+
|
225 |
+
.eva-course-header {
|
226 |
+
padding: 1.5rem;
|
227 |
+
cursor: pointer;
|
228 |
+
display: flex;
|
229 |
+
justify-content: space-between;
|
230 |
+
align-items: center;
|
231 |
+
border-bottom: 1px solid rgba(154, 30, 179, 0.3);
|
232 |
+
}
|
233 |
+
|
234 |
+
.eva-course-title {
|
235 |
+
font-size: 1.5rem;
|
236 |
+
color: var(--eva-purple);
|
237 |
+
text-transform: uppercase;
|
238 |
+
letter-spacing: 1px;
|
239 |
+
margin: 0;
|
240 |
+
}
|
241 |
+
|
242 |
+
.eva-course-toggle {
|
243 |
+
color: var(--eva-purple);
|
244 |
+
font-size: 1.5rem;
|
245 |
+
transition: var(--eva-transition);
|
246 |
+
}
|
247 |
+
|
248 |
+
.eva-course-content {
|
249 |
+
padding: 0 1.5rem;
|
250 |
+
max-height: 0;
|
251 |
+
overflow: hidden;
|
252 |
+
transition: var(--eva-transition);
|
253 |
+
}
|
254 |
+
|
255 |
+
.eva-course.active .eva-course-content {
|
256 |
+
padding: 1.5rem;
|
257 |
+
max-height: 1000px;
|
258 |
+
}
|
259 |
+
|
260 |
+
.eva-course.active .eva-course-toggle {
|
261 |
+
transform: rotate(180deg);
|
262 |
+
}
|
263 |
+
|
264 |
+
.eva-course-description {
|
265 |
+
margin-bottom: 1.5rem;
|
266 |
+
font-size: 0.9rem;
|
267 |
+
line-height: 1.6;
|
268 |
+
}
|
269 |
+
|
270 |
+
.eva-notebooks {
|
271 |
+
margin-top: 1rem;
|
272 |
+
}
|
273 |
+
|
274 |
+
.eva-notebook {
|
275 |
+
margin-bottom: 0.75rem;
|
276 |
+
padding: 0.5rem;
|
277 |
+
border-left: 2px solid var(--eva-blue);
|
278 |
+
transition: var(--eva-transition);
|
279 |
+
display: flex;
|
280 |
+
align-items: center;
|
281 |
+
}
|
282 |
+
|
283 |
+
.eva-notebook:hover {
|
284 |
+
background-color: rgba(0, 102, 255, 0.1);
|
285 |
+
padding-left: 1rem;
|
286 |
+
}
|
287 |
+
|
288 |
+
.eva-notebook a {
|
289 |
+
color: white;
|
290 |
+
text-decoration: none;
|
291 |
+
display: block;
|
292 |
+
font-size: 0.9rem;
|
293 |
+
flex-grow: 1;
|
294 |
+
}
|
295 |
+
|
296 |
+
.eva-notebook a:hover {
|
297 |
+
color: var(--eva-blue);
|
298 |
+
}
|
299 |
+
|
300 |
+
.eva-notebook-number {
|
301 |
+
color: var(--eva-blue);
|
302 |
+
font-size: 0.8rem;
|
303 |
+
margin-right: 0.5rem;
|
304 |
+
opacity: 0.7;
|
305 |
+
min-width: 24px;
|
306 |
+
}
|
307 |
+
|
308 |
+
.eva-button {
|
309 |
+
display: inline-block;
|
310 |
+
background-color: transparent;
|
311 |
+
color: var(--eva-green);
|
312 |
+
border: 1px solid var(--eva-green);
|
313 |
+
padding: 0.7rem 1.5rem;
|
314 |
+
text-decoration: none;
|
315 |
+
text-transform: uppercase;
|
316 |
+
font-size: 0.9rem;
|
317 |
+
letter-spacing: 1px;
|
318 |
+
transition: var(--eva-transition);
|
319 |
+
cursor: pointer;
|
320 |
+
border-radius: var(--eva-border-radius);
|
321 |
+
position: relative;
|
322 |
+
overflow: hidden;
|
323 |
+
}
|
324 |
+
|
325 |
+
.eva-button:hover {
|
326 |
+
background-color: var(--eva-green);
|
327 |
+
color: var(--eva-black);
|
328 |
+
}
|
329 |
+
|
330 |
+
.eva-button::after {
|
331 |
+
content: '';
|
332 |
+
position: absolute;
|
333 |
+
top: 0;
|
334 |
+
left: -100%;
|
335 |
+
width: 100%;
|
336 |
+
height: 100%;
|
337 |
+
background: linear-gradient(90deg, transparent, rgba(255, 255, 255, 0.2), transparent);
|
338 |
+
transition: 0.5s;
|
339 |
+
}
|
340 |
+
|
341 |
+
.eva-button:hover::after {
|
342 |
+
left: 100%;
|
343 |
+
}
|
344 |
+
|
345 |
+
.eva-cta {
|
346 |
+
background-color: var(--eva-terminal-bg);
|
347 |
+
border: 1px solid var(--eva-orange);
|
348 |
+
padding: 3rem 2rem;
|
349 |
+
margin: 4rem 0;
|
350 |
+
text-align: center;
|
351 |
+
border-radius: var(--eva-border-radius);
|
352 |
+
position: relative;
|
353 |
+
overflow: hidden;
|
354 |
+
}
|
355 |
+
|
356 |
+
.eva-cta h2 {
|
357 |
+
font-size: 2rem;
|
358 |
+
color: var(--eva-orange);
|
359 |
+
margin-bottom: 1.5rem;
|
360 |
+
text-transform: uppercase;
|
361 |
+
}
|
362 |
+
|
363 |
+
.eva-cta p {
|
364 |
+
max-width: 600px;
|
365 |
+
margin: 0 auto 2rem;
|
366 |
+
font-size: 1.1rem;
|
367 |
+
}
|
368 |
+
|
369 |
+
.eva-cta .eva-button {
|
370 |
+
color: var(--eva-orange);
|
371 |
+
border-color: var(--eva-orange);
|
372 |
+
}
|
373 |
+
|
374 |
+
.eva-cta .eva-button:hover {
|
375 |
+
background-color: var(--eva-orange);
|
376 |
+
color: var(--eva-black);
|
377 |
+
}
|
378 |
+
|
379 |
+
.eva-footer {
|
380 |
+
margin-top: 4rem;
|
381 |
+
padding-top: 2rem;
|
382 |
+
border-top: 2px solid var(--eva-green);
|
383 |
+
display: flex;
|
384 |
+
justify-content: space-between;
|
385 |
+
align-items: center;
|
386 |
+
flex-wrap: wrap;
|
387 |
+
gap: 2rem;
|
388 |
+
}
|
389 |
+
|
390 |
+
.eva-footer-links {
|
391 |
+
display: flex;
|
392 |
+
gap: 1.5rem;
|
393 |
+
}
|
394 |
+
|
395 |
+
.eva-footer-links a {
|
396 |
+
color: var(--eva-text);
|
397 |
+
text-decoration: none;
|
398 |
+
transition: var(--eva-transition);
|
399 |
+
}
|
400 |
+
|
401 |
+
.eva-footer-links a:hover {
|
402 |
+
color: var(--eva-green);
|
403 |
+
}
|
404 |
+
|
405 |
+
.eva-footer-copyright {
|
406 |
+
font-size: 0.9rem;
|
407 |
+
}
|
408 |
+
|
409 |
+
.eva-search {
|
410 |
+
position: relative;
|
411 |
+
margin-bottom: 3rem;
|
412 |
+
}
|
413 |
+
|
414 |
+
.eva-search input {
|
415 |
+
width: 100%;
|
416 |
+
padding: 1rem;
|
417 |
+
background-color: var(--eva-terminal-bg);
|
418 |
+
border: 1px solid var(--eva-green);
|
419 |
+
color: var(--eva-text);
|
420 |
+
font-family: 'Courier New', monospace;
|
421 |
+
font-size: 1rem;
|
422 |
+
border-radius: var(--eva-border-radius);
|
423 |
+
outline: none;
|
424 |
+
transition: var(--eva-transition);
|
425 |
+
}
|
426 |
+
|
427 |
+
.eva-search input:focus {
|
428 |
+
box-shadow: 0 0 10px rgba(0, 255, 0, 0.3);
|
429 |
+
}
|
430 |
+
|
431 |
+
.eva-search input::placeholder {
|
432 |
+
color: rgba(224, 224, 224, 0.5);
|
433 |
+
}
|
434 |
+
|
435 |
+
.eva-search-icon {
|
436 |
+
position: absolute;
|
437 |
+
right: 1rem;
|
438 |
+
top: 50%;
|
439 |
+
transform: translateY(-50%);
|
440 |
+
color: var(--eva-green);
|
441 |
+
font-size: 1.2rem;
|
442 |
+
}
|
443 |
+
|
444 |
+
@keyframes scanline {
|
445 |
+
0% {
|
446 |
+
transform: translateX(-100%);
|
447 |
+
}
|
448 |
+
100% {
|
449 |
+
transform: translateX(100%);
|
450 |
+
}
|
451 |
+
}
|
452 |
+
|
453 |
+
@keyframes blink {
|
454 |
+
0%, 100% {
|
455 |
+
opacity: 1;
|
456 |
+
}
|
457 |
+
50% {
|
458 |
+
opacity: 0;
|
459 |
+
}
|
460 |
+
}
|
461 |
+
|
462 |
+
.eva-cursor {
|
463 |
+
display: inline-block;
|
464 |
+
width: 10px;
|
465 |
+
height: 1.2em;
|
466 |
+
background-color: var(--eva-green);
|
467 |
+
margin-left: 2px;
|
468 |
+
animation: blink 1s infinite;
|
469 |
+
vertical-align: middle;
|
470 |
+
}
|
471 |
+
|
472 |
+
@media (max-width: 768px) {
|
473 |
+
.eva-courses {
|
474 |
+
grid-template-columns: 1fr;
|
475 |
+
}
|
476 |
+
|
477 |
+
.eva-header {
|
478 |
+
flex-direction: column;
|
479 |
+
align-items: flex-start;
|
480 |
+
padding: 1rem;
|
481 |
+
}
|
482 |
+
|
483 |
+
.eva-nav {
|
484 |
+
margin-top: 1rem;
|
485 |
+
flex-wrap: wrap;
|
486 |
+
}
|
487 |
+
|
488 |
+
.eva-hero {
|
489 |
+
padding: 2rem 1rem;
|
490 |
+
}
|
491 |
+
|
492 |
+
.eva-hero h1 {
|
493 |
+
font-size: 2rem;
|
494 |
+
}
|
495 |
+
|
496 |
+
.eva-features {
|
497 |
+
grid-template-columns: 1fr;
|
498 |
+
}
|
499 |
+
|
500 |
+
.eva-footer {
|
501 |
+
flex-direction: column;
|
502 |
+
align-items: center;
|
503 |
+
text-align: center;
|
504 |
+
}
|
505 |
+
}
|
506 |
+
|
507 |
+
</style>
|
508 |
+
</head>
|
509 |
+
<body>
|
510 |
+
<div class="eva-container">
|
511 |
+
<header class="eva-header">
|
512 |
+
<div class="eva-logo">MARIMO LEARN</div>
|
513 |
+
<nav class="eva-nav">
|
514 |
+
<a href="#features">Features</a>
|
515 |
+
<a href="#courses">Courses</a>
|
516 |
+
<a href="#contribute">Contribute</a>
|
517 |
+
<a href="https://docs.marimo.io" target="_blank">Documentation</a>
|
518 |
+
<a href="https://github.com/marimo-team/learn" target="_blank">GitHub</a>
|
519 |
+
</nav>
|
520 |
+
</header>
|
521 |
+
|
522 |
+
<section class="eva-hero">
|
523 |
+
<h1>Interactive Learning with Marimo<span class="eva-cursor"></span></h1>
|
524 |
+
<p>
|
525 |
+
A curated collection of educational notebooks covering computer science,
|
526 |
+
mathematics, data science, and more. Built with marimo - the reactive
|
527 |
+
Python notebook that makes data exploration delightful.
|
528 |
+
</p>
|
529 |
+
<a href="#courses" class="eva-button">Explore Courses</a>
|
530 |
+
</section>
|
531 |
+
|
532 |
+
<section id="features">
|
533 |
+
<h2 class="eva-section-title">Why Marimo Learn?</h2>
|
534 |
+
<div class="eva-features">
|
535 |
+
<div class="eva-feature">
|
536 |
+
<div class="eva-feature-icon"><i class="fas fa-bolt"></i></div>
|
537 |
+
<h3>Reactive Notebooks</h3>
|
538 |
+
<p>Experience the power of reactive programming with marimo notebooks that automatically update when dependencies change.</p>
|
539 |
+
</div>
|
540 |
+
<div class="eva-feature">
|
541 |
+
<div class="eva-feature-icon"><i class="fas fa-code"></i></div>
|
542 |
+
<h3>Learn by Doing</h3>
|
543 |
+
<p>Interactive examples and exercises help you understand concepts through hands-on practice.</p>
|
544 |
+
</div>
|
545 |
+
<div class="eva-feature">
|
546 |
+
<div class="eva-feature-icon"><i class="fas fa-graduation-cap"></i></div>
|
547 |
+
<h3>Comprehensive Courses</h3>
|
548 |
+
<p>From Python basics to advanced optimization techniques, our courses cover a wide range of topics.</p>
|
549 |
+
</div>
|
550 |
+
</div>
|
551 |
+
</section>
|
552 |
+
|
553 |
+
<section id="courses">
|
554 |
+
<h2 class="eva-section-title">Explore Courses</h2>
|
555 |
+
<div class="eva-search">
|
556 |
+
<input type="text" id="courseSearch" placeholder="Search courses and notebooks...">
|
557 |
+
<span class="eva-search-icon"><i class="fas fa-search"></i></span>
|
558 |
+
</div>
|
559 |
+
<div class="eva-courses">
|
560 |
+
<div class="eva-course" data-course-id="optimization">
|
561 |
+
<div class="eva-course-header">
|
562 |
+
<h2 class="eva-course-title">Optimization</h2>
|
563 |
+
<span class="eva-course-toggle"><i class="fas fa-chevron-down"></i></span>
|
564 |
+
</div>
|
565 |
+
<div class="eva-course-content">
|
566 |
+
<p class="eva-course-description">This collection of marimo notebooks teaches you the basics of convex optimization.</p>
|
567 |
+
<div class="eva-notebooks">
|
568 |
+
<div class="eva-notebook">
|
569 |
+
<span class="eva-notebook-number">01</span>
|
570 |
+
<a href="optimization\01_least_squares.html" data-notebook-title="Least Squares">Least Squares</a>
|
571 |
+
</div>
|
572 |
+
<div class="eva-notebook">
|
573 |
+
<span class="eva-notebook-number">02</span>
|
574 |
+
<a href="optimization\02_linear_program.html" data-notebook-title="Linear Program">Linear Program</a>
|
575 |
+
</div>
|
576 |
+
<div class="eva-notebook">
|
577 |
+
<span class="eva-notebook-number">03</span>
|
578 |
+
<a href="optimization\03_minimum_fuel_optimal_control.html" data-notebook-title="Minimum Fuel Optimal Control">Minimum Fuel Optimal Control</a>
|
579 |
+
</div>
|
580 |
+
<div class="eva-notebook">
|
581 |
+
<span class="eva-notebook-number">04</span>
|
582 |
+
<a href="optimization\04_quadratic_program.html" data-notebook-title="Quadratic Program">Quadratic Program</a>
|
583 |
+
</div>
|
584 |
+
<div class="eva-notebook">
|
585 |
+
<span class="eva-notebook-number">05</span>
|
586 |
+
<a href="optimization\05_portfolio_optimization.html" data-notebook-title="Portfolio Optimization">Portfolio Optimization</a>
|
587 |
+
</div>
|
588 |
+
<div class="eva-notebook">
|
589 |
+
<span class="eva-notebook-number">06</span>
|
590 |
+
<a href="optimization\06_convex_optimization.html" data-notebook-title="Convex Optimization">Convex Optimization</a>
|
591 |
+
</div>
|
592 |
+
<div class="eva-notebook">
|
593 |
+
<span class="eva-notebook-number">07</span>
|
594 |
+
<a href="optimization\07_sdp.html" data-notebook-title="Sdp">Sdp</a>
|
595 |
+
</div>
|
596 |
+
</div>
|
597 |
+
</div>
|
598 |
+
</div>
|
599 |
+
<div class="eva-course" data-course-id="polars">
|
600 |
+
<div class="eva-course-header">
|
601 |
+
<h2 class="eva-course-title">Polars</h2>
|
602 |
+
<span class="eva-course-toggle"><i class="fas fa-chevron-down"></i></span>
|
603 |
+
</div>
|
604 |
+
<div class="eva-course-content">
|
605 |
+
<p class="eva-course-description">_🚧 This collection is a work in progress. Please help us add notebooks!_</p>
|
606 |
+
<div class="eva-notebooks">
|
607 |
+
<div class="eva-notebook">
|
608 |
+
<span class="eva-notebook-number">01</span>
|
609 |
+
<a href="polars\01_why_polars.html" data-notebook-title="Why Polars">Why Polars</a>
|
610 |
+
</div>
|
611 |
+
<div class="eva-notebook">
|
612 |
+
<span class="eva-notebook-number">02</span>
|
613 |
+
<a href="polars\04_basic_operations.html" data-notebook-title="Basic Operations">Basic Operations</a>
|
614 |
+
</div>
|
615 |
+
<div class="eva-notebook">
|
616 |
+
<span class="eva-notebook-number">03</span>
|
617 |
+
<a href="polars\12_aggregations.html" data-notebook-title="Aggregations">Aggregations</a>
|
618 |
+
</div>
|
619 |
+
<div class="eva-notebook">
|
620 |
+
<span class="eva-notebook-number">04</span>
|
621 |
+
<a href="polars\14_user_defined_functions.html" data-notebook-title="User Defined Functions">User Defined Functions</a>
|
622 |
+
</div>
|
623 |
+
</div>
|
624 |
+
</div>
|
625 |
+
</div>
|
626 |
+
<div class="eva-course" data-course-id="probability">
|
627 |
+
<div class="eva-course-header">
|
628 |
+
<h2 class="eva-course-title">Probability</h2>
|
629 |
+
<span class="eva-course-toggle"><i class="fas fa-chevron-down"></i></span>
|
630 |
+
</div>
|
631 |
+
<div class="eva-course-content">
|
632 |
+
<p class="eva-course-description">🚧 _This collection is a work in progress. Check back later for new noteboks._</p>
|
633 |
+
<div class="eva-notebooks">
|
634 |
+
<div class="eva-notebook">
|
635 |
+
<span class="eva-notebook-number">01</span>
|
636 |
+
<a href="probability\01_sets.html" data-notebook-title="Sets">Sets</a>
|
637 |
+
</div>
|
638 |
+
<div class="eva-notebook">
|
639 |
+
<span class="eva-notebook-number">02</span>
|
640 |
+
<a href="probability\02_axioms.html" data-notebook-title="Axioms">Axioms</a>
|
641 |
+
</div>
|
642 |
+
<div class="eva-notebook">
|
643 |
+
<span class="eva-notebook-number">03</span>
|
644 |
+
<a href="probability\03_probability_of_or.html" data-notebook-title="Probability Of Or">Probability Of Or</a>
|
645 |
+
</div>
|
646 |
+
<div class="eva-notebook">
|
647 |
+
<span class="eva-notebook-number">04</span>
|
648 |
+
<a href="probability\04_conditional_probability.html" data-notebook-title="Conditional Probability">Conditional Probability</a>
|
649 |
+
</div>
|
650 |
+
<div class="eva-notebook">
|
651 |
+
<span class="eva-notebook-number">05</span>
|
652 |
+
<a href="probability\05_independence.html" data-notebook-title="Independence">Independence</a>
|
653 |
+
</div>
|
654 |
+
<div class="eva-notebook">
|
655 |
+
<span class="eva-notebook-number">06</span>
|
656 |
+
<a href="probability\06_probability_of_and.html" data-notebook-title="Probability Of And">Probability Of And</a>
|
657 |
+
</div>
|
658 |
+
<div class="eva-notebook">
|
659 |
+
<span class="eva-notebook-number">07</span>
|
660 |
+
<a href="probability\07_law_of_total_probability.html" data-notebook-title="Law Of Total Probability">Law Of Total Probability</a>
|
661 |
+
</div>
|
662 |
+
<div class="eva-notebook">
|
663 |
+
<span class="eva-notebook-number">08</span>
|
664 |
+
<a href="probability\08_bayes_theorem.html" data-notebook-title="Bayes Theorem">Bayes Theorem</a>
|
665 |
+
</div>
|
666 |
+
<div class="eva-notebook">
|
667 |
+
<span class="eva-notebook-number">09</span>
|
668 |
+
<a href="probability\09_random_variables.html" data-notebook-title="Random Variables">Random Variables</a>
|
669 |
+
</div>
|
670 |
+
<div class="eva-notebook">
|
671 |
+
<span class="eva-notebook-number">10</span>
|
672 |
+
<a href="probability\10_probability_mass_function.html" data-notebook-title="Probability Mass Function">Probability Mass Function</a>
|
673 |
+
</div>
|
674 |
+
<div class="eva-notebook">
|
675 |
+
<span class="eva-notebook-number">11</span>
|
676 |
+
<a href="probability\11_expectation.html" data-notebook-title="Expectation">Expectation</a>
|
677 |
+
</div>
|
678 |
+
</div>
|
679 |
+
</div>
|
680 |
+
</div>
|
681 |
+
<div class="eva-course" data-course-id="python">
|
682 |
+
<div class="eva-course-header">
|
683 |
+
<h2 class="eva-course-title">Python</h2>
|
684 |
+
<span class="eva-course-toggle"><i class="fas fa-chevron-down"></i></span>
|
685 |
+
</div>
|
686 |
+
<div class="eva-course-content">
|
687 |
+
<p class="eva-course-description">This collection of marimo notebooks is designed to teach you the basics of the Python programming language.</p>
|
688 |
+
<div class="eva-notebooks">
|
689 |
+
<div class="eva-notebook">
|
690 |
+
<span class="eva-notebook-number">01</span>
|
691 |
+
<a href="python\001_numbers.html" data-notebook-title="Numbers">Numbers</a>
|
692 |
+
</div>
|
693 |
+
<div class="eva-notebook">
|
694 |
+
<span class="eva-notebook-number">02</span>
|
695 |
+
<a href="python\002_strings.html" data-notebook-title="Strings">Strings</a>
|
696 |
+
</div>
|
697 |
+
<div class="eva-notebook">
|
698 |
+
<span class="eva-notebook-number">03</span>
|
699 |
+
<a href="python\003_collections.html" data-notebook-title="Collections">Collections</a>
|
700 |
+
</div>
|
701 |
+
<div class="eva-notebook">
|
702 |
+
<span class="eva-notebook-number">04</span>
|
703 |
+
<a href="python\004_conditional_logic.html" data-notebook-title="Conditional Logic">Conditional Logic</a>
|
704 |
+
</div>
|
705 |
+
<div class="eva-notebook">
|
706 |
+
<span class="eva-notebook-number">05</span>
|
707 |
+
<a href="python\005_loops.html" data-notebook-title="Loops">Loops</a>
|
708 |
+
</div>
|
709 |
+
<div class="eva-notebook">
|
710 |
+
<span class="eva-notebook-number">06</span>
|
711 |
+
<a href="python\007_advanced_collections.html" data-notebook-title="Advanced Collections">Advanced Collections</a>
|
712 |
+
</div>
|
713 |
+
<div class="eva-notebook">
|
714 |
+
<span class="eva-notebook-number">07</span>
|
715 |
+
<a href="python\008_functions.html" data-notebook-title="Functions">Functions</a>
|
716 |
+
</div>
|
717 |
+
<div class="eva-notebook">
|
718 |
+
<span class="eva-notebook-number">08</span>
|
719 |
+
<a href="python\009_modules.html" data-notebook-title="Modules">Modules</a>
|
720 |
+
</div>
|
721 |
+
<div class="eva-notebook">
|
722 |
+
<span class="eva-notebook-number">09</span>
|
723 |
+
<a href="python\010_exceptions.html" data-notebook-title="Exceptions">Exceptions</a>
|
724 |
+
</div>
|
725 |
+
</div>
|
726 |
+
</div>
|
727 |
+
</div>
|
728 |
+
</div>
|
729 |
+
</section>
|
730 |
+
|
731 |
+
<section id="contribute" class="eva-cta">
|
732 |
+
<h2>Contribute to Marimo Learn</h2>
|
733 |
+
<p>
|
734 |
+
Help us expand our collection of educational notebooks. Whether you're an expert in machine learning,
|
735 |
+
statistics, or any other field, your contributions are welcome!
|
736 |
+
</p>
|
737 |
+
<a href="https://github.com/marimo-team/learn" target="_blank" class="eva-button">
|
738 |
+
<i class="fab fa-github"></i> Contribute on GitHub
|
739 |
+
</a>
|
740 |
+
</section>
|
741 |
+
|
742 |
+
<footer class="eva-footer">
|
743 |
+
<div class="eva-footer-copyright">
|
744 |
+
© 2024 Marimo Learn. Built with <a href="https://marimo.io" target="_blank" style="color: var(--eva-green);">marimo</a>.
|
745 |
+
</div>
|
746 |
+
<div class="eva-footer-links">
|
747 |
+
<a href="https://marimo.io" target="_blank">Marimo Website</a>
|
748 |
+
<a href="https://docs.marimo.io" target="_blank">Documentation</a>
|
749 |
+
<a href="https://github.com/marimo-team/learn" target="_blank">GitHub</a>
|
750 |
+
</div>
|
751 |
+
</footer>
|
752 |
+
</div>
|
753 |
+
|
754 |
+
<script>
|
755 |
+
document.addEventListener('DOMContentLoaded', function() {
|
756 |
+
// Terminal typing effect for hero text
|
757 |
+
const heroTitle = document.querySelector('.eva-hero h1');
|
758 |
+
const heroText = document.querySelector('.eva-hero p');
|
759 |
+
const cursor = document.querySelector('.eva-cursor');
|
760 |
+
|
761 |
+
const originalTitle = heroTitle.textContent;
|
762 |
+
const originalText = heroText.textContent.trim();
|
763 |
+
|
764 |
+
heroTitle.textContent = '';
|
765 |
+
heroText.textContent = '';
|
766 |
+
|
767 |
+
let titleIndex = 0;
|
768 |
+
let textIndex = 0;
|
769 |
+
|
770 |
+
function typeTitle() {
|
771 |
+
if (titleIndex < originalTitle.length) {
|
772 |
+
heroTitle.textContent += originalTitle.charAt(titleIndex);
|
773 |
+
titleIndex++;
|
774 |
+
setTimeout(typeTitle, 50);
|
775 |
+
} else {
|
776 |
+
cursor.style.display = 'none';
|
777 |
+
setTimeout(typeText, 500);
|
778 |
+
}
|
779 |
+
}
|
780 |
+
|
781 |
+
function typeText() {
|
782 |
+
if (textIndex < originalText.length) {
|
783 |
+
heroText.textContent += originalText.charAt(textIndex);
|
784 |
+
textIndex++;
|
785 |
+
setTimeout(typeText, 20);
|
786 |
+
}
|
787 |
+
}
|
788 |
+
|
789 |
+
typeTitle();
|
790 |
+
|
791 |
+
// Course toggle functionality
|
792 |
+
const courseHeaders = document.querySelectorAll('.eva-course-header');
|
793 |
+
|
794 |
+
courseHeaders.forEach(header => {
|
795 |
+
header.addEventListener('click', () => {
|
796 |
+
const course = header.parentElement;
|
797 |
+
course.classList.toggle('active');
|
798 |
+
});
|
799 |
+
});
|
800 |
+
|
801 |
+
// Search functionality
|
802 |
+
const searchInput = document.getElementById('courseSearch');
|
803 |
+
const courses = document.querySelectorAll('.eva-course');
|
804 |
+
const notebooks = document.querySelectorAll('.eva-notebook');
|
805 |
+
|
806 |
+
searchInput.addEventListener('input', function() {
|
807 |
+
const searchTerm = this.value.toLowerCase();
|
808 |
+
|
809 |
+
if (searchTerm === '') {
|
810 |
+
// Reset all visibility
|
811 |
+
courses.forEach(course => {
|
812 |
+
course.style.display = 'block';
|
813 |
+
course.classList.remove('active');
|
814 |
+
});
|
815 |
+
|
816 |
+
notebooks.forEach(notebook => {
|
817 |
+
notebook.style.display = 'flex';
|
818 |
+
});
|
819 |
+
|
820 |
+
return;
|
821 |
+
}
|
822 |
+
|
823 |
+
// First hide all courses
|
824 |
+
courses.forEach(course => {
|
825 |
+
course.style.display = 'none';
|
826 |
+
course.classList.remove('active');
|
827 |
+
});
|
828 |
+
|
829 |
+
// Then show courses and notebooks that match the search
|
830 |
+
let hasResults = false;
|
831 |
+
|
832 |
+
notebooks.forEach(notebook => {
|
833 |
+
const notebookTitle = notebook.querySelector('a').getAttribute('data-notebook-title').toLowerCase();
|
834 |
+
const matchesSearch = notebookTitle.includes(searchTerm);
|
835 |
+
|
836 |
+
notebook.style.display = matchesSearch ? 'flex' : 'none';
|
837 |
+
|
838 |
+
if (matchesSearch) {
|
839 |
+
const course = notebook.closest('.eva-course');
|
840 |
+
course.style.display = 'block';
|
841 |
+
course.classList.add('active');
|
842 |
+
hasResults = true;
|
843 |
+
}
|
844 |
+
});
|
845 |
+
|
846 |
+
// Also search course titles
|
847 |
+
courses.forEach(course => {
|
848 |
+
const courseTitle = course.querySelector('.eva-course-title').textContent.toLowerCase();
|
849 |
+
const courseDescription = course.querySelector('.eva-course-description').textContent.toLowerCase();
|
850 |
+
|
851 |
+
if (courseTitle.includes(searchTerm) || courseDescription.includes(searchTerm)) {
|
852 |
+
course.style.display = 'block';
|
853 |
+
course.classList.add('active');
|
854 |
+
hasResults = true;
|
855 |
+
}
|
856 |
+
});
|
857 |
+
});
|
858 |
+
|
859 |
+
// Open the first course by default
|
860 |
+
if (courses.length > 0) {
|
861 |
+
courses[0].classList.add('active');
|
862 |
+
}
|
863 |
+
|
864 |
+
// Smooth scrolling for anchor links
|
865 |
+
document.querySelectorAll('a[href^="#"]').forEach(anchor => {
|
866 |
+
anchor.addEventListener('click', function(e) {
|
867 |
+
e.preventDefault();
|
868 |
+
|
869 |
+
const targetId = this.getAttribute('href');
|
870 |
+
const targetElement = document.querySelector(targetId);
|
871 |
+
|
872 |
+
if (targetElement) {
|
873 |
+
window.scrollTo({
|
874 |
+
top: targetElement.offsetTop - 100,
|
875 |
+
behavior: 'smooth'
|
876 |
+
});
|
877 |
+
}
|
878 |
+
});
|
879 |
+
});
|
880 |
+
});
|
881 |
+
</script>
|
882 |
+
</body>
|
883 |
+
</html>
|
_site/optimization/.nojekyll
ADDED
File without changes
|
_site/optimization/01_least_squares.html
ADDED
@@ -0,0 +1,83 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<!DOCTYPE html>
|
2 |
+
<html lang="en">
|
3 |
+
<head>
|
4 |
+
<meta charset="utf-8" />
|
5 |
+
<link rel="icon" href="./favicon.ico" />
|
6 |
+
<!-- Preload is necessary because we show these images when we disconnect from the server,
|
7 |
+
but at that point we cannot load these images from the server -->
|
8 |
+
<link rel="preload" href="./assets/gradient-yHQUC_QB.png" as="image" />
|
9 |
+
<link rel="preload" href="./assets/noise-60BoTA8O.png" as="image" />
|
10 |
+
<!-- Preload the fonts -->
|
11 |
+
<link rel="preload" href="./assets/Lora-VariableFont_wght-B2ootaw-.ttf" as="font" crossorigin="anonymous" />
|
12 |
+
<link rel="preload" href="./assets/PTSans-Regular-CxL0S8W7.ttf" as="font" crossorigin="anonymous" />
|
13 |
+
<link rel="preload" href="./assets/PTSans-Bold-D9fedIX3.ttf" as="font" crossorigin="anonymous" />
|
14 |
+
<link rel="preload" href="./assets/FiraMono-Regular-BTCkDNvf.ttf" as="font" crossorigin="anonymous" />
|
15 |
+
<link rel="preload" href="./assets/FiraMono-Medium-DU3aDxX5.ttf" as="font" crossorigin="anonymous" />
|
16 |
+
<link rel="preload" href="./assets/FiraMono-Bold-CLVRCuM9.ttf" as="font" crossorigin="anonymous" />
|
17 |
+
|
18 |
+
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
19 |
+
<meta name="theme-color" content="#000000" />
|
20 |
+
<meta name="description" content="a marimo app" />
|
21 |
+
<link rel="apple-touch-icon" href="./apple-touch-icon.png" />
|
22 |
+
<link rel="manifest" href="./manifest.json" />
|
23 |
+
|
24 |
+
<script data-marimo="true">
|
25 |
+
function __resizeIframe(obj) {
|
26 |
+
var scrollbarHeight = 20; // Max between windows, mac, and linux
|
27 |
+
|
28 |
+
function setHeight() {
|
29 |
+
var element = obj.contentWindow.document.documentElement;
|
30 |
+
// If there is no vertical scrollbar, we don't need to resize the iframe
|
31 |
+
if (element.scrollHeight === element.clientHeight) {
|
32 |
+
return;
|
33 |
+
}
|
34 |
+
|
35 |
+
// Create a new height that includes the scrollbar height if it's visible
|
36 |
+
var hasHorizontalScrollbar = element.scrollWidth > element.clientWidth;
|
37 |
+
var newHeight = element.scrollHeight + (hasHorizontalScrollbar ? scrollbarHeight : 0);
|
38 |
+
|
39 |
+
// Only update the height if it's different from the current height
|
40 |
+
if (obj.style.height !== `${newHeight}px`) {
|
41 |
+
obj.style.height = `${newHeight}px`;
|
42 |
+
}
|
43 |
+
}
|
44 |
+
|
45 |
+
// Resize the iframe to the height of the content and bottom scrollbar height
|
46 |
+
setHeight();
|
47 |
+
|
48 |
+
// Resize the iframe when the content changes
|
49 |
+
const resizeObserver = new ResizeObserver((entries) => {
|
50 |
+
setHeight();
|
51 |
+
});
|
52 |
+
resizeObserver.observe(obj.contentWindow.document.body);
|
53 |
+
}
|
54 |
+
</script>
|
55 |
+
<marimo-filename hidden>notebook.py</marimo-filename>
|
56 |
+
<marimo-mode data-mode='edit' hidden></marimo-mode>
|
57 |
+
<marimo-version data-version='0.11.9' hidden></marimo-version>
|
58 |
+
<marimo-user-config data-config='{"completion": {"activate_on_typing": true, "copilot": false}, "display": {"code_editor_font_size": 14, "dataframes": "rich", "cell_output": "above", "theme": "light", "default_width": "medium"}, "formatting": {"line_length": 79}, "keymap": {"preset": "default", "overrides": {}}, "runtime": {"auto_instantiate": true, "auto_reload": "off", "on_cell_change": "autorun", "watcher_on_save": "lazy", "output_max_bytes": 8000000, "std_stream_max_bytes": 1000000}, "save": {"autosave": "off", "autosave_delay": 1000, "format_on_save": false}, "package_management": {"manager": "pip"}, "server": {"browser": "default", "follow_symlink": false}, "snippets": {"custom_paths": [], "include_default_snippets": true}}' data-overrides='{}' hidden></marimo-user-config>
|
59 |
+
<marimo-app-config data-config='{"width": "compact"}' hidden></marimo-app-config>
|
60 |
+
<marimo-server-token data-token='123' hidden></marimo-server-token>
|
61 |
+
<title>01 least squares</title>
|
62 |
+
<script type="module" crossorigin src="./assets/index-BiV-b1K2.js"></script>
|
63 |
+
<link rel="stylesheet" crossorigin href="./assets/index-DkqMrX_B.css">
|
64 |
+
<marimo-wasm hidden=""></marimo-wasm>
|
65 |
+
<script>
|
66 |
+
if (window.location.protocol === 'file:') {
|
67 |
+
alert('Warning: This file must be served by an HTTP server to function correctly.');
|
68 |
+
}
|
69 |
+
</script>
|
70 |
+
|
71 |
+
<style>
|
72 |
+
#save-button {
|
73 |
+
display: none !important;
|
74 |
+
}
|
75 |
+
#filename-input {
|
76 |
+
display: none !important;
|
77 |
+
}
|
78 |
+
</style>
|
79 |
+
<marimo-code hidden="" data-show-code="false">import%20marimo%0A%0A__generated_with%20%3D%20%220.11.9%22%0Aapp%20%3D%20marimo.App()%0A%0A%0A%40app.cell%0Adef%20_()%3A%0A%20%20%20%20import%20marimo%20as%20mo%0A%20%20%20%20return%20(mo%2C)%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20r%22%22%22%0A%20%20%20%20%20%20%20%20%23%20Least%20squares%0A%0A%20%20%20%20%20%20%20%20In%20a%20least-squares%20problem%2C%20we%20have%20measurements%20%24A%20%5Cin%20%5Cmathcal%7BR%7D%5E%7Bm%20%5Ctimes%0A%20%20%20%20%20%20%20%20n%7D%24%20(i.e.%2C%20%24m%24%20rows%20and%20%24n%24%20columns)%20and%20%24b%20%5Cin%20%5Cmathcal%7BR%7D%5Em%24.%20We%20seek%20a%20vector%0A%20%20%20%20%20%20%20%20%24x%20%5Cin%20%5Cmathcal%7BR%7D%5E%7Bn%7D%24%20such%20that%20%24Ax%24%20is%20close%20to%20%24b%24.%20The%20matrices%20%24A%24%20and%20%24b%24%20are%20problem%20data%20or%20constants%2C%20and%20%24x%24%20is%20the%20variable%20we%20are%20solving%20for.%0A%0A%20%20%20%20%20%20%20%20Closeness%20is%20defined%20as%20the%20sum%20of%20the%20squared%20differences%3A%0A%0A%20%20%20%20%20%20%20%20%5C%5B%20%5Csum_%7Bi%3D1%7D%5Em%20(a_i%5ETx%20-%20b_i)%5E2%2C%20%5C%5D%0A%0A%20%20%20%20%20%20%20%20also%20known%20as%20the%20%24%5Cell_2%24-norm%20squared%2C%20%24%5C%7CAx%20-%20b%5C%7C_2%5E2%24.%0A%0A%20%20%20%20%20%20%20%20For%20example%2C%20we%20might%20have%20a%20dataset%20of%20%24m%24%20users%2C%20each%20represented%20by%20%24n%24%20features.%20Each%20row%20%24a_i%5ET%24%20of%20%24A%24%20is%20the%20feature%20vector%20for%20user%20%24i%24%2C%20while%20the%20corresponding%20entry%20%24b_i%24%20of%20%24b%24%20is%20the%20measurement%20we%20want%20to%20predict%20from%20%24a_i%5ET%24%2C%20such%20as%20ad%20spending.%20The%20prediction%20for%20user%20%24i%24%20is%20given%20by%20%24a_i%5ETx%24.%0A%0A%20%20%20%20%20%20%20%20We%20find%20the%20optimal%20value%20of%20%24x%24%20by%20solving%20the%20optimization%20problem%0A%0A%20%20%20%20%20%20%20%20%5C%5B%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Cbegin%7Barray%7D%7Bll%7D%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Ctext%7Bminimize%7D%20%20%20%26%20%5C%7CAx%20-%20b%5C%7C_2%5E2.%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Cend%7Barray%7D%0A%20%20%20%20%20%20%20%20%5C%5D%0A%0A%20%20%20%20%20%20%20%20Let%20%24x%5E%5Cstar%24%20denote%20the%20optimal%20%24x%24.%20The%20quantity%20%24r%20%3D%20Ax%5E%5Cstar%20-%20b%24%20is%20known%20as%20the%20residual.%20If%20%24%5C%7Cr%5C%7C_2%20%3D%200%24%2C%20we%20have%20a%20perfect%20fit.%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20r%22%22%22%0A%20%20%20%20%20%20%20%20%23%23%20Example%0A%0A%20%20%20%20%20%20%20%20In%20this%20example%2C%20we%20use%20the%20Python%20library%20%5BCVXPY%5D(https%3A%2F%2Fgithub.com%2Fcvxpy%2Fcvxpy)%20to%20construct%20and%20solve%20a%20least-squares%20problems.%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell%0Adef%20_()%3A%0A%20%20%20%20import%20cvxpy%20as%20cp%0A%20%20%20%20import%20numpy%20as%20np%0A%20%20%20%20return%20cp%2C%20np%0A%0A%0A%40app.cell%0Adef%20_()%3A%0A%20%20%20%20m%20%3D%2020%0A%20%20%20%20n%20%3D%2015%0A%20%20%20%20return%20m%2C%20n%0A%0A%0A%40app.cell%0Adef%20_(m%2C%20n%2C%20np)%3A%0A%20%20%20%20np.random.seed(0)%0A%20%20%20%20A%20%3D%20np.random.randn(m%2C%20n)%0A%20%20%20%20b%20%3D%20np.random.randn(m)%0A%20%20%20%20return%20A%2C%20b%0A%0A%0A%40app.cell%0Adef%20_(A%2C%20b%2C%20cp%2C%20n)%3A%0A%20%20%20%20x%20%3D%20cp.Variable(n)%0A%20%20%20%20objective%20%3D%20cp.sum_squares(A%20%40%20x%20-%20b)%0A%20%20%20%20problem%20%3D%20cp.Problem(cp.Minimize(objective))%0A%20%20%20%20optimal_value%20%3D%20problem.solve()%0A%20%20%20%20return%20objective%2C%20optimal_value%2C%20problem%2C%20x%0A%0A%0A%40app.cell%0Adef%20_(A%2C%20b%2C%20cp%2C%20mo%2C%20optimal_value%2C%20x)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20f%22%22%22%0A%20%20%20%20%20%20%20%20-%20The%20optimal%20value%20is%20**%7Boptimal_value%3A.04f%7D**.%0A%20%20%20%20%20%20%20%20-%20The%20optimal%20value%20of%20%24x%24%20is%20%7Bmo.as_html(list(x.value))%7D%0A%20%20%20%20%20%20%20%20-%20The%20norm%20of%20the%20residual%20is%20**%7Bcp.norm(A%20%40%20x%20-%20b%2C%20p%3D2).value%3A0.4f%7D**%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20r%22%22%22%0A%20%20%20%20%20%20%20%20%23%23%20Further%20reading%0A%0A%20%20%20%20%20%20%20%20For%20a%20primer%20on%20least%20squares%2C%20with%20many%20real-world%20examples%2C%20check%20out%20the%20free%20book%0A%20%20%20%20%20%20%20%20%5BVectors%2C%20Matrices%2C%20and%20Least%20Squares%5D(https%3A%2F%2Fweb.stanford.edu%2F~boyd%2Fvmls%2F)%2C%20which%20is%20used%20for%20undergraduate%20linear%20algebra%20education%20at%20Stanford.%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0Aif%20__name__%20%3D%3D%20%22__main__%22%3A%0A%20%20%20%20app.run()%0A</marimo-code></head>
|
80 |
+
<body>
|
81 |
+
<div id="root"></div>
|
82 |
+
</body>
|
83 |
+
</html>
|
_site/optimization/02_linear_program.html
ADDED
@@ -0,0 +1,83 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<!DOCTYPE html>
|
2 |
+
<html lang="en">
|
3 |
+
<head>
|
4 |
+
<meta charset="utf-8" />
|
5 |
+
<link rel="icon" href="./favicon.ico" />
|
6 |
+
<!-- Preload is necessary because we show these images when we disconnect from the server,
|
7 |
+
but at that point we cannot load these images from the server -->
|
8 |
+
<link rel="preload" href="./assets/gradient-yHQUC_QB.png" as="image" />
|
9 |
+
<link rel="preload" href="./assets/noise-60BoTA8O.png" as="image" />
|
10 |
+
<!-- Preload the fonts -->
|
11 |
+
<link rel="preload" href="./assets/Lora-VariableFont_wght-B2ootaw-.ttf" as="font" crossorigin="anonymous" />
|
12 |
+
<link rel="preload" href="./assets/PTSans-Regular-CxL0S8W7.ttf" as="font" crossorigin="anonymous" />
|
13 |
+
<link rel="preload" href="./assets/PTSans-Bold-D9fedIX3.ttf" as="font" crossorigin="anonymous" />
|
14 |
+
<link rel="preload" href="./assets/FiraMono-Regular-BTCkDNvf.ttf" as="font" crossorigin="anonymous" />
|
15 |
+
<link rel="preload" href="./assets/FiraMono-Medium-DU3aDxX5.ttf" as="font" crossorigin="anonymous" />
|
16 |
+
<link rel="preload" href="./assets/FiraMono-Bold-CLVRCuM9.ttf" as="font" crossorigin="anonymous" />
|
17 |
+
|
18 |
+
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
19 |
+
<meta name="theme-color" content="#000000" />
|
20 |
+
<meta name="description" content="a marimo app" />
|
21 |
+
<link rel="apple-touch-icon" href="./apple-touch-icon.png" />
|
22 |
+
<link rel="manifest" href="./manifest.json" />
|
23 |
+
|
24 |
+
<script data-marimo="true">
|
25 |
+
function __resizeIframe(obj) {
|
26 |
+
var scrollbarHeight = 20; // Max between windows, mac, and linux
|
27 |
+
|
28 |
+
function setHeight() {
|
29 |
+
var element = obj.contentWindow.document.documentElement;
|
30 |
+
// If there is no vertical scrollbar, we don't need to resize the iframe
|
31 |
+
if (element.scrollHeight === element.clientHeight) {
|
32 |
+
return;
|
33 |
+
}
|
34 |
+
|
35 |
+
// Create a new height that includes the scrollbar height if it's visible
|
36 |
+
var hasHorizontalScrollbar = element.scrollWidth > element.clientWidth;
|
37 |
+
var newHeight = element.scrollHeight + (hasHorizontalScrollbar ? scrollbarHeight : 0);
|
38 |
+
|
39 |
+
// Only update the height if it's different from the current height
|
40 |
+
if (obj.style.height !== `${newHeight}px`) {
|
41 |
+
obj.style.height = `${newHeight}px`;
|
42 |
+
}
|
43 |
+
}
|
44 |
+
|
45 |
+
// Resize the iframe to the height of the content and bottom scrollbar height
|
46 |
+
setHeight();
|
47 |
+
|
48 |
+
// Resize the iframe when the content changes
|
49 |
+
const resizeObserver = new ResizeObserver((entries) => {
|
50 |
+
setHeight();
|
51 |
+
});
|
52 |
+
resizeObserver.observe(obj.contentWindow.document.body);
|
53 |
+
}
|
54 |
+
</script>
|
55 |
+
<marimo-filename hidden>notebook.py</marimo-filename>
|
56 |
+
<marimo-mode data-mode='edit' hidden></marimo-mode>
|
57 |
+
<marimo-version data-version='0.11.9' hidden></marimo-version>
|
58 |
+
<marimo-user-config data-config='{"completion": {"activate_on_typing": true, "copilot": false}, "display": {"default_width": "medium", "theme": "light", "code_editor_font_size": 14, "dataframes": "rich", "cell_output": "above"}, "formatting": {"line_length": 79}, "keymap": {"preset": "default", "overrides": {}}, "runtime": {"auto_instantiate": true, "auto_reload": "off", "on_cell_change": "autorun", "watcher_on_save": "lazy", "output_max_bytes": 8000000, "std_stream_max_bytes": 1000000}, "save": {"autosave": "off", "autosave_delay": 1000, "format_on_save": false}, "package_management": {"manager": "pip"}, "server": {"browser": "default", "follow_symlink": false}, "snippets": {"custom_paths": [], "include_default_snippets": true}}' data-overrides='{}' hidden></marimo-user-config>
|
59 |
+
<marimo-app-config data-config='{"width": "compact"}' hidden></marimo-app-config>
|
60 |
+
<marimo-server-token data-token='123' hidden></marimo-server-token>
|
61 |
+
<title>02 linear program</title>
|
62 |
+
<script type="module" crossorigin src="./assets/index-BiV-b1K2.js"></script>
|
63 |
+
<link rel="stylesheet" crossorigin href="./assets/index-DkqMrX_B.css">
|
64 |
+
<marimo-wasm hidden=""></marimo-wasm>
|
65 |
+
<script>
|
66 |
+
if (window.location.protocol === 'file:') {
|
67 |
+
alert('Warning: This file must be served by an HTTP server to function correctly.');
|
68 |
+
}
|
69 |
+
</script>
|
70 |
+
|
71 |
+
<style>
|
72 |
+
#save-button {
|
73 |
+
display: none !important;
|
74 |
+
}
|
75 |
+
#filename-input {
|
76 |
+
display: none !important;
|
77 |
+
}
|
78 |
+
</style>
|
79 |
+
<marimo-code hidden="" data-show-code="false">import%20marimo%0A%0A__generated_with%20%3D%20%220.11.9%22%0Aapp%20%3D%20marimo.App()%0A%0A%0A%40app.cell%0Adef%20_()%3A%0A%20%20%20%20import%20marimo%20as%20mo%0A%20%20%20%20return%20(mo%2C)%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20r%22%22%22%0A%20%20%20%20%20%20%20%20%23%20Linear%20program%0A%0A%20%20%20%20%20%20%20%20A%20linear%20program%20is%20an%20optimization%20problem%20with%20a%20linear%20objective%20and%20affine%0A%20%20%20%20%20%20%20%20inequality%20constraints.%20A%20common%20standard%20form%20is%20the%20following%3A%0A%0A%20%20%20%20%20%20%20%20%5C%5B%20%20%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Cbegin%7Barray%7D%7Bll%7D%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Ctext%7Bminimize%7D%20%20%20%26%20c%5ETx%20%5C%5C%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Ctext%7Bsubject%20to%7D%20%26%20Ax%20%5Cleq%20b.%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Cend%7Barray%7D%0A%20%20%20%20%20%20%20%20%5C%5D%0A%0A%20%20%20%20%20%20%20%20Here%20%24A%20%5Cin%20%5Cmathcal%7BR%7D%5E%7Bm%20%5Ctimes%20n%7D%24%2C%20%24b%20%5Cin%20%5Cmathcal%7BR%7D%5Em%24%2C%20and%20%24c%20%5Cin%20%5Cmathcal%7BR%7D%5En%24%20are%20problem%20data%20and%20%24x%20%5Cin%20%5Cmathcal%7BR%7D%5E%7Bn%7D%24%20is%20the%20optimization%20variable.%20The%20inequality%20constraint%20%24Ax%20%5Cleq%20b%24%20is%20elementwise.%0A%0A%20%20%20%20%20%20%20%20For%20example%2C%20we%20might%20have%20%24n%24%20different%20products%2C%20each%20constructed%20out%20of%20%24m%24%20components.%20Each%20entry%20%24A_%7Bij%7D%24%20is%20the%20amount%20of%20component%20%24i%24%20required%20to%20build%20one%20unit%20of%20product%20%24j%24.%20Each%20entry%20%24b_i%24%20is%20the%20total%20amount%20of%20component%20%24i%24%20available.%20We%20lose%20%24c_j%24%20for%20each%20unit%20of%20product%20%24j%24%20(%24c_j%20%3C%200%24%20indicates%20profit).%20Our%20goal%20then%20is%20to%20choose%20how%20many%20units%20of%20each%20product%20%24j%24%20to%20make%2C%20%24x_j%24%2C%20in%20order%20to%20minimize%20loss%20without%20exceeding%20our%20budget%20for%20any%20component.%0A%0A%20%20%20%20%20%20%20%20In%20addition%20to%20a%20solution%20%24x%5E%5Cstar%24%2C%20we%20obtain%20a%20dual%20solution%20%24%5Clambda%5E%5Cstar%24.%20A%20positive%20entry%20%24%5Clambda%5E%5Cstar_i%24%20indicates%20that%20the%20constraint%20%24a_i%5ETx%20%5Cleq%20b_i%24%20holds%20with%20equality%20for%20%24x%5E%5Cstar%24%20and%20suggests%20that%20changing%20%24b_i%24%20would%20change%20the%20optimal%20value.%0A%0A%20%20%20%20%20%20%20%20**Why%20linear%20programming%3F**%20Linear%20programming%20is%20a%20way%20to%20achieve%20an%20optimal%20outcome%2C%20such%20as%20maximum%20utility%20or%20lowest%20cost%2C%20subject%20to%20a%20linear%20objective%20function%20and%20affine%20constraints.%20Developed%20in%20the%2020th%20century%2C%20linear%20programming%20is%20widely%20used%20today%20to%20solve%20problems%20in%20resource%20allocation%2C%20scheduling%2C%20transportation%2C%20and%20more.%20The%20discovery%20of%20polynomial-time%20algorithms%20to%20solve%20linear%20programs%20was%20of%20tremendous%20worldwide%20importance%20and%20entered%20the%20public%20discourse%2C%20even%20making%20the%20front%20page%20of%20the%20New%20York%20Times.%0A%0A%20%20%20%20%20%20%20%20In%20the%20late%2020th%20and%20early%2021st%20century%2C%20researchers%20generalized%20linear%20programming%20to%20a%20much%20wider%20class%20of%20problems%20called%20convex%20optimization%20problems.%20Nearly%20all%20convex%20optimization%20problems%20can%20be%20solved%20efficiently%20and%20reliably%2C%20and%20even%20more%20difficult%20problems%20are%20readily%20solved%20by%20a%20sequence%20of%20convex%20optimization%20problems.%20Today%2C%20convex%20optimization%20is%20used%20to%20fit%20machine%20learning%20models%2C%20land%20rockets%20in%20real-time%20at%20SpaceX%2C%20plan%20trajectories%20for%20self-driving%20cars%20at%20Waymo%2C%20execute%20many%20billions%20of%20dollars%20of%20financial%20trades%20a%20day%2C%20and%20much%20more.%0A%0A%20%20%20%20%20%20%20%20This%20marimo%20learn%20course%20uses%20CVXPY%2C%20a%20modeling%20language%20for%20convex%20optimization%20problems%20developed%20originally%20at%20Stanford%2C%20to%20construct%20and%20solve%20convex%20programs.%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20f%22%22%22%0A%20%20%20%20%20%20%20%20%7Bmo.image(%22https%3A%2F%2Fwww.debugmind.com%2Fwp-content%2Fuploads%2F2020%2F01%2Fspacex-1.jpg%22)%7D%0A%20%20%20%20%20%20%20%20_SpaceX%20solves%20convex%20optimization%20problems%20onboard%20to%20land%20its%20rockets%2C%20using%20CVXGEN%2C%20a%20code%20generator%20for%20quadratic%20programming%20developed%20at%20Stephen%20Boyd%E2%80%99s%20Stanford%20lab.%20Photo%20by%20SpaceX%2C%20licensed%20CC%20BY-NC%202.0._%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20r%22%22%22%0A%20%20%20%20%20%20%20%20%23%23%20Example%0A%0A%20%20%20%20%20%20%20%20Here%20we%20use%20CVXPY%20to%20construct%20and%20solve%20a%20linear%20program.%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell%0Adef%20_()%3A%0A%20%20%20%20import%20cvxpy%20as%20cp%0A%20%20%20%20import%20numpy%20as%20np%0A%20%20%20%20return%20cp%2C%20np%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(np)%3A%0A%20%20%20%20A%20%3D%20np.array(%0A%20%20%20%20%20%20%20%20%5B%0A%20%20%20%20%20%20%20%20%20%20%20%20%5B0.76103773%2C%200.12167502%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%5B0.44386323%2C%200.33367433%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%5B1.49407907%2C%20-0.20515826%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%5B0.3130677%2C%20-0.85409574%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%5B-2.55298982%2C%200.6536186%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%5B0.8644362%2C%20-0.74216502%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%5B2.26975462%2C%20-1.45436567%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%5B0.04575852%2C%20-0.18718385%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%5B1.53277921%2C%201.46935877%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%5B0.15494743%2C%200.37816252%5D%2C%0A%20%20%20%20%20%20%20%20%5D%0A%20%20%20%20)%0A%0A%20%20%20%20b%20%3D%20np.array(%0A%20%20%20%20%20%20%20%20%5B%0A%20%20%20%20%20%20%20%20%20%20%20%202.05062369%2C%0A%20%20%20%20%20%20%20%20%20%20%20%200.94934659%2C%0A%20%20%20%20%20%20%20%20%20%20%20%200.89559424%2C%0A%20%20%20%20%20%20%20%20%20%20%20%201.04389978%2C%0A%20%20%20%20%20%20%20%20%20%20%20%202.45035643%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20-0.95479445%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20-0.83801349%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20-0.26562529%2C%0A%20%20%20%20%20%20%20%20%20%20%20%202.35763652%2C%0A%20%20%20%20%20%20%20%20%20%20%20%200.98286942%2C%0A%20%20%20%20%20%20%20%20%5D%0A%20%20%20%20)%0A%20%20%20%20return%20A%2C%20b%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(r%22%22%22We've%20randomly%20generated%20problem%20data%20%24A%24%20and%20%24B%24.%20The%20vector%20for%20%24c%24%20is%20shown%20below.%20Try%20playing%20with%20the%20value%20of%20%24c%24%20by%20dragging%20the%20components%2C%20and%20see%20how%20the%20level%20curves%20change%20in%20the%20visualization%20below.%22%22%22)%0A%20%20%20%20return%0A%0A%0A%40app.cell%0Adef%20_(mo%2C%20np)%3A%0A%20%20%20%20from%20wigglystuff%20import%20Matrix%0A%0A%20%20%20%20c_widget%20%3D%20mo.ui.anywidget(Matrix(matrix%3Dnp.array(%5B%5B0.1%2C%20-0.2%5D%5D)%2C%20step%3D0.01))%0A%20%20%20%20c_widget%0A%20%20%20%20return%20Matrix%2C%20c_widget%0A%0A%0A%40app.cell%0Adef%20_(c_widget%2C%20np)%3A%0A%20%20%20%20c%20%3D%20np.array(c_widget.value%5B%22matrix%22%5D%5B0%5D)%0A%20%20%20%20return%20(c%2C)%0A%0A%0A%40app.cell%0Adef%20_(A%2C%20b%2C%20c%2C%20cp)%3A%0A%20%20%20%20x%20%3D%20cp.Variable(A.shape%5B1%5D)%0A%20%20%20%20prob%20%3D%20cp.Problem(cp.Minimize(c.T%20%40%20x)%2C%20%5BA%20%40%20x%20%3C%3D%20b%5D)%0A%20%20%20%20_%20%3D%20prob.solve()%0A%20%20%20%20x_star%20%3D%20x.value%0A%20%20%20%20return%20prob%2C%20x%2C%20x_star%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(r%22%22%22Below%2C%20we%20plot%20the%20feasible%20region%20of%20the%20problem%20%E2%80%94%20the%20intersection%20of%20the%20inequalities%20%E2%80%94%20and%20the%20level%20curves%20of%20the%20objective%20function.%20The%20optimal%20value%20%24x%5E%5Cstar%24%20is%20the%20point%20farthest%20in%20the%20feasible%20region%20in%20the%20direction%20%24-c%24.%22%22%22)%0A%20%20%20%20return%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(A%2C%20b%2C%20c%2C%20make_plot%2C%20x_star)%3A%0A%20%20%20%20make_plot(A%2C%20b%2C%20c%2C%20x_star)%0A%20%20%20%20return%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(np)%3A%0A%20%20%20%20import%20matplotlib.pyplot%20as%20plt%0A%0A%0A%20%20%20%20def%20make_plot(A%2C%20b%2C%20c%2C%20x_star)%3A%0A%20%20%20%20%20%20%20%20%23%20Define%20a%20grid%20over%20a%20region%20that%20covers%20the%20feasible%20set.%0A%20%20%20%20%20%20%20%20%23%20You%20might%20need%20to%20adjust%20these%20limits.%0A%20%20%20%20%20%20%20%20x_vals%20%3D%20np.linspace(-1%2C%201%2C%20400)%0A%20%20%20%20%20%20%20%20y_vals%20%3D%20np.linspace(1%2C%203%2C%20400)%0A%20%20%20%20%20%20%20%20X%2C%20Y%20%3D%20np.meshgrid(x_vals%2C%20y_vals)%0A%0A%20%20%20%20%20%20%20%20%23%20Flatten%20the%20grid%20points%20into%20an%20(N%2C2)%20array.%0A%20%20%20%20%20%20%20%20points%20%3D%20np.vstack(%5BX.ravel()%2C%20Y.ravel()%5D).T%0A%0A%20%20%20%20%20%20%20%20%23%20For%20each%20point%2C%20check%20if%20it%20satisfies%20all%20the%20constraints%3A%20A%20%40%20x%20%3C%3D%20b.%0A%20%20%20%20%20%20%20%20%23%20A%20dot%20product%3A%20shape%20of%20A%20%40%20x.T%20will%20be%20(m%2C%20N).%20We%20add%20a%20little%20tolerance.%0A%20%20%20%20%20%20%20%20feasible%20%3D%20np.all(np.dot(A%2C%20points.T)%20%3C%3D%20(b%5B%3A%2C%20None%5D%20%2B%201e-8)%2C%20axis%3D0)%0A%20%20%20%20%20%20%20%20feasible%20%3D%20feasible.reshape(X.shape)%0A%0A%20%20%20%20%20%20%20%20%23%20Create%20the%20figure%20with%20a%20white%20background.%0A%20%20%20%20%20%20%20%20fig%20%3D%20plt.figure(figsize%3D(8%2C%206)%2C%20facecolor%3D%22white%22)%0A%20%20%20%20%20%20%20%20ax%20%3D%20fig.add_subplot(111)%0A%20%20%20%20%20%20%20%20ax.set_facecolor(%22white%22)%0A%0A%20%20%20%20%20%20%20%20%23%20Plot%20the%20feasible%20region.%0A%20%20%20%20%20%20%20%20%23%20Since%20%22feasible%22%20is%20a%20boolean%20array%20(False%3D0%2C%20True%3D1)%2C%20we%20set%20contour%20levels%20so%20that%0A%20%20%20%20%20%20%20%20%23%20the%20region%20with%20value%201%20(feasible)%20gets%20filled.%0A%20%20%20%20%20%20%20%20ax.contourf(%0A%20%20%20%20%20%20%20%20%20%20%20%20X%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20Y%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20feasible.astype(float)%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20levels%3D%5B-0.5%2C%200.5%2C%201.5%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20colors%3D%5B%22white%22%2C%20%22gray%22%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20alpha%3D0.5%2C%0A%20%20%20%20%20%20%20%20)%0A%0A%20%20%20%20%20%20%20%20%23%20Plot%20level%20curves%20of%20the%20objective%20function%20c%5ET%20x.%0A%20%20%20%20%20%20%20%20%23%20Compute%20c%5ET%20x%20over%20the%20grid%3A%0A%20%20%20%20%20%20%20%20Z%20%3D%20c%5B0%5D%20*%20X%20%2B%20c%5B1%5D%20*%20Y%0A%20%20%20%20%20%20%20%20%23%20Choose%20several%20levels%20for%20the%20iso-cost%20lines.%0A%20%20%20%20%20%20%20%20levels%20%3D%20np.linspace(np.min(Z)%2C%20np.max(Z)%2C%2020)%0A%20%20%20%20%20%20%20%20contours%20%3D%20ax.contour(%0A%20%20%20%20%20%20%20%20%20%20%20%20X%2C%20Y%2C%20Z%2C%20levels%3Dlevels%2C%20colors%3D%22gray%22%2C%20linestyles%3D%22--%22%2C%20linewidths%3D1%0A%20%20%20%20%20%20%20%20)%0A%0A%20%20%20%20%20%20%20%20%23%20Draw%20the%20vector%20-c%20as%20an%20arrow%20starting%20at%20x_star.%0A%20%20%20%20%20%20%20%20norm_c%20%3D%20np.linalg.norm(c)%0A%20%20%20%20%20%20%20%20if%20norm_c%20%3E%200%3A%0A%20%20%20%20%20%20%20%20%20%20%20%20head_width%20%3D%20norm_c%20*%200.1%0A%20%20%20%20%20%20%20%20%20%20%20%20head_length%20%3D%20norm_c%20*%200.1%0A%20%20%20%20%20%20%20%20%20%20%20%20%23%20The%20arrow%20starts%20at%20x_star%20and%20points%20in%20the%20-c%20direction.%0A%20%20%20%20%20%20%20%20%20%20%20%20ax.arrow(%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20x_star%5B0%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20x_star%5B1%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20-c%5B0%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20-c%5B1%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20head_width%3Dhead_width%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20head_length%3Dhead_length%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20fc%3D%22black%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20ec%3D%22black%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20length_includes_head%3DTrue%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20)%0A%20%20%20%20%20%20%20%20%20%20%20%20%23%20Label%20the%20arrow%20near%20its%20tip.%0A%20%20%20%20%20%20%20%20%20%20%20%20ax.text(%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20x_star%5B0%5D%20-%20c%5B0%5D%20*%201.05%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20x_star%5B1%5D%20-%20c%5B1%5D%20*%201.05%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20r%22%24-c%24%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20color%3D%22black%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20fontsize%3D12%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20)%0A%0A%20%20%20%20%20%20%20%20%23%20Optionally%2C%20mark%20and%20label%20the%20point%20x_star.%0A%20%20%20%20%20%20%20%20ax.plot(x_star%5B0%5D%2C%20x_star%5B1%5D%2C%20%22ko%22%2C%20markersize%3D5)%0A%20%20%20%20%20%20%20%20ax.text(%0A%20%20%20%20%20%20%20%20%20%20%20%20x_star%5B0%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20x_star%5B1%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20r%22%24%5Cmathbf%7Bx%7D%5E%5Cstar%24%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20color%3D%22black%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20fontsize%3D12%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20verticalalignment%3D%22bottom%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20horizontalalignment%3D%22right%22%2C%0A%20%20%20%20%20%20%20%20)%0A%20%20%20%20%20%20%20%20%23%20Label%20the%20axes%20and%20set%20title.%0A%20%20%20%20%20%20%20%20ax.set_xlabel(%22%24x_1%24%22)%0A%20%20%20%20%20%20%20%20ax.set_ylabel(%22%24x_2%24%22)%0A%20%20%20%20%20%20%20%20ax.set_title(%22Feasible%20Region%20and%20Level%20Curves%20of%20%24c%5ETx%24%22)%0A%20%20%20%20%20%20%20%20ax.set_xlim(np.min(x_vals)%2C%20np.max(x_vals))%0A%20%20%20%20%20%20%20%20ax.set_ylim(np.min(y_vals)%2C%20np.max(y_vals))%0A%20%20%20%20%20%20%20%20return%20ax%0A%20%20%20%20return%20make_plot%2C%20plt%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo%2C%20prob%2C%20x)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20f%22%22%22%0A%20%20%20%20%20%20%20%20The%20optimal%20value%20is%20%7Bprob.value%3A.04f%7D.%0A%0A%20%20%20%20%20%20%20%20A%20solution%20%24x%24%20is%20%7Bmo.as_html(list(x.value))%7D%0A%20%20%20%20%20%20%20%20A%20dual%20solution%20is%20is%20%7Bmo.as_html(list(prob.constraints%5B0%5D.dual_value))%7D%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0Aif%20__name__%20%3D%3D%20%22__main__%22%3A%0A%20%20%20%20app.run()%0A</marimo-code></head>
|
80 |
+
<body>
|
81 |
+
<div id="root"></div>
|
82 |
+
</body>
|
83 |
+
</html>
|
_site/optimization/03_minimum_fuel_optimal_control.html
ADDED
@@ -0,0 +1,83 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<!DOCTYPE html>
|
2 |
+
<html lang="en">
|
3 |
+
<head>
|
4 |
+
<meta charset="utf-8" />
|
5 |
+
<link rel="icon" href="./favicon.ico" />
|
6 |
+
<!-- Preload is necessary because we show these images when we disconnect from the server,
|
7 |
+
but at that point we cannot load these images from the server -->
|
8 |
+
<link rel="preload" href="./assets/gradient-yHQUC_QB.png" as="image" />
|
9 |
+
<link rel="preload" href="./assets/noise-60BoTA8O.png" as="image" />
|
10 |
+
<!-- Preload the fonts -->
|
11 |
+
<link rel="preload" href="./assets/Lora-VariableFont_wght-B2ootaw-.ttf" as="font" crossorigin="anonymous" />
|
12 |
+
<link rel="preload" href="./assets/PTSans-Regular-CxL0S8W7.ttf" as="font" crossorigin="anonymous" />
|
13 |
+
<link rel="preload" href="./assets/PTSans-Bold-D9fedIX3.ttf" as="font" crossorigin="anonymous" />
|
14 |
+
<link rel="preload" href="./assets/FiraMono-Regular-BTCkDNvf.ttf" as="font" crossorigin="anonymous" />
|
15 |
+
<link rel="preload" href="./assets/FiraMono-Medium-DU3aDxX5.ttf" as="font" crossorigin="anonymous" />
|
16 |
+
<link rel="preload" href="./assets/FiraMono-Bold-CLVRCuM9.ttf" as="font" crossorigin="anonymous" />
|
17 |
+
|
18 |
+
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
19 |
+
<meta name="theme-color" content="#000000" />
|
20 |
+
<meta name="description" content="a marimo app" />
|
21 |
+
<link rel="apple-touch-icon" href="./apple-touch-icon.png" />
|
22 |
+
<link rel="manifest" href="./manifest.json" />
|
23 |
+
|
24 |
+
<script data-marimo="true">
|
25 |
+
function __resizeIframe(obj) {
|
26 |
+
var scrollbarHeight = 20; // Max between windows, mac, and linux
|
27 |
+
|
28 |
+
function setHeight() {
|
29 |
+
var element = obj.contentWindow.document.documentElement;
|
30 |
+
// If there is no vertical scrollbar, we don't need to resize the iframe
|
31 |
+
if (element.scrollHeight === element.clientHeight) {
|
32 |
+
return;
|
33 |
+
}
|
34 |
+
|
35 |
+
// Create a new height that includes the scrollbar height if it's visible
|
36 |
+
var hasHorizontalScrollbar = element.scrollWidth > element.clientWidth;
|
37 |
+
var newHeight = element.scrollHeight + (hasHorizontalScrollbar ? scrollbarHeight : 0);
|
38 |
+
|
39 |
+
// Only update the height if it's different from the current height
|
40 |
+
if (obj.style.height !== `${newHeight}px`) {
|
41 |
+
obj.style.height = `${newHeight}px`;
|
42 |
+
}
|
43 |
+
}
|
44 |
+
|
45 |
+
// Resize the iframe to the height of the content and bottom scrollbar height
|
46 |
+
setHeight();
|
47 |
+
|
48 |
+
// Resize the iframe when the content changes
|
49 |
+
const resizeObserver = new ResizeObserver((entries) => {
|
50 |
+
setHeight();
|
51 |
+
});
|
52 |
+
resizeObserver.observe(obj.contentWindow.document.body);
|
53 |
+
}
|
54 |
+
</script>
|
55 |
+
<marimo-filename hidden>notebook.py</marimo-filename>
|
56 |
+
<marimo-mode data-mode='edit' hidden></marimo-mode>
|
57 |
+
<marimo-version data-version='0.11.9' hidden></marimo-version>
|
58 |
+
<marimo-user-config data-config='{"completion": {"activate_on_typing": true, "copilot": false}, "display": {"default_width": "medium", "code_editor_font_size": 14, "dataframes": "rich", "cell_output": "above", "theme": "light"}, "formatting": {"line_length": 79}, "keymap": {"preset": "default", "overrides": {}}, "runtime": {"auto_instantiate": true, "auto_reload": "off", "on_cell_change": "autorun", "watcher_on_save": "lazy", "output_max_bytes": 8000000, "std_stream_max_bytes": 1000000}, "save": {"autosave": "off", "autosave_delay": 1000, "format_on_save": false}, "package_management": {"manager": "pip"}, "server": {"browser": "default", "follow_symlink": false}, "snippets": {"custom_paths": [], "include_default_snippets": true}}' data-overrides='{}' hidden></marimo-user-config>
|
59 |
+
<marimo-app-config data-config='{"width": "compact"}' hidden></marimo-app-config>
|
60 |
+
<marimo-server-token data-token='123' hidden></marimo-server-token>
|
61 |
+
<title>03 minimum fuel optimal control</title>
|
62 |
+
<script type="module" crossorigin src="./assets/index-BiV-b1K2.js"></script>
|
63 |
+
<link rel="stylesheet" crossorigin href="./assets/index-DkqMrX_B.css">
|
64 |
+
<marimo-wasm hidden=""></marimo-wasm>
|
65 |
+
<script>
|
66 |
+
if (window.location.protocol === 'file:') {
|
67 |
+
alert('Warning: This file must be served by an HTTP server to function correctly.');
|
68 |
+
}
|
69 |
+
</script>
|
70 |
+
|
71 |
+
<style>
|
72 |
+
#save-button {
|
73 |
+
display: none !important;
|
74 |
+
}
|
75 |
+
#filename-input {
|
76 |
+
display: none !important;
|
77 |
+
}
|
78 |
+
</style>
|
79 |
+
<marimo-code hidden="" data-show-code="false">import%20marimo%0A%0A__generated_with%20%3D%20%220.11.9%22%0Aapp%20%3D%20marimo.App()%0A%0A%0A%40app.cell%0Adef%20_()%3A%0A%20%20%20%20import%20marimo%20as%20mo%0A%20%20%20%20return%20(mo%2C)%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20r%22%22%22%0A%20%20%20%20%20%20%20%20%23%20Minimal%20fuel%20optimal%20control%0A%0A%20%20%20%20%20%20%20%20This%20notebook%20includes%20an%20application%20of%20linear%20programming%20to%20controlling%20a%0A%20%20%20%20%20%20%20%20physical%20system%2C%20adapted%20from%20%5BConvex%0A%20%20%20%20%20%20%20%20Optimization%5D(https%3A%2F%2Fweb.stanford.edu%2F~boyd%2Fcvxbook%2F)%20by%20Boyd%20and%20Vandenberghe.%0A%0A%20%20%20%20%20%20%20%20We%20consider%20a%20linear%20dynamical%20system%20with%20state%20%24x(t)%20%5Cin%20%5Cmathbf%7BR%7D%5En%24%2C%20for%20%24t%20%3D%200%2C%20%5Cldots%2C%20T%24.%20At%20each%20time%20step%20%24t%20%3D%200%2C%20%5Cldots%2C%20T%20-%201%24%2C%20an%20actuator%20or%20input%20signal%20%24u(t)%24%20is%20applied%2C%20affecting%20the%20state.%20The%20dynamics%0A%20%20%20%20%20%20%20%20of%20the%20system%20is%20given%20by%20the%20linear%20recurrence%0A%0A%20%20%20%20%20%20%20%20%5C%5B%0A%20%20%20%20%20%20%20%20%20%20%20%20x(t%20%2B%201)%20%3D%20Ax(t)%20%2B%20bu(t)%2C%20%5Cquad%20t%20%3D%200%2C%20%5Cldots%2C%20T%20-%201%2C%0A%20%20%20%20%20%20%20%20%5C%5D%0A%0A%20%20%20%20%20%20%20%20where%20%24A%20%5Cin%20%5Cmathbf%7BR%7D%5E%7Bn%20%5Ctimes%20n%7D%24%20and%20%24b%20%5Cin%20%5Cmathbf%7BR%7D%5En%24%20are%20given%20and%20encode%20how%20the%20system%20evolves.%20The%20initial%20state%20%24x(0)%24%20is%20also%20given.%0A%0A%20%20%20%20%20%20%20%20The%20_minimum%20fuel%20optimal%20control%20problem_%20is%20to%20choose%20the%20inputs%20%24u(0)%2C%20%5Cldots%2C%20u(T%20-%201)%24%20so%20as%20to%20achieve%0A%20%20%20%20%20%20%20%20a%20given%20desired%20state%20%24x_%5Ctext%7Bdes%7D%20%3D%20x(T)%24%20while%20minimizing%20the%20total%20fuel%20consumed%0A%0A%20%20%20%20%20%20%20%20%5C%5B%0A%20%20%20%20%20%20%20%20F%20%3D%20%5Csum_%7Bt%3D0%7D%5E%7BT%20-%201%7D%20f(u(t)).%0A%20%20%20%20%20%20%20%20%5C%5D%0A%0A%20%20%20%20%20%20%20%20The%20function%20%24f%20%3A%20%5Cmathbf%7BR%7D%20%5Cto%20%5Cmathbf%7BR%7D%24%20tells%20us%20how%20much%20fuel%20is%20consumed%20as%20a%20function%20of%20the%20input%2C%20and%20is%20given%20by%0A%0A%20%20%20%20%20%20%20%20%5C%5B%0A%20%20%20%20%20%20%20%20%20%20%20%20f(a)%20%3D%20%5Cbegin%7Bcases%7D%0A%20%20%20%20%20%20%20%20%20%20%20%20%7Ca%7C%20%26%20%7Ca%7C%20%5Cleq%201%20%5C%5C%0A%20%20%20%20%20%20%20%20%20%20%20%202%7Ca%7C%20-%201%20%26%20%7Ca%7C%20%3E%201.%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Cend%7Bcases%7D%0A%20%20%20%20%20%20%20%20%5C%5D%0A%0A%20%20%20%20%20%20%20%20This%20means%20the%20fuel%20use%20is%20proportional%20to%20the%20magnitude%20of%20the%20signal%20between%20%24-1%24%20and%20%241%24%2C%20but%20for%20larger%20signals%20the%20marginal%20fuel%20efficiency%20is%20half.%0A%0A%20%20%20%20%20%20%20%20**This%20notebook.**%20In%20this%20notebook%20we%20use%20CVXPY%20to%20formulate%20the%20minimum%20fuel%20optimal%20control%20problem%20as%20a%20linear%20program.%20The%20notebook%20lets%20you%20play%20with%20the%20initial%20and%20target%20states%2C%20letting%20you%20see%20how%20they%20affect%20the%20planned%20trajectory%20of%20inputs%20%24u%24.%0A%0A%20%20%20%20%20%20%20%20First%2C%20we%20create%20the%20**problem%20data**.%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell%0Adef%20_()%3A%0A%20%20%20%20import%20numpy%20as%20np%0A%20%20%20%20return%20(np%2C)%0A%0A%0A%40app.cell%0Adef%20_()%3A%0A%20%20%20%20n%2C%20T%20%3D%203%2C%2030%0A%20%20%20%20return%20T%2C%20n%0A%0A%0A%40app.cell%0Adef%20_(np)%3A%0A%20%20%20%20A%20%3D%20np.array(%5B%5B-1%2C%200.4%2C%200.8%5D%2C%20%5B1%2C%200%2C%200%5D%2C%20%5B0%2C%201%2C%200%5D%5D)%0A%20%20%20%20b%20%3D%20np.array(%5B%5B1%2C%200%2C%200.3%5D%5D).T%0A%20%20%20%20return%20A%2C%20b%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo%2C%20n%2C%20np)%3A%0A%20%20%20%20import%20wigglystuff%0A%0A%20%20%20%20x0_widget%20%3D%20mo.ui.anywidget(wigglystuff.Matrix(np.zeros((1%2C%20n))))%0A%20%20%20%20xdes_widget%20%3D%20mo.ui.anywidget(wigglystuff.Matrix(np.array(%5B%5B7%2C%202%2C%20-6%5D%5D)))%0A%0A%20%20%20%20_a%20%3D%20mo.md(%0A%20%20%20%20%20%20%20%20rf%22%22%22%0A%0A%20%20%20%20%20%20%20%20Choose%20a%20value%20for%20%24x_0%24%20...%0A%0A%20%20%20%20%20%20%20%20%7Bx0_widget%7D%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%0A%20%20%20%20_b%20%3D%20mo.md(%0A%20%20%20%20%20%20%20%20rf%22%22%22%0A%20%20%20%20%20%20%20%20...%20and%20for%20%24x_%5Ctext%7B%7Bdes%7D%7D%24%0A%0A%20%20%20%20%20%20%20%20%7Bxdes_widget%7D%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%0A%20%20%20%20mo.hstack(%5B_a%2C%20_b%5D%2C%20justify%3D%22space-around%22)%0A%20%20%20%20return%20wigglystuff%2C%20x0_widget%2C%20xdes_widget%0A%0A%0A%40app.cell%0Adef%20_(x0_widget%2C%20xdes_widget)%3A%0A%20%20%20%20x0%20%3D%20x0_widget.matrix%0A%20%20%20%20xdes%20%3D%20xdes_widget.matrix%0A%20%20%20%20return%20x0%2C%20xdes%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(r%22%22%22**Next%2C%20we%20specify%20the%20problem%20as%20a%20linear%20program%20using%20CVXPY.**%20This%20problem%20is%20linear%20because%20the%20objective%20and%20constraints%20are%20affine.%20(In%20fact%2C%20the%20objective%20is%20piecewise%20affine%2C%20but%20CVXPY%20rewrites%20it%20to%20be%20affine%20for%20you.)%22%22%22)%0A%20%20%20%20return%0A%0A%0A%40app.cell%0Adef%20_()%3A%0A%20%20%20%20import%20cvxpy%20as%20cp%0A%20%20%20%20return%20(cp%2C)%0A%0A%0A%40app.cell%0Adef%20_(A%2C%20T%2C%20b%2C%20cp%2C%20mo%2C%20n%2C%20x0%2C%20xdes)%3A%0A%20%20%20%20X%2C%20u%20%3D%20cp.Variable(shape%3D(n%2C%20T%20%2B%201))%2C%20cp.Variable(shape%3D(1%2C%20T))%0A%0A%20%20%20%20objective%20%3D%20cp.sum(cp.maximum(cp.abs(u)%2C%202%20*%20cp.abs(u)%20-%201))%0A%20%20%20%20constraints%20%3D%20%5B%0A%20%20%20%20%20%20%20%20X%5B%3A%2C%201%3A%5D%20%3D%3D%20A%20%40%20X%5B%3A%2C%20%3A-1%5D%20%2B%20b%20%40%20u%2C%0A%20%20%20%20%20%20%20%20X%5B%3A%2C%200%5D%20%3D%3D%20x0%2C%0A%20%20%20%20%20%20%20%20X%5B%3A%2C%20-1%5D%20%3D%3D%20xdes%2C%0A%20%20%20%20%5D%0A%0A%20%20%20%20fuel_used%20%3D%20cp.Problem(cp.Minimize(objective)%2C%20constraints).solve()%0A%20%20%20%20mo.md(f%22Achieved%20a%20fuel%20usage%20of%20%7Bfuel_used%3A.02f%7D.%20%F0%9F%9A%80%22)%0A%20%20%20%20return%20X%2C%20constraints%2C%20fuel_used%2C%20objective%2C%20u%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20%20%20%20%20Finally%2C%20we%20plot%20the%20chosen%20inputs%20over%20time.%0A%0A%20%20%20%20%20%20%20%20**%F0%9F%8C%8A%20Try%20it!**%20Change%20the%20initial%20and%20desired%20states%3B%20how%20do%20fuel%20usage%20and%20controls%20change%3F%20Can%20you%20explain%20what%20you%20see%3F%20You%20can%20also%20try%20experimenting%20with%20the%20value%20of%20%24T%24.%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell%0Adef%20_(plot_solution%2C%20u)%3A%0A%20%20%20%20plot_solution(u)%0A%20%20%20%20return%0A%0A%0A%40app.cell%0Adef%20_(T%2C%20cp%2C%20np)%3A%0A%20%20%20%20def%20plot_solution(u%3A%20cp.Variable)%3A%0A%20%20%20%20%20%20%20%20import%20matplotlib.pyplot%20as%20plt%0A%0A%20%20%20%20%20%20%20%20plt.step(np.arange(T)%2C%20u.T.value)%0A%20%20%20%20%20%20%20%20plt.axis(%22tight%22)%0A%20%20%20%20%20%20%20%20plt.xlabel(%22%24t%24%22)%0A%20%20%20%20%20%20%20%20plt.ylabel(%22%24u%24%22)%0A%20%20%20%20%20%20%20%20return%20plt.gca()%0A%20%20%20%20return%20(plot_solution%2C)%0A%0A%0Aif%20__name__%20%3D%3D%20%22__main__%22%3A%0A%20%20%20%20app.run()%0A</marimo-code></head>
|
80 |
+
<body>
|
81 |
+
<div id="root"></div>
|
82 |
+
</body>
|
83 |
+
</html>
|
_site/optimization/04_quadratic_program.html
ADDED
@@ -0,0 +1,83 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<!DOCTYPE html>
|
2 |
+
<html lang="en">
|
3 |
+
<head>
|
4 |
+
<meta charset="utf-8" />
|
5 |
+
<link rel="icon" href="./favicon.ico" />
|
6 |
+
<!-- Preload is necessary because we show these images when we disconnect from the server,
|
7 |
+
but at that point we cannot load these images from the server -->
|
8 |
+
<link rel="preload" href="./assets/gradient-yHQUC_QB.png" as="image" />
|
9 |
+
<link rel="preload" href="./assets/noise-60BoTA8O.png" as="image" />
|
10 |
+
<!-- Preload the fonts -->
|
11 |
+
<link rel="preload" href="./assets/Lora-VariableFont_wght-B2ootaw-.ttf" as="font" crossorigin="anonymous" />
|
12 |
+
<link rel="preload" href="./assets/PTSans-Regular-CxL0S8W7.ttf" as="font" crossorigin="anonymous" />
|
13 |
+
<link rel="preload" href="./assets/PTSans-Bold-D9fedIX3.ttf" as="font" crossorigin="anonymous" />
|
14 |
+
<link rel="preload" href="./assets/FiraMono-Regular-BTCkDNvf.ttf" as="font" crossorigin="anonymous" />
|
15 |
+
<link rel="preload" href="./assets/FiraMono-Medium-DU3aDxX5.ttf" as="font" crossorigin="anonymous" />
|
16 |
+
<link rel="preload" href="./assets/FiraMono-Bold-CLVRCuM9.ttf" as="font" crossorigin="anonymous" />
|
17 |
+
|
18 |
+
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
19 |
+
<meta name="theme-color" content="#000000" />
|
20 |
+
<meta name="description" content="a marimo app" />
|
21 |
+
<link rel="apple-touch-icon" href="./apple-touch-icon.png" />
|
22 |
+
<link rel="manifest" href="./manifest.json" />
|
23 |
+
|
24 |
+
<script data-marimo="true">
|
25 |
+
function __resizeIframe(obj) {
|
26 |
+
var scrollbarHeight = 20; // Max between windows, mac, and linux
|
27 |
+
|
28 |
+
function setHeight() {
|
29 |
+
var element = obj.contentWindow.document.documentElement;
|
30 |
+
// If there is no vertical scrollbar, we don't need to resize the iframe
|
31 |
+
if (element.scrollHeight === element.clientHeight) {
|
32 |
+
return;
|
33 |
+
}
|
34 |
+
|
35 |
+
// Create a new height that includes the scrollbar height if it's visible
|
36 |
+
var hasHorizontalScrollbar = element.scrollWidth > element.clientWidth;
|
37 |
+
var newHeight = element.scrollHeight + (hasHorizontalScrollbar ? scrollbarHeight : 0);
|
38 |
+
|
39 |
+
// Only update the height if it's different from the current height
|
40 |
+
if (obj.style.height !== `${newHeight}px`) {
|
41 |
+
obj.style.height = `${newHeight}px`;
|
42 |
+
}
|
43 |
+
}
|
44 |
+
|
45 |
+
// Resize the iframe to the height of the content and bottom scrollbar height
|
46 |
+
setHeight();
|
47 |
+
|
48 |
+
// Resize the iframe when the content changes
|
49 |
+
const resizeObserver = new ResizeObserver((entries) => {
|
50 |
+
setHeight();
|
51 |
+
});
|
52 |
+
resizeObserver.observe(obj.contentWindow.document.body);
|
53 |
+
}
|
54 |
+
</script>
|
55 |
+
<marimo-filename hidden>notebook.py</marimo-filename>
|
56 |
+
<marimo-mode data-mode='edit' hidden></marimo-mode>
|
57 |
+
<marimo-version data-version='0.11.9' hidden></marimo-version>
|
58 |
+
<marimo-user-config data-config='{"completion": {"activate_on_typing": true, "copilot": false}, "display": {"dataframes": "rich", "code_editor_font_size": 14, "theme": "light", "default_width": "medium", "cell_output": "above"}, "formatting": {"line_length": 79}, "keymap": {"preset": "default", "overrides": {}}, "runtime": {"auto_instantiate": true, "auto_reload": "off", "on_cell_change": "autorun", "watcher_on_save": "lazy", "output_max_bytes": 8000000, "std_stream_max_bytes": 1000000}, "save": {"autosave": "off", "autosave_delay": 1000, "format_on_save": false}, "package_management": {"manager": "pip"}, "server": {"browser": "default", "follow_symlink": false}, "snippets": {"custom_paths": [], "include_default_snippets": true}}' data-overrides='{}' hidden></marimo-user-config>
|
59 |
+
<marimo-app-config data-config='{"width": "compact"}' hidden></marimo-app-config>
|
60 |
+
<marimo-server-token data-token='123' hidden></marimo-server-token>
|
61 |
+
<title>04 quadratic program</title>
|
62 |
+
<script type="module" crossorigin src="./assets/index-BiV-b1K2.js"></script>
|
63 |
+
<link rel="stylesheet" crossorigin href="./assets/index-DkqMrX_B.css">
|
64 |
+
<marimo-wasm hidden=""></marimo-wasm>
|
65 |
+
<script>
|
66 |
+
if (window.location.protocol === 'file:') {
|
67 |
+
alert('Warning: This file must be served by an HTTP server to function correctly.');
|
68 |
+
}
|
69 |
+
</script>
|
70 |
+
|
71 |
+
<style>
|
72 |
+
#save-button {
|
73 |
+
display: none !important;
|
74 |
+
}
|
75 |
+
#filename-input {
|
76 |
+
display: none !important;
|
77 |
+
}
|
78 |
+
</style>
|
79 |
+
<marimo-code hidden="" data-show-code="false">import%20marimo%0A%0A__generated_with%20%3D%20%220.11.9%22%0Aapp%20%3D%20marimo.App()%0A%0A%0A%40app.cell%0Adef%20_()%3A%0A%20%20%20%20import%20marimo%20as%20mo%0A%20%20%20%20return%20(mo%2C)%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20r%22%22%22%0A%20%20%20%20%20%20%20%20%23%20Quadratic%20program%0A%0A%20%20%20%20%20%20%20%20A%20quadratic%20program%20is%20an%20optimization%20problem%20with%20a%20quadratic%20objective%20and%0A%20%20%20%20%20%20%20%20affine%20equality%20and%20inequality%20constraints.%20A%20common%20standard%20form%20is%20the%0A%20%20%20%20%20%20%20%20following%3A%0A%0A%20%20%20%20%20%20%20%20%5C%5B%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Cbegin%7Barray%7D%7Bll%7D%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Ctext%7Bminimize%7D%20%20%20%26%20(1%2F2)x%5ETPx%20%2B%20q%5ETx%5C%5C%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Ctext%7Bsubject%20to%7D%20%26%20Gx%20%5Cleq%20h%20%5C%5C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%26%20Ax%20%3D%20b.%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Cend%7Barray%7D%0A%20%20%20%20%20%20%20%20%5C%5D%0A%0A%20%20%20%20%20%20%20%20Here%20%24P%20%5Cin%20%5Cmathcal%7BS%7D%5E%7Bn%7D_%2B%24%2C%20%24q%20%5Cin%20%5Cmathcal%7BR%7D%5En%24%2C%20%24G%20%5Cin%20%5Cmathcal%7BR%7D%5E%7Bm%20%5Ctimes%20n%7D%24%2C%20%24h%20%5Cin%20%5Cmathcal%7BR%7D%5Em%24%2C%20%24A%20%5Cin%20%5Cmathcal%7BR%7D%5E%7Bp%20%5Ctimes%20n%7D%24%2C%20and%20%24b%20%5Cin%20%5Cmathcal%7BR%7D%5Ep%24%20are%20problem%20data%20and%20%24x%20%5Cin%20%5Cmathcal%7BR%7D%5E%7Bn%7D%24%20is%20the%20optimization%20variable.%20The%20inequality%20constraint%20%24Gx%20%5Cleq%20h%24%20is%20elementwise.%0A%0A%20%20%20%20%20%20%20%20**Why%20quadratic%20programming%3F**%20Quadratic%20programs%20are%20convex%20optimization%20problems%20that%20generalize%20both%20least-squares%20and%20linear%20programming.They%20can%20be%20solved%20efficiently%20and%20reliably%2C%20even%20in%20real-time.%0A%0A%20%20%20%20%20%20%20%20**An%20example%20from%20finance.**%20A%20simple%20example%20of%20a%20quadratic%20program%20arises%20in%20finance.%20Suppose%20we%20have%20%24n%24%20different%20stocks%2C%20an%20estimate%20%24r%20%5Cin%20%5Cmathcal%7BR%7D%5En%24%20of%20the%20expected%20return%20on%20each%20stock%2C%20and%20an%20estimate%20%24%5CSigma%20%5Cin%20%5Cmathcal%7BS%7D%5E%7Bn%7D_%2B%24%20of%20the%20covariance%20of%20the%20returns.%20Then%20we%20solve%20the%20optimization%20problem%0A%0A%20%20%20%20%20%20%20%20%5C%5B%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Cbegin%7Barray%7D%7Bll%7D%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Ctext%7Bminimize%7D%20%20%20%26%20(1%2F2)x%5ET%5CSigma%20x%20-%20r%5ETx%5C%5C%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Ctext%7Bsubject%20to%7D%20%26%20x%20%5Cgeq%200%20%5C%5C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%26%20%5Cmathbf%7B1%7D%5ETx%20%3D%201%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Cend%7Barray%7D%0A%20%20%20%20%20%20%20%20%5C%5D%0A%0A%20%20%20%20%20%20%20%20to%20find%20a%20nonnegative%20portfolio%20allocation%20%24x%20%5Cin%20%5Cmathcal%7BR%7D%5En_%2B%24%20that%20optimally%20balances%20expected%20return%20and%20variance%20of%20return.%0A%0A%20%20%20%20%20%20%20%20When%20we%20solve%20a%20quadratic%20program%2C%20in%20addition%20to%20a%20solution%20%24x%5E%5Cstar%24%2C%20we%20obtain%20a%20dual%20solution%20%24%5Clambda%5E%5Cstar%24%20corresponding%20to%20the%20inequality%20constraints.%20A%20positive%20entry%20%24%5Clambda%5E%5Cstar_i%24%20indicates%20that%20the%20constraint%20%24g_i%5ETx%20%5Cleq%20h_i%24%20holds%20with%20equality%20for%20%24x%5E%5Cstar%24%20and%20suggests%20that%20changing%20%24h_i%24%20would%20change%20the%20optimal%20value.%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20r%22%22%22%0A%20%20%20%20%20%20%20%20%23%23%20Example%0A%0A%20%20%20%20%20%20%20%20In%20this%20example%2C%20we%20use%20CVXPY%20to%20construct%20and%20solve%20a%20quadratic%20program.%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell%0Adef%20_()%3A%0A%20%20%20%20import%20cvxpy%20as%20cp%0A%20%20%20%20import%20numpy%20as%20np%0A%20%20%20%20return%20cp%2C%20np%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%22%22%22First%20we%20generate%20synthetic%20data.%20In%20this%20problem%2C%20we%20don't%20include%20equality%20constraints%2C%20only%20inequality.%22%22%22)%0A%20%20%20%20return%0A%0A%0A%40app.cell%0Adef%20_(np)%3A%0A%20%20%20%20m%20%3D%204%0A%20%20%20%20n%20%3D%202%0A%0A%20%20%20%20np.random.seed(1)%0A%20%20%20%20q%20%3D%20np.random.randn(n)%0A%20%20%20%20G%20%3D%20np.random.randn(m%2C%20n)%0A%20%20%20%20h%20%3D%20G%20%40%20np.random.randn(n)%0A%20%20%20%20return%20G%2C%20h%2C%20m%2C%20n%2C%20q%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo%2C%20np)%3A%0A%20%20%20%20import%20wigglystuff%0A%0A%20%20%20%20P_widget%20%3D%20mo.ui.anywidget(%0A%20%20%20%20%20%20%20%20wigglystuff.Matrix(np.array(%5B%5B4.0%2C%20-1.4%5D%2C%20%5B-1.4%2C%204%5D%5D)%2C%20step%3D0.1)%0A%20%20%20%20)%0A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20f%22%22%22%0A%20%20%20%20%20%20%20%20The%20quadratic%20form%20%24P%24%20is%20equal%20to%20the%20symmetrized%20version%20of%20this%0A%20%20%20%20%20%20%20%20matrix%3A%0A%0A%20%20%20%20%20%20%20%20%7BP_widget.center()%7D%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%20P_widget%2C%20wigglystuff%0A%0A%0A%40app.cell%0Adef%20_(P_widget%2C%20np)%3A%0A%20%20%20%20P%20%3D%200.5%20*%20(np.array(P_widget.matrix)%20%2B%20np.array(P_widget.matrix).T)%0A%20%20%20%20return%20(P%2C)%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(r%22%22%22Next%2C%20we%20specify%20the%20problem.%20Notice%20that%20we%20use%20the%20%60quad_form%60%20function%20from%20CVXPY%20to%20create%20the%20quadratic%20form%20%24x%5ETPx%24.%22%22%22)%0A%20%20%20%20return%0A%0A%0A%40app.cell%0Adef%20_(G%2C%20P%2C%20cp%2C%20h%2C%20n%2C%20q)%3A%0A%20%20%20%20x%20%3D%20cp.Variable(n)%0A%0A%20%20%20%20problem%20%3D%20cp.Problem(%0A%20%20%20%20%20%20%20%20cp.Minimize((1%20%2F%202)%20*%20cp.quad_form(x%2C%20P)%20%2B%20q.T%20%40%20x)%2C%0A%20%20%20%20%20%20%20%20%5BG%20%40%20x%20%3C%3D%20h%5D%2C%0A%20%20%20%20)%0A%20%20%20%20_%20%3D%20problem.solve()%0A%20%20%20%20return%20problem%2C%20x%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo%2C%20problem%2C%20x)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20f%22%22%22%0A%20%20%20%20%20%20%20%20The%20optimal%20value%20is%20%7Bproblem.value%3A.04f%7D.%0A%0A%20%20%20%20%20%20%20%20A%20solution%20%24x%24%20is%20%7Bmo.as_html(list(x.value))%7D%0A%20%20%20%20%20%20%20%20A%20dual%20solution%20is%20is%20%7Bmo.as_html(list(problem.constraints%5B0%5D.dual_value))%7D%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell%0Adef%20_(G%2C%20P%2C%20h%2C%20plot_contours%2C%20q%2C%20x)%3A%0A%20%20%20%20plot_contours(P%2C%20G%2C%20h%2C%20q%2C%20x.value)%0A%20%20%20%20return%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20r%22%22%22%0A%20%20%20%20%20%20%20%20In%20this%20plot%2C%20the%20gray%20shaded%20region%20is%20the%20feasible%20region%20(points%20satisfying%20the%20inequality)%2C%20and%20the%20ellipses%20are%20level%20curves%20of%20the%20quadratic%20form.%0A%0A%20%20%20%20%20%20%20%20**%F0%9F%8C%8A%20Try%20it!**%20Try%20changing%20the%20entries%20of%20%24P%24%20above%20with%20your%20mouse.%20How%20do%20the%0A%20%20%20%20%20%20%20%20level%20curves%20and%20the%20optimal%20value%20of%20%24x%24%20change%3F%20Can%20you%20explain%20what%20you%20see%3F%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(P%2C%20mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20rf%22%22%22%0A%20%20%20%20%20%20%20%20The%20above%20contour%20lines%20were%20generated%20with%0A%0A%20%20%20%20%20%20%20%20%5C%5B%0A%20%20%20%20%20%20%20%20P%3D%20%5Cbegin%7B%7Bbmatrix%7D%7D%0A%20%20%20%20%20%20%20%20%7BP%5B0%2C%200%5D%3A.01f%7D%20%26%20%7BP%5B0%2C%201%5D%3A.01f%7D%20%5C%5C%0A%20%20%20%20%20%20%20%20%7BP%5B1%2C%200%5D%3A.01f%7D%20%26%20%7BP%5B1%2C%201%5D%3A.01f%7D%20%5C%5C%0A%20%20%20%20%20%20%20%20%5Cend%7B%7Bbmatrix%7D%7D%0A%20%20%20%20%20%20%20%20%5C%5D%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(np)%3A%0A%20%20%20%20def%20plot_contours(P%2C%20G%2C%20h%2C%20q%2C%20x_star)%3A%0A%20%20%20%20%20%20%20%20import%20matplotlib.pyplot%20as%20plt%0A%0A%20%20%20%20%20%20%20%20%23%20Create%20a%20grid%20of%20x%20and%20y%20values.%0A%20%20%20%20%20%20%20%20x%20%3D%20np.linspace(-5%2C%205%2C%20400)%0A%20%20%20%20%20%20%20%20y%20%3D%20np.linspace(-5%2C%205%2C%20400)%0A%20%20%20%20%20%20%20%20X%2C%20Y%20%3D%20np.meshgrid(x%2C%20y)%0A%0A%20%20%20%20%20%20%20%20%23%20Compute%20the%20quadratic%20form%20Q(x%2C%20y)%20%3D%20a*x%5E2%20%2B%202*b*x*y%20%2B%20c*y%5E2.%0A%20%20%20%20%20%20%20%20%23%20Here%2C%20a%20%3D%20P%5B0%2C0%5D%2C%20b%20%3D%20P%5B0%2C1%5D%20(and%20P%5B1%2C0%5D)%2C%20c%20%3D%20P%5B1%2C1%5D%0A%20%20%20%20%20%20%20%20Z%20%3D%20(%0A%20%20%20%20%20%20%20%20%20%20%20%200.5%20*%20(P%5B0%2C%200%5D%20*%20X**2%20%2B%202%20*%20P%5B0%2C%201%5D%20*%20X%20*%20Y%20%2B%20P%5B1%2C%201%5D%20*%20Y**2)%0A%20%20%20%20%20%20%20%20%20%20%20%20%2B%20q%5B0%5D%20*%20X%0A%20%20%20%20%20%20%20%20%20%20%20%20%2B%20q%5B1%5D%20*%20Y%0A%20%20%20%20%20%20%20%20)%0A%0A%20%20%20%20%20%20%20%20%23%20---%20Evaluate%20the%20constraints%20on%20the%20grid%20---%0A%20%20%20%20%20%20%20%20%23%20We%20stack%20X%20and%20Y%20to%20get%20a%20list%20of%20(x%2Cy)%20points.%0A%20%20%20%20%20%20%20%20points%20%3D%20np.vstack(%5BX.ravel()%2C%20Y.ravel()%5D).T%0A%0A%20%20%20%20%20%20%20%20%23%20Start%20with%20all%20points%20feasible%0A%20%20%20%20%20%20%20%20feasible%20%3D%20np.ones(points.shape%5B0%5D%2C%20dtype%3Dbool)%0A%0A%20%20%20%20%20%20%20%20%23%20Apply%20the%20inequality%20constraints%20Gx%20%3C%3D%20h.%0A%20%20%20%20%20%20%20%20%23%20Each%20row%20of%20G%20and%20corresponding%20h%20defines%20a%20condition.%0A%20%20%20%20%20%20%20%20for%20i%20in%20range(G.shape%5B0%5D)%3A%0A%20%20%20%20%20%20%20%20%20%20%20%20%23%20For%20a%20given%20point%20x%2C%20the%20condition%20is%3A%20G%5Bi%2C0%5D*x%20%2B%20G%5Bi%2C1%5D*y%20%3C%3D%20h%5Bi%5D%0A%20%20%20%20%20%20%20%20%20%20%20%20feasible%20%26%3D%20points.dot(G%5Bi%5D)%20%3C%3D%20h%5Bi%5D%20%2B%201e-8%20%20%23%20small%20tolerance%0A%20%20%20%20%20%20%20%20%23%20Reshape%20the%20boolean%20mask%20back%20to%20grid%20shape.%0A%20%20%20%20%20%20%20%20feasible_grid%20%3D%20feasible.reshape(X.shape)%0A%0A%20%20%20%20%20%20%20%20%23%20---%20Plot%20the%20feasible%20region%20and%20contour%20lines---%0A%20%20%20%20%20%20%20%20plt.figure(figsize%3D(8%2C%206))%0A%0A%20%20%20%20%20%20%20%20%23%20Use%20contourf%20to%20fill%20the%20region%20where%20feasible_grid%20is%20True.%0A%20%20%20%20%20%20%20%20%23%20We%20define%20two%20levels%2C%20so%20that%20points%20that%20are%20True%20(feasible)%20get%20one%0A%20%20%20%20%20%20%20%20%23%20color.%0A%20%20%20%20%20%20%20%20plt.contourf(%0A%20%20%20%20%20%20%20%20%20%20%20%20X%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20Y%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20feasible_grid%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20levels%3D%5B-0.5%2C%200.5%2C%201.5%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20colors%3D%5B%22white%22%2C%20%22gray%22%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20alpha%3D0.5%2C%0A%20%20%20%20%20%20%20%20)%0A%0A%20%20%20%20%20%20%20%20contours%20%3D%20plt.contour(X%2C%20Y%2C%20Z%2C%20levels%3D10%2C%20cmap%3D%22viridis%22)%0A%20%20%20%20%20%20%20%20plt.clabel(contours%2C%20inline%3DTrue%2C%20fontsize%3D8)%0A%20%20%20%20%20%20%20%20plt.title(%22Feasible%20region%20and%20level%20curves%22)%0A%20%20%20%20%20%20%20%20plt.xlabel(%22%24x_1%24%22)%0A%20%20%20%20%20%20%20%20plt.ylabel(%22%24y_2%24%22)%0A%20%20%20%20%20%20%20%20%23%20plt.colorbar(contours%2C%20label%3D'Q(x%2C%20y)')%0A%0A%20%20%20%20%20%20%20%20ax%20%3D%20plt.gca()%0A%20%20%20%20%20%20%20%20%23%20Optionally%2C%20mark%20and%20label%20the%20point%20x_star.%0A%20%20%20%20%20%20%20%20ax.plot(x_star%5B0%5D%2C%20x_star%5B1%5D%2C%20%22ko%22%2C%20markersize%3D5)%0A%20%20%20%20%20%20%20%20ax.text(%0A%20%20%20%20%20%20%20%20%20%20%20%20x_star%5B0%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20x_star%5B1%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20r%22%24%5Cmathbf%7Bx%7D%5E%5Cstar%24%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20color%3D%22black%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20fontsize%3D12%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20verticalalignment%3D%22bottom%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20horizontalalignment%3D%22right%22%2C%0A%20%20%20%20%20%20%20%20)%0A%20%20%20%20%20%20%20%20return%20plt.gca()%0A%20%20%20%20return%20(plot_contours%2C)%0A%0A%0Aif%20__name__%20%3D%3D%20%22__main__%22%3A%0A%20%20%20%20app.run()%0A</marimo-code></head>
|
80 |
+
<body>
|
81 |
+
<div id="root"></div>
|
82 |
+
</body>
|
83 |
+
</html>
|
_site/optimization/05_portfolio_optimization.html
ADDED
@@ -0,0 +1,83 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<!DOCTYPE html>
|
2 |
+
<html lang="en">
|
3 |
+
<head>
|
4 |
+
<meta charset="utf-8" />
|
5 |
+
<link rel="icon" href="./favicon.ico" />
|
6 |
+
<!-- Preload is necessary because we show these images when we disconnect from the server,
|
7 |
+
but at that point we cannot load these images from the server -->
|
8 |
+
<link rel="preload" href="./assets/gradient-yHQUC_QB.png" as="image" />
|
9 |
+
<link rel="preload" href="./assets/noise-60BoTA8O.png" as="image" />
|
10 |
+
<!-- Preload the fonts -->
|
11 |
+
<link rel="preload" href="./assets/Lora-VariableFont_wght-B2ootaw-.ttf" as="font" crossorigin="anonymous" />
|
12 |
+
<link rel="preload" href="./assets/PTSans-Regular-CxL0S8W7.ttf" as="font" crossorigin="anonymous" />
|
13 |
+
<link rel="preload" href="./assets/PTSans-Bold-D9fedIX3.ttf" as="font" crossorigin="anonymous" />
|
14 |
+
<link rel="preload" href="./assets/FiraMono-Regular-BTCkDNvf.ttf" as="font" crossorigin="anonymous" />
|
15 |
+
<link rel="preload" href="./assets/FiraMono-Medium-DU3aDxX5.ttf" as="font" crossorigin="anonymous" />
|
16 |
+
<link rel="preload" href="./assets/FiraMono-Bold-CLVRCuM9.ttf" as="font" crossorigin="anonymous" />
|
17 |
+
|
18 |
+
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
19 |
+
<meta name="theme-color" content="#000000" />
|
20 |
+
<meta name="description" content="a marimo app" />
|
21 |
+
<link rel="apple-touch-icon" href="./apple-touch-icon.png" />
|
22 |
+
<link rel="manifest" href="./manifest.json" />
|
23 |
+
|
24 |
+
<script data-marimo="true">
|
25 |
+
function __resizeIframe(obj) {
|
26 |
+
var scrollbarHeight = 20; // Max between windows, mac, and linux
|
27 |
+
|
28 |
+
function setHeight() {
|
29 |
+
var element = obj.contentWindow.document.documentElement;
|
30 |
+
// If there is no vertical scrollbar, we don't need to resize the iframe
|
31 |
+
if (element.scrollHeight === element.clientHeight) {
|
32 |
+
return;
|
33 |
+
}
|
34 |
+
|
35 |
+
// Create a new height that includes the scrollbar height if it's visible
|
36 |
+
var hasHorizontalScrollbar = element.scrollWidth > element.clientWidth;
|
37 |
+
var newHeight = element.scrollHeight + (hasHorizontalScrollbar ? scrollbarHeight : 0);
|
38 |
+
|
39 |
+
// Only update the height if it's different from the current height
|
40 |
+
if (obj.style.height !== `${newHeight}px`) {
|
41 |
+
obj.style.height = `${newHeight}px`;
|
42 |
+
}
|
43 |
+
}
|
44 |
+
|
45 |
+
// Resize the iframe to the height of the content and bottom scrollbar height
|
46 |
+
setHeight();
|
47 |
+
|
48 |
+
// Resize the iframe when the content changes
|
49 |
+
const resizeObserver = new ResizeObserver((entries) => {
|
50 |
+
setHeight();
|
51 |
+
});
|
52 |
+
resizeObserver.observe(obj.contentWindow.document.body);
|
53 |
+
}
|
54 |
+
</script>
|
55 |
+
<marimo-filename hidden>notebook.py</marimo-filename>
|
56 |
+
<marimo-mode data-mode='edit' hidden></marimo-mode>
|
57 |
+
<marimo-version data-version='0.11.9' hidden></marimo-version>
|
58 |
+
<marimo-user-config data-config='{"completion": {"activate_on_typing": true, "copilot": false}, "display": {"cell_output": "above", "default_width": "medium", "dataframes": "rich", "code_editor_font_size": 14, "theme": "light"}, "formatting": {"line_length": 79}, "keymap": {"preset": "default", "overrides": {}}, "runtime": {"auto_instantiate": true, "auto_reload": "off", "on_cell_change": "autorun", "watcher_on_save": "lazy", "output_max_bytes": 8000000, "std_stream_max_bytes": 1000000}, "save": {"autosave": "off", "autosave_delay": 1000, "format_on_save": false}, "package_management": {"manager": "pip"}, "server": {"browser": "default", "follow_symlink": false}, "snippets": {"custom_paths": [], "include_default_snippets": true}}' data-overrides='{}' hidden></marimo-user-config>
|
59 |
+
<marimo-app-config data-config='{"width": "compact"}' hidden></marimo-app-config>
|
60 |
+
<marimo-server-token data-token='123' hidden></marimo-server-token>
|
61 |
+
<title>05 portfolio optimization</title>
|
62 |
+
<script type="module" crossorigin src="./assets/index-BiV-b1K2.js"></script>
|
63 |
+
<link rel="stylesheet" crossorigin href="./assets/index-DkqMrX_B.css">
|
64 |
+
<marimo-wasm hidden=""></marimo-wasm>
|
65 |
+
<script>
|
66 |
+
if (window.location.protocol === 'file:') {
|
67 |
+
alert('Warning: This file must be served by an HTTP server to function correctly.');
|
68 |
+
}
|
69 |
+
</script>
|
70 |
+
|
71 |
+
<style>
|
72 |
+
#save-button {
|
73 |
+
display: none !important;
|
74 |
+
}
|
75 |
+
#filename-input {
|
76 |
+
display: none !important;
|
77 |
+
}
|
78 |
+
</style>
|
79 |
+
<marimo-code hidden="" data-show-code="false">import%20marimo%0A%0A__generated_with%20%3D%20%220.11.9%22%0Aapp%20%3D%20marimo.App()%0A%0A%0A%40app.cell%0Adef%20_()%3A%0A%20%20%20%20import%20marimo%20as%20mo%0A%20%20%20%20return%20(mo%2C)%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(r%22%22%22%23%20Portfolio%20optimization%22%22%22)%0A%20%20%20%20return%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20r%22%22%22%0A%20%20%20%20%20%20%20%20In%20this%20example%20we%20show%20how%20to%20use%20CVXPY%20to%20design%20a%20financial%20portfolio%3B%20this%20is%20called%20_portfolio%20optimization_.%0A%0A%20%20%20%20%20%20%20%20In%20portfolio%20optimization%20we%20have%20some%20amount%20of%20money%20to%20invest%20in%20any%20of%20%24n%24%20different%20assets.%0A%20%20%20%20%20%20%20%20We%20choose%20what%20fraction%20%24w_i%24%20of%20our%20money%20to%20invest%20in%20each%20asset%20%24i%24%2C%20%24i%3D1%2C%20%5Cldots%2C%20n%24.%20The%20goal%20is%20to%20maximize%20return%20of%20the%20portfolio%20while%20minimizing%20risk.%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20r%22%22%22%0A%20%20%20%20%20%20%20%20%23%23%20Asset%20returns%20and%20risk%0A%0A%20%20%20%20%20%20%20%20We%20will%20only%20model%20investments%20held%20for%20one%20period.%20The%20initial%20prices%20are%20%24p_i%20%3E%200%24.%20The%20end%20of%20period%20prices%20are%20%24p_i%5E%2B%20%3E0%24.%20The%20asset%20(fractional)%20returns%20are%20%24r_i%20%3D%20(p_i%5E%2B-p_i)%2Fp_i%24.%20The%20portfolio%20(fractional)%20return%20is%20%24R%20%3D%20r%5ETw%24.%0A%0A%20%20%20%20%20%20%20%20A%20common%20model%20is%20that%20%24r%24%20is%20a%20random%20variable%20with%20mean%20%24%7B%5Cbf%20E%7Dr%20%3D%20%5Cmu%24%20and%20covariance%20%24%7B%5Cbf%20E%7B(r-%5Cmu)(r-%5Cmu)%5ET%7D%7D%20%3D%20%5CSigma%24.%0A%20%20%20%20%20%20%20%20It%20follows%20that%20%24R%24%20is%20a%20random%20variable%20with%20%24%7B%5Cbf%20E%7DR%20%3D%20%5Cmu%5ET%20w%24%20and%20%24%7B%5Cbf%20var%7D(R)%20%3D%20w%5ET%5CSigma%20w%24.%20In%20real-world%20applications%2C%20%24%5Cmu%24%20and%20%24%5CSigma%24%20are%20estimated%20from%20data%20and%20models%2C%20and%20%24w%24%20is%20chosen%20using%20a%20library%20like%20CVXPY.%0A%0A%20%20%20%20%20%20%20%20%24%7B%5Cbf%20E%7DR%24%20is%20the%20(mean)%20*return*%20of%20the%20portfolio.%20%24%7B%5Cbf%20var%7D(R)%24%20is%20the%20*risk*%20of%20the%20portfolio.%20Portfolio%20optimization%20has%20two%20competing%20objectives%3A%20high%20return%20and%20low%20risk.%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20r%22%22%22%0A%20%20%20%20%20%20%20%20%23%23%20Classical%20(Markowitz)%20portfolio%20optimization%0A%0A%20%20%20%20%20%20%20%20Classical%20(Markowitz)%20portfolio%20optimization%20solves%20the%20optimization%20problem%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20r%22%22%22%0A%20%20%20%20%20%20%20%20%24%24%0A%20%20%20%20%20%20%20%20%5Cbegin%7Barray%7D%7Bll%7D%20%5Ctext%7Bmaximize%7D%20%26%20%5Cmu%5ET%20w%20-%20%5Cgamma%20w%5ET%5CSigma%20w%5C%5C%0A%20%20%20%20%20%20%20%20%5Ctext%7Bsubject%20to%7D%20%26%20%7B%5Cbf%201%7D%5ET%20w%20%3D%201%2C%20w%20%5Cgeq%200%2C%0A%20%20%20%20%20%20%20%20%5Cend%7Barray%7D%0A%20%20%20%20%20%20%20%20%24%24%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20r%22%22%22%0A%20%20%20%20%20%20%20%20where%20%24w%20%5Cin%20%7B%5Cbf%20R%7D%5En%24%20is%20the%20optimization%20variable%20and%20%24%5Cgamma%20%3E0%24%20is%20a%20constant%20called%20the%20*risk%20aversion%20parameter*.%20The%20constraint%20%24%5Cmathbf%7B1%7D%5ETw%20%3D%201%24%20says%20the%20portfolio%20weight%20vector%20must%20sum%20to%201%2C%20and%20%24w%20%5Cgeq%200%24%20says%20that%20we%20can't%20invest%20a%20negative%20amount%20into%20any%20asset.%0A%0A%20%20%20%20%20%20%20%20The%20objective%20%24%5Cmu%5ETw%20-%20%5Cgamma%20w%5ET%5CSigma%20w%24%20is%20the%20*risk-adjusted%20return*.%20Varying%20%24%5Cgamma%24%20gives%20the%20optimal%20*risk-return%20trade-off*.%0A%20%20%20%20%20%20%20%20We%20can%20get%20the%20same%20risk-return%20trade-off%20by%20fixing%20return%20and%20minimizing%20risk.%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20r%22%22%22%0A%20%20%20%20%20%20%20%20%23%23%20Example%0A%0A%20%20%20%20%20%20%20%20In%20the%20following%20code%20we%20compute%20and%20plot%20the%20optimal%20risk-return%20trade-off%20for%20%2410%24%20assets.%20First%20we%20generate%20random%20problem%20data%20%24%5Cmu%24%20and%20%24%5CSigma%24.%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell%0Adef%20_()%3A%0A%20%20%20%20import%20numpy%20as%20np%0A%20%20%20%20return%20(np%2C)%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo%2C%20np)%3A%0A%20%20%20%20import%20wigglystuff%0A%0A%20%20%20%20mu_widget%20%3D%20mo.ui.anywidget(%0A%20%20%20%20%20%20%20%20wigglystuff.Matrix(%0A%20%20%20%20%20%20%20%20%20%20%20%20np.array(%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%5B%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%5B1.6%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%5B0.6%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%5B0.5%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%5B1.1%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%5B0.9%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%5B2.3%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%5B1.7%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%5B0.7%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%5B0.9%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%5B0.3%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%5D%0A%20%20%20%20%20%20%20%20%20%20%20%20)%0A%20%20%20%20%20%20%20%20)%0A%20%20%20%20)%0A%0A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20rf%22%22%22%0A%20%20%20%20%20%20%20%20The%20value%20of%20%24%5Cmu%24%20is%20%0A%0A%20%20%20%20%20%20%20%20%7Bmu_widget.center()%7D%0A%0A%20%20%20%20%20%20%20%20_Try%20changing%20the%20entries%20of%20%24%5Cmu%24%20and%20see%20how%20the%20plots%20below%20change._%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%20mu_widget%2C%20wigglystuff%0A%0A%0A%40app.cell%0Adef%20_(mu_widget%2C%20np)%3A%0A%20%20%20%20np.random.seed(1)%0A%20%20%20%20n%20%3D%2010%0A%20%20%20%20mu%20%3D%20np.array(mu_widget.matrix)%0A%20%20%20%20Sigma%20%3D%20np.random.randn(n%2C%20n)%0A%20%20%20%20Sigma%20%3D%20Sigma.T.dot(Sigma)%0A%20%20%20%20return%20Sigma%2C%20mu%2C%20n%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%22%22%22Next%2C%20we%20solve%20the%20problem%20for%20100%20different%20values%20of%20%24%5Cgamma%24%22%22%22)%0A%20%20%20%20return%0A%0A%0A%40app.cell%0Adef%20_(Sigma%2C%20mu%2C%20n)%3A%0A%20%20%20%20import%20cvxpy%20as%20cp%0A%0A%20%20%20%20w%20%3D%20cp.Variable(n)%0A%20%20%20%20gamma%20%3D%20cp.Parameter(nonneg%3DTrue)%0A%20%20%20%20ret%20%3D%20mu.T%20%40%20w%0A%20%20%20%20risk%20%3D%20cp.quad_form(w%2C%20Sigma)%0A%20%20%20%20prob%20%3D%20cp.Problem(cp.Maximize(ret%20-%20gamma%20*%20risk)%2C%20%5Bcp.sum(w)%20%3D%3D%201%2C%20w%20%3E%3D%200%5D)%0A%20%20%20%20return%20cp%2C%20gamma%2C%20prob%2C%20ret%2C%20risk%2C%20w%0A%0A%0A%40app.cell%0Adef%20_(cp%2C%20gamma%2C%20np%2C%20prob%2C%20ret%2C%20risk)%3A%0A%20%20%20%20_SAMPLES%20%3D%20100%0A%20%20%20%20risk_data%20%3D%20np.zeros(_SAMPLES)%0A%20%20%20%20ret_data%20%3D%20np.zeros(_SAMPLES)%0A%20%20%20%20gamma_vals%20%3D%20np.logspace(-2%2C%203%2C%20num%3D_SAMPLES)%0A%20%20%20%20for%20_i%20in%20range(_SAMPLES)%3A%0A%20%20%20%20%20%20%20%20gamma.value%20%3D%20gamma_vals%5B_i%5D%0A%20%20%20%20%20%20%20%20prob.solve()%0A%20%20%20%20%20%20%20%20risk_data%5B_i%5D%20%3D%20cp.sqrt(risk).value%0A%20%20%20%20%20%20%20%20ret_data%5B_i%5D%20%3D%20ret.value%0A%20%20%20%20return%20gamma_vals%2C%20ret_data%2C%20risk_data%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%22%22%22Plotted%20below%20are%20the%20risk%20return%20tradeoffs%20for%20two%20values%20of%20%24%5Cgamma%24%20(blue%20squares)%2C%20and%20the%20risk%20return%20tradeoffs%20for%20investing%20fully%20in%20each%20asset%20(red%20circles)%22%22%22)%0A%20%20%20%20return%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(Sigma%2C%20cp%2C%20gamma_vals%2C%20mu%2C%20n%2C%20ret_data%2C%20risk_data)%3A%0A%20%20%20%20import%20matplotlib.pyplot%20as%20plt%0A%0A%20%20%20%20markers_on%20%3D%20%5B29%2C%2040%5D%0A%20%20%20%20fig%20%3D%20plt.figure()%0A%20%20%20%20ax%20%3D%20fig.add_subplot(111)%0A%20%20%20%20plt.plot(risk_data%2C%20ret_data%2C%20%22g-%22)%0A%20%20%20%20for%20marker%20in%20markers_on%3A%0A%20%20%20%20%20%20%20%20plt.plot(risk_data%5Bmarker%5D%2C%20ret_data%5Bmarker%5D%2C%20%22bs%22)%0A%20%20%20%20%20%20%20%20ax.annotate(%0A%20%20%20%20%20%20%20%20%20%20%20%20%22%24%5C%5Cgamma%20%3D%20%25.2f%24%22%20%25%20gamma_vals%5Bmarker%5D%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20xy%3D(risk_data%5Bmarker%5D%20%2B%200.08%2C%20ret_data%5Bmarker%5D%20-%200.03)%2C%0A%20%20%20%20%20%20%20%20)%0A%20%20%20%20for%20_i%20in%20range(n)%3A%0A%20%20%20%20%20%20%20%20plt.plot(cp.sqrt(Sigma%5B_i%2C%20_i%5D).value%2C%20mu%5B_i%5D%2C%20%22ro%22)%0A%20%20%20%20plt.xlabel(%22Standard%20deviation%22)%0A%20%20%20%20plt.ylabel(%22Return%22)%0A%20%20%20%20plt.show()%0A%20%20%20%20return%20ax%2C%20fig%2C%20marker%2C%20markers_on%2C%20plt%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20r%22%22%22%0A%20%20%20%20%20%20%20%20We%20plot%20below%20the%20return%20distributions%20for%20the%20two%20risk%20aversion%20values%20marked%20on%20the%20trade-off%20curve.%0A%20%20%20%20%20%20%20%20Notice%20that%20the%20probability%20of%20a%20loss%20is%20near%200%20for%20the%20low%20risk%20value%20and%20far%20above%200%20for%20the%20high%20risk%20value.%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(gamma%2C%20gamma_vals%2C%20markers_on%2C%20np%2C%20plt%2C%20prob%2C%20ret%2C%20risk)%3A%0A%20%20%20%20import%20scipy.stats%20as%20spstats%0A%0A%20%20%20%20plt.figure()%0A%20%20%20%20for%20midx%2C%20_idx%20in%20enumerate(markers_on)%3A%0A%20%20%20%20%20%20%20%20gamma.value%20%3D%20gamma_vals%5B_idx%5D%0A%20%20%20%20%20%20%20%20prob.solve()%0A%20%20%20%20%20%20%20%20x%20%3D%20np.linspace(-2%2C%205%2C%201000)%0A%20%20%20%20%20%20%20%20plt.plot(%0A%20%20%20%20%20%20%20%20%20%20%20%20x%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20spstats.norm.pdf(x%2C%20ret.value%2C%20risk.value)%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20label%3D%22%24%5C%5Cgamma%20%3D%20%25.2f%24%22%20%25%20gamma.value%2C%0A%20%20%20%20%20%20%20%20)%0A%20%20%20%20plt.xlabel(%22Return%22)%0A%20%20%20%20plt.ylabel(%22Density%22)%0A%20%20%20%20plt.legend(loc%3D%22upper%20right%22)%0A%20%20%20%20plt.show()%0A%20%20%20%20return%20midx%2C%20spstats%2C%20x%0A%0A%0Aif%20__name__%20%3D%3D%20%22__main__%22%3A%0A%20%20%20%20app.run()%0A</marimo-code></head>
|
80 |
+
<body>
|
81 |
+
<div id="root"></div>
|
82 |
+
</body>
|
83 |
+
</html>
|
_site/optimization/06_convex_optimization.html
ADDED
@@ -0,0 +1,83 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<!DOCTYPE html>
|
2 |
+
<html lang="en">
|
3 |
+
<head>
|
4 |
+
<meta charset="utf-8" />
|
5 |
+
<link rel="icon" href="./favicon.ico" />
|
6 |
+
<!-- Preload is necessary because we show these images when we disconnect from the server,
|
7 |
+
but at that point we cannot load these images from the server -->
|
8 |
+
<link rel="preload" href="./assets/gradient-yHQUC_QB.png" as="image" />
|
9 |
+
<link rel="preload" href="./assets/noise-60BoTA8O.png" as="image" />
|
10 |
+
<!-- Preload the fonts -->
|
11 |
+
<link rel="preload" href="./assets/Lora-VariableFont_wght-B2ootaw-.ttf" as="font" crossorigin="anonymous" />
|
12 |
+
<link rel="preload" href="./assets/PTSans-Regular-CxL0S8W7.ttf" as="font" crossorigin="anonymous" />
|
13 |
+
<link rel="preload" href="./assets/PTSans-Bold-D9fedIX3.ttf" as="font" crossorigin="anonymous" />
|
14 |
+
<link rel="preload" href="./assets/FiraMono-Regular-BTCkDNvf.ttf" as="font" crossorigin="anonymous" />
|
15 |
+
<link rel="preload" href="./assets/FiraMono-Medium-DU3aDxX5.ttf" as="font" crossorigin="anonymous" />
|
16 |
+
<link rel="preload" href="./assets/FiraMono-Bold-CLVRCuM9.ttf" as="font" crossorigin="anonymous" />
|
17 |
+
|
18 |
+
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
19 |
+
<meta name="theme-color" content="#000000" />
|
20 |
+
<meta name="description" content="a marimo app" />
|
21 |
+
<link rel="apple-touch-icon" href="./apple-touch-icon.png" />
|
22 |
+
<link rel="manifest" href="./manifest.json" />
|
23 |
+
|
24 |
+
<script data-marimo="true">
|
25 |
+
function __resizeIframe(obj) {
|
26 |
+
var scrollbarHeight = 20; // Max between windows, mac, and linux
|
27 |
+
|
28 |
+
function setHeight() {
|
29 |
+
var element = obj.contentWindow.document.documentElement;
|
30 |
+
// If there is no vertical scrollbar, we don't need to resize the iframe
|
31 |
+
if (element.scrollHeight === element.clientHeight) {
|
32 |
+
return;
|
33 |
+
}
|
34 |
+
|
35 |
+
// Create a new height that includes the scrollbar height if it's visible
|
36 |
+
var hasHorizontalScrollbar = element.scrollWidth > element.clientWidth;
|
37 |
+
var newHeight = element.scrollHeight + (hasHorizontalScrollbar ? scrollbarHeight : 0);
|
38 |
+
|
39 |
+
// Only update the height if it's different from the current height
|
40 |
+
if (obj.style.height !== `${newHeight}px`) {
|
41 |
+
obj.style.height = `${newHeight}px`;
|
42 |
+
}
|
43 |
+
}
|
44 |
+
|
45 |
+
// Resize the iframe to the height of the content and bottom scrollbar height
|
46 |
+
setHeight();
|
47 |
+
|
48 |
+
// Resize the iframe when the content changes
|
49 |
+
const resizeObserver = new ResizeObserver((entries) => {
|
50 |
+
setHeight();
|
51 |
+
});
|
52 |
+
resizeObserver.observe(obj.contentWindow.document.body);
|
53 |
+
}
|
54 |
+
</script>
|
55 |
+
<marimo-filename hidden>notebook.py</marimo-filename>
|
56 |
+
<marimo-mode data-mode='edit' hidden></marimo-mode>
|
57 |
+
<marimo-version data-version='0.11.9' hidden></marimo-version>
|
58 |
+
<marimo-user-config data-config='{"completion": {"activate_on_typing": true, "copilot": false}, "display": {"dataframes": "rich", "default_width": "medium", "cell_output": "above", "code_editor_font_size": 14, "theme": "light"}, "formatting": {"line_length": 79}, "keymap": {"preset": "default", "overrides": {}}, "runtime": {"auto_instantiate": true, "auto_reload": "off", "on_cell_change": "autorun", "watcher_on_save": "lazy", "output_max_bytes": 8000000, "std_stream_max_bytes": 1000000}, "save": {"autosave": "off", "autosave_delay": 1000, "format_on_save": false}, "package_management": {"manager": "pip"}, "server": {"browser": "default", "follow_symlink": false}, "snippets": {"custom_paths": [], "include_default_snippets": true}}' data-overrides='{}' hidden></marimo-user-config>
|
59 |
+
<marimo-app-config data-config='{"width": "compact"}' hidden></marimo-app-config>
|
60 |
+
<marimo-server-token data-token='123' hidden></marimo-server-token>
|
61 |
+
<title>06 convex optimization</title>
|
62 |
+
<script type="module" crossorigin src="./assets/index-BiV-b1K2.js"></script>
|
63 |
+
<link rel="stylesheet" crossorigin href="./assets/index-DkqMrX_B.css">
|
64 |
+
<marimo-wasm hidden=""></marimo-wasm>
|
65 |
+
<script>
|
66 |
+
if (window.location.protocol === 'file:') {
|
67 |
+
alert('Warning: This file must be served by an HTTP server to function correctly.');
|
68 |
+
}
|
69 |
+
</script>
|
70 |
+
|
71 |
+
<style>
|
72 |
+
#save-button {
|
73 |
+
display: none !important;
|
74 |
+
}
|
75 |
+
#filename-input {
|
76 |
+
display: none !important;
|
77 |
+
}
|
78 |
+
</style>
|
79 |
+
<marimo-code hidden="" data-show-code="false">import%20marimo%0A%0A__generated_with%20%3D%20%220.11.9%22%0Aapp%20%3D%20marimo.App()%0A%0A%0A%40app.cell%0Adef%20_()%3A%0A%20%20%20%20import%20marimo%20as%20mo%0A%20%20%20%20return%20(mo%2C)%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20r%22%22%22%0A%20%20%20%20%20%20%20%20%23%20Convex%20optimization%0A%0A%20%20%20%20%20%20%20%20In%20the%20previous%20tutorials%2C%20we%20learned%20about%20least%20squares%2C%20linear%20programming%2C%0A%20%20%20%20%20%20%20%20and%20quadratic%20programming%2C%20and%20saw%20applications%20of%20each.%20We%20also%20learned%20that%20these%20problem%0A%20%20%20%20%20%20%20%20classes%20can%20be%20solved%20efficiently%20and%20reliably%20using%20CVXPY.%20That's%20because%20these%20problem%20classes%20are%20a%20special%0A%20%20%20%20%20%20%20%20case%20of%20a%20more%20general%20class%20of%20tractable%20problems%2C%20called%20**convex%20optimization%20problems.**%0A%0A%20%20%20%20%20%20%20%20A%20convex%20optimization%20problem%20is%20an%20optimization%20problem%20that%20minimizes%20a%20convex%0A%20%20%20%20%20%20%20%20function%2C%20subject%20to%20affine%20equality%20constraints%20and%20convex%20inequality%0A%20%20%20%20%20%20%20%20constraints%20(%24f_i(x)%5Cleq%200%24%2C%20where%20%24f_i%24%20is%20a%20convex%20function).%0A%0A%20%20%20%20%20%20%20%20**CVXPY.**%20CVXPY%20lets%20you%20specify%20and%20solve%20any%20convex%20optimization%20problem%2C%0A%20%20%20%20%20%20%20%20abstracting%20away%20the%20more%20specific%20problem%20classes.%20You%20start%20with%20CVXPY's%20**atomic%20functions**%2C%20like%20%60cp.exp%60%2C%20%60cp.log%60%2C%20and%20%60cp.square%60%2C%20and%20compose%20them%20to%20build%20more%20complex%20convex%20functions.%20As%20long%20as%20the%20functions%20are%20composed%20in%20the%20right%20way%20%E2%80%94%20as%20long%20as%20they%20are%20%22DCP-compliant%22%20%E2%80%94%20%20your%20resulting%20problem%20will%20be%20convex%20and%20solvable%20by%20CVXPY.%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20r%22%22%22%0A%20%20%20%20%20%20%20%20**%F0%9F%9B%91%20Stop!**%20Before%20proceeding%2C%20read%20the%20CVXPY%20docs%20to%20learn%20about%20atomic%20functions%20and%20the%20DCP%20ruleset%3A%0A%0A%20%20%20%20%20%20%20%20https%3A%2F%2Fwww.cvxpy.org%2Ftutorial%2Findex.html%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(r%22%22%22**Is%20my%20problem%20DCP-compliant%3F**%20Below%20is%20a%20sample%20CVXPY%20problem.%20It%20is%20DCP-compliant.%20Try%20typing%20in%20other%20problems%20and%20seeing%20if%20they%20are%20DCP-compliant.%20If%20you%20know%20your%20problem%20is%20convex%2C%20there%20exists%20a%20way%20to%20express%20it%20in%20a%20DCP-compliant%20way.%22%22%22)%0A%20%20%20%20return%0A%0A%0A%40app.cell%0Adef%20_(mo)%3A%0A%20%20%20%20import%20cvxpy%20as%20cp%0A%20%20%20%20import%20numpy%20as%20np%0A%0A%20%20%20%20x%20%3D%20cp.Variable(3)%0A%20%20%20%20P_sqrt%20%3D%20np.random.randn(3%2C%203)%0A%0A%20%20%20%20objective%20%3D%20cp.log(np.random.randn(3)%20%40%20x)%20-%20cp.sum_squares(P_sqrt%20%40%20x)%0A%20%20%20%20constraints%20%3D%20%5Bx%20%3E%3D%200%2C%20cp.sum(x)%20%3D%3D%201%5D%0A%20%20%20%20problem%20%3D%20cp.Problem(cp.Maximize(objective)%2C%20constraints)%0A%20%20%20%20mo.md(f%22Is%20my%20problem%20DCP%3F%20%60%7Bproblem.is_dcp()%7D%60%22)%0A%20%20%20%20return%20P_sqrt%2C%20constraints%2C%20cp%2C%20np%2C%20objective%2C%20problem%2C%20x%0A%0A%0A%40app.cell%0Adef%20_(problem)%3A%0A%20%20%20%20problem.solve()%0A%20%20%20%20return%0A%0A%0A%40app.cell%0Adef%20_(x)%3A%0A%20%20%20%20x.value%0A%20%20%20%20return%0A%0A%0Aif%20__name__%20%3D%3D%20%22__main__%22%3A%0A%20%20%20%20app.run()%0A</marimo-code></head>
|
80 |
+
<body>
|
81 |
+
<div id="root"></div>
|
82 |
+
</body>
|
83 |
+
</html>
|
_site/optimization/07_sdp.html
ADDED
@@ -0,0 +1,83 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<!DOCTYPE html>
|
2 |
+
<html lang="en">
|
3 |
+
<head>
|
4 |
+
<meta charset="utf-8" />
|
5 |
+
<link rel="icon" href="./favicon.ico" />
|
6 |
+
<!-- Preload is necessary because we show these images when we disconnect from the server,
|
7 |
+
but at that point we cannot load these images from the server -->
|
8 |
+
<link rel="preload" href="./assets/gradient-yHQUC_QB.png" as="image" />
|
9 |
+
<link rel="preload" href="./assets/noise-60BoTA8O.png" as="image" />
|
10 |
+
<!-- Preload the fonts -->
|
11 |
+
<link rel="preload" href="./assets/Lora-VariableFont_wght-B2ootaw-.ttf" as="font" crossorigin="anonymous" />
|
12 |
+
<link rel="preload" href="./assets/PTSans-Regular-CxL0S8W7.ttf" as="font" crossorigin="anonymous" />
|
13 |
+
<link rel="preload" href="./assets/PTSans-Bold-D9fedIX3.ttf" as="font" crossorigin="anonymous" />
|
14 |
+
<link rel="preload" href="./assets/FiraMono-Regular-BTCkDNvf.ttf" as="font" crossorigin="anonymous" />
|
15 |
+
<link rel="preload" href="./assets/FiraMono-Medium-DU3aDxX5.ttf" as="font" crossorigin="anonymous" />
|
16 |
+
<link rel="preload" href="./assets/FiraMono-Bold-CLVRCuM9.ttf" as="font" crossorigin="anonymous" />
|
17 |
+
|
18 |
+
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
19 |
+
<meta name="theme-color" content="#000000" />
|
20 |
+
<meta name="description" content="a marimo app" />
|
21 |
+
<link rel="apple-touch-icon" href="./apple-touch-icon.png" />
|
22 |
+
<link rel="manifest" href="./manifest.json" />
|
23 |
+
|
24 |
+
<script data-marimo="true">
|
25 |
+
function __resizeIframe(obj) {
|
26 |
+
var scrollbarHeight = 20; // Max between windows, mac, and linux
|
27 |
+
|
28 |
+
function setHeight() {
|
29 |
+
var element = obj.contentWindow.document.documentElement;
|
30 |
+
// If there is no vertical scrollbar, we don't need to resize the iframe
|
31 |
+
if (element.scrollHeight === element.clientHeight) {
|
32 |
+
return;
|
33 |
+
}
|
34 |
+
|
35 |
+
// Create a new height that includes the scrollbar height if it's visible
|
36 |
+
var hasHorizontalScrollbar = element.scrollWidth > element.clientWidth;
|
37 |
+
var newHeight = element.scrollHeight + (hasHorizontalScrollbar ? scrollbarHeight : 0);
|
38 |
+
|
39 |
+
// Only update the height if it's different from the current height
|
40 |
+
if (obj.style.height !== `${newHeight}px`) {
|
41 |
+
obj.style.height = `${newHeight}px`;
|
42 |
+
}
|
43 |
+
}
|
44 |
+
|
45 |
+
// Resize the iframe to the height of the content and bottom scrollbar height
|
46 |
+
setHeight();
|
47 |
+
|
48 |
+
// Resize the iframe when the content changes
|
49 |
+
const resizeObserver = new ResizeObserver((entries) => {
|
50 |
+
setHeight();
|
51 |
+
});
|
52 |
+
resizeObserver.observe(obj.contentWindow.document.body);
|
53 |
+
}
|
54 |
+
</script>
|
55 |
+
<marimo-filename hidden>notebook.py</marimo-filename>
|
56 |
+
<marimo-mode data-mode='edit' hidden></marimo-mode>
|
57 |
+
<marimo-version data-version='0.11.9' hidden></marimo-version>
|
58 |
+
<marimo-user-config data-config='{"completion": {"activate_on_typing": true, "copilot": false}, "display": {"theme": "light", "code_editor_font_size": 14, "dataframes": "rich", "cell_output": "above", "default_width": "medium"}, "formatting": {"line_length": 79}, "keymap": {"preset": "default", "overrides": {}}, "runtime": {"auto_instantiate": true, "auto_reload": "off", "on_cell_change": "autorun", "watcher_on_save": "lazy", "output_max_bytes": 8000000, "std_stream_max_bytes": 1000000}, "save": {"autosave": "off", "autosave_delay": 1000, "format_on_save": false}, "package_management": {"manager": "pip"}, "server": {"browser": "default", "follow_symlink": false}, "snippets": {"custom_paths": [], "include_default_snippets": true}}' data-overrides='{}' hidden></marimo-user-config>
|
59 |
+
<marimo-app-config data-config='{"width": "compact"}' hidden></marimo-app-config>
|
60 |
+
<marimo-server-token data-token='123' hidden></marimo-server-token>
|
61 |
+
<title>07 sdp</title>
|
62 |
+
<script type="module" crossorigin src="./assets/index-BiV-b1K2.js"></script>
|
63 |
+
<link rel="stylesheet" crossorigin href="./assets/index-DkqMrX_B.css">
|
64 |
+
<marimo-wasm hidden=""></marimo-wasm>
|
65 |
+
<script>
|
66 |
+
if (window.location.protocol === 'file:') {
|
67 |
+
alert('Warning: This file must be served by an HTTP server to function correctly.');
|
68 |
+
}
|
69 |
+
</script>
|
70 |
+
|
71 |
+
<style>
|
72 |
+
#save-button {
|
73 |
+
display: none !important;
|
74 |
+
}
|
75 |
+
#filename-input {
|
76 |
+
display: none !important;
|
77 |
+
}
|
78 |
+
</style>
|
79 |
+
<marimo-code hidden="" data-show-code="false">import%20marimo%0A%0A__generated_with%20%3D%20%220.11.9%22%0Aapp%20%3D%20marimo.App()%0A%0A%0A%40app.cell%0Adef%20_()%3A%0A%20%20%20%20import%20marimo%20as%20mo%0A%20%20%20%20return%20(mo%2C)%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(r%22%22%22%23%20Semidefinite%20program%22%22%22)%0A%20%20%20%20return%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20r%22%22%22%0A%20%20%20%20%20%20%20%20_This%20notebook%20introduces%20an%20advanced%20topic._%20A%20semidefinite%20program%20(SDP)%20is%20an%20optimization%20problem%20of%20the%20form%0A%0A%20%20%20%20%20%20%20%20%5C%5B%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Cbegin%7Barray%7D%7Bll%7D%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Ctext%7Bminimize%7D%20%20%20%26%20%5Cmathbf%7Btr%7D(CX)%20%5C%5C%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Ctext%7Bsubject%20to%7D%20%26%20%5Cmathbf%7Btr%7D(A_iX)%20%3D%20b_i%2C%20%5Cquad%20i%3D1%2C%5Cldots%2Cp%20%5C%5C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%26%20X%20%5Csucceq%200%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Cend%7Barray%7D%0A%20%20%20%20%20%20%20%20%5C%5D%0A%0A%20%20%20%20%20%20%20%20where%20%24%5Cmathbf%7Btr%7D%24%20is%20the%20trace%20function%2C%20%24X%20%5Cin%20%5Cmathcal%7BS%7D%5E%7Bn%7D%24%20is%20the%20optimization%20variable%20and%20%24C%2C%20A_1%2C%20%5Cldots%2C%20A_p%20%5Cin%20%5Cmathcal%7BS%7D%5E%7Bn%7D%24%2C%20and%20%24b_1%2C%20%5Cldots%2C%20b_p%20%5Cin%20%5Cmathcal%7BR%7D%24%20are%20problem%20data%2C%20and%20%24X%20%5Csucceq%200%24%20is%20a%20matrix%20inequality.%20Here%20%24%5Cmathcal%7BS%7D%5E%7Bn%7D%24%20denotes%20the%20set%20of%20%24n%24-by-%24n%24%20symmetric%20matrices.%0A%0A%20%20%20%20%20%20%20%20**Example.**%20An%20example%20of%20an%20SDP%20is%20to%20complete%20a%20covariance%20matrix%20%24%5Ctilde%20%5CSigma%20%5Cin%20%5Cmathcal%7BS%7D%5E%7Bn%7D_%2B%24%20with%20missing%20entries%20%24M%20%5Csubset%20%5C%7B1%2C%5Cldots%2Cn%5C%7D%20%5Ctimes%20%5C%7B1%2C%5Cldots%2Cn%5C%7D%24%3A%0A%0A%20%20%20%20%20%20%20%20%5C%5B%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Cbegin%7Barray%7D%7Bll%7D%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Ctext%7Bminimize%7D%20%20%20%26%200%20%5C%5C%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Ctext%7Bsubject%20to%7D%20%26%20%5CSigma_%7Bij%7D%20%3D%20%5Ctilde%20%5CSigma_%7Bij%7D%2C%20%5Cquad%20(i%2Cj)%20%5Cnotin%20M%20%5C%5C%0A%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%26%20%5CSigma%20%5Csucceq%200%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%5Cend%7Barray%7D%0A%20%20%20%20%20%20%20%20%5C%5D%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell(hide_code%3DTrue)%0Adef%20_(mo)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20r%22%22%22%0A%20%20%20%20%20%20%20%20%23%23%20Example%0A%0A%20%20%20%20%20%20%20%20In%20the%20following%20code%2C%20we%20show%20how%20to%20specify%20and%20solve%20an%20SDP%20with%20CVXPY.%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell%0Adef%20_()%3A%0A%20%20%20%20import%20cvxpy%20as%20cp%0A%20%20%20%20import%20numpy%20as%20np%0A%20%20%20%20return%20cp%2C%20np%0A%0A%0A%40app.cell%0Adef%20_(np)%3A%0A%20%20%20%20%23%20Generate%20a%20random%20SDP.%0A%20%20%20%20n%20%3D%203%0A%20%20%20%20p%20%3D%203%0A%20%20%20%20np.random.seed(1)%0A%20%20%20%20C%20%3D%20np.random.randn(n%2C%20n)%0A%20%20%20%20A%20%3D%20%5B%5D%0A%20%20%20%20b%20%3D%20%5B%5D%0A%20%20%20%20for%20i%20in%20range(p)%3A%0A%20%20%20%20%20%20%20%20A.append(np.random.randn(n%2C%20n))%0A%20%20%20%20%20%20%20%20b.append(np.random.randn())%0A%20%20%20%20return%20A%2C%20C%2C%20b%2C%20i%2C%20n%2C%20p%0A%0A%0A%40app.cell%0Adef%20_(A%2C%20C%2C%20b%2C%20cp%2C%20n%2C%20p)%3A%0A%20%20%20%20%23%20Create%20a%20symmetric%20matrix%20variable.%0A%20%20%20%20X%20%3D%20cp.Variable((n%2C%20n)%2C%20symmetric%3DTrue)%0A%0A%20%20%20%20%23%20The%20operator%20%3E%3E%20denotes%20matrix%20inequality%2C%20with%20X%20%3E%3E%200%20constraining%20X%0A%20%20%20%20%23%20to%20be%20positive%20semidefinite%0A%20%20%20%20constraints%20%3D%20%5BX%20%3E%3E%200%5D%0A%20%20%20%20constraints%20%2B%3D%20%5Bcp.trace(A%5Bi%5D%20%40%20X)%20%3D%3D%20b%5Bi%5D%20for%20i%20in%20range(p)%5D%0A%20%20%20%20prob%20%3D%20cp.Problem(cp.Minimize(cp.trace(C%20%40%20X))%2C%20constraints)%0A%20%20%20%20_%20%3D%20prob.solve()%0A%20%20%20%20return%20X%2C%20constraints%2C%20prob%0A%0A%0A%40app.cell%0Adef%20_(X%2C%20mo%2C%20prob%2C%20wigglystuff)%3A%0A%20%20%20%20mo.md(%0A%20%20%20%20%20%20%20%20f%22%22%22%0A%20%20%20%20%20%20%20%20The%20optimal%20value%20is%20%7Bprob.value%3A0.4f%7D.%0A%0A%20%20%20%20%20%20%20%20A%20solution%20for%20%24X%24%20is%20(rounded%20to%20the%20nearest%20decimal)%20is%3A%20%0A%0A%20%20%20%20%20%20%20%20%7Bmo.ui.anywidget(wigglystuff.Matrix(X.value)).center()%7D%0A%20%20%20%20%20%20%20%20%22%22%22%0A%20%20%20%20)%0A%20%20%20%20return%0A%0A%0A%40app.cell%0Adef%20_()%3A%0A%20%20%20%20import%20wigglystuff%0A%20%20%20%20return%20(wigglystuff%2C)%0A%0A%0Aif%20__name__%20%3D%3D%20%22__main__%22%3A%0A%20%20%20%20app.run()%0A</marimo-code></head>
|
80 |
+
<body>
|
81 |
+
<div id="root"></div>
|
82 |
+
</body>
|
83 |
+
</html>
|
_site/optimization/android-chrome-192x192.png
ADDED
![]() |
_site/optimization/android-chrome-512x512.png
ADDED
![]() |
_site/optimization/apple-touch-icon.png
ADDED
![]() |
_site/optimization/assets/ConnectedDataExplorerComponent-D39SA74B.js
ADDED
The diff for this file is too large to render.
See raw diff
|
|
_site/optimization/assets/FiraMono-Bold-CLVRCuM9.ttf
ADDED
Binary file (202 kB). View file
|
|
_site/optimization/assets/FiraMono-Medium-DU3aDxX5.ttf
ADDED
Binary file (169 kB). View file
|
|
_site/optimization/assets/FiraMono-Regular-BTCkDNvf.ttf
ADDED
Binary file (170 kB). View file
|
|
_site/optimization/assets/KaTeX_AMS-Regular-BQhdFMY1.woff2
ADDED
Binary file (28.1 kB). View file
|
|
_site/optimization/assets/KaTeX_AMS-Regular-DMm9YOAa.woff
ADDED
Binary file (33.5 kB). View file
|
|
_site/optimization/assets/KaTeX_AMS-Regular-DRggAlZN.ttf
ADDED
Binary file (63.6 kB). View file
|
|
_site/optimization/assets/KaTeX_Caligraphic-Bold-ATXxdsX0.ttf
ADDED
Binary file (12.4 kB). View file
|
|
_site/optimization/assets/KaTeX_Caligraphic-Bold-BEiXGLvX.woff
ADDED
Binary file (7.72 kB). View file
|
|
_site/optimization/assets/KaTeX_Caligraphic-Bold-Dq_IR9rO.woff2
ADDED
Binary file (6.91 kB). View file
|
|
_site/optimization/assets/KaTeX_Caligraphic-Regular-CTRA-rTL.woff
ADDED
Binary file (7.66 kB). View file
|
|
_site/optimization/assets/KaTeX_Caligraphic-Regular-Di6jR-x-.woff2
ADDED
Binary file (6.91 kB). View file
|
|
_site/optimization/assets/KaTeX_Caligraphic-Regular-wX97UBjC.ttf
ADDED
Binary file (12.3 kB). View file
|
|
_site/optimization/assets/KaTeX_Fraktur-Bold-BdnERNNW.ttf
ADDED
Binary file (19.6 kB). View file
|
|
_site/optimization/assets/KaTeX_Fraktur-Bold-BsDP51OF.woff
ADDED
Binary file (13.3 kB). View file
|
|
_site/optimization/assets/KaTeX_Fraktur-Bold-CL6g_b3V.woff2
ADDED
Binary file (11.3 kB). View file
|
|
_site/optimization/assets/KaTeX_Fraktur-Regular-CB_wures.ttf
ADDED
Binary file (19.6 kB). View file
|
|
_site/optimization/assets/KaTeX_Fraktur-Regular-CTYiF6lA.woff2
ADDED
Binary file (11.3 kB). View file
|
|
_site/optimization/assets/KaTeX_Fraktur-Regular-Dxdc4cR9.woff
ADDED
Binary file (13.2 kB). View file
|
|
_site/optimization/assets/KaTeX_Main-Bold-Cx986IdX.woff2
ADDED
Binary file (25.3 kB). View file
|
|
_site/optimization/assets/KaTeX_Main-Bold-Jm3AIy58.woff
ADDED
Binary file (29.9 kB). View file
|
|
_site/optimization/assets/KaTeX_Main-Bold-waoOVXN0.ttf
ADDED
Binary file (51.3 kB). View file
|
|
_site/optimization/assets/KaTeX_Main-BoldItalic-DxDJ3AOS.woff2
ADDED
Binary file (16.8 kB). View file
|
|
_site/optimization/assets/KaTeX_Main-BoldItalic-DzxPMmG6.ttf
ADDED
Binary file (33 kB). View file
|
|
_site/optimization/assets/KaTeX_Main-BoldItalic-SpSLRI95.woff
ADDED
Binary file (19.4 kB). View file
|
|
_site/optimization/assets/KaTeX_Main-Italic-3WenGoN9.ttf
ADDED
Binary file (33.6 kB). View file
|
|
_site/optimization/assets/KaTeX_Main-Italic-BMLOBm91.woff
ADDED
Binary file (19.7 kB). View file
|
|
_site/optimization/assets/KaTeX_Main-Italic-NWA7e6Wa.woff2
ADDED
Binary file (17 kB). View file
|
|
_site/optimization/assets/KaTeX_Main-Regular-B22Nviop.woff2
ADDED
Binary file (26.3 kB). View file
|
|
_site/optimization/assets/KaTeX_Main-Regular-Dr94JaBh.woff
ADDED
Binary file (30.8 kB). View file
|
|
_site/optimization/assets/KaTeX_Main-Regular-ypZvNtVU.ttf
ADDED
Binary file (53.6 kB). View file
|
|
_site/optimization/assets/KaTeX_Math-BoldItalic-B3XSjfu4.ttf
ADDED
Binary file (31.2 kB). View file
|
|
_site/optimization/assets/KaTeX_Math-BoldItalic-CZnvNsCZ.woff2
ADDED
Binary file (16.4 kB). View file
|
|
_site/optimization/assets/KaTeX_Math-BoldItalic-iY-2wyZ7.woff
ADDED
Binary file (18.7 kB). View file
|
|
_site/optimization/assets/KaTeX_Math-Italic-DA0__PXp.woff
ADDED
Binary file (18.7 kB). View file
|
|
_site/optimization/assets/KaTeX_Math-Italic-flOr_0UB.ttf
ADDED
Binary file (31.3 kB). View file
|
|