我有一个结构数组,其中结构具有三个整数字段。它按其中一个字段(例如 F)排序,我想要一种对此字段进行二分搜索的方法,即,形式为 binarySearch(mystruct[] myarray, int val) 的函数,它返回索引其中字段 F = val 的结构的结构。我知道有一个现有的 Array.BinarySearch(T[] array, T value) 函数,但它只允许搜索与数组中的类型相同的类型 T 。这意味着如果我想搜索某个值,我需要创建一个新的结构并将字段 F 设置为该值,以便我可以将其传递给该函数。我不认为会有显着的性能开销,但它看起来很难看。我能想到的另一种方法是自己实现该功能,但是当存在如此类似的东西时,这也显得不优雅。有什么更好的方法或者首选哪种方法的建议吗?
I have an array of structs, where the struct has three integer fields. It is sorted by one of the fields, say F, and I want a way to do a binary search with respect to this field, that is, a function of the form binarySearch(mystruct[] myarray, int val) which returns the index of the structure in which field F = val. I know that there is an existing Array.BinarySearch(T[] array, T value) function, but it only allows to search for a type T that is the same as the types in the array. This means that if I want to search with respect to a value, I need to create a new struct and set field F to that value just so that I can pass it to this function. I don't think there would be significant performance overhead but it seems ugly. The other way I can think is to implement the function myself, but this also seems inelegant when something so similar exists. Any suggestions for a better way or which way would be preferred?
发布评论
评论(3)
您可以为您的结构实现
IComparable
以在字段 (F) 上进行比较,也可以为您的结构创建一个IComparer<>
来进行比较将该字段传递给 Array.BinarySearch() 。因此,
可以这样调用:
或者一个单独的
IComparer
:可以这样调用:
第一个代码较少,但它将排序与类型强烈耦合,如果您想要根据情况改变排序(即允许多个排序顺序),这并不理想。后者提供了更大的灵活性,因为您可以提供独立于结构的多个不同的顺序,但它确实需要一个额外的类。
更新
如果您不想创建一个虚拟结构进行比较,您可以实现
IComparable
,如下所示:可以这样调用:
但是同样,这将类的实现与给定的比较类型紧密联系在一起,我认为这比使用虚拟搜索目标更难看,但这是我个人的偏好。就我个人而言,特别是在初始化语法尽可能短的情况下,我更喜欢使用虚拟目标:
UPDATE:您可以支持
int
和MyStruct
比较,但它很快就会变得混乱,这就是为什么我个人再次建议使用虚拟结构来避免令人头痛的问题:You can either implement
IComparable<T>
for your struct to compare on the field (F), or you can create anIComparer<>
for your struct that will compare based on that field and pass that intoArray.BinarySearch()
.So either:
Which can be called as:
Or a separate
IComparer<MyStruct>
:Which can be called like:
The first has less code, but it strongly couples ordering to the type, which if you want to alter ordering based on situation (that is, allow multiple sort orders), this isn't ideal. The latter gives more flexibility in that you can provide multiple different orders independent of the struct, but it does require an extra class.
UPDATE
If you don't want to create a dummy struct to compare against, you can implement
IComparable
like:Which could be called like:
But again, this strongly ties your implementation of the class to a given comparison type, which I think is uglier than using a dummy search target, but that's my personal preference. Personally, especially with initializer syntax being as short as it is, I prefer to use the dummy target:
UPDATE: You can support both
int
andMyStruct
comparissons, but it gets messy quickly, which is why I personally, again, recommend using the dummy struct to avoid the headaches:如果您的结构实现了 IComparable,您可以使用:
如 http://msdn 中所述。 microsoft.com/en-us/library/y15ef976.aspx
您可以使用 IComparable 将结构与对象进行比较,以便您可以传入值而不是新值结构。在 CompareTo 的实现中,您可以将任何值与字段值进行比较,从而允许您说“我的结构应被视为大于/小于该数字”。
编辑:
这是您的结构的 CompareTo 示例:
If your struct implements IComparable, you can use:
as described in http://msdn.microsoft.com/en-us/library/y15ef976.aspx
You can compare a struct to an object with IComparable, so you can pass in the value intead of a new struct. In your implementation of CompareTo, you can compare any value with the field value, allowing you to say 'My struct should be considered greater/less than this number'.
EDIT:
Here is an example of CompareTo for your struct:
一种方法是创建一个自定义
IComparer
,它仅根据该字段的值比较结构的实例,并将其传递给BinarySearch
的此重载(您还需要创建一个“虚拟”结构实例来比较)。这可能是最纯粹的解决方案。然而,实际上,您可以使用 LINQ 投影到字段值的集合中,并对其进行二分搜索;结果索引将与您搜索结构集合本身相同。例如:
在上面的代码中,
F
是字段的值,42
是您要搜索的值(其类型将是F 的类型) )。这不会那么快,但您不需要编写任何代码,并且速度很可能与您的情况无关。
更新:澄清一下:显然,由于投影操作,上面的代码将是 O(n),因此在投影后使用二分搜索一次是愚蠢的(您可以简单地进行线性搜索)。但是,如果您打算进行多次搜索,那么它可能会开始有意义。
我绝对不建议在您的结构中覆盖
Equals
,除非实例之间的比较旨在减少到应用程序中各处比较F
成员。One way would be to create a custom
IComparer<T>
that compares instances of your struct based only on the value of that field and pass it to this overload ofBinarySearch
(you will also need to create a "dummy" struct instance to compare to). This is probably the purest solution.However, as a practical matter you can use LINQ to project into a collection of field values and binary search into that; the resulting index will be the same as if you had searched the struct collection itself. For example:
In the code above,
F
is the vame of the field and42
is the value you are searching for (its type would be the type ofF
). This is not going to be as fast, but you don't need to write any code and speed could very well be irrelevant in your case.Update: To clarify: obviously, the code above will be O(n) due to the projection operation so using binary search once after projecting like that is silly (you can simply do a linear search instead). However, if you intend to make multiple searches then it might start making sense.
I would definitely not recommend overriding
Equals
in your struct unless comparisons between instances are meant to be reduced to comparingF
members everywhere in your application.