C基础 算法实现层面套路
引言 - 从实践狗讲起
理论到实践(有了算法到实现) 中间有很多过程. 算法方面本人啥也不懂, 只能说说实现方面. 例如下面
一个普通的插入排序.
//
// 插入排序默认从大到小
//
extern void sort_insert_int(int a[], int len) {
int i, j;
for (i = ; i < len; ++i) {
int key = a[j = i];
// 从小到大
while (j > && a[j - ] < key) {
a[j] = a[j - ];
--j;
}
a[j] = key;
}
}
这时候有人就想了, 那数组是 double 的, 那怎么弄了. 也有一种解决方案
#define sort_insert(a, len) \
_Generic(a
, int * : sort_insert_int
, double * : sort_insert_double
, default : sort_insert_default) (a, len)
是不是有所启发. 当然了. 对于上面是使用从大到小封装. 那如果需要从小到大呢. 可以这么做
static inline int _compare_2(const int left, const int key) {
return left - key;
} extern void sort_insert_2(int a[], int len,
int compare(const int left, const int key)) {
int i, j;
for (i = ; i < len; ++i) {
int key = a[j = i];
while (j > && compare(a[j - ], key) < ) {
a[j] = a[j - ];
--j;
}
a[j] = key;
}
}
单独把比较的行为抽象出来, 注册进去. 是不是很开心.
前言 - 细致一点封装
也许到这里会更开心. 既然能通过高科技泛型模拟出来. 那我们不也可以使用旧科技弄弄.
typedef int (* compare_f)(const void * left, const void * key); static inline int _compare_3(const void * left, const void * key) {
return *(int *)left - *(int *)key;
} extern void sort_insert_3_(void * data, size_t ez, int len, compare_f compare) {
char * a = data;
void * key;
int i, j; if ((key = malloc(ez)) == NULL)
return; for (i = ; i < len; ++i) {
memcpy(key, &a[i * ez], ez);
for (j = i; j > && compare(&a[(j - ) * ez], key) < ; --j)
memcpy(&a[j * ez], &a[(j - ) * ez], ez);
if (j != i)
memcpy(&a[j * ez], key, ez);
} free(key);
} #define sort_insert_3(a, len, compare) \
sort_insert_3_(a, sizeof(*(a)), len, (compare_f)compare)
是不是很巧妙, 一切都编程 void * 了. 当然了如果使用 C99 版本以上, 或者说用高版本的 GCC.
可以写的更好.
extern void sort_insert_4_(void * data, size_t ez, int len, compare_f compare) {
char * a = data;
char key[ez];
int i, j; for (i = ; i < len; ++i) {
memcpy(key, &a[i * ez], ez);
for (j = i; j > && compare(&a[(j - ) * ez], key) < ; --j)
memcpy(&a[j * ez], &a[(j - ) * ez], ez);
if (j != i)
memcpy(&a[j * ez], key, ez);
}
}
这里用了 C99 的 VLA 特性. 不知道细心的同学是否和思考. GCC 是怎么实现 VLA 可变长数组呢.
拨开云雾见青天, 我们不妨来个实验验证一哈. 看下面测试代码
#include <stdio.h>
#include <stdlib.h> /*
* file : vla.c
* make : gcc -g -Wall -O2 -o vla.exe vla.c
*
*/
int main(int argc, char * argv[]) {
char a[];
int b = ;
char c[b];
int * d = malloc(sizeof(int));
if (d == NULL)
exit(EXIT_FAILURE);
*d = ;
char e[*d]; printf("%p : char a[10]\n", a);
printf("%p : int b\n", &b);
printf("%p : char c[b]\n", c);
printf("%p : int * d\n", d);
printf("%p : char e[*d]\n", e); free(d);
return EXIT_SUCCESS;
}
最终输出结果是
通过地址匹配对于 vla 可变数组, GCC是放在栈上的. 所有可以预测, 当可变数组大小太大. 函数栈会直接崩溃.
如果你有想法, 那么就去实现它, 多数简单我们还是能独立捉出来滴~~
正文 - 通用套路
还有一种套路, 采用宏模板去实现, 简单提一下这个思路. 看下面代码
#if defined(__T) #define __f(type) sort_insert_##type
#define __F(type) __f(type) static void __F(__T) (__T a[], int len, int compare(const __T left, const __T key)) {
int i, j;
for (i = ; i < (int)len; ++i) {
__T key = a[j = i];
while (j > && compare(a[j - ], key) < ) {
a[j] = a[j - ];
--j;
}
a[j] = key;
}
} #endif
一般而言上面模板函数都会封装在一个局部文件中使用的时候也很方便, 例如下面这样
// 定义部分, 声明和定义分离可以自己搞
#undef __T
#define __T int
#include "sort_insert.c" // 使用部分和普通函数无异
sort_insert_int(a, LEN(a), _compare_2);
当然除了上面一种基于文件的函数模板. 还用一种纯基于函数宏的函数模板实现.
#define sort_insert_definition(T) \
static void sort_insert_##T (T a[], int len, int compare(const T left, const T key)) { \
int i, j; \
for (i = ; i < len; ++i) { \
T key = a[j = i]; \
while (j > && compare(a[j - ], key) < ) { \
a[j] = a[j - ]; \
--j; \
} \
a[j] = key; \
} \
} sort_insert_definition(int)
使用还是一样 sort_insert_int(a, LEN(a), _compare_2); 跑起来. 第一种函数模板, 在嵌入式用的多.
第二种在实战中用的多, 对于处理各种算法相关的代码很普遍. 到这里应该可以理解上面那些
C 封装中一个小函数存在的套路.
后记 - 路越来越窄, 越来越清晰
错误是可以纠正的, 欢迎指正 ~ 表示感谢哈哈
<<啥时候成为津门第一呀>> : http://music.163.com/#/mv?id=197148
对不起 ~ 什么都明白的好晚 ~
C基础 算法实现层面套路的更多相关文章
- PHP基础算法
1.首先来画个菱形玩玩,很多人学C时在书上都画过,咱们用PHP画下,画了一半. 思路:多少行for一次,然后在里面空格和星号for一次. <?php for($i=0;$i<=3;$i++ ...
- 10个经典的C语言面试基础算法及代码
10个经典的C语言面试基础算法及代码作者:码农网 – 小峰 原文地址:http://www.codeceo.com/article/10-c-interview-algorithm.html 算法是一 ...
- Java基础算法集50题
最近因为要准备实习,还有一个蓝桥杯的编程比赛,所以准备加强一下算法这块,然后百度了一下java基础算法,看到的都是那50套题,那就花了差不多三个晚自习的时间吧,大体看了一遍,做了其中的27道题,有一些 ...
- 贝叶斯公式由浅入深大讲解—AI基础算法入门
1 贝叶斯方法 长久以来,人们对一件事情发生或不发生的概率,只有固定的0和1,即要么发生,要么不发生,从来不会去考虑某件事情发生的概率有多大,不发生的概率又是多大.而且概率虽然未知,但最起码是一个确定 ...
- 贝叶斯公式由浅入深大讲解—AI基础算法入门【转】
本文转载自:https://www.cnblogs.com/zhoulujun/p/8893393.html 1 贝叶斯方法 长久以来,人们对一件事情发生或不发生的概率,只有固定的0和1,即要么发生, ...
- java入门学习(3)—循环,选择,基础算法,API概念
1.顺序结构:也就是顺着程序的前后关系,依次执行.2.选择分支:利用if..else , / switch(){case [ 这个必须是常量]:}; / if..else if….. ….else.. ...
- Java - 冒泡排序的基础算法(尚学堂第七章数组)
/** * 冒泡排序的基础算法 */ import java.util.Arrays; public class TestBubbleSort1 { public static void main(S ...
- c/c++面试总结---c语言基础算法总结2
c/c++面试总结---c语言基础算法总结2 算法是程序设计的灵魂,好的程序一定是根据合适的算法编程完成的.所有面试过程中重点在考察应聘者基础算法的掌握程度. 上一篇讲解了5中基础的算法,需要在面试之 ...
- c/c++面试指导---c语言基础算法总结1
c语言基础算法总结 1 初学者学习任何一门编程语言都必须要明确,重点是学习编程方法和编程思路,不是学习语法规则,语法规则是为编程实现提供服务和支持.所以只要认真的掌握了c语言编程方法,在学习其它的语 ...
随机推荐
- 【bzoj1026】[SCOI2009]windy数 数位dp
题目描述 windy定义了一种windy数.不含前导零且相邻两个数字之差至少为2的正整数被称为windy数. windy想知道,在A和B之间,包括A和B,总共有多少个windy数? 输入 包含两个整数 ...
- linq的decimal类型保存到数据库只保存到小数点后两位的问题
今天的一个decimal类型保存到数据的问题困扰了我很长时间,最后就是一个小小的设置问题解决······坑······深坑···· 话不多说,直接说问题,在说答案: 问题:linq当采用EF的DbCo ...
- [洛谷P4345][SHOI2015]超能粒子炮·改
题目大意:给你$n,k$,求:$$\sum\limits_{i=0}^k\binom n i\pmod{2333}$$题解:令$p=2333,f(n,k)\equiv\sum\limits_{i=0} ...
- apply的理解和数组降维
func.apply(thisObj, [argArray] ); apply方法用来改变函数执行时的this指向,后面的参数是一个类数组对象,可以是数组,arguments,甚至一个有length属 ...
- AtCoder Regular Contest 081 E - Don't Be a Subsequence(字符串DP)
引用自:onion_cyc 字符串DP一直不是强项...以后没思路的题就想DP和网络流23333333 f[i]表示从i开始的后缀非子序列的最短长度 pos[i][j]表示从i开始的j字符最早出现位 ...
- jquery 操作实例一
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"% ...
- 一个完整的upstart脚本分析
基本概念可以了解 http://www.mike.org.cn/articles/understand-upstart/ http://blog.fens.me/linux-upstart/ http ...
- FreeRTOS - 程序开发阶段建议
1.创建任务.定时器等都需要耗用分配给FreeRTOS的heap,由于RAM有限,分配作为FreeRTOS的heap量有限,一不小心就不够用了,所以应该有检测任务.定时器等是否创建成功,如下图: 2. ...
- MongoDB入门(1)- MongoDB简介
什么是MongoDB NoSQL NoSQL systems are also sometimes called "Not only SQL" to emphasize that ...
- LightOJ 1284 - Lights inside 3D Grid 概率/期望/二项式定理
题意:给你一个长宽高为x,y,z的长方体,里面每个格子放了灯,再给你k次选取任意长方体形状的区块,对其内所有灯开或关操作,初始为关,问亮灯数量的期望值. 题解:首先考虑选取区块的概率,使某个灯在被选取 ...