こんにちは、働くC#プログラマーのさんさめです。
WPFで独自のレイアウトを持ったコントロールを作る場合、
ユーザーコントロールとして作る方法と、
カスタムコントロールとして作る方法があります。
たとえば、こちらの記事や、あちらの記事ではその違いについて書かれており、
パフォーマンスにも言及しています。
ユーザーコントロールの方が遅く、
カスタムコントロールの方が速いそうです。
data:image/s3,"s3://crabby-images/3b30b/3b30b64a0069e9d306945081bfdfc80fadbc0115" alt="さんさめ"
実際どんなもんなんだろう…?
気にするほどなのか…?
このような疑問が出てきたので、
簡単なサンプルを作って実際に試してみました。
結論から言うと、
「シンプルなコントロールの場合、
カスタムコントロールの構築は
ユーザーコントロールの約5倍高速。
ただし、別の場所で処理がかかる」
という結果になりました。
以下、条件や比較に使ったコードなど詳細を記録として残しておきます。
比較に使ったコントロール
まず、比較に使ったコントロールを紹介します。
「ボタンの横に任意のコメントを付けられる」
というとてもシンプルなコントロールです。
data:image/s3,"s3://crabby-images/f110e/f110ead88600b4fa9773fc01fd5f0e1937fd8700" alt=""
xamlでは以下のように書いています。
data:image/s3,"s3://crabby-images/b99f5/b99f5539dd8b04c891d1cfcedd7c33a43bf4b1ed" alt=""
data:image/s3,"s3://crabby-images/3b30b/3b30b64a0069e9d306945081bfdfc80fadbc0115" alt="さんさめ"
前回の記事で、
カスタムコントロールへの移植サンプル
として使ったものですな
これを、ユーザーコントロール版と
カスタムコントロール版、それぞれ用意しました。
ユーザーコントロール版のコード詳細
ユーザーコントロール版のxamlは以下の通りです。
<UserControl x:Class="WpfCustomControlLibrary1.UserButtonWithComment"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="clr-namespace:WpfCustomControlLibrary1">
<DockPanel>
<TextBlock DockPanel.Dock="Right"
Text="{Binding Comment, RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type local:UserButtonWithComment}}}"/>
<Button Content="{Binding ButtonText, RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type local:UserButtonWithComment}}}"
Command="{Binding Command, RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type local:UserButtonWithComment}}}"/>
</DockPanel>
</UserControl>
ビハインドは以下のようになっています。
public partial class UserButtonWithComment : UserControl
{
public UserButtonWithComment()
{
InitializeComponent();
}
public ICommand Command
{
get { return (ICommand)GetValue(CommandProperty); }
set { SetValue(CommandProperty, value); }
}
// Using a DependencyProperty as the backing store for Command. This enables animation, styling, binding, etc...
public static readonly DependencyProperty CommandProperty =
DependencyProperty.Register("Command", typeof(ICommand), typeof(UserButtonWithComment), new PropertyMetadata(null));
public string Comment
{
get { return (string)GetValue(CommentProperty); }
set { SetValue(CommentProperty, value); }
}
// Using a DependencyProperty as the backing store for Comment. This enables animation, styling, binding, etc...
public static readonly DependencyProperty CommentProperty =
DependencyProperty.Register("Comment", typeof(string), typeof(UserButtonWithComment), new PropertyMetadata(string.Empty));
public string ButtonText
{
get { return (string)GetValue(ButtonTextProperty); }
set { SetValue(ButtonTextProperty, value); }
}
// Using a DependencyProperty as the backing store for ButtonText. This enables animation, styling, binding, etc...
public static readonly DependencyProperty ButtonTextProperty =
DependencyProperty.Register("ButtonText", typeof(string), typeof(UserButtonWithComment), new PropertyMetadata(string.Empty));
}
カスタムコントロール版のコード詳細
カスタムコントロール版のxamlは以下です。
独立した専用ファイルというわけではなく、
Generic.xamlにStyleだけ書いてあります。
<ResourceDictionary
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="clr-namespace:WpfCustomControlLibrary1">
<Style TargetType="{x:Type local:CustomButtonWithComment}">
<Setter Property="Template">
<Setter.Value>
<ControlTemplate>
<DockPanel>
<TextBlock DockPanel.Dock="Right"
Text="{Binding Comment, RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type local:CustomButtonWithComment}}}"/>
<Button Content="{Binding ButtonText, RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type local:CustomButtonWithComment}}}"
Command="{Binding Command, RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type local:CustomButtonWithComment}}}"/>
</DockPanel>
</ControlTemplate>
</Setter.Value>
</Setter>
</Style>
</ResourceDictionary>
コードは以下の通りです。
ユーザーコントロール版とほぼ一緒ですね。
public class CustomButtonWithComment : Control
{
static CustomButtonWithComment()
{
DefaultStyleKeyProperty.OverrideMetadata(typeof(CustomButtonWithComment), new FrameworkPropertyMetadata(typeof(CustomButtonWithComment)));
}
public ICommand Command
{
get { return (ICommand)GetValue(CommandProperty); }
set { SetValue(CommandProperty, value); }
}
// Using a DependencyProperty as the backing store for Command. This enables animation, styling, binding, etc...
public static readonly DependencyProperty CommandProperty =
DependencyProperty.Register("Command", typeof(ICommand), typeof(CustomButtonWithComment), new PropertyMetadata(null));
public string Comment
{
get { return (string)GetValue(CommentProperty); }
set { SetValue(CommentProperty, value); }
}
// Using a DependencyProperty as the backing store for Comment. This enables animation, styling, binding, etc...
public static readonly DependencyProperty CommentProperty =
DependencyProperty.Register("Comment", typeof(string), typeof(CustomButtonWithComment), new PropertyMetadata(string.Empty));
public string ButtonText
{
get { return (string)GetValue(ButtonTextProperty); }
set { SetValue(ButtonTextProperty, value); }
}
// Using a DependencyProperty as the backing store for ButtonText. This enables animation, styling, binding, etc...
public static readonly DependencyProperty ButtonTextProperty =
DependencyProperty.Register("ButtonText", typeof(string), typeof(CustomButtonWithComment), new PropertyMetadata(string.Empty));
}
比較コードの詳細
さて、この2つのコントロールをそれぞれ
配置したWindowを用意します。
といっても、1個や2個置いただけでは、
誤差レベルになってしまいそうなので、
どーんと2000個置いてみます。
xamlは以下の通りです…
と、言いたいところですが、
画像で雰囲気だけ感じ取ってください(笑)
まずは、カスタムコントロールの方。
data:image/s3,"s3://crabby-images/84b17/84b178a18d75672c3d8d5a11a10fd2ec1b7113b3" alt=""
data:image/s3,"s3://crabby-images/6b09a/6b09aa2790f0bcdb2fbac8e292247ea675855443" alt=""
次に、ユーザーコントロールの方です。
行番号から異常さが伝わってきます。
data:image/s3,"s3://crabby-images/d555f/d555faebe3cdc02680d6c48fda57da3c25fdfaec" alt=""
data:image/s3,"s3://crabby-images/006d7/006d7c40afa0df193d03f52baa55a75b1c2719c3" alt=""
そして、それぞれのビハインドに次のようなコードを仕込みます。
public MainWindowUser()
{
var sw = new Stopwatch();
sw.Start();
InitializeComponent();
sw.Stop();
this.Title = $"{nameof(MainWindowUser)} - {sw.ElapsedMilliseconds.ToString()} ミリ秒";
}
要は、「InitializeComponentにどれくらいかかるか?」
を計測しているわけです。
これを、App.xaml.csに以下のように記述して
2つのウィンドウを出して比較してみます。
protected override void OnStartup(StartupEventArgs e)
{
base.OnStartup(e);
var view = new Window();
view.Show();
var sw = new Stopwatch();
var customView = new MainWindowCustom();
sw.Start();
customView.Show();
Debug.WriteLine($"Custom Show : {sw.ElapsedMilliseconds}ミリ秒");
var userView = new MainWindowUser();
sw.Restart();
userView.Show();
Debug.WriteLine($"User Show : {sw.ElapsedMilliseconds}ミリ秒");
}
インスタンス生成後の、
WindowのShowにかかる時間を測定しています。
始めにおもむろに空のWindowを
作ってShowしているのには理由があります。
最初、それぞれのMainWindowを作るだけの
コードにしていたのですが、
先に作ったWindowは処理時間が不当に長く
下駄をはかされている感じでした。
初めて生成したWindowに対して、
WPFフレームワーク側が何か行っているのでしょうか?
なんとなく、Application.ShutdownMode
に関連していそう(MainWindowと位置付けている?)な気もしますが、
詳細は調べていません。
さて、実行結果は以下のようになりました。
data:image/s3,"s3://crabby-images/2d06f/2d06fc18d34d0f76242b28d14c5b424921c313b3" alt=""
data:image/s3,"s3://crabby-images/0a4a2/0a4a280564f177f5cdb7a9fde68cccd3a3d9a144" alt=""
本当はたくさんやった方が良いのでしょうが、
正確な比較がしたいわけではなかったので、
数回試して傾向が変わらないことを確認して満足しました。
ユーザーコントロールとカスタムコントロールそれぞれ負荷の考察
InitializeComponentに関しては、
カスタムコントロールの方が5倍ほど高速です。
パフォーマンスプロファイラーで見てみましたが、
「解析」の部分がそれに相当するみたいですね。
data:image/s3,"s3://crabby-images/b551d/b551d52901d49a0db88b6ad2ab073ac40b039a5c" alt=""
data:image/s3,"s3://crabby-images/ef005/ef005dcb52ed457bea8b8b5f0173727da308237c" alt=""
つまり、xamlパースについては、
圧倒的にカスタムコントロールに軍配が上がります。
一方で、Showについては
なぜかユーザーコントロールの方が速いです。
5倍低速だった遅れを完全に取り戻してます。
data:image/s3,"s3://crabby-images/cd954/cd9544c20cc5b505285e299abfaaeb318e6ee341" alt=""
data:image/s3,"s3://crabby-images/34ed5/34ed5e823955dccbee238f11564ff5bf701d966a" alt=""
しかし、やってることは
StackPanelにひたすら平置きしているだけなので、
原因があるとしたら、要素のサイズが
より下側の要素から先に決定しているとかでしょうか。
カスタムコントロールの場合、
レイアウトが定まるのがTemplateが適用されてからなので、
それによりレイアウト再計算が走っているのかもしれません。
だとすると、
レイアウト再計算が走らないような配置方法だったら、
カスタムコントロールの長所を
最大限に生かせるのかもしれません。
data:image/s3,"s3://crabby-images/3b30b/3b30b64a0069e9d306945081bfdfc80fadbc0115" alt="さんさめ"
より詳しく調査が必要そう…
まとめ
まとめです。
- ユーザーコントロールとカスタムコントロールに
パフォーマンスの差があるか調べた - カスタムコントロールの構築は
ユーザーコントロールの約5倍高速 - InitializeComponentは速くなったが、
レイアウト計算が遅くなり結果はトントン - レイアウト計算が走らないような配置なら
利点がありそう
最後までお読みいただき、ありがとうございました。
関連記事
この記事を読んで、
カスタムコントロールとの比較をしてみたくなった方は、
【WPF】ユーザーコントロールをカスタムコントロールに変える手順
を参考にすれば既存ユーザーコントロールの移植ができます。
ぜひ、あなたの知見をいただければと思います。
コメント