{"id":1998,"date":"2019-05-27T21:00:04","date_gmt":"2019-05-27T18:00:04","guid":{"rendered":"http:\/\/demensdeum.com\/blog\/?p=1998"},"modified":"2024-12-16T22:32:37","modified_gmt":"2024-12-16T19:32:37","slug":"decorator-pattern","status":"publish","type":"post","link":"https:\/\/demensdeum.com\/blog\/pt\/2019\/05\/27\/decorator-pattern\/","title":{"rendered":"Decorador de padr\u00f5es"},"content":{"rendered":"<p>O padr\u00e3o Decorator refere-se a padr\u00f5es de projeto estrutural.<br \/><img loading=\"lazy\" decoding=\"async\" class=\"alignnone size-full wp-image-2012\" src=\"https:\/\/demensdeum.com\/blog\/wp-content\/uploads\/2019\/05\/pexels-photo-348601.jpeg\" alt=\"\" width=\"380\" height=\"252\" \/><br \/>\nO decorador \u00e9 usado como alternativa \u00e0 heran\u00e7a para estender a funcionalidade das classes.<br \/>Existe a tarefa de expandir a funcionalidade do aplicativo dependendo do tipo de produto. O cliente exige tr\u00eas tipos de produto &#8211; B\u00e1sico, Profissional, Final.<br \/>B\u00e1sico&#8211; conta o n\u00famero de caracteres, Profissional &#8211; recursos Basic + imprime texto em letras mai\u00fasculas, Ultimate &#8211; B\u00e1sico + Profissional + imprime texto dizendo ULTIMATE.<br \/>N\u00f3s o implementamos usando heran\u00e7a:<\/p>\n<div class=\"hcb_wrap\">\n<pre class=\"prism line-numbers lang-unknown\" data-lang=\"unknown\"><code>protocol Feature {\n\tfunc textOperation(text: String)\n}\n\nclass BasicVersionFeature: Feature {\n\tfunc textOperation(text: String) {\n\t\tprint(\"\\(text.count)\")\n\t}\n}\n\nclass ProfessionalVersionFeature: BasicVersionFeature {\n\toverride func textOperation(text: String) {\n\t\tsuper.textOperation(text: text)\n\t\tprint(\"\\(text.uppercased())\")\n\t}\n}\n\nclass UltimateVersionFeature: ProfessionalVersionFeature {\n\toverride func textOperation(text: String) {\n\t\tsuper.textOperation(text: text)\n\t\tprint(\"ULTIMATE: \\(text)\")\n\t}\n}\n\nlet textToFormat = \"Hello Decorator\"\n\nlet basicProduct = BasicVersionFeature()\nbasicProduct.textOperation(text: textToFormat)\n\nlet professionalProduct = ProfessionalVersionFeature()\nprofessionalProduct.textOperation(text: textToFormat)\n\nlet ultimateProduct = UltimateVersionFeature()\nultimateProduct.textOperation(text: textToFormat)\n<\/code><\/pre>\n<\/div>\n<p>Agora existe a necessidade de implementa\u00e7\u00e3o do produto \u201cUltimate Light\u201d &#8211; Basic + Ultimate, mas sem os recursos da vers\u00e3o Professional. O primeiro OH! acontece, porque&#8230; voc\u00ea ter\u00e1 que criar uma classe separada para uma tarefa t\u00e3o simples e duplicar o c\u00f3digo.<br \/>Vamos continuar a implementa\u00e7\u00e3o usando heran\u00e7a:<\/p>\n<div class=\"hcb_wrap\">\n<pre class=\"prism line-numbers lang-unknown\" data-lang=\"unknown\"><code>protocol Feature {\n\tfunc textOperation(text: String)\n}\n\nclass BasicVersionFeature: Feature {\n\tfunc textOperation(text: String) {\n\t\tprint(\"\\(text.count)\")\n\t}\n}\n\nclass ProfessionalVersionFeature: BasicVersionFeature {\n\toverride func textOperation(text: String) {\n\t\tsuper.textOperation(text: text)\n\t\tprint(\"\\(text.uppercased())\")\n\t}\n}\n\nclass UltimateVersionFeature: ProfessionalVersionFeature {\n\toverride func textOperation(text: String) {\n\t\tsuper.textOperation(text: text)\n\t\tprint(\"ULTIMATE: \\(text)\")\n\t}\n}\n\nclass UltimateLightVersionFeature: BasicVersionFeature {\n\toverride func textOperation(text: String) {\n\t\tsuper.textOperation(text: text)\n\t\tprint(\"ULTIMATE: \\(text)\")\t\n\t}\n}\n\nlet textToFormat = \"Hello Decorator\"\n\nlet basicProduct = BasicVersionFeature()\nbasicProduct.textOperation(text: textToFormat)\n\nlet professionalProduct = ProfessionalVersionFeature()\nprofessionalProduct.textOperation(text: textToFormat)\n\nlet ultimateProduct = UltimateVersionFeature()\nultimateProduct.textOperation(text: textToFormat)\n\nlet ultimateLightProduct = UltimateLightVersionFeature()\nultimateLightProduct.textOperation(text: textToFormat)\n<\/code><\/pre>\n<\/div>\n<p>O exemplo pode ser desenvolvido para maior clareza, mas mesmo agora a complexidade de suportar um sistema baseado em uma base de heran\u00e7a \u00e9 vis\u00edvel &#8211; complicado e falta de flexibilidade.<br \/>Um decorador \u00e9 um conjunto de protocolos que descreve a funcionalidade, uma classe abstrata contendo uma refer\u00eancia a uma inst\u00e2ncia concreta filha da classe decoradora que estende a funcionalidade.<br \/>Vamos reescrever o exemplo acima usando o padr\u00e3o:<\/p>\n<div class=\"hcb_wrap\">\n<pre class=\"prism line-numbers lang-unknown\" data-lang=\"unknown\"><code>protocol Feature {\n\tfunc textOperation(text: String)\n}\n\nclass FeatureDecorator: Feature {\n\tprivate var feature: Feature?\n\t\n\tinit(feature: Feature? = nil) {\n\t\tself.feature = feature\n\t}\n\t\n\tfunc textOperation(text: String) {\n\t\tfeature?.textOperation(text: text)\n\t}\n}\n\nclass BasicVersionFeature: FeatureDecorator {\n\toverride func textOperation(text: String) {\n\t\tsuper.textOperation(text: text)\n\t\tprint(\"\\(text.count)\")\n\t}\n}\n\nclass ProfessionalVersionFeature: FeatureDecorator {\n\toverride func textOperation(text: String) {\n\t\tsuper.textOperation(text: text)\n\t\tprint(\"\\(text.uppercased())\")\n\t}\n}\n\nclass UltimateVersionFeature: FeatureDecorator {\n\toverride func textOperation(text: String) {\n\t\tsuper.textOperation(text: text)\n\t\tprint(\"ULTIMATE: \\(text)\")\n\t}\n}\n\nlet textToFormat = \"Hello Decorator\"\n\nlet basicProduct = BasicVersionFeature(feature: UltimateVersionFeature())\nbasicProduct.textOperation(text: textToFormat)\n\nlet professionalProduct = ProfessionalVersionFeature(feature: UltimateVersionFeature())\nprofessionalProduct.textOperation(text: textToFormat)\n\nlet ultimateProduct = BasicVersionFeature(feature: UltimateVersionFeature(feature: ProfessionalVersionFeature()))\nultimateProduct.textOperation(text: textToFormat)\n\nlet ultimateLightProduct = BasicVersionFeature(feature: UltimateVersionFeature())\nultimateLightProduct.textOperation(text: textToFormat)\n<\/code><\/pre>\n<\/div>\n<p>Agora podemos criar varia\u00e7\u00f5es de qualquer tipo de produto &#8211; basta inicializar os tipos combinados na fase de inicializa\u00e7\u00e3o do aplicativo; o exemplo abaixo \u00e9 a cria\u00e7\u00e3o da vers\u00e3o Ultimate + Professional:<\/p>\n<div class=\"hcb_wrap\">\n<pre class=\"prism line-numbers lang-unknown\" data-lang=\"unknown\"><code>ultimateProfessionalProduct.textOperation(text: textToFormat)\n<\/code><\/pre>\n<\/div>\n<h3>Fontes<\/h3>\n<p><a href=\"https:\/\/refactoring.guru\/ru\/design-patterns\/decorator\" target=\"_blank\" rel=\"noopener\">https:\/\/refactoring.guru\/ru\/design-patterns\/decorator <\/a><\/p>\n<h3>C\u00f3digo fonte<\/h3>\n<p><a href=\"https:\/\/gitlab.com\/demensdeum\/patterns\" target=\"_blank\" rel=\"noopener\">https:\/\/gitlab.com\/demensdeum\/patterns<\/a><\/p>\n","protected":false},"excerpt":{"rendered":"<p>O padr\u00e3o Decorator refere-se a padr\u00f5es de projeto estrutural. O decorador \u00e9 usado como alternativa \u00e0 heran\u00e7a para estender a funcionalidade das classes.Existe a tarefa de expandir a funcionalidade do aplicativo dependendo do tipo de produto. O cliente exige tr\u00eas tipos de produto &#8211; B\u00e1sico, Profissional, Final.B\u00e1sico&#8211; conta o n\u00famero de caracteres, Profissional &#8211; recursos<a class=\"more-link\" href=\"https:\/\/demensdeum.com\/blog\/pt\/2019\/05\/27\/decorator-pattern\/\">Continue reading <span class=\"screen-reader-text\">&#8220;Decorador de padr\u00f5es&#8221;<\/span><\/a><\/p>\n","protected":false},"author":1,"featured_media":0,"comment_status":"open","ping_status":"open","sticky":false,"template":"","format":"standard","meta":{"_monsterinsights_skip_tracking":false,"_monsterinsights_sitenote_active":false,"_monsterinsights_sitenote_note":"","_monsterinsights_sitenote_category":0,"_jetpack_memberships_contains_paid_content":false,"footnotes":""},"categories":[61,52],"tags":[104,95],"class_list":["post-1998","post","type-post","status-publish","format-standard","hentry","category-techie","category-tutorials","tag-decorator","tag-patterns","entry"],"translation":{"provider":"WPGlobus","version":"3.0.2","language":"pt","enabled_languages":["en","ru","zh","de","fr","ja","pt"],"languages":{"en":{"title":true,"content":true,"excerpt":false},"ru":{"title":true,"content":true,"excerpt":false},"zh":{"title":true,"content":true,"excerpt":false},"de":{"title":true,"content":true,"excerpt":false},"fr":{"title":true,"content":true,"excerpt":false},"ja":{"title":true,"content":true,"excerpt":false},"pt":{"title":true,"content":true,"excerpt":false}}},"jetpack_featured_media_url":"","jetpack_sharing_enabled":true,"_links":{"self":[{"href":"https:\/\/demensdeum.com\/blog\/pt\/wp-json\/wp\/v2\/posts\/1998","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/demensdeum.com\/blog\/pt\/wp-json\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/demensdeum.com\/blog\/pt\/wp-json\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/demensdeum.com\/blog\/pt\/wp-json\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/demensdeum.com\/blog\/pt\/wp-json\/wp\/v2\/comments?post=1998"}],"version-history":[{"count":22,"href":"https:\/\/demensdeum.com\/blog\/pt\/wp-json\/wp\/v2\/posts\/1998\/revisions"}],"predecessor-version":[{"id":3957,"href":"https:\/\/demensdeum.com\/blog\/pt\/wp-json\/wp\/v2\/posts\/1998\/revisions\/3957"}],"wp:attachment":[{"href":"https:\/\/demensdeum.com\/blog\/pt\/wp-json\/wp\/v2\/media?parent=1998"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/demensdeum.com\/blog\/pt\/wp-json\/wp\/v2\/categories?post=1998"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/demensdeum.com\/blog\/pt\/wp-json\/wp\/v2\/tags?post=1998"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}