代表性平等的递归解决方案
在这种情况下,我们不是检查数组是否相等(见下文),而是检查表示为字符串然后连接起来的数字是否相等。例如:
[1,2,3,4,5] === [12,34,5] === [123,45]
这个想法是我们减少/折叠每个数组,用表示的字符串作为我们的最终值,将空字符串作为我们的初始值,获取每个整数,将其转换为字符串,并在我们的末尾连接它结果。然后我们发现自己有两个字符串,我们比较它们是否相等。
请注意,我的解决方案存在几个问题。
首先,我没有对字符串操作做任何边界检查,所以下面的代码会愉快地践踏你的记忆。
我们也不跟踪结果的结尾,这意味着每个连接都需要遍历结果字符串,这不是一种有效的方法。
给出的 itoa 实现的一个限制是它对于最大的负整数不能正常工作。
但总体思路保持不变。用一种很好的编程语言,你可以写:
(eq?
(reduce concat "" (map tostr [1,2,3,4,5] ))
(reduce concat "" (map tostr [1,2,3,4,5] )))
或同等学历; 但这是 C,所以我们必须努力做到:
#include <assert.h>
#include <string.h>
#define SAME 0
#define DIFFER 1
void reverse(char s[]);
void itoa(int n, char s[]);
void reduce_to_string(char *result, int values[], int values_len){
char tmpstr[256] = "";
if( values_len == 0 ){ return; }
// convert the first number to a string, writing the representation to tmpstr
itoa(values[0], tmpstr);
// concatenate the first number with the accumulated string
result = strcat(result, tmpstr);
// recur with a smaller array.
return reduce_to_string(result, &values[1], (values_len-1));
}
int compare_representation(int a[], int a_len, int b[], int b_len){
char a_as_string[512] = "";
char b_as_string[512] = "";
reduce_to_string(a_as_string, a, a_len);
reduce_to_string(b_as_string, b, b_len);
if( 0 == strcmp(a_as_string, b_as_string) ){ return SAME;}
return DIFFER;
}
int main(void){
int b[] = {6,7,8,9};
int b_len = 4;
int c[] = {1,2,3,4,5};
int c_len = 5;
int d[] = {67, 89};
int d_len = 2;
assert(SAME == compare_representation(b,b_len,d,d_len));
assert(DIFFER == compare_representation(b,b_len,c,c_len));
return 0;
}
/*
* The following are from K&R C, second edition.
*/
/* reverse: reverse string s in place. page 62 */
void reverse(char s[])
{
int i, j;
char c;
for (i = 0, j = strlen(s)-1; i<j; i++, j--) {
c = s[i];
s[i] = s[j];
s[j] = c;
}
}
/* itoa: convert n to characters in s. page 64*/
void itoa(int n, char s[])
{
int i, sign;
if ((sign = n) < 0) /* record sign */
n = -n; /* make n positive */
i = 0;
do { /* generate digits in reverse order */
s[i++] = n % 10 + '0'; /* get next digit */
} while ((n /= 10) > 0); /* delete it */
if (sign < 0)
s[i++] = '-';
s[i] = '\0';
reverse(s);
}
语法解释:
如果您是 C 新手,使用&a[1]
数组 'a' 的尾部可能不清楚。
(当我说尾巴时,我指的是除第一个之外的所有其他元素。)
将其分解为英语会&a[1]
说:“数组'a'中第二个值的地址”。为什么这行得通?因为在 C 语言中,数组只不过是指向用于存储变量的内存开头的指针,因此获取第二个变量的地址本质上会为您提供一个少一个元素的数组。为了确保我们考虑到数组更小,我们还在将数组长度变量“a_len”传递给下一个函数调用之前减少它。
递归解和迭代解(用于数组相等)
这两个compare_*
函数都在两个整数数组之间进行比较,检查数组是否相同。
#include <assert.h>
#define SAME 0
#define DIFFER 1
int compare_recursive(int a[], int a_len, int b[], int b_len){
if( a_len != b_len ){ return DIFFER; }
if( a_len == 0 && b_len == 0 ){ return SAME; }
if(a[0] == b[0]){
return compare_recursive(&a[1], (a_len-1), &b[1], (b_len-1) );
}else{
return DIFFER;
}
}
int compare_iterative(int a[], int a_len, int b[], int b_len){
int i;
if( a_len != b_len ){ return DIFFER; }
for(i = 0; i < a_len; i++){
if( a[i] != b[i] ){ return DIFFER; }
}
return SAME;
}
int main(void){
int a[] = {1,2,3,4,5};
int a_len = 5;
int b[] = {6,7,8,9};
int b_len = 4;
int c[] = {1,2,3,4,5};
int c_len = 5;
assert(DIFFER == compare_recursive(a,a_len,b,b_len));
assert(SAME == compare_recursive(a,a_len,c,c_len));
assert(DIFFER == compare_iterative(a,a_len,b,b_len));
assert(SAME == compare_iterative(a,a_len,c,c_len));
return 0;
}