所以我试图将java Bag类移植到C#。我知道我在这里做错了,因为在foreach语句中对此代码的测试只产生前两个添加的项目,但这不是我的主要问题。
当我尝试向Bag类添加索引器时,编译器会抛出一个CS0102,抱怨Bag<Item>
已经包含Item
的定义。但是我可以创建一个方法public Item Get(int i)
,它完全可以做同样的事情。为什么会发生这种情况?如何为此类创建索引器?
编辑确切的错误是:Bag.cs(15,15):错误CS0102:类型Algorithms4e.ch1.Bag<Item>' already contains a definition for
项目'(CS0102)
正如旁注,我知道Bag类不应该使用索引器,但这是事物的原理;我应该能够为任何类添加索引器吗?
我在Ubuntu 14.04.2 LTS下运行Mono C#编译器版本3.2.8.0,如果这有帮助的话。
如果您需要更多信息,请告知我们,或者如果我发布此信息,那么这是正确的开始。我很乐意更新这个问题。
public class Bag<Item> : IEnumerable<Item> {
private int N; // number of elements in bag
private Node<Item> first; // beginning of bag
// helper linked list class
private class Node<T> {
public T item;
public Node<T> next;
}
/**
* Initializes an empty bag.
*/
public Bag() {
first = null;
N = 0;
}
/**
* Is this bag empty?
* @return true if this bag is empty; false otherwise
*/
public bool isEmpty() {
return first == null;
}
/**
* Returns the number of items in this bag.
* @return the number of items in this bag
*/
public int size() {
return N;
}
/**
* Adds the item to this bag.
* @param item the item to add to this bag
*/
public void Add(Item item) {
Node<Item> oldfirst = first;
first = new Node<Item>();
first.item = item;
first.next = oldfirst;
N++;
}
public Item Get(int i)
{
return ((ListIterator<Item>)GetEnumerator ())[i];
}
public Item this[int i]
{
get {
return ((ListIterator<Item>)GetEnumerator ())[i];
}
}
/**
* Returns an iterator that iterates over the items in the bag in arbitrary order.
* @return an iterator that iterates over the items in the bag in arbitrary order
*/
public IEnumerator<Item> GetEnumerator() {
return new ListIterator<Item>(first);
}
System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator()
{
return GetEnumerator();
}
// an iterator, doesn't implement remove() since it's optional
private class ListIterator<T> : IEnumerator<T> {
private Node<T> current;
private Node<T> first;
public ListIterator(Node<T> first) {
this.first = first;
current = first;
}
public T GetEnumerator() {
if (!MoveNext ())
throw new Exception ("No such element");
T item = current.item;
//current = current.next;
return item;
}
public T this[int index]
{
get {
Node<T> temp = first;
for (int i = 0; i < index; i++) {
temp = temp.next;
}
return temp.item;
}
}
public T Current
{
get { return current.item; }
}
object IEnumerator.Current
{
get { return Current; }
}
public void Dispose()
{
current = null;
}
public void Reset()
{
current = first;
}
public bool MoveNext()
{
bool res = (current != null);
current = current.next;
return (res);
}
}
}
答案 0 :(得分:1)
如果你想让它变得通用,为什么你有T和Item尝试使它全部T
public class Bag<T> : IEnumerable<T>
答案 1 :(得分:1)
答案 2 :(得分:1)
您正在遇到此问题,因为索引器的默认属性名称为Item
。
如果您已按照naming guidlines for type parameters并命名类型参数TItem
(或仅T
),则不会遇到此问题。
但是如果您确实需要将类型参数调用Item
,则可以使用IndexerName attribute更改索引器属性的名称:
public class Bag<Item> : IEnumerable<Item>
{
...
[IndexerName("MyIndexer")]
public Item this[int i]
{
get
{
return ((ListIterator<Item>)GetEnumerator ())[i];
}
}
...
}