Mengurutkan yang dapat diamati dan mengembalikan objek yang sama yang diurutkan dapat dilakukan dengan menggunakan metode ekstensi. Untuk koleksi yang lebih besar hati-hati terhadap jumlah koleksi yang berubah pemberitahuan.
Saya telah memperbarui kode saya untuk meningkatkan kinerja dan menangani duplikat (terima kasih kepada nawfal karena menyoroti kinerja yang buruk dari aslinya meskipun itu bekerja dengan baik pada contoh data asli). Observable dipartisi menjadi setengah yang diurutkan kiri dan setengah kanan yang tidak diurutkan, di mana setiap kali item minimum (seperti yang ditemukan dalam daftar yang diurutkan) digeser ke akhir partisi yang diurutkan dari yang tidak diurutkan. Kasus terburuk O (n). Pada dasarnya semacam seleksi (Lihat di bawah untuk keluaran).
public static void Sort<T>(this ObservableCollection<T> collection)
where T : IComparable<T>, IEquatable<T>
{
List<T> sorted = collection.OrderBy(x => x).ToList();
int ptr = 0;
while (ptr < sorted.Count - 1)
{
if (!collection[ptr].Equals(sorted[ptr]))
{
int idx = search(collection, ptr+1, sorted[ptr]);
collection.Move(idx, ptr);
}
ptr++;
}
}
public static int search<T>(ObservableCollection<T> collection, int startIndex, T other)
{
for (int i = startIndex; i < collection.Count; i++)
{
if (other.Equals(collection[i]))
return i;
}
return -1; // decide how to handle error case
}
penggunaan: Sampel dengan pengamat (menggunakan kelas Person agar tetap sederhana)
public class Person:IComparable<Person>,IEquatable<Person>
{
public string Name { get; set; }
public int Age { get; set; }
public int CompareTo(Person other)
{
if (this.Age == other.Age) return 0;
return this.Age.CompareTo(other.Age);
}
public override string ToString()
{
return Name + " aged " + Age;
}
public bool Equals(Person other)
{
if (this.Name.Equals(other.Name) && this.Age.Equals(other.Age)) return true;
return false;
}
}
static void Main(string[] args)
{
Console.WriteLine("adding items...");
var observable = new ObservableCollection<Person>()
{
new Person {Name = "Katy", Age = 51},
new Person {Name = "Jack", Age = 12},
new Person {Name = "Bob", Age = 13},
new Person {Name = "Alice", Age = 39},
new Person {Name = "John", Age = 14},
new Person {Name = "Mary", Age = 41},
new Person {Name = "Jane", Age = 20},
new Person {Name = "Jim", Age = 39},
new Person {Name = "Sue", Age = 5},
new Person {Name = "Kim", Age = 19}
};
//what do observers see?
observable.CollectionChanged += (sender, e) =>
{
Console.WriteLine(
e.OldItems[0] + " move from " + e.OldStartingIndex + " to " + e.NewStartingIndex);
int i = 0;
foreach (var person in sender as ObservableCollection<Person>)
{
if (i == e.NewStartingIndex)
{
Console.Write("(" + (person as Person).Age + "),");
}
else
{
Console.Write((person as Person).Age + ",");
}
i++;
}
Console.WriteLine();
};
Detail kemajuan pengurutan yang menunjukkan bagaimana koleksi diputar:
Sue aged 5 move from 8 to 0
(5),51,12,13,39,14,41,20,39,19,
Jack aged 12 move from 2 to 1
5,(12),51,13,39,14,41,20,39,19,
Bob aged 13 move from 3 to 2
5,12,(13),51,39,14,41,20,39,19,
John aged 14 move from 5 to 3
5,12,13,(14),51,39,41,20,39,19,
Kim aged 19 move from 9 to 4
5,12,13,14,(19),51,39,41,20,39,
Jane aged 20 move from 8 to 5
5,12,13,14,19,(20),51,39,41,39,
Alice aged 39 move from 7 to 6
5,12,13,14,19,20,(39),51,41,39,
Jim aged 39 move from 9 to 7
5,12,13,14,19,20,39,(39),51,41,
Mary aged 41 move from 9 to 8
5,12,13,14,19,20,39,39,(41),51,
Kelas Person mengimplementasikan IComparable dan IEquatable yang terakhir digunakan untuk meminimalkan perubahan pada koleksi sehingga dapat mengurangi jumlah pemberitahuan perubahan yang dimunculkan
- EDIT Mengurutkan koleksi yang sama tanpa membuat salinan baru *
Untuk mengembalikan ObservableCollection, panggil .ToObservableCollection di * sortOC * menggunakan eg [implementasi ini] [1].
**** orig jawaban - ini membuat koleksi baru **** Anda dapat menggunakan linq seperti yang diilustrasikan oleh metode doSort di bawah ini. Cuplikan kode cepat: menghasilkan
3: xey 6: fty 7: aaa
Alternatifnya, Anda dapat menggunakan metode ekstensi pada koleksi itu sendiri
var sortedOC = _collection.OrderBy(i => i.Key);
private void doSort()
{
ObservableCollection<Pair<ushort, string>> _collection =
new ObservableCollection<Pair<ushort, string>>();
_collection.Add(new Pair<ushort,string>(7,"aaa"));
_collection.Add(new Pair<ushort, string>(3, "xey"));
_collection.Add(new Pair<ushort, string>(6, "fty"));
var sortedOC = from item in _collection
orderby item.Key
select item;
foreach (var i in sortedOC)
{
Debug.WriteLine(i);
}
}
public class Pair<TKey, TValue>
{
private TKey _key;
public TKey Key
{
get { return _key; }
set { _key = value; }
}
private TValue _value;
public TValue Value
{
get { return _value; }
set { _value = value; }
}
public Pair(TKey key, TValue value)
{
_key = key;
_value = value;
}
public override string ToString()
{
return this.Key + ":" + this.Value;
}
}