XAML 給資源起個好名字 用 StaticResource 起一個別名

来源:https://www.cnblogs.com/lindexi/archive/2022/03/22/16037770.html
-Advertisement-
Play Games

十天前,我發佈了對.NET Core程式進行瘦身的開源軟體Zack.DotNetTrimmer,與.NET Core內置的剪裁器相比,Zack.DotNetTrimmer不僅對程式的剪裁效果更好,而且還支持WPF、WinForm程式。 很多朋友對於這個開源項目的原理很感興趣,因此我將通過這篇文章為大 ...


本文来和大家聊一下关于 XAML 资源的定义的事情,和开发技术关系不大,更多的是开发的思路

在稍微大一点的项目里,肯定 XAML 资源是少不了的。对于 XAML 资源,行业里讨论多(非小白讨论)的是关于资源的复用和初始化时机,加载时机的问题。本文聊的是资源的复用以及变更资源的开发模式

如果团队里有一位开发者是命名高手,那么问题将特别小。或者说这个项目只有一位开发者在开发,也不需要后续维护,那这个问题也特别的小。然而在很多团队里面,都有很多开发者,同时开发者的命名能力也有一定的差距

如果没有给资源一个好的命名,自然,每次用起来的时候,都会遇到一个问题,那就是某个资源找不到的问题。一旦找不到资源,自然就只能重新定义资源了。或者说我以为某个资源是我此模块能用的,然而此资源却非通用设计,也许下个版本就被更改,于是我的模块就因为某个资源的变更而挖坑

如下面的例子:

我需要开发一个应用,此应用有多个页面组合。由于技术侧的问题,我不能将每个页面都使用相同的控件,需要采用不同的控件。但是为了界面的美观,尽管使用的不同的控件,依然也需要保持相似页面布局方式。按照惯例,假定定义页面的内边距,假设默认的页边距是 12 单位大小,定义为资源可以如何写?如以下的资源定义

<Thickness x:Key="TwelveThickness">12,12,12,12</Thickness>

可以看到,定义资源为 12 单位的 Thickness 有点过于具体了。如果后续设计师想改为 16 个单位呢?那此时就是一个选择了,要么将此资源改为 16 的值,但是保留 TwelveThickness 这个词,让其他的开发者看到这里虽然写着是 Twelve 但实际上是 16 个单位。要么就是改值和改资源名,说不定就是炸鱼的改动了。要么就是再新加一个资源名,也许这样垃圾就加一了

如果有一个具体的命名呢?命名和数值无关的呢?例如命名为 MainPagePadding 呢?这个感觉是不错的,给主页面使用的 Padding 值。但是不够抽象,如果我用的是 FxxPage 的呢,这个也需要一个内边距,那用哪个资源好呢

如果定义的资源命名是 DefaultPagePadding 呢,或者 PagePadding 呢,是不是就更抽象了一层,不仅 MainPage 能用,也合适 FxxPage 使用

但是定义的抽象等级,除非是命名高手,否则一次性叫对一个命名还是很有难度的。而有一些资源,又是十分冲突的。既是非常通用的,却也许会被变更。例如颜色,定义颜色画刷的时候,资源的重复存在的问题核心就是开发模式上的寻找困难和被其他开发者更改带来的锅,在性能上的问题就是非托管资源的占用增加,没有复用原有的画刷。但是颜色的定义,是会在迭代被变更的

从技术侧的一个解决方法是采用 StaticResource 来进行资源的引用,相当于给资源一个别名的方式。再定义一个资源,引用原先的资源

例如有一个红色是默认的主题红色,最好的定义是 Brush.SolidColorBrush.RedThemeBrush 的资源名。然而此资源名特别具体,如果作为主页面的背景色的时候,此时将会因为太过具体而不合适。毕竟后续如果我还有其他的页面,似乎用来做背景的,耦合了具体的红主题色是不合适的

那如果再定义一个画刷主题,称为 MainPageBackgroundBrush 呢?自然,重复定义的画刷就是重复的资源,不合适

好在可以使用 StaticResource 的方式,使用静态资源引用,从而让资源被重新定向,如下面代码,定义了一个主题颜色

        <SolidColorBrush x:Key="Brush.SolidColorBrush.RedThemeBrush" Color="#FFC10606"/>

接下来可以采用 StaticResource 引用此颜色,定义一个默认的页面的背景画刷

        <StaticResource x:Key="DefaultPageBackgroundBrush" ResourceKey="Brush.SolidColorBrush.RedThemeBrush"/>

