Мне нужно разрешить моему расширению конвейера контента использовать шаблон, похожий на фабрику. Я начинаю со словаря типа:
public delegate T Mapper<T>(MapFactory<T> mf, XElement d);
public class MapFactory<T>
{
Dictionary<string, Mapper<T>> map = new Dictionary<string, Mapper<T>>();
public void Add(string s, Mapper<T> m)
{
map.Add(s, m);
}
public T Get(XElement xe)
{
if (xe == null) throw new ArgumentNullException(
"Invalid document");
var key = xe.Name.ToString();
if (!map.ContainsKey(key)) throw new ArgumentException(
key + " is not a valid key.");
return map[key](this, xe);
}
public IEnumerable<T> GetAll(XElement xe)
{
if (xe == null) throw new ArgumentNullException(
"Invalid document");
foreach (var e in xe.Elements())
{
var val = e.Name.ToString();
if (map.ContainsKey(val))
yield return map[val](this, e);
}
}
}
Вот один тип объекта, который я хочу сохранить:
public partial class TestContent
{
// Test type
public string title;
// Once test if true
public bool once;
// Parameters
public Dictionary<string, object> args;
public TestContent()
{
title = string.Empty;
args = new Dictionary<string, object>();
}
public TestContent(XElement xe)
{
title = xe.Name.ToString();
args = new Dictionary<string, object>();
xe.ParseAttribute("once", once);
}
}
XElement.ParseAttribute - это метод расширения, который работает, как и следовало ожидать. Возвращает логическое значение, которое является истинным в случае успеха.
Проблема в том, что у меня есть много разных типов тестов, каждый из которых заполняет объект уникальным для конкретного теста способом. Имя элемента является ключом к словарю MapFactory. Этот тип теста, хотя и нетипичный, иллюстрирует мою проблему.
public class LogicTest : TestBase
{
string opkey;
List<TestBase> items;
public override bool Test(BehaviorArgs args)
{
if (items == null) return false;
if (items.Count == 0) return false;
bool result = items[0].Test(args);
for (int i = 1; i < items.Count; i++)
{
bool other = items[i].Test(args);
switch (opkey)
{
case "And":
result &= other;
if (!result) return false;
break;
case "Or":
result |= other;
if (result) return true;
break;
case "Xor":
result ^= other;
break;
case "Nand":
result = !(result & other);
break;
case "Nor":
result = !(result | other);
break;
default:
result = false;
break;
}
}
return result;
}
public static TestContent Build(MapFactory<TestContent> mf, XElement xe)
{
var result = new TestContent(xe);
string key = "Or";
xe.GetAttribute("op", key);
result.args.Add("key", key);
var names = mf.GetAll(xe).ToList();
if (names.Count() < 2) throw new ArgumentException(
"LogicTest requires at least two entries.");
result.args.Add("items", names);
return result;
}
}
Мой настоящий код более сложен, так как на фабрике есть два словаря: один превращает XElement в тип контента для записи, а другой используется читателем для создания реальных игровых объектов.
Мне нужно построить эти фабрики в коде, потому что они отображают строки для делегатов. У меня есть служба, которая содержит несколько таких заводов. Миссия состоит в том, чтобы сделать эти фабричные классы доступными для процессора контента. Ни сам процессор, ни контекст, который он использует в качестве параметра, не имеют известных хуков для присоединения IServiceProvider или его эквивалента.
Есть идеи?