WPF布朗运动:用线程更新状态

时间:2012-10-31 07:55:02

标签: c# wpf

这是WPF中的一个小布朗运动演示:

using System;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Media;
using System.Windows.Shapes;
using System.Windows.Threading;
using System.Threading;

namespace WpfBrownianMotion
{
    static class MiscUtils
    {
        public static double Clamp(this double n, int low, double high)
        {
            return Math.Min(Math.Max(n, low), high);
        }
    }

    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();

            Width = 500;
            Height = 500;

            var canvas = new Canvas();

            Content = canvas;

            var transform = new TranslateTransform(250, 250);

            var circle = new Ellipse()
            {
                Width = 25,
                Height = 25,
                Fill = Brushes.PowderBlue,
                RenderTransform = transform
            };

            canvas.Children.Add(circle);

            var random = new Random();

            var thread =
                new Thread(
                    () =>
                    {
                        while (true)
                        {
                            Dispatcher.Invoke(
                                DispatcherPriority.Normal,
                                (ThreadStart)delegate()
                                {
                                    transform.X += -1 + random.Next(3);
                                    transform.Y += -1 + random.Next(3);

                                    transform.X = transform.X.Clamp(0, 499);
                                    transform.Y = transform.Y.Clamp(0, 499);
                                });
                        }
                    });

            thread.Start();

            Closed += (s, e) => thread.Abort();
        }
    }
}

我的问题是这个。在没有使用标准WPF动画工具的情况下,上述方法是使用ThreadDispatcher建议的方法吗?

一般来说,我有时会有需要更新和渲染的状态,并且动画设施不适合。所以我需要一种方法在一个单独的线程中进行更新和渲染。只是想知道上述方法是否正确。

1 个答案:

答案 0 :(得分:0)

上述评论中的Clemens建议使用DispatcherTimer。实际上,这确实简化了代码。这是采用这种方法的版本:

using System;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Media;
using System.Windows.Shapes;
using System.Windows.Threading;

namespace WpfBrownianMotion
{
    static class MiscUtils
    {
        public static double Clamp(this double n, int low, double high)
        {
            return Math.Min(Math.Max(n, low), high);
        }
    }

    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();

            Width = 500;
            Height = 500;

            var canvas = new Canvas();

            Content = canvas;

            var transform = new TranslateTransform(250, 250);

            var circle = new Ellipse()
            {
                Width = 25,
                Height = 25,
                Fill = Brushes.PowderBlue,
                RenderTransform = transform
            };

            canvas.Children.Add(circle);

            var random = new Random();

            var timer = new DispatcherTimer();

            timer.Tick += (s, e) =>
                {
                    transform.X += -1 + random.Next(3);
                    transform.Y += -1 + random.Next(3);

                    transform.X = transform.X.Clamp(0, 499);
                    transform.Y = transform.Y.Clamp(0, 499);
                };

            timer.Start();
        }
    }
}