要点:
fgetc
,fgets
,gets
, gets(char* s)
memcpy
、 memmove
qsort
, void qsort(void *base, size_t nitems, size_t size, int (*compar)(const void *, const void *))
基本原理:
从前向后(或从后向前)两两对比并按照目标顺序交换位置。每一趟均可确定未排序序列中最值元素的最终位置并使得下一趟排序中少进行一次比较。对于 n
个元素的序列需要执行 n-1
趟,对比
void bubble_sort(int nums[], int n) {
// 排序共计循环 n-1 趟
for(int trips = 0; trips < n - 1; trips++) {
// 对前n-trips个元素进行冒泡,比较nums[i]和nums[i+1]的大小,i \in [0, n - trips - 2]
for(int i = 0; i < n - trips - 1; i++) {
if(nums[i] > nums[i + 1])
{
int temp = nums[i];
nums[i] = nums[i + 1];
nums[i + 1] = temp;
}
}
}
}
基本原理:
使用分治的基本思想,选择一个元素作为枢轴元素(pivot),通常选择第一个或者最后一个元素,并通过元素交换使得所有左侧元素小于等于枢轴、右侧元素大于等于枢轴;并对枢轴左右两侧序列递归进行快速排序。
具体细节:
// 选择枢轴元素、处理数组并确定枢轴最终位置
int partition(int nums[], int low, int high)
{
int pivot = nums[low];
while(low < high)
{
while(low < high && nums[high] >= pivot) high --;
nums[low] = nums[high];
while(low < high && nums[low] <= pivot) low ++;
nums[high] = nums[low];
}
// 此时 low = high ,且为pivot的最终index
nums[low] = pivot;
return low;
}
void quick_sort(int nums[], int low, int high)
{
if(low < high)
{
// 选择枢轴元素并处理数组,返回枢轴元素最终位置
int pivot = partition(nums, low, high);
// 对枢轴左右未排序序列递归进行快排
quick_sort(nums, low, pivot - 1);
quick_sort(nums, pivot + 1, high);
}
}
int binary_search(int nums[], int n, int target) {
int l_index = 0, r_index = n - 1;
while(l_index <= r_index)
{
int mid = (l_index + r_index) / 2;
if(nums[mid] == target)
return mid;
else if(nums[mid] > target)
r_index = mid - 1;
else
l_index = mid + 1;
}
return -1;
}
// 递归free链表
void destory_list(node_t* list)
{
if(list != NULL)
{
destory_list(list -> next);
free(list);
}
}
辗转相除法是用于求两个非负整数 a
、 b
的最大公约数(gcd、Greatest Common Divisor)的方法,其基本逻辑如下:
注意:
a
大于等于 b
。// a must geq than b.
int gcd(int a, int b)
{
int mod = a % b;
if(mod)
return gcd(b, mod);
else
return b;
}
用二者之中小值乘最大公约数即可。
递归不是很好处理,直接双指针。
typedef struct node
{
int id;
struct node* next;
} node_t;
node_t* reverseList(node_t* list) {
// 窗口中存储上一节点和当前节点
node_t* prev = NULL;
node_t* curr = list;
while(curr)
{
// 缓存下一个要处理的节点
node_t *next = curr -> next;
// 将当前节点拆出,使其等于上一个节点
curr -> next = prev;
// 后移窗口
prev = curr;
curr = next;
}
return prev;
}
基本原理:
每次将一个待排序的记录按关键字大小插入到前面已经排好序的序列中,直到全部记录插入完毕。
即当从左向右、从小到大排序时,已排序序列位于左侧,然后依次将右侧未排序序列中第一个元素插入到左侧已排序序列中。
void insert_sort(int array[], int n)
{
for(int i = 1; i < n; i++)
{
for(int j = i - 1; j >= 0; j--)
{
if(array[j] > array[i])
{
swap(&array[i], &array[i]);
} else {
break;
}
}
}
}
->
和 .
运算符优先级低于解引用运算符 *
,故必须 (*p).num
。p++
优先级高于解引用运算符 *
,故 a = *p++
等价于 a = *(p++)
,又由于延迟自增,故该表达式等价于 a = *p; p++;
(大部分都必背,整理列表只是剔除重复习题)
2023计算机学院:
2018年计算机学院:
\0
问题sizeof
)直接在预处理期完成。做P267的2015年第4题。
下列代码:
for(int i = 0; nums[i] != temp; i++)
{
printf("%d", i);
}
是什么程序结构?使用显示结构语言该如何表示?并标出条件跳转和强制跳转。
答案:
在黄迪明的《C语言程序设计》一书中,程序被划分为三种基本结构:
因此该程序的结构为循环结构,当 nums[i] != temp
时执行循环,当 nums[i] == temp
时结束循环。
其流程图为:
显示表示为:
int i = 0;
L1:
if(nums[i] != temp)
goto L2; // 条件跳转
else
goto L3; // 条件跳转
L2:
printf("%d", i);
goto L1; // 强制跳转
L3:
假设 arr
为整形数组, num
和 item
为整形变量, item
是否在数组中,如果程序片段为:
for(num = N; arr[num] != item; num--)
printf("%d", num);
可能会引发什么异常?
答:
num
从 arr[num] == item
。arr
中不存在 item
,则 num
会不停递减,直到递减到 num = -1
时进行 arr[num] != item
表达式判断时触发异常。给出两个字符串,判断第一个字符串中有几个内容等于第二个字符串的子串。
例如给定字符串A ababaaabbb
,字符串B ab
。
memcmp
函数)int calculate_substr(char *A, char *B)
{
int len_A = strlen(A); //计算字符串A、B的长度
int len_B = strlen(B);
int substr_num = 0; //A的子串数量
if(lenA < len_B)
return substr_num;
for(int i = 0; i < len_A - len_B + 1; i++)
{
if(!memcmp(&A[i], B, len_B)) //指针也可以使用字符串引索
substr_num++;
}
return substr_num;
}
strstr
函数),如果需要简化边界条件则会增加搜索次数(如下)int calculate_substr(char *A, char *B)
{
int *start_pos = A; //字符串A的起始搜索地址
int len_B = strlen(B); //计算字符串B的长度
int substr_num = 0; //A的子串数量
while(start_pos = strstr(start_pos, B))
{
substr_num ++;
start_pos ++; //实际上可以加2,但是会使边界条件复杂化
}
return substr_num;
}
给出一个英语文件作为输入,统计有多少不同单次还有每个单次的数量,要求使用结构体数组完成,可以用 strcmp
函数判断字符串是否已经存在。
数组法:
#define MAX_WORD_LEN 32
#define MAX_WORD_NUM 1024
#include <string.h>
#include <stdio.h>
typedef struct {
char word[MAX_WORD_LEN];
int show_times;
} word_t;
word_t library[MAX_WORD_NUM] = { 0 };
int search_word(char word[])
{
for(int i = 0; i < MAX_WORD_NUM; i++)
{
if(library[i].show_times == 0)
{
return -1;
} else if(!strcmp(library[i].word, word))
{
return i;
}
}
}
int count_word(char word[])
{
int word_id = search_word(word);
if(word_id > 0)
{
library[word_id].show_times++;
return word_id;
} else {
for(int i = 0; i < MAX_WORD_NUM; i++)
{
if(library[i].show_times == 0)
{
strcpy(library[i], word);
library[i].show_times = 1;
return i;
}
}
return -1
}
}
int process_file(char path[])
{
FILE *f = fopen(path, "r");
if(f == NULL)
{
printf("open file: %s failed.\r\n", path);
return -1;
} else {
char word[MAX_WORD_LEN] = { 0 };
while(fgets(word, MAX_WORD_LEN, f)) //fgets函数当遇到换行符时会自动分割并返回word指针,遇到EOF时会返回NULL。
{
count_word(word);
memset(word, 0x00, MAX_WORD_LEN);
}
return 0;
}
}
// C语言的命令行参数中,第一个参数通常为可执行文件的路径,第二个参数开始才是命令行传入的参数
int main(int argc, char argv[])
{
// load English file.
if(!process_file(argv[1]))
return -1;
// A demostration of search a word "hello"
char test_word[] = "hello";
int index = search_word(test_word);
if(index >= 0)
printf("The show times of word %s is %d\r\n", test_word, library[index]);
else
printf("The show times of word %s is 0", test_word);
return 0;
}
结构体中存有学生姓名、学号、分数三个信息。要求:
#include <stdio.h>
typedef struct{
char name[16];
int id;
int score;
} student_t;
// 冒泡排序所用比较函数,值为学生a的分数减去学生b的分数
int compare(const student_t *a, const student_t *b)
{
return a->score - b->score;
}
// 冒泡排序的实现,传入student_t数组和数组大小n,以及比较函数的指针
void bubble_sort(student_t s[], int n, int *cmp(const void* a, const void* b))
{
// 执行n-1趟
for(int trips = 0; trips < n - 1, trips++)
{
// 将元素s[i]和元素s[i+1]做对比,i从0到n - trips - 1
for(int i = 0; i < n - trips - 1; i++)
{
if(cmp(&s[i], &s[i+1]) > 0)
{
student_t temp = s[i]; // 可以直接使用swap(&s[i], &s[i+1])
s[i] = s[i+1];
s[i+1] = temp;
}
}
}
}
// 二分查找,返回值为score的学生的index。但是只能查找到一个为score的学生
int binary_search(student_t s[], int n, int score)
{
int l_index = 0, r_index = n - 1;
while(l_index <= r_index)
{
int mid = (l_index + r_index) / 2;
if(s[mid].score == score)
return mid;
else if(s[mid].score > score)
l_index = mid + 1;
else
r_index = mid - 1;
}
return -1;
}
// 查找所有分数为score的学生,假设caller负责调用free函数
student_t *search_score(student_t s[], int n, int score, int* result_nums)
{
int index = binary_search(s, n, socre);
int l_index = index, r_index = index;
*result_nums = 0;
if(index < 0)
return NULL;
while(l_index > 0 && s[l_index - 1].score == score)
l_index --;
while(r_index < n - 1 && s[r_index + 1].score == score)
r_index ++;
*result_nums = r_index - l_index + 1;
student_t *result = malloc(sizeof(student_t) * (*result_nums));
memcpy(result, &s[l_index], sizeof(student_t) * (*result_nums));
return result;
}
int main()
{
int n = 100; // 给定学生数量
int score = 89; // 目标分数
// 假设题中所给学生数据由create_student_info提供。
student_t *s_list = create_student_info(n);
bubble_sort(s_list, n, compare);
int result_nums = 0;
student_t *result = search_score(s_list, n, score, &result_nums);
for(int i = 0; i < result_nums; i++)
{
printf("name: %s, id: %d, score: %d",
(result + i) -> name, (result + i) -> id, (result + i) -> score);
}
return 0;
}
#include <stdio.h>
#include <time.h>
typedef struct node {
int id;
struct node* next;
} node_t;
int get_rand()
{
return rand()%10;
}
// 传入链表和目标id;使用含头结点的链表
void move_node_to_head(node_t* nodelist, int id)
{
node_t* pre = NULL;
node_t* cur = *nodelist;
// find target
while(cur -> next && cur -> id != id)
{
pre = cur;
cur = cur -> next;
}
// move node to head;
if(cur -> id == id)
{
pre -> next = cur -> next;
cur -> next = nodelist -> next;
nodelist -> next = cur;
return;
} else {
// 移动失败,在合法调用时也不存在该情况。
return;
}
}
node_t* create_list()
{
node_t* head = calloc(sizeof(node_t), 1);
node_t* curr = head;
for(int i = 1; i <= 10; i++)
{
curr -> next = calloc(sizeof(node_t), 1);
curr = curr -> next;
curr -> id = i;
}
return head;
}
// 递归free链表
void destory_list(node_t* list)
{
if(list != NULL)
{
destory_list(list -> next);
free(list);
}
}
int main()
{
//设置随机数种子
srand(time());
int m = abs(rand());
node_t* list = create_list();
for(int i = 0; i < m; i++)
{
move_node_to_head(list, get_rand());
}
destory_list(list);
return 0;
}
是一个冒泡排序的改错。无题干信息。
给定5个数,求这5个数的最小公倍数。
// 辗转相除法求最大公倍数,要求a 必须大于等于 b
int gcd(int a, int b)
{
int mod = a % b;
if(mod)
return gcd(b, mod);
else
return b;
}
// 求两个数的最小公倍数
int lcm(int a, int b)
{
if(a > b)
return b * gcd(a, b);
else
return a * gcd(b, a);
}
// 求数组的最小公倍数,要求n>=2。
int LCM(int array[], int n)
{
if(n < 2)
return -1;
// 先求前两个数的lcm
int current_lcm = lcm(array[0], array[1]);
// 求第三个数开始的lcm
for(int i = 2; i < n; i++)
{
current_lcm = lcm(array[i], current_lcm);
}
return current_lcm;
}
在字符串 a
的指定位置 pos
插入字符串 b
。
#include <string.h>
int insert_str(char a[], const char b[], int pos)
{
// 先计算字符串a、b的长度
int a_len = strlen(a);
int b_len = strlen(b);
// 将a字符串的pos位后的字符串后移,由于目标内存和源内存可能存在内存区域重合,故只可使用 `memmove` 而不可使用 `strcpy` 。
memmove(a + pos + strlen(b) * sizeof(char), a + pos, strlen(b) * sizeof(char));
// 将字符串b拷贝到字符串a的指定位置
memcpy(b, a + pos, strlen(b) * sizeof(char));
return 0;
}
给定三个函数,回答:
函数1:
for(int i = 0; i < 100; i++)
{
a[i] = rand() % 100 + 1;
for(j = 0; j < i; j++)
if(a[i] == a[j])
i--;
}
函数2:
int tag[100] = { 0 };
for(int i = 0; i < 100; i++)
{
a[i] = rand() % 100 + 1;
if(tag[a[i]] == 1)
i--;
else
tag[a[i]] = 1;
}
函数3:
for(int i = 0; i < n; i++)
{
a[i] = i + 1;
for(int j = 0; j < 100; j++)
swap(&a[rand()%100], &a[rand()%100]);
}
函数1:
a[100]
中输出值域为 [1, 100]
的随机数,并保证数组中没有重复元素。函数2:
函数3:
改错题,资料没给题干,故手写一遍。
先排序,再找中位数。
注意:
n = 0
的情况。int compare(int *a, int *b)
return a - b;
float calc_mid_num(int array[], int n)
{
qsort(array, sizeof(int), n, compare);
if(n == 0)
return -1;
if(n % 2)
return array[n/2];
else
return (array[n/2] + array[(n/2) - 1]);
}
回忆版题目均未给空,以下内容均为从零手写
给定结构体 student
,其中存有 score
、 age
、 height
,现在对其按照如下规则进行升序排序:
score
。score
相等时比较 age
。score
和 age
均相等时比较 height
。typedef struct student {
int score;
int age;
int height;
} student_t;
int compare(student_t *a, student_t *b)
{
if(a -> score != b -> score)
{
return a -> score - b -> score;
} else if(a -> age != b -> age)
{
return a -> age - b -> age;
} else {
return a -> height - b -> height;
}
}
int bubble_sort(student_t s[], int n, int (*cmp)(student_t *a, student_t *b))
{
// 比较n-1趟
for(int trips = 0; trips < n - 1; trips ++)
{
// 进行该趟的冒泡排序,将第i号引索元素和其后元素做对比,排序区间为 i \in [0, n - trips - 2]
for(int i = 0; i < n - trips - 1; i ++)
{
if(cmp(&s[i], &s[i + 1]))
swap(&s[i], &s[i + 1])
}
}
return 0;
}
int sort_student(student_t s[], int n)
{
return bubble_sort(s, n, compare);
}
// 删除字符串str的第pos位开始的长度为len的子串,pos从0计数,删除内容包含pos位
char * del_substr(char *str, int pos, int len)
{
int str_len = strlen(str);
if(pos > str_len)
{
return str;
} else if(pos + len >= str_len) {
memset(str, 0x00, str_len - pos);
return str;
} else {
memset(str, 0x00, len);
memmove(str + pos, str + pos + len, str_len - pos - len);
memset(str + str_len - len, 0x00, len);
return str;
}
}
LeetCode 92
给定链表L,输入m,n。实现从链表的第m个节点到低n个节点逆序。
挺难,考虑类数组法。
typedef struct node
{
int id;
struct node* next;
} node_t;
// 无头结点的链表转置
node_t* reverse(node_t * const list)
{
node_t *prev = NULL;
node_t *curr = list;
while(curr) {
// 暂存下一节点
node_t *next = curr -> next;
// 将链表指针方向转置
curr -> next = prev;
// 移动窗口
prev = curr;
curr = next;
}
return prev;
}
// 翻转链表list的第start节点到第end节点的部分,使用带头结点的链表。
// start和end从1开始计算。
node_t* partical_reverse(node_t* const list, int start, int end)
{
// 假设传入链表以及对应的start、end如下:
// | 1 | -> ··· -> | a | -> | b | -> ··· -> | c | -> | d | -> ··· -> | e | -> NULL
// ↑ ↑ ↑
// list start end
// 则定义如下关系的指针:
// | 1 | -> ··· -> | a | -> | b | -> ··· -> | c | -> | d | -> ··· -> | e | -> NULL
// ↑ ↑ ↑ ↑ ↑
// list prev front rear after
node_t *curr = list, *prev = NULL, *front = NULL, *rear = NULL, *after = NULL;
// 随后获取上述指针的值,curr当作遍历指针
int i = 0;
while(curr)
{
// index从1开始
i++;
if(i == start - 1)
prev = curr;
if(i == start)
front = curr;
if(i == end)
rear = curr;
if(i == end + 1)
after = curr;
curr = curr -> next;
}
// 其中 prev 和 after 可能为 NULL 。
// 截断 front 到 rear 的部分。
rear -> next = NULL;
reverse(front);
// 拼回去
front -> next = after;
// 注意prev可能为null
if(prev == NULL)
{
return rear;
} else {
prev -> next = rear;
return list;
}
}
题目不全,假设要求写一个程序分别计算:
double func1(int n)
{
double sum = 0;
for(int i = 1; i <= n; i++)
sum += (2.0 * i - 1.0) / (2.0 * i + 1.0);
return sum;
}
double func2(int n)
{
double sum = 0;
for(int i = 1; i <= n; i++)
sum += (2.0 * i) / (2.0 * i + 1.0);
return sum;
}
给定循环数组 Array[n]
,已知其有序,但不知其向哪有序,也不知队头在哪,求数组的最小值,并要求时间复杂度为
与Leetcode 128类似
程序思路:
求以下代码的输出:
void main()
{
int a[] = { 1, 7, 12, 15 };
int *p1 = a, *p2 = p1++;
*p1 += *p2++;
printf("%d %d", *p1, *p2);
}
注意点:
*p2 = p1++
是先赋值再自增a = *p++
,后置自增优先级比解引用高,等价于条目2a = *(p++)
,先对 p
解引用取值,随后 p
自增a = (*p)++
,对 p
解引用,并对 p
指向的对象自增void main()
{
int a[] = { 1, 7, 12, 15 };
int *p1 = a;
*p2 = p1++; // 此时p1指向a[1],p2指向a[0]
*p1 += *p2++; // a[1] += *p,p++
printf("%d %d", *p1, *p2);
}
判断语句下列代码是否正确:
typedef struct
{
int age;
int score;
} student_t;
int main()
{
student_t stu;
student_t *p = &stu;
*p.age = 23;
return 0;
}
错误,因为 .
运算符取成员的优先级大于 *
解引用的优先级,故应更改为:
(*p).age = 23;
错误点:
strlen(str)
+1\0
要有耐心,算法可能不是普通人写的正常思路算法,不一定为较优解,故要慢慢猜。
#include <stdio.h>
void main()
{
int i, j, k, d, flag1, flag2;
scanf("%d", %d);
printf("d = %d\n", d);
for(i = 1; i <= 100; i++)
{
j = i;
flag1 = 0;
while((j > 0) && (!flag1))
{
k = j % 10;
j = j / 10;
if(k == d)
flag1 = 1;
}
if(flag1)
{
j = i * i;
flag2 = 0;
while((j > 0) && (!flag2))
{
k = j % 10;
j = j / 10;
if(k == d)
flag2 = 1;
}
}
if(flag2)
printf("%-5d %-5d\n", i, i * i);
}
}
char
、short
)参与运算时,会被自动提升为 int
或 unsigned int
,尽可能避免表达式中精度丢失问题。有两种方法:
long var = (long)(float_value + 0.5);
double round(double);
函数,先将float值进行四舍五入转换double的整数,随后利用截断赋值给long。该函数声明位于 math.h
但该函数在部分编译器(如GCC)下不支持。long var = round(float_value);
数组越界是指访问数组时,访问到了该数组以外的内存空间,因此可以分为如下几类:
见2017年题目3.3 找第k大的数。
做P267的2015年第4题。
题目要求:
atoi
函数)实现:
void encrypt(char *str, char *key)
{
int offset = atoi(key); // atoi为stdlib.h中定义的将字符串转为整形的一个方法。
offset %= 256; // 保证offset范围为[0, 255]
while(*str != '\0')
{
if('A' <= *str && *str <= 'Z')
{
// 此时为大写字符
*str = (*str + offset) % 'A';
} else if('a' <= *str && *str <= 'z'){
// 此时为小写字符
*str = (*str + offset) % 'a';
}
str++;
}
}
输入若干个整数,逆序构建双向链表。该整数序列以0结束。
typedef struct Node
{
int value;
struct Node* prev;
struct Node* next;
} Node_t;
int verify(Node_t *node)
{
int node_nums = 0;
while(node)
{
node_nums++;
printf("Index: %d, value: %d\r\n", node_nums, node -> value);
node = node -> next;
}
return node_nums;
}
int main()
{
int value = 0;
Node_t *head = NULL;
Node_t *rear = NULL;
while(1)
{
scanf("%d", &value);
if(value == 0)
break;
if(!head)
{
head = malloc(sizeof(Node_t));
head -> value = value;
head -> prev = NULL;
head -> next = NULL;
rear = head;
} else {
head -> prev = malloc(sizeof(Node_t));
head -> prev -> next = head;
head = head -> prev;
head -> value = value;
head -> prev = NULL;
}
}
int node_nums = verify(head);
printf("node nums: %d\r\n", node_nums);
return 0;
}
同2023简答题1.1
同2023简答1.3
同2015年第4题
求级数
double calculate(int n)
{
// 直接计算第一项
double sum = 1;
// 每一项的阶乘分母
double mul = 1;
for(int i = 1; i <= n; i++)
{
mul *= i;
double t = pow(-1, i) * pow(x, i) / mul;
sum += t;
}
return sum;
}
for(int i = 12; i < 999; i++)
{
t = 0;
for(x = 0; x < 10; x++)
f[x] = 0;
for(j = 345; j < 9999; j++)
{
__________;
s[0] = a;
s[1] = b;
s[2] = c;
for(x = 0; x < 3; x++)
{
y = s[x];
__________ {
int t = y % 10;
f[t] ++;
__________;
}
for(x = 0; x < 10; x++)
{
if(__________)
t++;
}
__________;
__________;
}
}
}
题干:
int low = ___;
int high = ___;
do{
i = low;
_______;
_______;
do{
_______;
_______;
_______;
} while(___);
_______;
if(i == k)
return ___;
if(i > k)
_______;
if(i < k)
_______;
} while(low < high);
return _______;
要点:
答案:
int low = 0;
int high = n - 1;
do{
i = low;
j = high;
pivot = array[i];
do{
while(array[j--] > pivot);
while(array[i++] < pivot);
swap(&array[j], &array[i]);
} while(i < j);
array[i] = array[high];
if(i == k)
return array[i];
if(i > k)
high = i - 1;
if(i < k)
high = i + 1;
} while(low < high);
return a[low];
要点:
题干:
根据下列代码,计算各区域变量的变量名、大小以及偏移量。其中 char
占2字节, int
占4字节,指针占2字节,内存按1字节编址。每个区域相对地址从0开始。
int num = 2;
void main()
{
char str1[10] = { "UESTC" };
char *str2 = "CHENGDU";
static int year = 2015;
char c;
}
void func(int m)
{
int n = 10;
return n;
}
则应当绘制如下表格:
内存区域 | 变量名或常量 | 占用大小 | 偏移量 | 备注(该列非答案) |
---|---|---|---|---|
常量区 | 2 | 4 | 0 | |
10 | 4 | 2 | 数组大小也要计入常量 | |
"UESTC" | 12 | 4 | ||
"CHENGDU" | 16 | 10 | ||
2015 | 4 | 18 | ||
10 | 4 | 20 | ||
全局变量区 | num | 4 | 0 | |
静态变量区 | year | 4 | 0 | |
main函数 | str1 | 20 | 0 | |
str2 | 4 | 10 | ||
c | 2 | 12 | ||
func函数 | m | 4 | 0 | |
n | 4 | 2 |
注意变量存储区域类型: