У меня была похожая проблема только вчера, когда я пытался протестировать свой собственный класс создания DependencyProperty. Я наткнулся на этот вопрос и заметил, что не было реального решения для отмены регистрации свойств зависимостей. Поэтому я немного покопался, используя Red Gate .NET Reflector , чтобы посмотреть, что я могу придумать.
Глядя на перегрузки DependencyProperty.Register
, все они, похоже, указывают на DependencyProperty.RegisterCommon
. Этот метод состоит из двух частей:
Сначала проверьте, если недвижимость уже зарегистрирована
FromNameKey key = new FromNameKey(name, ownerType);
lock (Synchronized)
{
if (PropertyFromName.Contains(key))
{
throw new ArgumentException(SR.Get("PropertyAlreadyRegistered",
new object[] { name, ownerType.Name }));
}
}
Второе, регистрация свойства DependencyProperty
DependencyProperty dp =
new DependencyProperty(name, propertyType, ownerType,
defaultMetadata, validateValueCallback);
defaultMetadata.Seal(dp, null);
//...Yada yada...
lock (Synchronized)
{
PropertyFromName[key] = dp;
}
Обе части сосредоточены вокруг DependencyProperty.PropertyFromName
, HashTable. Я также заметил DependencyProperty.RegisteredPropertyList
, ItemStructList<DependencyProperty>
, но не видел, где он используется. Однако, ради безопасности, я решил, что постараюсь убрать это, если это возможно.
Итак, я получил следующий код, который позволил мне «отменить регистрацию» свойства зависимости.
private void RemoveDependency(DependencyProperty prop)
{
var registeredPropertyField = typeof(DependencyProperty).
GetField("RegisteredPropertyList", BindingFlags.NonPublic | BindingFlags.Static);
object list = registeredPropertyField.GetValue(null);
var genericMeth = list.GetType().GetMethod("Remove");
try
{
genericMeth.Invoke(list, new[] { prop });
}
catch (TargetInvocationException)
{
Console.WriteLine("Does not exist in list");
}
var propertyFromNameField = typeof(DependencyProperty).
GetField("PropertyFromName", BindingFlags.NonPublic | BindingFlags.Static);
var propertyFromName = (Hashtable)propertyFromNameField.GetValue(null);
object keyToRemove = null;
foreach (DictionaryEntry item in propertyFromName)
{
if (item.Value == prop)
keyToRemove = item.Key;
}
if (keyToRemove != null)
propertyFromName.Remove(keyToRemove);
}
Это сработало достаточно для того, чтобы я смог выполнить свои тесты без исключения «AlreadyRegistered». Тем не менее, я настоятельно рекомендую вам не использовать это в каком-либо производственном коде. Вероятно, есть причина, по которой MSFT решила не использовать формальный способ отмены регистрации свойства зависимости и попыталась пойти против этого. просто просит неприятностей.