Chinaunix首页 | 论坛 | 博客
  • 博客访问: 225528
  • 博文数量: 136
  • 博客积分: 2919
  • 博客等级: 少校
  • 技术积分: 1299
  • 用 户 组: 普通用户
  • 注册时间: 2011-03-11 09:08
文章分类

全部博文(136)

文章存档

2013年(1)

2011年(135)

我的朋友

分类: C/C++

2011-03-15 09:19:05

2.3 Libraries

2011-03-10 Thu

2.3.1 Quicksort
  • The standard libraries for C and C++ include sort functions that should be robust against adverse inputs, and tuned to run as fast as possible.
  • Library routines are prepared to son any data type, but in return we must adapt to their interface.
  • In C, the library function is named qsort, and we need to provide a comparison function to be called by qsort whenever it needs to compare two values. Since the values might be of any type, the comparison function is handed two void* pointers to the data items to be compared. The function casts the pointers to the proper type, extracts the data values, compares them, and returns the result (negative, zero or positive according to whether the first value is less than, equal to, or greater than the second).

Here's an implementation for sorting an array of strings, which is a common case. We define a function scmp to cast the arguments and call strcmp to do the comparison.

  1. /* scmp: string compare of *p1 and *p2 */
  2. int scmp (const void *p1, const void *p2)
  3. {
  4.     char *v1, *v2;

  5.     v1 = *(char **)p1;
  6.     v2 = *(char **)p2;
  7.     return strcmp(v1, v2);
  8. }
  • We can't use strcmp directly as the comparison function because qsort passes the address of each entry in the array, &str[i] (of type char**), not str[i] (of type char*).

To sort elements str1 through str[N-1] of an array of strings, qsort must be called with the array, its length, the size of the items being sorted, and the comparison function:

char *str[N];

qsort(str, N, sizeof(str[0]), scmp);

Here's a similar function icmp for comparing integers:

  1. /* icmp: integer compare of *p1 and *p2 */
  2. int icmp(const void *p1, const void *p2)
  3. {
  4.         int v1, v2;

  5.         v1 = *(int *) p1;
  6.         v2 = *(int *) p2;
  7.         if (v1 < v2)
  8.                 return -1;
  9.         else if (v1 == v2)
  10.                 return 0;
  11.         else
  12.                 return 1;
  13. }

Again, the call to qsort requires the array, its length, the size of the items being sorted, and the comparison function:

int arr[N];

qsort(arr, N, sizeof(arr[0]), icmp);
2.3.2 Binary Search

2011-03-11 Fri
ANSI C also defines a binary search routine, bsearch. Like qsort, bsearch requires a pointer to a comparison function (often the same one used for qsort); it returns a pointer to the matching element or NULL if not found. Here is out HTML loopup routine, rewritten to use bsearch:

  1. /* lookupV1: use bsearch to find name in tab, return index */
  2. int lookupV1(char *name, Nameval tab[], int ntab)
  3. {
  4.         Nameval key, *np;

  5.         key.name = name;
  6.         key.value = 0; /* unused; anthing will do */
  7.         np = (Nameval *)bsearch(&key, tab, ntab, sizeof(tab[0]), nvcmp);

  8.         if (np == NULL)
  9.                 return -1;
  10.         else
  11.                 return np-tab;
  12. }

As with qsort, the comparison routine receives the address of the items to be compared, so the key must have that type; in this example, we need to construct a fake Nameeval entry that is passed to the comparison routine itself is a function nvcmp that compares two Nameval items by calling strcmp on their string components, ignoring their values:

  1. /* nvcmp: compare two Nameval names */
  2. int nvcmp(const void *va, const void *vb)
  3. {
  4.         const Nameval *a, *b;
  5.         a = (Nameval *)va;
  6.         b = (Nameval *)vb;
  7.         return strcmp(a->name, b->name);
  8. }

This is analogous to scmp but differs because the strings are stored as members of a structure.

It's a good idea to use bsearch instead of writing your own. Over the years, binary search has proven suprisingly hard for programmers to get right.

The standard C++ library has a generic algorithm called sort that guarantees O(nlog n) behavior.

  1. int arr[N];
  2. sort(arr, arr+N);

The C++ library also has generic binary search routines, with similar notational advantages.

阅读(418) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~