9

RTB でテキストをフォーマットすると、パフォーマンスが低下します。

<Grid>
    <Grid.RowDefinitions>
        <RowDefinition Height="Auto"/>
        <RowDefinition/>
    </Grid.RowDefinitions>
    <StackPanel Orientation="Horizontal">
        <Button Click="ApplyFormatClick">ApplyFormat</Button>
        <TextBlock x:Name="Time"/>
    </StackPanel>

    <RichTextBox x:Name="Rtb" Grid.Row="1">
        <RichTextBox.Document>
            <FlowDocument>
                <Paragraph>
                    <Run>
                        Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum
                    </Run>
                </Paragraph>
            </FlowDocument>
        </RichTextBox.Document>
    </RichTextBox>
</Grid>

コードビハインド:

private readonly SolidColorBrush _blueBrush = Brushes.Blue;
private void ApplyFormatClick(object sender, RoutedEventArgs e)
{
    Stopwatch stopwatch = Stopwatch.StartNew();
    FlowDocument doc = Rtb.Document;
    TextRange range = new TextRange(doc.ContentStart, doc.ContentEnd);
    range.ClearAllProperties();
    int i = 0;
    while (true)
    {
        TextPointer p1 = range.Start.GetPositionAtOffset(i);
        i++;
        TextPointer p2 = range.Start.GetPositionAtOffset(i);
        if (p2 == null)
            break;
        TextRange tempRange = new TextRange(p1, p2);
        tempRange.ApplyPropertyValue(TextElement.ForegroundProperty, _blueBrush);
        tempRange.ApplyPropertyValue(TextElement.FontWeightProperty, FontWeights.Bold);
        i++;
    }
    Time.Text = "Formatting took: " + stopwatch.ElapsedMilliseconds + " ms, number of characters: " + range.Text.Length;
}

フォーマットの適用には 1 秒以上かかり、プロファイリングするときの原因は次のとおりです。

tempRange.ApplyPropertyValue(TextElement.ForegroundProperty, _blueBrush);
tempRange.ApplyPropertyValue(TextElement.FontWeightProperty, FontWeights.Bold);

プロファイラーの結果は、私にはかなり不透明です。

FlowDocument以前にand を使用したことがないRichTextBoxので、おそらくこれは非常に間違っています。

最終結果は、編集可能な正規表現に基づいてテキスト内の一致を強調表示する VS 検索置換に似たものになることを意図しています。

これをスピードアップするために別の方法で何ができますか? ( Github のサンプル)

4

3 に答える 3

10

FlowDocument新しいフォーマット ( MSDN Magazine August 2007: WPF Flexible Content Display With Flow Documents ; または最新の MSDN 記事Flow Document Overviewを確認できます) を使用して手動で構成することをお勧めします。これにより、パフォーマンスが劇的に向上します。たとえば、以下のように手動で実行すると、私のマシンでは 52 ミリ秒で結果が得られますが、使用ApplyPropertyValueすると 1266 ミリ秒かかります。

private readonly SolidColorBrush _blueBrush = Brushes.Blue;

private void ApplyFormatClick(object sender, RoutedEventArgs e)
{
    Stopwatch stopwatch = Stopwatch.StartNew();
    FlowDocument doc = Rtb.Document;
    TextRange range = new TextRange(doc.ContentStart, doc.ContentEnd);
    Paragraph para = new Paragraph();
    string rangetem = range.Text;
    range.ClearAllProperties();

    for(int i=0; i<rangetem.Count();i+=2)
    {
        Span s = new Span() { Foreground = _blueBrush };
        Bold b = new Bold();
        s.Inlines.Add(rangetem[i].ToString());
        b.Inlines.Add(s);
        para.Inlines.Add(b);
        if(i+1<rangetem.Count())
        {
            para.Inlines.Add(rangetem[i + 1].ToString());
        }
    }
    doc.Blocks.Clear();
    doc.Blocks.Add(para);

    Time.Text = "Formatting took: " + stopwatch.ElapsedMilliseconds + " ms, number of characters: " + range.Text.Length;
}
于 2013-07-25T10:01:04.780 に答える
7

RichTextBox から戻る機能が必要ない場合は、IsUndoEnabled を false に設定することで、パフォーマンスをもう少し絞り込むことができます。

<RichTextBox IsUndoEnabled="False">
于 2013-07-26T01:58:22.803 に答える
3

次の方法を使用すると、テキストの書式設定のパフォーマンスを大幅に向上させることができます。

rtb.BeginChange();

/// Your formating logic

rtb.EndChange();
于 2014-01-07T19:43:15.203 に答える