我有一个基类O
和两个派生类:A : O
和B : O
,如下所示:
public abstract class O
{
public O(string val)
{
this.Value = val;
}
private string value;
public string Value
{
get { return this.value; }
set
{
this.value = value;
RaiseValueChanged();
}
}
protected delegate void ValueChangedEventHandler();
protected event ValueChangedEventHandler ValueChanged;
protected void RaiseValueChanged()
{ if (ValueChanged != null) ValueChanged(); }
}
public sealed class A : O
{
public A(string val)
: base(val) { }
}
public sealed class B : O
{
public B(string val, A instance)
: base(val)
{
instanceA = instance;
}
private A instanceA;
public A InstanceA
{
get { return instanceA; }
set
{
instanceA = value;
}
}
public void MyMethod()
{
//Some stuff..
}
}
当B.instanceA.ValueChanged
被提出时,我想调用B.MyMethod();
,但当我发现我不能这样做时,我很惊讶:
public sealed class B : O
{
public B(string val, A instance)
: base(val)
{
instanceA = instance;
//instanceA.ValueChanged += MyMethod;
}
private A instanceA;
public A InstanceA
{
get { return instanceA; }
set
{
//instanceA.ValueChanged -= MyMethod;
instanceA = value;
//instanceA.ValueChanged += MyMethod;
}
}
public void MyMethod()
{
//Some stuff..
}
}
根据MSDN:
protected关键字是成员访问修饰符。受保护的成员可在其类内以及派生类实例中访问。
但是,尽管A
和B
都是从O
派生的类,但它们不能使用彼此的基保护成员。
有什么建议可以解决我的问题吗?(而不是使事件public
)
如果O
、A
和B
都在同一个程序集中,如果不希望它是公共的,则可以使用internal
。否则,在这种情况下,您只能访问public
。
当您尝试以这种方式访问基类成员时,您只能通过public
API访问它,而不能通过protected
访问它。您可以通过尝试以下操作亲眼看到这一点:
public sealed class B : O
{
public B(string val, O instance)
: base(val)
{
instanceO = instance;
instanceO.ValueChanged += MyMethod;
}
private O instanceO;
}
这仍然会产生编译错误。您可以在B
中访问ValueChanged
,但只能通过B
的实例访问,如引用中所述:
protected关键字是成员访问修饰符。受保护的成员可在其类内通过派生类实例访问
(重点是我的)。
但是,尽管
A
和B
都是从O
派生的类,但它们不能使用彼此的基保护成员。
这是意料之中的事。
受保护的成员可以在其类内以及通过派生类实例进行访问。
A
派生自O
,而不是B
,因此它不可能看到B
的受保护成员(反之亦然)。
如果你想让这些类能够看到彼此的成员/事件等,那么你需要将它们公开。
您有instanceA作为类B的字段。Protected用于在派生类内部访问,但不能在派生类外部访问。所以只有一个解决方案——使用public。
将其公开并尝试此操作。
这将保证两件事。
首先,事件处理程序附加操作的调用者是从O、派生的类的成员(或静态)方法
其次,作为事件处理程序传递的委托是从O.派生的类的成员(或静态)方法
public delegate void ValueChangedEventHandler();
private event ValueChangedEventHandler valueChanged;
public event ValueChangedEventHandler ValueChanged
{
add
{
if (!typeof(O).IsAssignableFrom(value.Method.DeclaringType))
{
throw new ArgumentException();
}
if (!typeof(O).IsAssignableFrom(new StackTrace().GetFrame(1).GetMethod().DeclaringType))
{
throw new ArgumentException();
}
valueChanged += value;
}
remove
{
valueChanged -= value;
}
}