WPF 雜談——Binding表達式

来源:http://www.cnblogs.com/hayasi/archive/2017/06/05/6940545.html
-Advertisement-
Play Games

不管是定義控制項還是用戶控制項都會用到一個功能——綁定(Binding)。書面的叫法:元素綁定。意思就是讓綁定的元素實現數據同步。在筆者看來WPF引入這一個功能實在是太完美了。編程更加的具體化。特別是跟MVVM模式的配合,那叫完美。筆者不是學術派的。全面性的講述的話那是不現實。就從筆者的使用經驗來談Bi ...


不管是定義控制項還是用戶控制項都會用到一個功能——綁定(Binding)。書面的叫法:元素綁定。意思就是讓綁定的元素實現數據同步。在筆者看來WPF引入這一個功能實在是太完美了。編程更加的具體化。特別是跟MVVM模式的配合,那叫完美。筆者不是學術派的。全面性的講述的話那是不現實。就從筆者的使用經驗來談Binding吧。

最普通的使用方式,他的目標元素是控制項上的DataContext對象。如下:

 <TextBlock Grid.Column="0" Text="{Binding DishName}" Style="{StaticResource TakingDishDishNameTextStyle}" />

DataContext這個屬性是在FrameworkElement類上面的。也就是說大部分的控制項上都會有自己的DataContext的。那麼我們一般只有在最外層設置DataContext屬性。為了更加清楚的瞭解DataContext綁定。筆者做了一個簡單的例子。筆者給最外面的Window設置了DataContext值。同時也給他的內部的Grid也設置了DataContext值。但是他們倆個不是同一個對象類型只是屬性相同而以。如下

<Window x:Class="Wpf.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:aomi="http://aomiwpf.com/ModernUI"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:local ="clr-namespace:Wpf"
        Title="MainWindow" Height="350" Width="525">

    <Window.DataContext>
        <local:WindDataContext />
    </Window.DataContext>
    <Grid>
        <Grid.DataContext>
            <local:GridDataContext />
        </Grid.DataContext>
        <TextBlock Text="{Binding  TestName}"></TextBlock>
    </Grid>
</Window>

執行結果:

實驗可以證明標準的綁定方式的目標元素是DataContext。他會去找當前綁定元素最接近的DataContext。我們在來一個假設——如果GridDataContext類裡面屬性TestName換成TestName1的話,又是什麼樣子呢?如下

 1 public class GridDataContext : NotifyPropertyChanged
 2     {
 3         private string _testName1 ="GridDataContext";
 4 
 5         public string TestName1
 6         {
 7             set
 8             {
 9 
10                 if (this._testName1 != value)
11                 {
12                     this._testName1 = value;
13                     OnPropertyChanged("TestName1");
14                 }
15             }
16             get { return this._testName1; }
17         }
18     }

執行結果:

不好意思!筆者以為他會去找Window的DataContext的屬性TestName。顯然他不會。又說明瞭一點,他只會去接近的DataContext裡面找。不會一個直一個的往上面去找。

值得註意的是如果上面只是寫{Binding}的話,那就是把當前的DataContext綁定過來。而不是他的屬性。

在開發過程中,我們往往希望某個元素能跟另一個元素上面的屬性進行綁定。只要另一個元素屬性改變就會通知某個元素一起改變。這個時候就是不得不用下麵的方式來了。

{Binding ElementName=SomeThingName, Path=Text} 

ElementName:表示元素的名稱。

Path:表示元素對象的屬性。

事實上我們可以想到一個問題。綁定是不是只能一方影響一方呢。這就是綁定的裡面要用到的模式。如下

{Binding ElementName=SomeThindName, Path=Text,Mode=TwoWay}

TwoWay:導致對源屬性或目標屬性的更改可自動更新對方。

OneWay: 當綁定源(源)更改時,更新綁定目標(目標)屬性。

OneTime:當應用程式啟動或數據上下文更改時,更新綁定目標。

