Я не думаю, что тебе defaultdictздесь вообще нужно . Почему бы просто не использовать dict.setdefaultметод?
>>> d = {}
>>> d.setdefault('p', C('p')).v
'p'
Это, конечно, создаст много экземпляров C. Если это проблема, я думаю, подойдет более простой подход:
>>> d = {}
>>> if 'e' not in d: d['e'] = C('e')
defaultdictНасколько я понимаю, это будет быстрее, чем любая другая альтернатива.
ETA относительно скорости inтеста по сравнению с использованием предложения try-except:
>>> def g():
d = {}
if 'a' in d:
return d['a']
>>> timeit.timeit(g)
0.19638929363557622
>>> def f():
d = {}
try:
return d['a']
except KeyError:
return
>>> timeit.timeit(f)
0.6167065411074759
>>> def k():
d = {'a': 2}
if 'a' in d:
return d['a']
>>> timeit.timeit(k)
0.30074866358404506
>>> def p():
d = {'a': 2}
try:
return d['a']
except KeyError:
return
>>> timeit.timeit(p)
0.28588609450770264