"use strict";(self.webpackChunkfi=self.webpackChunkfi||[]).push([[3184],{3905:(e,t,a)=>{a.d(t,{Zo:()=>h,kt:()=>k});var n=a(7294);function r(e,t,a){return t in e?Object.defineProperty(e,t,{value:a,enumerable:!0,configurable:!0,writable:!0}):e[t]=a,e}function s(e,t){var a=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),a.push.apply(a,n)}return a}function i(e){for(var t=1;t=0||(r[a]=e[a]);return r}(e,t);if(Object.getOwnPropertySymbols){var s=Object.getOwnPropertySymbols(e);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(e,a)&&(r[a]=e[a])}return r}var o=n.createContext({}),m=function(e){var t=n.useContext(o),a=t;return e&&(a="function"==typeof e?e(t):i(i({},t),e)),a},h=function(e){var t=m(e.components);return n.createElement(o.Provider,{value:t},e.children)},c="mdxType",d={inlineCode:"code",wrapper:function(e){var t=e.children;return n.createElement(n.Fragment,{},t)}},p=n.forwardRef((function(e,t){var a=e.components,r=e.mdxType,s=e.originalType,o=e.parentName,h=l(e,["components","mdxType","originalType","parentName"]),c=m(a),p=r,k=c["".concat(o,".").concat(p)]||c[p]||d[p]||s;return a?n.createElement(k,i(i({ref:t},h),{},{components:a})):n.createElement(k,i({ref:t},h))}));function k(e,t){var a=arguments,r=t&&t.mdxType;if("string"==typeof e||r){var s=a.length,i=new Array(s);i[0]=p;var l={};for(var o in t)hasOwnProperty.call(t,o)&&(l[o]=t[o]);l.originalType=e,l[c]="string"==typeof e?e:r,i[1]=l;for(var m=2;m{a.r(t),a.d(t,{assets:()=>V,contentTitle:()=>x,default:()=>W,frontMatter:()=>O,metadata:()=>U,toc:()=>M});var n=a(7462),r=a(7294),s=a(3905),i=a(6010),l=a(2466),o=a(6550),m=a(1980),h=a(7392),c=a(12);function d(e){return function(e){return r.Children.map(e,(e=>{if(!e||(0,r.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}(e).map((e=>{let{props:{value:t,label:a,attributes:n,default:r}}=e;return{value:t,label:a,attributes:n,default:r}}))}function p(e){const{values:t,children:a}=e;return(0,r.useMemo)((()=>{const e=t??d(a);return function(e){const t=(0,h.l)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,a])}function k(e){let{value:t,tabValues:a}=e;return a.some((e=>e.value===t))}function g(e){let{queryString:t=!1,groupId:a}=e;const n=(0,o.k6)(),s=function(e){let{queryString:t=!1,groupId:a}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!a)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return a??null}({queryString:t,groupId:a});return[(0,m._X)(s),(0,r.useCallback)((e=>{if(!s)return;const t=new URLSearchParams(n.location.search);t.set(s,e),n.replace({...n.location,search:t.toString()})}),[s,n])]}function u(e){const{defaultValue:t,queryString:a=!1,groupId:n}=e,s=p(e),[i,l]=(0,r.useState)((()=>function(e){let{defaultValue:t,tabValues:a}=e;if(0===a.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!k({value:t,tabValues:a}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${a.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const n=a.find((e=>e.default))??a[0];if(!n)throw new Error("Unexpected error: 0 tabValues");return n.value}({defaultValue:t,tabValues:s}))),[o,m]=g({queryString:a,groupId:n}),[h,d]=function(e){let{groupId:t}=e;const a=function(e){return e?`docusaurus.tab.${e}`:null}(t),[n,s]=(0,c.Nk)(a);return[n,(0,r.useCallback)((e=>{a&&s.set(e)}),[a,s])]}({groupId:n}),u=(()=>{const e=o??h;return k({value:e,tabValues:s})?e:null})();(0,r.useLayoutEffect)((()=>{u&&l(u)}),[u]);return{selectedValue:i,selectValue:(0,r.useCallback)((e=>{if(!k({value:e,tabValues:s}))throw new Error(`Can't select invalid tab value=${e}`);l(e),m(e),d(e)}),[m,d,s]),tabValues:s}}var b=a(2389);const N={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};function y(e){let{className:t,block:a,selectedValue:s,selectValue:o,tabValues:m}=e;const h=[],{blockElementScrollPositionUntilNextRender:c}=(0,l.o5)(),d=e=>{const t=e.currentTarget,a=h.indexOf(t),n=m[a].value;n!==s&&(c(t),o(n))},p=e=>{let t=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const a=h.indexOf(e.currentTarget)+1;t=h[a]??h[0];break}case"ArrowLeft":{const a=h.indexOf(e.currentTarget)-1;t=h[a]??h[h.length-1];break}}t?.focus()};return r.createElement("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,i.Z)("tabs",{"tabs--block":a},t)},m.map((e=>{let{value:t,label:a,attributes:l}=e;return r.createElement("li",(0,n.Z)({role:"tab",tabIndex:s===t?0:-1,"aria-selected":s===t,key:t,ref:e=>h.push(e),onKeyDown:p,onClick:d},l,{className:(0,i.Z)("tabs__item",N.tabItem,l?.className,{"tabs__item--active":s===t})}),a??t)})))}function f(e){let{lazy:t,children:a,selectedValue:n}=e;const s=(Array.isArray(a)?a:[a]).filter(Boolean);if(t){const e=s.find((e=>e.props.value===n));return e?(0,r.cloneElement)(e,{className:"margin-top--md"}):null}return r.createElement("div",{className:"margin-top--md"},s.map(((e,t)=>(0,r.cloneElement)(e,{key:t,hidden:e.props.value!==n}))))}function R(e){const t=u(e);return r.createElement("div",{className:(0,i.Z)("tabs-container",N.tabList)},r.createElement(y,(0,n.Z)({},e,t)),r.createElement(f,(0,n.Z)({},e,t)))}function w(e){const t=(0,b.Z)();return r.createElement(R,(0,n.Z)({key:String(t)},e))}const Z={tabItem:"tabItem_Ymn6"};function v(e){let{children:t,hidden:a,className:n}=e;return r.createElement("div",{role:"tabpanel",className:(0,i.Z)(Z.tabItem,n),hidden:a},t)}const O={id:"rules",title:"On the rules of the red-black tree",description:"Shower thoughts on the rules of the red-black tree.\n",tags:["red-black trees","balanced trees"],last_update:{date:new Date("2023-06-10T00:00:00.000Z")}},x=void 0,U={unversionedId:"rb-trees/rules",id:"rb-trees/rules",title:"On the rules of the red-black tree",description:"Shower thoughts on the rules of the red-black tree.\n",source:"@site/ib002/08-rb-trees/2023-06-10-rules.md",sourceDirName:"08-rb-trees",slug:"/rb-trees/rules",permalink:"/ib002/rb-trees/rules",draft:!1,editUrl:"https://gitlab.com/mfocko/blog/tree/main/ib002/08-rb-trees/2023-06-10-rules.md",tags:[{label:"red-black trees",permalink:"/ib002/tags/red-black-trees"},{label:"balanced trees",permalink:"/ib002/tags/balanced-trees"}],version:"current",lastUpdatedAt:1686355200,formattedLastUpdatedAt:"Jun 10, 2023",frontMatter:{id:"rules",title:"On the rules of the red-black tree",description:"Shower thoughts on the rules of the red-black tree.\n",tags:["red-black trees","balanced trees"],last_update:{date:"2023-06-10T00:00:00.000Z"}},sidebar:"autogeneratedBar",previous:{title:"Pou\u017eit\xed \u010derveno-\u010dern\xfdch strom\u016f",permalink:"/ib002/rb-trees/applications"},next:{title:"Graphs",permalink:"/ib002/category/graphs"}},V={},M=[{value:"Introduction",id:"introduction",level:2},{value:"1\xaa Every node is either red or black.",id:"1\xaa-every-node-is-either-red-or-black",level:2},{value:"Do I really need the nodes to be explicitly colored?",id:"do-i-really-need-the-nodes-to-be-explicitly-colored",level:3},{value:"Black height",id:"black-height",level:4},{value:"Isomorphic trees",id:"isomorphic-trees",level:4},{value:"2\xaa The root is black.",id:"2\xaa-the-root-is-black",level:2},{value:"3\xaa Every leaf (nil) is black.",id:"3\xaa-every-leaf-nil-is-black",level:2},{value:"4\xaa If a node is red, then both its children are black.",id:"4\xaa-if-a-node-is-red-then-both-its-children-are-black",level:2},{value:"5\xaa For each node, all simple paths from the node to descendant leaves contain the same number of black nodes.",id:"5\xaa-for-each-node-all-simple-paths-from-the-node-to-descendant-leaves-contain-the-same-number-of-black-nodes",level:2}],j={toc:M},E="wrapper";function W(e){let{components:t,...r}=e;return(0,s.kt)(E,(0,n.Z)({},j,r,{components:t,mdxType:"MDXLayout"}),(0,s.kt)("h2",{id:"introduction"},"Introduction"),(0,s.kt)("p",null,"Have you ever thought about the red-black tree rules in more depth? Why are they\nformulated the way they are? How come they keep the tree balanced? Let's go through\neach of the red-black tree rules and try to change, break and contemplate about\nthem."),(0,s.kt)("p",null,"We expect that you are familiar with the following set of the rules",(0,s.kt)("sup",{parentName:"p",id:"fnref-1"},(0,s.kt)("a",{parentName:"sup",href:"#fn-1",className:"footnote-ref"},"1")),":"),(0,s.kt)("ol",null,(0,s.kt)("li",{parentName:"ol"},"Every node is either red or black."),(0,s.kt)("li",{parentName:"ol"},"The root is black."),(0,s.kt)("li",{parentName:"ol"},"Every leaf (",(0,s.kt)("inlineCode",{parentName:"li"},"nil"),") is black."),(0,s.kt)("li",{parentName:"ol"},"If a node is red, then both its children are black."),(0,s.kt)("li",{parentName:"ol"},"For each node, all simple paths from the node to descendant leaves contain the\nsame number of black nodes.")),(0,s.kt)("p",null,"Each section will go into ",(0,s.kt)("em",{parentName:"p"},"reasonable")," details of each rule."),(0,s.kt)("h2",{id:"1\xaa-every-node-is-either-red-or-black"},"1\xaa Every node is either red or black."),(0,s.kt)("p",null,"OK\u2026 This one is very simple. It is just a definition and is used in all other\nrules. Not much to talk about here. Or is there?"),(0,s.kt)("h3",{id:"do-i-really-need-the-nodes-to-be-explicitly-colored"},"Do I really need the nodes to be explicitly colored?"),(0,s.kt)("p",null,"The answer is no. Balancing of the red-black trees is \u201cenforced\u201d by the 4th and\n5th rule in the enumeration above. There are many ways you can avoid using colors."),(0,s.kt)("h4",{id:"black-height"},"Black height"),(0,s.kt)("p",null,"We mentioned the 4th and 5th rule and that it enforces the balancing. What does\nit mean for us?"),(0,s.kt)("p",null,"Well, we definitely do not have to use the colors, which even as a ",(0,s.kt)("em",{parentName:"p"},"boolean")," flag\nwould take at least 1 byte of space (and usually even more), cause\u2026 well, it is\neasier for the CPU to work with words rather than single bits."),(0,s.kt)("p",null,"We could use the black height, couldn't we? It would mean more memory used, cause\nit should be ideally big and unsigned. Can we tell the color of a node from the\nblack height? Of course we can, if my child has the same black height as I do,\nit means that there was no black node added on the path between us and therefore\nmy child would be colored red."),(0,s.kt)("p",null,"Example of a red-black tree that keeps count of black nodes on paths to the\nleaves follows:"),(0,s.kt)("p",null,(0,s.kt)("img",{alt:"Red-black tree with black height",src:a(5021).Z+"#gh-light-mode-only",width:"923",height:"539"}),"\n",(0,s.kt)("img",{alt:"Red-black tree with black height",src:a(9493).Z+"#gh-dark-mode-only",width:"923",height:"539"})),(0,s.kt)("p",null,"We mark the ",(0,s.kt)("em",{parentName:"p"},"black heights")," in superscript. You can see that all leaves have the\nblack height equal to ",(0,s.kt)("span",{parentName:"p",className:"math math-inline"},(0,s.kt)("span",{parentName:"span",className:"katex"},(0,s.kt)("span",{parentName:"span",className:"katex-mathml"},(0,s.kt)("math",{parentName:"span",xmlns:"http://www.w3.org/1998/Math/MathML"},(0,s.kt)("semantics",{parentName:"math"},(0,s.kt)("mrow",{parentName:"semantics"},(0,s.kt)("mn",{parentName:"mrow"},"1")),(0,s.kt)("annotation",{parentName:"semantics",encoding:"application/x-tex"},"1")))),(0,s.kt)("span",{parentName:"span",className:"katex-html","aria-hidden":"true"},(0,s.kt)("span",{parentName:"span",className:"base"},(0,s.kt)("span",{parentName:"span",className:"strut",style:{height:"0.6444em"}}),(0,s.kt)("span",{parentName:"span",className:"mord"},"1"))))),". Let's take a look at some of the interesting cases:"),(0,s.kt)("ul",null,(0,s.kt)("li",{parentName:"ul"},(0,s.kt)("p",{parentName:"li"},"If we take a look at the node with ",(0,s.kt)("span",{parentName:"p",className:"math math-inline"},(0,s.kt)("span",{parentName:"span",className:"katex"},(0,s.kt)("span",{parentName:"span",className:"katex-mathml"},(0,s.kt)("math",{parentName:"span",xmlns:"http://www.w3.org/1998/Math/MathML"},(0,s.kt)("semantics",{parentName:"math"},(0,s.kt)("mrow",{parentName:"semantics"},(0,s.kt)("mtext",{parentName:"mrow"},"key"),(0,s.kt)("mo",{parentName:"mrow"},"="),(0,s.kt)("mn",{parentName:"mrow"},"9")),(0,s.kt)("annotation",{parentName:"semantics",encoding:"application/x-tex"},"\\text{key} = 9")))),(0,s.kt)("span",{parentName:"span",className:"katex-html","aria-hidden":"true"},(0,s.kt)("span",{parentName:"span",className:"base"},(0,s.kt)("span",{parentName:"span",className:"strut",style:{height:"0.8889em",verticalAlign:"-0.1944em"}}),(0,s.kt)("span",{parentName:"span",className:"mord text"},(0,s.kt)("span",{parentName:"span",className:"mord"},"key")),(0,s.kt)("span",{parentName:"span",className:"mspace",style:{marginRight:"0.2778em"}}),(0,s.kt)("span",{parentName:"span",className:"mrel"},"="),(0,s.kt)("span",{parentName:"span",className:"mspace",style:{marginRight:"0.2778em"}})),(0,s.kt)("span",{parentName:"span",className:"base"},(0,s.kt)("span",{parentName:"span",className:"strut",style:{height:"0.6444em"}}),(0,s.kt)("span",{parentName:"span",className:"mord"},"9"))))),", we can see that it is\ncoloured red and its black height is 1, because it is a leaf."),(0,s.kt)("p",{parentName:"li"},"Let's look at its parent (node with ",(0,s.kt)("span",{parentName:"p",className:"math math-inline"},(0,s.kt)("span",{parentName:"span",className:"katex"},(0,s.kt)("span",{parentName:"span",className:"katex-mathml"},(0,s.kt)("math",{parentName:"span",xmlns:"http://www.w3.org/1998/Math/MathML"},(0,s.kt)("semantics",{parentName:"math"},(0,s.kt)("mrow",{parentName:"semantics"},(0,s.kt)("mtext",{parentName:"mrow"},"key"),(0,s.kt)("mo",{parentName:"mrow"},"="),(0,s.kt)("mn",{parentName:"mrow"},"8")),(0,s.kt)("annotation",{parentName:"semantics",encoding:"application/x-tex"},"\\text{key} = 8")))),(0,s.kt)("span",{parentName:"span",className:"katex-html","aria-hidden":"true"},(0,s.kt)("span",{parentName:"span",className:"base"},(0,s.kt)("span",{parentName:"span",className:"strut",style:{height:"0.8889em",verticalAlign:"-0.1944em"}}),(0,s.kt)("span",{parentName:"span",className:"mord text"},(0,s.kt)("span",{parentName:"span",className:"mord"},"key")),(0,s.kt)("span",{parentName:"span",className:"mspace",style:{marginRight:"0.2778em"}}),(0,s.kt)("span",{parentName:"span",className:"mrel"},"="),(0,s.kt)("span",{parentName:"span",className:"mspace",style:{marginRight:"0.2778em"}})),(0,s.kt)("span",{parentName:"span",className:"base"},(0,s.kt)("span",{parentName:"span",className:"strut",style:{height:"0.6444em"}}),(0,s.kt)("span",{parentName:"span",className:"mord"},"8"))))),"). On its left side it has\n",(0,s.kt)("inlineCode",{parentName:"p"},"nil")," and on its right side the ",(0,s.kt)("span",{parentName:"p",className:"math math-inline"},(0,s.kt)("span",{parentName:"span",className:"katex"},(0,s.kt)("span",{parentName:"span",className:"katex-mathml"},(0,s.kt)("math",{parentName:"span",xmlns:"http://www.w3.org/1998/Math/MathML"},(0,s.kt)("semantics",{parentName:"math"},(0,s.kt)("mrow",{parentName:"semantics"},(0,s.kt)("mn",{parentName:"mrow"},"9")),(0,s.kt)("annotation",{parentName:"semantics",encoding:"application/x-tex"},"9")))),(0,s.kt)("span",{parentName:"span",className:"katex-html","aria-hidden":"true"},(0,s.kt)("span",{parentName:"span",className:"base"},(0,s.kt)("span",{parentName:"span",className:"strut",style:{height:"0.6444em"}}),(0,s.kt)("span",{parentName:"span",className:"mord"},"9"))))),". And its black height is still ",(0,s.kt)("span",{parentName:"p",className:"math math-inline"},(0,s.kt)("span",{parentName:"span",className:"katex"},(0,s.kt)("span",{parentName:"span",className:"katex-mathml"},(0,s.kt)("math",{parentName:"span",xmlns:"http://www.w3.org/1998/Math/MathML"},(0,s.kt)("semantics",{parentName:"math"},(0,s.kt)("mrow",{parentName:"semantics"},(0,s.kt)("mn",{parentName:"mrow"},"1")),(0,s.kt)("annotation",{parentName:"semantics",encoding:"application/x-tex"},"1")))),(0,s.kt)("span",{parentName:"span",className:"katex-html","aria-hidden":"true"},(0,s.kt)("span",{parentName:"span",className:"base"},(0,s.kt)("span",{parentName:"span",className:"strut",style:{height:"0.6444em"}}),(0,s.kt)("span",{parentName:"span",className:"mord"},"1"))))),", cause\nexcept for the ",(0,s.kt)("inlineCode",{parentName:"p"},"nil")," leaves, there are no other black nodes."),(0,s.kt)("p",{parentName:"li"},"We can clearly see that if a node has the same black height as its parent, it\nis a red node.")),(0,s.kt)("li",{parentName:"ul"},(0,s.kt)("p",{parentName:"li"},"Now let's take a look at the root with ",(0,s.kt)("span",{parentName:"p",className:"math math-inline"},(0,s.kt)("span",{parentName:"span",className:"katex"},(0,s.kt)("span",{parentName:"span",className:"katex-mathml"},(0,s.kt)("math",{parentName:"span",xmlns:"http://www.w3.org/1998/Math/MathML"},(0,s.kt)("semantics",{parentName:"math"},(0,s.kt)("mrow",{parentName:"semantics"},(0,s.kt)("mtext",{parentName:"mrow"},"key"),(0,s.kt)("mo",{parentName:"mrow"},"="),(0,s.kt)("mn",{parentName:"mrow"},"3")),(0,s.kt)("annotation",{parentName:"semantics",encoding:"application/x-tex"},"\\text{key} = 3")))),(0,s.kt)("span",{parentName:"span",className:"katex-html","aria-hidden":"true"},(0,s.kt)("span",{parentName:"span",className:"base"},(0,s.kt)("span",{parentName:"span",className:"strut",style:{height:"0.8889em",verticalAlign:"-0.1944em"}}),(0,s.kt)("span",{parentName:"span",className:"mord text"},(0,s.kt)("span",{parentName:"span",className:"mord"},"key")),(0,s.kt)("span",{parentName:"span",className:"mspace",style:{marginRight:"0.2778em"}}),(0,s.kt)("span",{parentName:"span",className:"mrel"},"="),(0,s.kt)("span",{parentName:"span",className:"mspace",style:{marginRight:"0.2778em"}})),(0,s.kt)("span",{parentName:"span",className:"base"},(0,s.kt)("span",{parentName:"span",className:"strut",style:{height:"0.6444em"}}),(0,s.kt)("span",{parentName:"span",className:"mord"},"3"))))),". It has a black height\nof 3. Both of its children are black nodes and have black height of 2."),(0,s.kt)("p",{parentName:"li"},"We can see that if a node has its height 1 lower than its parent, it is a black\nnode."),(0,s.kt)("p",{parentName:"li"},"The reasoning behind it is rather simple, we count the black nodes all the way\nto the leaves, therefore if my parent has a higher black height, it means that\non the path from me to my parent there is a black node, but the only node added\nis me, therefore I must be black."))),(0,s.kt)("h4",{id:"isomorphic-trees"},"Isomorphic trees"),(0,s.kt)("p",null,"One of the other ways to avoid using color is storing the red-black tree in some\nisomorphic tree. The structure of 2-3-4 tree allows us to avoid using the color\ncompletely. This is a bit different approach, cause we would be basically using\ndifferent tree, so we keep this note in just as a \u201chack\u201d."),(0,s.kt)("h2",{id:"2\xaa-the-root-is-black"},"2\xaa The root is black."),(0,s.kt)("p",null,"This rule might seem like a very important one, but overall is not. You can safely\nomit this rule, but you also need to deal with the consequences."),(0,s.kt)("p",null,"Let's refresh our memory with the algorithm of ",(0,s.kt)("em",{parentName:"p"},"insert fixup"),":"),(0,s.kt)("pre",null,(0,s.kt)("code",{parentName:"pre"},"WHILE z.p.color == Red\n IF z.p == z.p.p.left\n y = z.p.p.right\n\n IF y.color == Red\n z.p.color = Black\n y.color = Black\n z.p.p.color = Red\n z = z.p.p\n ELSE\n IF z == z.p.right\n z = z.p\n Left-Rotate(T, z)\n z.p.color = Black\n z.p.p.color = Red\n Right-Rotate(T, z.p.p)\n ELSE (same as above with \u201cright\u201d and \u201cleft\u201d exchanged)\n\nT.root.color = Black\n")),(0,s.kt)("admonition",{type:"tip"},(0,s.kt)("p",{parentName:"admonition"},"If you have tried to implement any of the more complex data structures, such as\nred-black trees, etc., in a statically typed language that also checks you for\n",(0,s.kt)("inlineCode",{parentName:"p"},"NULL"),"-correctness (e.g. ",(0,s.kt)("em",{parentName:"p"},"mypy")," or even C# with nullable reference types), you\nmight have run into numerous issues in the cases where you are 100% sure that you\ncannot obtain ",(0,s.kt)("inlineCode",{parentName:"p"},"NULL")," because of the invariants, but the static type checking\ndoesn't know that."),(0,s.kt)("p",{parentName:"admonition"},"The issue we hit with the ",(0,s.kt)("em",{parentName:"p"},"insert fixup")," is very similar.")),(0,s.kt)("p",null,"You might not realize the issue at the first sight, but the algorithm described\nwith the pseudocode above expects that the root of the red-black tree is black by\nboth relying on the invariant in the algorithm and afterwards by enforcing the\nblack root property."),(0,s.kt)("p",null,"If we decide to omit this condition, we need to address it in the pseudocodes\naccordingly."),(0,s.kt)("table",null,(0,s.kt)("thead",{parentName:"table"},(0,s.kt)("tr",{parentName:"thead"},(0,s.kt)("th",{parentName:"tr",align:"center"},"Usual algorithm with black root"),(0,s.kt)("th",{parentName:"tr",align:"center"},"Allowing red root"))),(0,s.kt)("tbody",{parentName:"table"},(0,s.kt)("tr",{parentName:"tbody"},(0,s.kt)("td",{parentName:"tr",align:"center"},(0,s.kt)("img",{alt:"1\xaa insertion",src:a(3416).Z+"#gh-light-mode-only",width:"179",height:"155"}),(0,s.kt)("img",{alt:"1\xaa insertion",src:a(4522).Z+"#gh-dark-mode-only",width:"179",height:"155"})),(0,s.kt)("td",{parentName:"tr",align:"center"},(0,s.kt)("img",{alt:"1\xaa insertion",src:a(477).Z+"#gh-light-mode-only",width:"179",height:"155"}),(0,s.kt)("img",{alt:"1\xaa insertion",src:a(8125).Z+"#gh-dark-mode-only",width:"179",height:"155"}))),(0,s.kt)("tr",{parentName:"tbody"},(0,s.kt)("td",{parentName:"tr",align:"center"},(0,s.kt)("img",{alt:"2\xaa insertion",src:a(6850).Z+"#gh-light-mode-only",width:"227",height:"251"}),(0,s.kt)("img",{alt:"2\xaa insertion",src:a(8983).Z+"#gh-dark-mode-only",width:"227",height:"251"})),(0,s.kt)("td",{parentName:"tr",align:"center"},(0,s.kt)("img",{alt:"2\xaa insertion",src:a(446).Z+"#gh-light-mode-only",width:"227",height:"251"}),(0,s.kt)("img",{alt:"2\xaa insertion",src:a(9905).Z+"#gh-dark-mode-only",width:"227",height:"251"}))),(0,s.kt)("tr",{parentName:"tbody"},(0,s.kt)("td",{parentName:"tr",align:"center"},(0,s.kt)("img",{alt:"3\xaa insertion",src:a(9657).Z+"#gh-light-mode-only",width:"371",height:"251"}),(0,s.kt)("img",{alt:"3\xaa insertion",src:a(9296).Z+"#gh-dark-mode-only",width:"371",height:"251"})),(0,s.kt)("td",{parentName:"tr",align:"center"},(0,s.kt)("img",{alt:"3\xaa insertion",src:a(841).Z+"#gh-light-mode-only",width:"371",height:"251"}),(0,s.kt)("img",{alt:"3\xaa insertion",src:a(7659).Z+"#gh-dark-mode-only",width:"371",height:"251"}))),(0,s.kt)("tr",{parentName:"tbody"},(0,s.kt)("td",{parentName:"tr",align:"center"},(0,s.kt)("img",{alt:"4\xaa insertion",src:a(515).Z+"#gh-light-mode-only",width:"419",height:"347"}),(0,s.kt)("img",{alt:"4\xaa insertion",src:a(9123).Z+"#gh-dark-mode-only",width:"419",height:"347"})),(0,s.kt)("td",{parentName:"tr",align:"center"},(0,s.kt)("img",{alt:"4\xaa insertion",src:a(2658).Z+"#gh-light-mode-only",width:"419",height:"347"}),(0,s.kt)("img",{alt:"4\xaa insertion",src:a(2979).Z+"#gh-dark-mode-only",width:"419",height:"347"}))),(0,s.kt)("tr",{parentName:"tbody"},(0,s.kt)("td",{parentName:"tr",align:"center"},(0,s.kt)("img",{alt:"5\xaa insertion",src:a(8747).Z+"#gh-light-mode-only",width:"419",height:"347"}),(0,s.kt)("img",{alt:"5\xaa insertion",src:a(7518).Z+"#gh-dark-mode-only",width:"419",height:"347"})),(0,s.kt)("td",{parentName:"tr",align:"center"},(0,s.kt)("img",{alt:"5\xaa insertion",src:a(2014).Z+"#gh-light-mode-only",width:"419",height:"347"}),(0,s.kt)("img",{alt:"5\xaa insertion",src:a(5959).Z+"#gh-dark-mode-only",width:"419",height:"347"}))),(0,s.kt)("tr",{parentName:"tbody"},(0,s.kt)("td",{parentName:"tr",align:"center"},(0,s.kt)("img",{alt:"6\xaa insertion",src:a(8317).Z+"#gh-light-mode-only",width:"563",height:"347"}),(0,s.kt)("img",{alt:"6\xaa insertion",src:a(2193).Z+"#gh-dark-mode-only",width:"563",height:"347"})),(0,s.kt)("td",{parentName:"tr",align:"center"},(0,s.kt)("img",{alt:"6\xaa insertion",src:a(162).Z+"#gh-light-mode-only",width:"563",height:"347"}),(0,s.kt)("img",{alt:"6\xaa insertion",src:a(1105).Z+"#gh-dark-mode-only",width:"563",height:"347"}))),(0,s.kt)("tr",{parentName:"tbody"},(0,s.kt)("td",{parentName:"tr",align:"center"},(0,s.kt)("img",{alt:"7\xaa insertion",src:a(8204).Z+"#gh-light-mode-only",width:"563",height:"443"}),(0,s.kt)("img",{alt:"7\xaa insertion",src:a(6986).Z+"#gh-dark-mode-only",width:"563",height:"443"})),(0,s.kt)("td",{parentName:"tr",align:"center"},(0,s.kt)("img",{alt:"7\xaa insertion",src:a(355).Z+"#gh-light-mode-only",width:"563",height:"443"}),(0,s.kt)("img",{alt:"7\xaa insertion",src:a(4300).Z+"#gh-dark-mode-only",width:"563",height:"443"}))),(0,s.kt)("tr",{parentName:"tbody"},(0,s.kt)("td",{parentName:"tr",align:"center"},(0,s.kt)("img",{alt:"8\xaa insertion",src:a(345).Z+"#gh-light-mode-only",width:"635",height:"443"}),(0,s.kt)("img",{alt:"8\xaa insertion",src:a(1988).Z+"#gh-dark-mode-only",width:"635",height:"443"})),(0,s.kt)("td",{parentName:"tr",align:"center"},(0,s.kt)("img",{alt:"8\xaa insertion",src:a(2530).Z+"#gh-light-mode-only",width:"635",height:"443"}),(0,s.kt)("img",{alt:"8\xaa insertion",src:a(363).Z+"#gh-dark-mode-only",width:"635",height:"443"}))),(0,s.kt)("tr",{parentName:"tbody"},(0,s.kt)("td",{parentName:"tr",align:"center"},(0,s.kt)("img",{alt:"9\xaa insertion",src:a(6464).Z+"#gh-light-mode-only",width:"755",height:"443"}),(0,s.kt)("img",{alt:"9\xaa insertion",src:a(437).Z+"#gh-dark-mode-only",width:"755",height:"443"})),(0,s.kt)("td",{parentName:"tr",align:"center"},(0,s.kt)("img",{alt:"9\xaa insertion",src:a(6039).Z+"#gh-light-mode-only",width:"755",height:"443"}),(0,s.kt)("img",{alt:"9\xaa insertion",src:a(3122).Z+"#gh-dark-mode-only",width:"755",height:"443"}))))),(0,s.kt)("h2",{id:"3\xaa-every-leaf-nil-is-black"},"3\xaa Every leaf (",(0,s.kt)("inlineCode",{parentName:"h2"},"nil"),") is black."),(0,s.kt)("p",null,"Now, this rule is a funny one. What does this imply and can I interpret this in\nsome other way? Let's go through some of the possible ways I can look at this and\nhow would they affect the other rules and balancing."),(0,s.kt)("p",null,"We will experiment with the following tree:\n",(0,s.kt)("img",{src:a(6769).Z+"#gh-light-mode-only",width:"899",height:"539"}),"\n",(0,s.kt)("img",{src:a(4086).Z+"#gh-dark-mode-only",width:"899",height:"539"})),(0,s.kt)("p",null,"We should start by counting the black nodes from root to the ",(0,s.kt)("inlineCode",{parentName:"p"},"nil")," leaves based\non the rules. We have multiple similar paths, so we will pick only the interesting\nones."),(0,s.kt)("ol",null,(0,s.kt)("li",{parentName:"ol"},"What happens if we do not count the ",(0,s.kt)("inlineCode",{parentName:"li"},"nil")," leaves?"),(0,s.kt)("li",{parentName:"ol"},"What happens if we consider leaves the nodes with ",(0,s.kt)("em",{parentName:"li"},"no descendants"),", i.e. both\nof node's children are ",(0,s.kt)("inlineCode",{parentName:"li"},"nil"),"?"),(0,s.kt)("li",{parentName:"ol"},"What happens if we do not count the ",(0,s.kt)("inlineCode",{parentName:"li"},"nil")," leaves, but consider nodes with at\nleast one ",(0,s.kt)("inlineCode",{parentName:"li"},"nil")," descendant as leaves?")),(0,s.kt)("table",null,(0,s.kt)("thead",{parentName:"table"},(0,s.kt)("tr",{parentName:"thead"},(0,s.kt)("th",{parentName:"tr",align:"right"},"path"),(0,s.kt)("th",{parentName:"tr",align:"right"},"black nodes"),(0,s.kt)("th",{parentName:"tr",align:"right"},"1\xaa idea"),(0,s.kt)("th",{parentName:"tr",align:"right"},"2\xaa idea"),(0,s.kt)("th",{parentName:"tr",align:"right"},"3\xaa idea"))),(0,s.kt)("tbody",{parentName:"table"},(0,s.kt)("tr",{parentName:"tbody"},(0,s.kt)("td",{parentName:"tr",align:"right"},(0,s.kt)("inlineCode",{parentName:"td"},"3 \u2192 1 \u2192 0 \u2192 nil")),(0,s.kt)("td",{parentName:"tr",align:"right"},"4"),(0,s.kt)("td",{parentName:"tr",align:"right"},"3"),(0,s.kt)("td",{parentName:"tr",align:"right"},"4"),(0,s.kt)("td",{parentName:"tr",align:"right"},"3")),(0,s.kt)("tr",{parentName:"tbody"},(0,s.kt)("td",{parentName:"tr",align:"right"},(0,s.kt)("inlineCode",{parentName:"td"},"3 \u2192 5 \u2192 7 \u2192 8 \u2192 nil")),(0,s.kt)("td",{parentName:"tr",align:"right"},"4"),(0,s.kt)("td",{parentName:"tr",align:"right"},"3"),(0,s.kt)("td",{parentName:"tr",align:"right"},"-"),(0,s.kt)("td",{parentName:"tr",align:"right"},"3")),(0,s.kt)("tr",{parentName:"tbody"},(0,s.kt)("td",{parentName:"tr",align:"right"},(0,s.kt)("inlineCode",{parentName:"td"},"3 \u2192 5 \u2192 7 \u2192 8 \u2192 9 \u2192 nil")),(0,s.kt)("td",{parentName:"tr",align:"right"},"4"),(0,s.kt)("td",{parentName:"tr",align:"right"},"3"),(0,s.kt)("td",{parentName:"tr",align:"right"},"4"),(0,s.kt)("td",{parentName:"tr",align:"right"},"3")))),(0,s.kt)("p",null,"First idea is very easy to execute and it is also very easy to argue about its\ncorrectness. It is correct, because we just subtract one from each of the paths.\nThis affects ",(0,s.kt)("strong",{parentName:"p"},"all")," paths and therefore results in global decrease by one."),(0,s.kt)("p",null,"Second idea is a bit more complicated. We count the ",(0,s.kt)("inlineCode",{parentName:"p"},"nil"),"s, so the count is ",(0,s.kt)("span",{parentName:"p",className:"math math-inline"},(0,s.kt)("span",{parentName:"span",className:"katex"},(0,s.kt)("span",{parentName:"span",className:"katex-mathml"},(0,s.kt)("math",{parentName:"span",xmlns:"http://www.w3.org/1998/Math/MathML"},(0,s.kt)("semantics",{parentName:"math"},(0,s.kt)("mrow",{parentName:"semantics"},(0,s.kt)("mn",{parentName:"mrow"},"4")),(0,s.kt)("annotation",{parentName:"semantics",encoding:"application/x-tex"},"4")))),(0,s.kt)("span",{parentName:"span",className:"katex-html","aria-hidden":"true"},(0,s.kt)("span",{parentName:"span",className:"base"},(0,s.kt)("span",{parentName:"span",className:"strut",style:{height:"0.6444em"}}),(0,s.kt)("span",{parentName:"span",className:"mord"},"4"))))),"\nas it should be. However, there is one difference. Second path no longer satisfies\nthe condition of a ",(0,s.kt)("em",{parentName:"p"},"leaf"),". Technically it relaxes the 5th rule, because we leave\nout some of the nodes. We should probably avoid that."),(0,s.kt)("admonition",{type:"caution"},(0,s.kt)("p",{parentName:"admonition"},"With the second idea, you may also feel that we are \u201cbending\u201d the rules a bit,\nespecially the definition of the \u201cleaf\u201d nodes."),(0,s.kt)("p",{parentName:"admonition"},"Given the definition of the red-black tree, where ",(0,s.kt)("inlineCode",{parentName:"p"},"nil")," is considered to be an\nexternal node, we have decided that bending it a bit just to stir a thought about\nit won't hurt anybody. \ud83d\ude09")),(0,s.kt)("h2",{id:"4\xaa-if-a-node-is-red-then-both-its-children-are-black"},"4\xaa If a node is red, then both its children are black."),(0,s.kt)("p",null,"This rule might seem rather silly on the first look, but there are 2 important\nfunctions:"),(0,s.kt)("ol",null,(0,s.kt)("li",{parentName:"ol"},"it allows the algorithms to ",(0,s.kt)("em",{parentName:"li"},"\u201cnotice\u201d")," that something went wrong (i.e. the\ntree needs to be rebalanced), and"),(0,s.kt)("li",{parentName:"ol"},"it holds the balancing and height of the tree ",(0,s.kt)("em",{parentName:"li"},"\u201cin check\u201d")," (with the help of\nthe 5th rule).")),(0,s.kt)("p",null,"When we have a look at the algorithms that are used for fixing up the red-black\ntree after an insertion or deletion, we will notice that all the algorithms need\nis the color of the node. "),(0,s.kt)("blockquote",null,(0,s.kt)("p",{parentName:"blockquote"},"How come it is the only thing that we need?\nHow come such na\xefve thing can be enough?")),(0,s.kt)("p",null,"Let's say we perform an insertion into the tree\u2026 We go with the usual and pretty\nprimitive insertion into the binary-search tree and then, if needed, we \u201cfix up\u201d\nbroken invariants. ",(0,s.kt)("em",{parentName:"p"},"How can that be enough?")," With each insertion and deletion we\nmaintain the invariants, therefore if we break them with one operation, there's\nonly one path on which the invariants were ",(0,s.kt)("em",{parentName:"p"},"felled"),". If we know that rest of the\ntree is correct, it allows us to fix the issues just by propagating it to the\nroot and ",(0,s.kt)("em",{parentName:"p"},"abusing")," the siblings (which are, of course, correct red-black\nsubtrees) to fix or at least partially mitigate the issues and propagate them\nfurther."),(0,s.kt)("p",null,"Let's assume that we do not enforce this rule, you can see how it breaks the\nbalancing of the tree below."),(0,s.kt)(w,{mdxType:"Tabs"},(0,s.kt)(v,{value:"enforcing",label:"Enforcing this rule",mdxType:"TabItem"},(0,s.kt)("p",null,(0,s.kt)("img",{src:a(772).Z+"#gh-light-mode-only",width:"755",height:"347"}),"\n",(0,s.kt)("img",{src:a(8663).Z+"#gh-dark-mode-only",width:"755",height:"347"}))),(0,s.kt)(v,{value:"omitting",label:"Omitting this rule",mdxType:"TabItem"},(0,s.kt)("p",null,(0,s.kt)("img",{src:a(4130).Z+"#gh-light-mode-only",width:"803",height:"443"}),"\n",(0,s.kt)("img",{src:a(2683).Z+"#gh-dark-mode-only",width:"803",height:"443"})))),(0,s.kt)("p",null,"We can create a ",(0,s.kt)("strong",{parentName:"p"},"big")," subtree with only red nodes and ",(0,s.kt)("strong",{parentName:"p"},"even")," when keeping\nthe rest of the rules maintained, it will break the time complexity. It stops us\nfrom \u201chacking\u201d the black height requirement laid by the 5th rule."),(0,s.kt)("h2",{id:"5\xaa-for-each-node-all-simple-paths-from-the-node-to-descendant-leaves-contain-the-same-number-of-black-nodes"},"5\xaa For each node, all simple paths from the node to descendant leaves contain the same number of black nodes."),(0,s.kt)("p",null,"As it was mentioned, with the 4th rule they hold the balancing of the red-black\ntree."),(0,s.kt)("admonition",{type:"tip"},(0,s.kt)("p",{parentName:"admonition"},"An important observation here is the fact that the red-black tree is a\n",(0,s.kt)("strong",{parentName:"p"},"height"),"-balanced tree.")),(0,s.kt)("p",null,"Enforcing this rule (together with the 4th rule) keeps the tree balanced:"),(0,s.kt)("ol",null,(0,s.kt)("li",{parentName:"ol"},"4th rule makes sure we can't \u201chack\u201d this requirement."),(0,s.kt)("li",{parentName:"ol"},"This rule ensures that we have \u201csimilar\u201d",(0,s.kt)("sup",{parentName:"li",id:"fnref-2"},(0,s.kt)("a",{parentName:"sup",href:"#fn-2",className:"footnote-ref"},"2"))," length to each of the leaves.")),(0,s.kt)("admonition",{title:"AVL tree",type:"tip"},(0,s.kt)("p",{parentName:"admonition"},"You might have heard about an ",(0,s.kt)("em",{parentName:"p"},"AVL tree")," before. It is the first self-balanced\ntree to be ever introduced and works in a very similar nature as the red-black\ntree, the only difference is that it does not deal with the ",(0,s.kt)("em",{parentName:"p"},"black height"),", but\nthe height in general."),(0,s.kt)("p",{parentName:"admonition"},"If you were to compare AVL with the red-black tree, you can say that AVL is much\nmore strict while red-black tree can still maintain the same asymptotic time\ncomplexity for the operations, but having more relaxed rules.")),(0,s.kt)("div",{className:"footnotes"},(0,s.kt)("hr",{parentName:"div"}),(0,s.kt)("ol",{parentName:"div"},(0,s.kt)("li",{parentName:"ol",id:"fn-1"},"CORMEN, Thomas. Introduction to algorithms. Cambridge, Mass: MIT Press, 2009. isbn 9780262033848.",(0,s.kt)("a",{parentName:"li",href:"#fnref-1",className:"footnote-backref"},"\u21a9")),(0,s.kt)("li",{parentName:"ol",id:"fn-2"},"red nodes still exist",(0,s.kt)("a",{parentName:"li",href:"#fnref-2",className:"footnote-backref"},"\u21a9")))))}W.isMDXComponent=!0},4086:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/rb_dark-c025d61dee7913262c86277087751328.png"},9493:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/rb_height_dark-75a70ddff74e5e1aff7e9986221b5687.png"},5021:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/rb_height_light-36fa69317ced094d7bb7b0fdf32cb3fe.png"},6769:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/rb_light-f53bee3b32ddb2e7a4249828bc03b1a4.png"},8663:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/correct_dark-bbd8d4c1796b145025fed5b6dff03b84.png"},772:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/correct_light-bc4770146072f748be4a5aa11abf3a0c.png"},2683:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/incorrect_dark-9b8b3be328ffad83233de4536c120016.png"},4130:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/incorrect_light-e787e568e9a1528dcac5bf55ef29fdaa.png"},4522:(e,t,a)=>{a.d(t,{Z:()=>n});const n=""},3416:(e,t,a)=>{a.d(t,{Z:()=>n});const n=""},8983:(e,t,a)=>{a.d(t,{Z:()=>n});const n=""},6850:(e,t,a)=>{a.d(t,{Z:()=>n});const n=""},9296:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/br_2_dark-e8c35bc37b250271cf480f71904c15a7.png"},9657:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/br_2_light-5e6a1d53e559a30e5fb86ee019229bbd.png"},9123:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/br_3_dark-a2b8248c182059b67c703f75f58f3784.png"},515:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/br_3_light-7be3bbcb08f8b7182a1c719693a47615.png"},7518:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/br_4_dark-eddf4c315becc51f89b0967320f132d8.png"},8747:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/br_4_light-d72dfa633794ec97eddce8e3a4b02660.png"},2193:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/br_5_dark-e24da2d7a3fb2ee63ac8e1ea9c2d45a8.png"},8317:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/br_5_light-efb3568bf4aadb19a9dcc57e748f89d8.png"},6986:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/br_6_dark-160fd071a93e279a5339c7976745f8b1.png"},8204:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/br_6_light-b090675f7b30b574af44d667b083e9b7.png"},1988:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/br_7_dark-35ede8f297484f4305ea7fd23cbddc49.png"},345:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/br_7_light-018e13c41ce1fc6257c4c65748aaae27.png"},437:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/br_8_dark-b0f871ab182d06edb5c29bb490ad70bc.png"},6464:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/br_8_light-81ac4c8a3988bd43c66f563cd4799d61.png"},8125:(e,t,a)=>{a.d(t,{Z:()=>n});const n=""},477:(e,t,a)=>{a.d(t,{Z:()=>n});const n=""},9905:(e,t,a)=>{a.d(t,{Z:()=>n});const n=""},446:(e,t,a)=>{a.d(t,{Z:()=>n});const n=""},7659:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/rr_2_dark-e8c35bc37b250271cf480f71904c15a7.png"},841:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/rr_2_light-5e6a1d53e559a30e5fb86ee019229bbd.png"},2979:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/rr_3_dark-4e8b2ca938738395e438b7fc2fc5dfe4.png"},2658:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/rr_3_light-51521ba414ff3a3530ed0109cfab799d.png"},5959:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/rr_4_dark-3cc1c8d4b39707d2a51b51f4f1b29dc8.png"},2014:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/rr_4_light-631a0b3947be21a12b3a489f0cd0c3c4.png"},1105:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/rr_5_dark-9028cc10e78c05cb669d5d438dcbf93f.png"},162:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/rr_5_light-6b8f80047906eb1f58472d231eb9b12a.png"},4300:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/rr_6_dark-160fd071a93e279a5339c7976745f8b1.png"},355:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/rr_6_light-b090675f7b30b574af44d667b083e9b7.png"},363:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/rr_7_dark-35ede8f297484f4305ea7fd23cbddc49.png"},2530:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/rr_7_light-018e13c41ce1fc6257c4c65748aaae27.png"},3122:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/rr_8_dark-b0f871ab182d06edb5c29bb490ad70bc.png"},6039:(e,t,a)=>{a.d(t,{Z:()=>n});const n=a.p+"assets/images/rr_8_light-81ac4c8a3988bd43c66f563cd4799d61.png"}}]);