再定义具体的主页面的背景画刷

        <StaticResource x:Key="MainPageBackgroundBrush" ResourceKey="DefaultPageBackgroundBrush"/>

于是在引用的时候,可以在主页面,只引用主页面的背景画刷

        <Grid x:Name="MainPage" Background="{StaticResource MainPageBackgroundBrush}"></Grid>

这样做的一个优势在于,让资源的定义是一层层定义的。解决了开发侧的重复资源定义,又想资源重复定义方便改动的时候相互不影响,又想着不重复定义方便要改可以一起改的问题

如以上的代码,相当于将资源的定义分为三层。最底层是 Brush.SolidColorBrush.RedThemeBrush 定义一个和业务无关的画刷,第二层是 DefaultPageBackgroundBrush 定义和页面有关的默认画刷,最上层是 MainPageBackgroundBrush 定义和主页面相关的画刷

这些画刷都是相同的一个资源,只是靠 StaticResource 进行引用

如果后续准备改主页面的画刷,而其他的页面保持不动。那么可以明确了解到,只需要改 MainPageBackgroundBrush 的资源值即可,影响不到其他的页面。而如果期望是全部页面的背景色都换成某个其他的颜色,只需要改 DefaultPageBackgroundBrush 即可。如果是设计师想要改整个应用的红色主题色,那就改 Brush.SolidColorBrush.RedThemeBrush 的颜色

如此的设计可以比较方便解决比较大的项目的资源引用问题。核心的思想就是依靠引用的方式,将资源的定义分层,按照抽象的不同,分为不同层的定义。定义资源的命名方式最好都是一组一组的,一个抽象层里面有很多组,每一组之间的命名都是非常相似的。如此就很方便资源的管理。这只是一个思想,不使用 StaticResource 也可以,如换成绑定的方式也可以

博客园博客只做备份,博客发布就不再更新,如果想看最新博客,请到 https://blog.lindexi.com/

知识共享许可协议
本作品采用知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议进行许可。欢迎转载、使用、重新发布,但务必保留文章署名[林德熙](http://blog.csdn.net/lindexi_gd)(包含链接:http://blog.csdn.net/lindexi_gd ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请与我[联系](mailto:[email protected])。
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • Hibernate 是一個開放源代碼的對象關係映射框架,它對 JDBC 進行了非常輕量級的對象封裝,它將 POJO 與資料庫表建立映射關係,是一個全自動的 orm 框架 ...
  • Java程式設計基礎之面向對象(下) (補充了上的一些遺漏的知識,同時加入了自己的筆記的ヾ(•ω•`)o) (至於為什麼分P,啊大概是為了自己查筆記方便(?)應該是(〃` 3′〃)) (但是u1s1,學完了面向對象後反而更懵逼,下一步先刷演算法吧,然後Java的學習也跟上,今年爭取考完二級證書(o-ω ...
  • (Java 常見排序演算法) 彙總 序號 排序演算法 平均時間 最好情況 最差情況 穩定度 額外空間 備註 相對時間 1 冒泡演算法 O(n2) O(n) O(n2) 穩定 O(1) n 越小越好 182 ms 2 選擇演算法 O(n2) O(n2) O(n2) 不穩定 O(1) n 越小越好 53 ms ...
  • hello大家好呀,我是小樓~ 今天又帶來一次性能優化的分享,這是我剛進公司時接手的祖傳(壞笑)項目,這個項目在我的文章中屢次被提及,我在它上面做了很多的性能優化,比如《記一次提升18倍的性能優化》這篇文章,比較偏向某個細節的優化,本文更偏向巨集觀上的性能優化,可以說是個老演員了。 背景 為了新朋友能 ...
  • 一、概述 date代表一個特定的時間,精確到毫秒 二、構造方法 方法名說明 public Date() 分配一個Date對象,並初始化,以便它代表被分配的時間,精確到毫秒 public Date(long date) 分配一個Date對象,並將其初始化為表示從標準基準時間起指定的毫秒數 import ...
  • .net core IdentityServer4 認證授權 token ...
  • 1.什麼是Blazor? 有什麼優勢? ASP.NET Core Blazor 簡介 Blazor 是一個使用 Blazor 生成互動式客戶端 Web UI 的框架: 使用 C# 代替 JavaScript 來創建信息豐富的互動式 UI。 共用使用 .NET 編寫的伺服器端和客戶端應用邏輯。 將 U ...
  • 配置源的同步 IOptionsMonitor 使用 //以下demo演示使用IOptionsMonitor重新載入配置並當重新載入配置是執行回調函數 var configuration = new ConfigurationBuilder().AddJsonFile(path: "profile.j ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...