OneWayToSource:當目標屬性更改時更新源屬性。

以上的用法算是比較常用的。也是比較簡單的。不如讓我們看一下開源項目裡面的一個綁定表達式吧。如下

<Button Command="{Binding Source={x:Static SystemCommands.MinimizeWindowCommand}}" ToolTip="{x:Static modernui:Resources.Minimize}" Style="{StaticResource SystemButton}">
   <Button.Content>
      <Grid Width="13" Height="12" RenderTransform="1,0,0,1,0,1">
         <Path Data="M0,6 L8,6 Z" Width="8" Height="7" VerticalAlignment="Center" HorizontalAlignment="Center"
                  Stroke="{Binding Foreground, RelativeSource={RelativeSource Mode=FindAncestor, AncestorType=Button}}" StrokeThickness="2"  />
      </Grid>
  </Button.Content>
</Button>

不知道大家看得明不明白。上面的意思是從父節點Button的Foreground和當前Path的Stroke綁定在一起。主要的關鍵在AncestorType。用於指定父親的類型。Mode是一個RelativeSourceMode類型。他有四個值。如下。

PreviousData: 用於數據列表,意指以前的數據項。即是數據集合上面的顯示。不包括控制項。

TemplatedParent:用於模板上的綁定。

Self:元素自己本身上的屬性相綁定。

FindAncestor:用於查找父級元素。

只要這樣子一講解就可以理解RelativeSource用於指定相對的源元素。即是目標元素。

事實上,上面的表達式還有一種可能用到的寫法。即是多出了一個用於限制父級的深度(AncestorLevel)。如下

{Binding RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type ItemsControl}, AncestorLevel=2}, Path=Name}

註意:如果想對綁定的值進行修改的話,要用轉化器。如下

{Binding ElementName=SomeThindName, Path=Text,Mode=TwoWay,Converter=XXXConverter}

在開發自定義控制項的時候,我們會經常用到一個表達式。如下

  Width="{TemplateBinding Width}"

上面的寫法只是一種縮寫。完整的如下

 Width="{Binding RelativeSource={RelativeSource TemplatedParent}, Path=Width}"

可以說上面的內容是筆者最常用到的。接下來讓我們在去看一些綁定其他的內容點。即是那些不常見的內容。

1.StringFormat的功能。相當於string.format功能一樣子。舉個列子。如果我們要在金額的前加上“¥”的時候,可以用一下。如下 

 <TextBlock Text="{Binding MoneyText, StringFormat=¥{0}}" />

如果不是這樣子做的話,你就不得不給“¥”一個TextBlock來顯示,或是MoneyText變成string類型,然後設置值裡面加上¥。但是筆者這裡卻是double類型的。所以用StringFormat的功能有就可以完美的決解了顯示“¥”的問題。

執行結果:

2.TargetNullValue的功能,用於綁定目標是一個null值的時候,要顯示的內容。如下筆者給NullName賦null。

 <TextBlock Text="{Binding NullName, TargetNullValue=aomi}" />

執行結果:

3.FallbackValue的功能,用於綁定目標是發生錯誤的時候,要顯示的內容。如下

<TextBlock Text="{Binding NullName, FallbackValue=aomi}" />

執行結果筆者就不貼了。

文章最後。在來說明一個不常用的功能——PriorityBinding。這個功能筆者不好說。只能讓讀者們自行體會吧。他主要用於在載入時間比較多的時候,要顯示的信息。比如顯示“正在載入中...”。筆者做了例子吧。

Xaml:

<Window x:Class="Wpf.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:aomi="http://aomiwpf.com/ModernUI"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:local ="clr-namespace:Wpf"
        Title="MainWindow" Height="350" Width="525">

    <Window.DataContext>
        <local:MainViewModel />
    </Window.DataContext>
    <Grid>
        <TextBlock>
            <TextBlock.Text>
                <PriorityBinding>
                    <Binding Path="UserName" IsAsync="True"></Binding>
                    <Binding Path="LoadingName"></Binding>
                </PriorityBinding>
            </TextBlock.Text>
        </TextBlock>
    </Grid>
