फोटो ब्राउज़र WPF एप्लिकेशन - प्रदर्शन में सुधार
मैं एक व्यक्तिगत परियोजना के रूप में WPF फोटो ब्रॉसर ऐप पर काम कर रहा हूं। मुझे लगता है कि थम्ब हैंडलिंग के लिए मेरी वर्तमान रणनीति खराब है क्योंकि मैं हर बार पूरी छवि में पढ़ता हूं - यह दिखाता है कि कुछ छवियों के साथ एक फ़ोल्डर ब्राउज़ करते समय। यदि कोई ऐसा बदलाव सुझा सकता है जो कोड के इस पहलू को सुधार सकता है, या कुछ और, तो यह बहुत उपयोगी होगा।
कोड का एक पतला नीचे संस्करण नीचे है। यह एक फ़ोल्डर लेता है और इसके भीतर थंबनेल और छवियों के नाम प्रदर्शित करता है। (पूर्ण ऐप में छवियों पर विभिन्न ऑपरेशन करने के लिए बटन भी हैं।)
MainWindow.xaml.cs
using System.Collections.ObjectModel;
using System.ComponentModel;
using System.IO;
using System.Runtime.CompilerServices;
using System.Windows;
namespace PhotoBrowser
{
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
this.DataContext = new MainWindowViewModel();
}
}
class MainWindowViewModel : INotifyPropertyChanged
{
public event PropertyChangedEventHandler PropertyChanged;
protected void OnPropertyChanged([CallerMemberName] string name = null)
{
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(name));
}
private string _folder;
public string Folder
{
get => _folder;
set
{
_folder = value;
if (Directory.Exists(_folder))
{
var filesInFolder = Directory.EnumerateFiles(_folder);
var files_ = new ObservableCollection<FileItem>();
foreach (string file in filesInFolder)
{
files_.Add(new FileItem(file, false));
}
Files = files_;
}
else
{
Files = new ObservableCollection<FileItem>();
}
OnPropertyChanged();
}
}
private ObservableCollection<FileItem> _files;
public ObservableCollection<FileItem> Files
{
get => _files;
set
{
_files = value;
OnPropertyChanged();
}
}
public MainWindowViewModel() { }
}
}
मेनविंडो। Xaml
<Window x:Class="PhotoBrowser.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:local="clr-namespace:PhotoBrowser"
mc:Ignorable="d"
Title="Photo Browser" Height="800" Width="800" MinHeight="300" MinWidth="400">
<Grid HorizontalAlignment="Stretch" Height="Auto" Margin="10,10,10,10" VerticalAlignment="Stretch" Width="Auto" ShowGridLines="False">
<Grid.RowDefinitions>
<RowDefinition Height="35"/>
<RowDefinition Height="1*" />
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="35"/>
<ColumnDefinition />
</Grid.ColumnDefinitions>
<TextBlock Text="Folder"
Grid.Column="0" HorizontalAlignment="Stretch" VerticalAlignment="Center"/>
<TextBox x:Name="Folder"
Text="{Binding Folder, UpdateSourceTrigger=PropertyChanged}"
Grid.Column="1"
HorizontalAlignment="Stretch" Height="25" TextWrapping="NoWrap" Margin="5,5,5,5"
VerticalAlignment="Center"/>
<ListBox x:Name="FilesInCurrentFolder"
Grid.Column="0" Grid.ColumnSpan="2" Grid.Row="1" HorizontalAlignment="Left" VerticalAlignment="Stretch"
BorderBrush="Black" Height="Auto" Width="772" SelectionMode="Extended"
ItemsSource="{Binding Files, UpdateSourceTrigger=PropertyChanged}" >
<ListBox.ItemTemplate>
<DataTemplate>
<StackPanel Orientation="Horizontal">
<Image Source="{Binding Thumbnail}" Width="100" Height="100"/>
<TextBlock Text="{Binding Name}" VerticalAlignment="Center" />
</StackPanel>
</DataTemplate>
</ListBox.ItemTemplate>
<ListBox.Resources>
<Style TargetType="ListBoxItem">
<Setter Property="IsSelected" Value="{Binding Mode=TwoWay, Path=IsSelected, UpdateSourceTrigger=PropertyChanged}" />
</Style>
</ListBox.Resources>
</ListBox>
</Grid>
</Window>
FileItem.cs
using System;
using System.ComponentModel;
using System.Drawing;
using System.IO;
using System.Runtime.CompilerServices;
using System.Windows.Media.Imaging;
namespace PhotoBrowser
{
public class FileItem : INotifyPropertyChanged
{
public event PropertyChangedEventHandler PropertyChanged;
protected void OnPropertyChanged([CallerMemberName] string name = null)
{
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(name));
}
private string _fileName;
public string FileName
{
get => _fileName;
set
{
_fileName = value;
OnPropertyChanged();
OnPropertyChanged("Name");
}
}
private string _name;
public string Name
{
get => Path.GetFileName(_fileName);
set
{
_name = value;
_fileName = Path.Combine(Path.GetDirectoryName(_fileName), value);
OnPropertyChanged();
OnPropertyChanged("FileName");
}
}
private bool _isSelected;
public bool IsSelected
{
get => _isSelected;
set
{
_isSelected = value;
OnPropertyChanged();
}
}
private BitmapSource _thumbnail;
public BitmapSource Thumbnail
{
get => _thumbnail;
set
{
_thumbnail = value;
OnPropertyChanged();
}
}
public FileItem(string fileName)
{
this.FileName = fileName;
GetThumbnail();
}
public FileItem(string fileName, bool isSelected) : this(fileName)
{
this.IsSelected = isSelected;
}
public void GetThumbnail()
{
Image image = new Bitmap(FileName);
image = image.GetThumbnailImage(100, 100, () => false, IntPtr.Zero);
Thumbnail = new BitmapImage(new Uri(FileName));
}
}
}
जवाब
मुझे लगता है कि आपकी कुछ संपत्ति बसने वाले बहुत अधिक कर रहे हैं। उदाहरण के लिए MainWindow
, Folder
प्रॉपर्टी सेटर सिर्फ फ़ोल्डर का नाम सेट नहीं करता है , बल्कि उस फ़ोल्डर के भीतर सभी फाइलों को एन्यूमरेट और इकट्ठा करने में भी समय लगता है। यह सिंगल रिस्पॉन्सिबिलिटी प्रिंसिपल और शायद लेफ्ट एस्टनमेंट के सिद्धांत के खिलाफ है । मैंने इस तरह के अनुप्रयोगों का उपयोग किया है और निराश हो जाता हूं कि जब मैं करना चाहता था तो एक फ़ोल्डर में नेविगेट किया गया था लेकिन प्रत्येक क्लिक फ़ोल्डर से रीड को मजबूर करने के लिए लगता है। संक्षेप में, एक फ़ोल्डर सेट करना केवल एक फ़ोल्डर सेट करना चाहिए, और उस फ़ोल्डर में फ़ाइलों की गणना करना एक अलग कार्रवाई होनी चाहिए।
प्रदर्शन कुछ async कॉल को लागू करने और somethings स्ट्रीमिंग रखने से लाभ हो सकता है। आप के साथ स्ट्रीम करते हैं EnumerateFiles
लेकिन आप उन्हें सूची में जोड़ते हैं। यह एक प्रदर्शन हिट ले सकता है। मुझे लगता है कि आप तुरंत फ़ाइलों को पढ़ने की जरूरत है, और आप उस पल तक उन्हें पढ़ने में देरी की सटीक सटीक जांच करेंगे।
आपको इससे जुड़े किसी भी async तरीकों को पढ़ना चाहिए BitMap
। मैं इस पोस्ट को कैसे C # के साथ अतुल्यकालिक रूप से एक बिटमैप लोड करने के लिए कैसे खोजें । और आप फ़ाइलों को एन्यूमरेट करने या थंबनेल लोड करने के लिए बैकग्राउंड टास्क का उपयोग करने पर विचार कर सकते हैं।