使用 bsearch() 获取多个键实例
Getting multiple Instances of key using bsearch()
有没有一种方法可以实现 bsearch()
来查找密钥的多个实例。
例如:(obj*)bsearch(key=r,arr,elements,sizeof(obj),(int(*)(const void*, const void*)bcompare);
我目前编写的代码只能找到第一个实例,并且由于它的工作原理而无法继续超过第一个找到的实例。
getline(target,81);
if(strcmp(target,"exit") == 0 || strcmp(target, "") == 0) break;
p = (Info*)bsearch(target,list,num,sizeof(Info),(int(*)(const void*, const void*))bcompare);
int foundIndex = (int)(p-list);
if(!p){
err_dis1_win();
clrscr();
}
else{
display_record(p);
cout << "\n\n found at index " << foundIndex << "\n";
getch();
clrscr();
}
变量:
- p - 是指向 class Info
对象的指针
- target - char
的 arr
- list - obj
的 arr
- foundIndex - 找到的元素索引
- Info - 从基础 class
派生 class
**比较函数
int bcompare(char *a,Info *b){
return(strncmpi(a, b -> get_name(), strlen(a)));
}
我不能使用其他方法,例如 std::find
或编写我自己的二进制搜索函数,必须使用 bsearch()
我试过在 else 块内循环,比较函数使用变量 foundIndex,以及在 return 值上使用 while 循环遍历对象列表 arr。有没有办法从特定索引开始。我感谢任何帮助。我不是在寻找代码,而是在寻找正确方向的总体推动力。谢谢。
警告 - 当前代码按预期编译和运行,但是,我想要的功能,我自己无法弄清楚。 Google 并在 Whosebug 上搜索未产生相关问题。
由于bsearch()
return只有一项,我将"find multiple instances of key"解释为"find the first instance of a key"。然后调用者可以从该项目向前遍历数组以处理与键匹配的每个项目,直到它到达末尾或到达不匹配的项目。
如果您必须使用标准库的 bsearch()
函数并说服它找到与给定键匹配的第一个项目,那么您真正需要使用的只是您提供的比较函数。 bsearch()
将根据该函数 return 与键匹配的项目,但如果有多个项目匹配,则无法保证将 return 编辑哪一个。那么,您必须确保只有您想要的项目匹配。
您可以通过适当实施比较功能来解决这个问题,但存在一个重大问题。该函数在某些情况下需要评估指定给它的项目之前的项目,但它不得尝试检查数组第一项之前的项目。 bsearch()
本身并不向比较函数传达任何有关数组边界的信息。
至少有两种可能的解决方案,但都不是很好。
- 将数组下界存储在函数可以访问的某个众所周知的位置。例如,如果比较函数是静态成员函数,那么您可能会使用其 class 的静态变量。但这不是线程安全的。您可以对线程局部变量做类似的事情,但即便如此,它也很丑陋。无论哪种方式,您都必须确保在调用
bsearch()
之前适当地设置该变量,这也很丑陋。
或
- 确保您永远不会
bsearch()
第一项。 您可以这样做的一种方法是初步检查第一项是否匹配(但不是通过比较函数),并在匹配的情况下直接使用它而不是调用 bsearch()
。我会把它包装在一个方法中,我自己,如果你不能这样做,那么要求手动使用这样的调用规则也是丑陋的。
选择以上之一后,您可以实现一个比较功能,除了查看指定项的键外,还查看前一项的键。沿着这些线的东西(假设第二种选择):
struct my_item {
int key;
void *data;
};
// bsearch() passes the target item as the first argument, and the one to compare
// to it as the second
int compare_items(const void *to_find, const void *to_check) {
const struct my_item *to_find_item = (const struct my_item *) to_find;
const struct my_item *to_check_item = (const struct my_item *) to_check;
// Check first how the key members are ordered
if (to_find_item->key < to_check_item->key) {
return -1;
} else if (to_find_item->key > to_check_item->key) {
return 1;
} else {
// The key members match, so check whether we're looking at the first
// such item.
const struct my_item *previous_item = to_check_item - 1;
// If the previous item's key does match, then we know the item we're
// looking for is an earlier one than we are presently checking.
return (previous_item->key == to_check_item->key) ? -1 : 0;
}
}
有没有一种方法可以实现 bsearch()
来查找密钥的多个实例。
例如:(obj*)bsearch(key=r,arr,elements,sizeof(obj),(int(*)(const void*, const void*)bcompare);
我目前编写的代码只能找到第一个实例,并且由于它的工作原理而无法继续超过第一个找到的实例。
getline(target,81);
if(strcmp(target,"exit") == 0 || strcmp(target, "") == 0) break;
p = (Info*)bsearch(target,list,num,sizeof(Info),(int(*)(const void*, const void*))bcompare);
int foundIndex = (int)(p-list);
if(!p){
err_dis1_win();
clrscr();
}
else{
display_record(p);
cout << "\n\n found at index " << foundIndex << "\n";
getch();
clrscr();
}
变量:
- p - 是指向 class Info 对象的指针
- target - char 的 arr
- list - obj 的 arr
- foundIndex - 找到的元素索引
- Info - 从基础 class 派生 class
**比较函数
int bcompare(char *a,Info *b){
return(strncmpi(a, b -> get_name(), strlen(a)));
}
我不能使用其他方法,例如 std::find
或编写我自己的二进制搜索函数,必须使用 bsearch()
我试过在 else 块内循环,比较函数使用变量 foundIndex,以及在 return 值上使用 while 循环遍历对象列表 arr。有没有办法从特定索引开始。我感谢任何帮助。我不是在寻找代码,而是在寻找正确方向的总体推动力。谢谢。
警告 - 当前代码按预期编译和运行,但是,我想要的功能,我自己无法弄清楚。 Google 并在 Whosebug 上搜索未产生相关问题。
由于bsearch()
return只有一项,我将"find multiple instances of key"解释为"find the first instance of a key"。然后调用者可以从该项目向前遍历数组以处理与键匹配的每个项目,直到它到达末尾或到达不匹配的项目。
如果您必须使用标准库的 bsearch()
函数并说服它找到与给定键匹配的第一个项目,那么您真正需要使用的只是您提供的比较函数。 bsearch()
将根据该函数 return 与键匹配的项目,但如果有多个项目匹配,则无法保证将 return 编辑哪一个。那么,您必须确保只有您想要的项目匹配。
您可以通过适当实施比较功能来解决这个问题,但存在一个重大问题。该函数在某些情况下需要评估指定给它的项目之前的项目,但它不得尝试检查数组第一项之前的项目。 bsearch()
本身并不向比较函数传达任何有关数组边界的信息。
至少有两种可能的解决方案,但都不是很好。
- 将数组下界存储在函数可以访问的某个众所周知的位置。例如,如果比较函数是静态成员函数,那么您可能会使用其 class 的静态变量。但这不是线程安全的。您可以对线程局部变量做类似的事情,但即便如此,它也很丑陋。无论哪种方式,您都必须确保在调用
bsearch()
之前适当地设置该变量,这也很丑陋。
或
- 确保您永远不会
bsearch()
第一项。 您可以这样做的一种方法是初步检查第一项是否匹配(但不是通过比较函数),并在匹配的情况下直接使用它而不是调用bsearch()
。我会把它包装在一个方法中,我自己,如果你不能这样做,那么要求手动使用这样的调用规则也是丑陋的。
选择以上之一后,您可以实现一个比较功能,除了查看指定项的键外,还查看前一项的键。沿着这些线的东西(假设第二种选择):
struct my_item {
int key;
void *data;
};
// bsearch() passes the target item as the first argument, and the one to compare
// to it as the second
int compare_items(const void *to_find, const void *to_check) {
const struct my_item *to_find_item = (const struct my_item *) to_find;
const struct my_item *to_check_item = (const struct my_item *) to_check;
// Check first how the key members are ordered
if (to_find_item->key < to_check_item->key) {
return -1;
} else if (to_find_item->key > to_check_item->key) {
return 1;
} else {
// The key members match, so check whether we're looking at the first
// such item.
const struct my_item *previous_item = to_check_item - 1;
// If the previous item's key does match, then we know the item we're
// looking for is an earlier one than we are presently checking.
return (previous_item->key == to_check_item->key) ? -1 : 0;
}
}