</Window>

ViewModel:

 public class MainViewModel : NotifyPropertyChanged
    {
        private string _userName ="Haya";
        private string _loadingName = "正在載入中...";
        public string UserName
        {
            set
            {

                if (this._userName != value)
                {
                    this._userName = value;
                    OnPropertyChanged("UserName");
                }
            }
            get {
                Thread.Sleep(7000);
                return this._userName;
            }
        }

        public string LoadingName
        {
            set
            {

                if (this._loadingName != value)
                {
                    this._loadingName = value;
                    OnPropertyChanged("LoadingName");
                }
            }
            get { return this._loadingName; }
        }
    }

執行結果:

七秒後:

本章的內容比較簡單。筆者只是講述了常用的一些知識點。但是必不是說就這些了。例如Binding還關係到Xml的綁定和集合的綁定功能。讀者們可以自行去找一下資料。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1.需要工具 Instsrv.exe(可以給系統安裝和刪除服務) Srvany.exe(可以讓程式以服務的方式運行) 2.運行cmd,輸入註冊服務命令 "instsrv.exe完整路徑" "服務名稱" "srvany.exe完整路徑" D:\instsrv.exe ObjProjcet D:\srv ...
  • 一、RabbitMQ基本概念和原理 1.AMQP,即Advanced Message Queuing Protocol,高級消息隊列協議,是應用層協議的一個開放標準,為面向消息的中間件設計。 2.RabbitMQ是一個開源的AMQP實現,伺服器端用Erlang語言編寫。 3.Channel Chan ...
  • 正則表達式的本質是使用一系列特殊字元模式,來表示某一類字元串。正則表達式無疑是處理文本最有力的工具,而.NET的System.dll類庫提供的System.Text.RegularExpressions.Regex類實現了驗證正則表達式的方法。Regex 類表示不可變(只讀)的正則表達式。它還包含各 ...
  • 本文原創,轉載請註明出處:http://www.cnblogs.com/AdvancePikachu/p/6944870.html 唉喲,這次厲害咯,網上搜羅了好久,終於被我找到漢化的保存對話框了,根據網上的一些前輩總結的內容,做了一些修改, 先放個效果圖: 首先需要定義一個OpenFileName ...
  • 公司業務量比較大,接了很多項目,為了縮短開發周期老闆讓我牽頭搭建了一個快速開發平臺。 我們主要的業務是做OA、CRM、ERP一類的管理系統,一個通用的後臺搭出來,再配合一些快速開發的組件開發效率能提高很多。 另外老闆一再強調要支持APP開發,一次開發能部署到安卓和iOS上。 作為開篇之作,先介紹一下 ...
  • 對數據的查詢,刪除等基本操作是任何編程語言都會涉及到的基礎,因此,研究了一下C#中比較常用的數據操作類型,並順手做個筆記. List查詢時,若是處理比較大的數據則使用HashSet<T>類,因為List是基於線性表操作的.但其內嵌了二分查找(BinarySearch),因此,也可以在存儲完之後進行排 ...
  • 終於審核通過了,開通了我的博客,雖然這麼多年,已經長期在從事java相關的架構工作,但是,其實我內心,喜歡的語言一直是c#,可惜,很多時候光靠一人的力量,改變不了什麼,現在的我,打算拋開自己的工作,做自己真正喜歡的事情,我開這個博客的目的主要就是為了用c#來開發大型游戲,我的第一個目標,打算使用c# ...
  • 克隆對象在開發過程中經常會遇到,有些時候需要淺克隆,有些時候需要深克隆,具體它們之間有什麼區別,以及實現方式有哪些,在這裡總結一下。 實現深克隆有以下幾種方法。 手動 代碼如下: 反射 代碼如下: 擴展方法: 1 public static class DeepCopyHelper 2 { 3 pu ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...