- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我想按 ID 对指针数组进行排序。但是 qsort 无法工作,因为我缺乏使用指针的经验。
typedef struct block{
int Id;
char * name;
} block;
typedef struct {
block ** data;
int size_array;
} database;
if( ( database = malloc(sizeof(database)) ) == NULL ) {
printf("Error: malloc failed\n");
exit(EXIT_FAILURE);
}
if( ( database->data = malloc( sizeof( block * ) * database->size_array ) ) == NULL ) {
exit(EXIT_FAILURE);
}
for( i = 0; i < database->size_array; i++ ) {
DB->data[i] = NULL;
}
我正在尝试使用 qsort 进行排序,但我显然做错了什么。
int compare(const void * a, const void * b ){
const block * eval1 = a;
const block * eval2 = b;
if (eval1->Id < eval2->Id){
return(-1);
}
else if (eval1->Id > eval2->Id)
return 1;
return 0;
}
调用代码:
qsort(database->data, database->size_array, sizeof(int), compare);
for(i = 0; i<DB->database->size_array; i++) {
if(database->data[i] != NULL)
printf("Id: %d\n", database->data[i]->Id);
}
通过使用 for 循环,我发现排序失败了。
插入函数:
void insertE(int Id, char * name){
block * item = malloc(sizeof(block));
item->name = malloc(strlen(name)+1);
strcpy(item->name, name);
item->Id = Id;
}
当前输出:
Id: 13
Id: 243
Id: 121
Id: 87
.
.
.
最佳答案
当您使用qsort()
(或bsearch()
或其他类似函数)时,传递给您的比较函数的指针属于“指向数组的指针”类型元素类型'。如果传递一个int
的数组,比较函数传递的是int *
(伪装成void *
)。因此,如果您有一个 block *
数组,则传递给比较器的类型是一个 block **
(伪装成 void *
) .
因此你需要这样的东西(尽管也有其他的写法):
int compare(const void *a, const void *b)
{
const block *eval1 = *(block **)a;
const block *eval2 = *(block **)b;
if (eval1->Id < eval2->Id)
return(-1);
else if (eval1->Id > eval2->Id)
return 1;
return 0;
}
但是,还有其他问题。您对 qsort
的调用是可疑的;数组元素的大小是sizeof(db->data[0])
(假设db
是database *
类型的变量),这也是 sizeof(block *)
。通常,特别是在 64 位 Unix 和 Windows 系统上,sizeof(int) != sizeof(block *)
。您可以在 sizeof(int) == sizeof(void *)
平台上使用 sizeof(int)
,该类别包括大多数 32 位系统。因此,您还需要修复对 qsort()
的调用:
database *db = …initialization/allocation…;
qsort(db->data, db->size_array, sizeof(db->data[0]), compare);
任何残留问题很可能与 block 指针数组的填充方式有关。这是一个 MCVE ( Minimal, Complete, Verifiable Example ),它显示了对 block 指针数组的排序。它假定您有可用的 C99 或 C11 编译器 — 它使用 C90 中没有的“复合文字”。
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
typedef struct block
{
int Id;
char *name;
} block;
typedef struct
{
block **data;
int size_array;
} database;
static int compare(const void *a, const void *b)
{
const block *eval1 = *(block **)a;
const block *eval2 = *(block **)b;
if (eval1->Id < eval2->Id)
return(-1);
else if (eval1->Id > eval2->Id)
return 1;
return 0;
}
static void dump_blocks(const char *tag, int nblocks, block * *blocks)
{
printf("%s:\n", tag);
for (int i = 0; i < nblocks; i++)
printf("Id: %3d - %s\n", blocks[i]->Id, blocks[i]->name);
putchar('\n');
}
int main(void)
{
block *b[] =
{
&(block){232, "RDQDLY" },
&(block){347, "XRZDMGJAZ" },
&(block){827, "QBYCVGQ" },
&(block){790, "VXSPDUX" },
&(block){245, "QRZEGGKAHD" },
&(block){717, "YGKRPIGFM" },
&(block){691, "SREIUBHVS" },
&(block){754, "ZCFLESX" },
&(block){868, "WESFFWMJ" },
&(block){291, "QCSAGIHQJ" },
};
database *db = &(database){ &b[0], sizeof(b) / sizeof(b[0]) };
dump_blocks("Before sort", db->size_array, db->data);
qsort(db->data, db->size_array, sizeof(db->data[0]), compare);
dump_blocks("After sort", db->size_array, db->data);
return 0;
}
dump_blocks()
函数遵循我发现非常有用的模式:该函数采用一个字符串标记,该标记首先被打印以标识正在转储的输出集,然后打印所有相关信息来自数据结构(如果合适,使用其他 dump_xxxxx()
函数)。然后可以在多个地方调用它。如果需要,我提供一个 FILE *fp
输出流作为第一个参数;这里似乎没有必要。它还可以在函数末尾使用 fflush(fp);
或 fflush(stdout);
或者 fflush(0);
来确保产生输出。这在调试崩溃的程序时很有帮助。请注意,输出以换行符终止,以帮助确保它们及时出现。
Before sort:
Id: 232 - RDQDLY
Id: 347 - XRZDMGJAZ
Id: 827 - QBYCVGQ
Id: 790 - VXSPDUX
Id: 245 - QRZEGGKAHD
Id: 717 - YGKRPIGFM
Id: 691 - SREIUBHVS
Id: 754 - ZCFLESX
Id: 868 - WESFFWMJ
Id: 291 - QCSAGIHQJ
After sort:
Id: 232 - RDQDLY
Id: 245 - QRZEGGKAHD
Id: 291 - QCSAGIHQJ
Id: 347 - XRZDMGJAZ
Id: 691 - SREIUBHVS
Id: 717 - YGKRPIGFM
Id: 754 - ZCFLESX
Id: 790 - VXSPDUX
Id: 827 - QBYCVGQ
Id: 868 - WESFFWMJ
关于c - 如何对使用结构的指针数组进行排序?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40962375/
我目前正在尝试基于哈希表构建字典。逻辑是:有一个名为 HashTable 的结构,其中包含以下内容: HashFunc HashFunc; PrintFunc PrintEntry; CompareF
如果我有一个指向结构/对象的指针,并且该结构/对象包含另外两个指向其他对象的指针,并且我想删除“包含这两个指针的对象而不破坏它所持有的指针”——我该怎么做这样做吗? 指向对象 A 的指针(包含指向对象
像这样的代码 package main import "fmt" type Hello struct { ID int Raw string } type World []*Hell
我有一个采用以下格式的 CSV: Module, Topic, Sub-topic 它需要能够导入到具有以下格式的 MySQL 数据库中: CREATE TABLE `modules` ( `id
通常我使用类似的东西 copy((uint8_t*)&POD, (uint8_t*)(&POD + 1 ), back_inserter(rawData)); copy((uint8_t*)&PODV
错误 : 联合只能在具有兼容列类型的表上执行。 结构(层:字符串,skyward_number:字符串,skyward_points:字符串)<> 结构(skyward_number:字符串,层:字符
我有一个指向结构的指针数组,我正在尝试使用它们进行 while 循环。我对如何准确初始化它并不完全有信心,但我一直这样做: Entry *newEntry = malloc(sizeof(Entry)
我正在学习 C,我的问题可能很愚蠢,但我很困惑。在这样的函数中: int afunction(somevariables) { if (someconditions)
我现在正在做一项编程作业,我并没有真正完全掌握链接,因为我们还没有涉及它。但是我觉得我需要它来做我想做的事情,因为数组还不够 我创建了一个结构,如下 struct node { float coef;
给定以下代码片段: #include #include #define MAX_SIZE 15 typedef struct{ int touchdowns; int intercepti
struct contact list[3]; int checknullarray() { for(int x=0;x<10;x++) { if(strlen(con
这个问题在这里已经有了答案: 关闭 11 年前。 Possible Duplicate: Empty “for” loop in Facebook ajax what does AJAX call
我刚刚在反射器中浏览了一个文件,并在结构构造函数中看到了这个: this = new Binder.SyntaxNodeOrToken(); 我以前从未见过该术语。有人能解释一下这个赋值在 C# 中的
我经常使用字符串常量,例如: DICT_KEY1 = 'DICT_KEY1' DICT_KEY2 = 'DICT_KEY2' ... 很多时候我不介意实际的文字是什么,只要它们是独一无二的并且对人类读
我是 C 的新手,我不明白为什么下面的代码不起作用: typedef struct{ uint8_t a; uint8_t* b; } test_struct; test_struct
您能否制作一个行为类似于内置类之一的结构,您可以在其中直接分配值而无需调用属性? 前任: RoundedDouble count; count = 5; 而不是使用 RoundedDouble cou
这是我的代码: #include typedef struct { const char *description; float value; int age; } swag
在创建嵌套列表时,我认为 R 具有对列表元素有用的命名结构。我有一个列表列表,并希望应用包含在任何列表中的每个向量的函数。 lapply这样做但随后剥离了列表的命名结构。我该怎么办 lapply嵌套列
我正在做一个用于学习目的的个人组织者,我从来没有使用过 XML,所以我不确定我的解决方案是否是最好的。这是我附带的 XML 文件的基本结构:
我是新来的 nosql概念,所以当我开始学习时 PouchDB ,我找到了这个转换表。我的困惑是,如何PouchDB如果可以说我有多个表,是否意味着我需要创建多个数据库?因为根据我在 pouchdb
我是一名优秀的程序员,十分优秀!