From 05050c4414823135194fde85f2a45aef0e756ca8 Mon Sep 17 00:00:00 2001 From: Nguyen Ba Ngoc Date: Sun, 14 Jan 2024 12:27:01 +0700 Subject: [PATCH] demo --- algo/algo.h | 10 +- all/cgen.h | 10 +- tests/cgen/topk_demo.c | 1014 +--------------------------------------- 3 files changed, 14 insertions(+), 1020 deletions(-) diff --git a/algo/algo.h b/algo/algo.h index 3700910..f54fceb 100644 --- a/algo/algo.h +++ b/algo/algo.h @@ -103,11 +103,11 @@ static void heap_shift_down(void *a, int n, int sz, int i, compare_fnt cmp) { } static void heap_shift_up(void *a, int n, int sz, int i, compare_fnt cmp) { - int j = HTOP(i); - while (i > 0 && cmp(a + j * sz, a + i * sz) < 0) { - vswap(a + i * sz, a + j * sz, sz); - i = j; - j = HTOP(i); + int top = HTOP(i); + while (i > 0 && cmp(a + top * sz, a + i * sz) < 0) { + vswap(a + i * sz, a + top * sz, sz); + i = top; + top = HTOP(top); } } diff --git a/all/cgen.h b/all/cgen.h index 261eb74..5a85ff8 100644 --- a/all/cgen.h +++ b/all/cgen.h @@ -166,11 +166,11 @@ static void heap_shift_down(void *a, int n, int sz, int i, compare_fnt cmp) { } } static void heap_shift_up(void *a, int n, int sz, int i, compare_fnt cmp) { - int j = HTOP(i); - while (i > 0 && cmp(a + j * sz, a + i * sz) < 0) { - vswap(a + i * sz, a + j * sz, sz); - i = j; - j = HTOP(i); + int top = HTOP(i); + while (i > 0 && cmp(a + top * sz, a + i * sz) < 0) { + vswap(a + i * sz, a + top * sz, sz); + i = top; + top = HTOP(top); } } static void heap_make(void *a, int n, int sz, compare_fnt cmp) { diff --git a/tests/cgen/topk_demo.c b/tests/cgen/topk_demo.c index b66b341..ec0a26a 100644 --- a/tests/cgen/topk_demo.c +++ b/tests/cgen/topk_demo.c @@ -1,1010 +1,4 @@ -/** (C) Nguyễn Bá Ngọc 2023 - * cgen version:3.6.9 - */ -#ifndef CGEN_H_ -#define CGEN_H_ - -/***** ./base/flog.h *****/ -#ifndef BASE_FLOG_H_ -#define BASE_FLOG_H_ -#include -#include -#include -#define FLOG_VERSION "1.0.0"; -#ifdef CGEN_DEBUG - #define FLOG(...) _flog(__FILE__, __LINE__, __VA_ARGS__) -#else - #define FLOG(...) -#endif -static inline void _flog(const char *file, int line, const char *fmt, ...) { - const char *filename = (strrchr(file, '/') ? strrchr(file, '/') + 1 : file); - fprintf(stderr, "%s:%d: ", filename, line); - va_list args; - va_start(args, fmt); - vfprintf(stderr, fmt, args); - va_end(args); - fprintf(stderr, "\n"); - fflush(stderr); -} -#endif - -/***** ./base/fnt.h *****/ -#ifndef BASE_FNT_H_ -#define BASE_FNT_H_ -#include -#include -typedef int (*compare_fnt)(const void *p1, const void *p2); -static inline int cmpi(const void *p1, const void *p2) { - return *(const int*)p1 - *(const int*)p2; -} -static inline int rcmpi(const void *p1, const void *p2) { - return *(const int*)p2 - *(const int*)p1; -} -static inline int cmpd(const void *p1, const void *p2) { - double v1 = *(const double *)p1, v2 = *(const double *)p2; - if (v1 < v2) { - return - 1; - } else if (v1 > v2) { - return 1; - } - return 0; -} -static inline int rcmpd(const void *p1, const void *p2) { - double v2 = *(const double *)p1, v1 = *(const double *)p2; - if (v1 < v2) { - return - 1; - } else if (v1 > v2) { - return 1; - } - return 0; -} -static inline int cmps(const void *p1, const void *p2) { - char *const *s1 = p1, *const *s2 = p2; - return strcmp(*s1, *s2); -} -static inline int rcmps(const void *p1, const void *p2) { - char *const *s2 = p1, *const *s1 = p2; - return strcmp(*s1, *s2); -} -static inline void frees(char *s) { - free(s); -} -#endif - -/***** ./algo/algo.h *****/ -#ifndef ALGO_ALGO_H_ -#define ALGO_ALGO_H_ -#include -static inline void vswap(void *o1, void *o2, int sz) { - char tmp, *p = o1, *q = o2; - for (int i = 0; i < sz; ++i) { - tmp = *p; - *p = *q; - *q = tmp; - ++p; - ++q; - } -} -static void q2sort(void *a, int n, int sz, compare_fnt cmp) { - if (n <= 1) { - return; - } - void *left = a, *right = a + (n - 2) * sz; - void *v = a + (n - 1) * sz; - for (;;) { - while (cmp(v, left) > 0) { - left += sz; - } - while (cmp(right, v) > 0) { - if (right == a) { - break; - } - right -= sz; - } - if (left >= right) { - break; - } - vswap(left, right, sz); - left += sz; - right -= sz; - } - vswap(left, v, sz); - right = left + sz; - left = left - sz; - q2sort(a, (left - a)/sz + 1, sz, cmp); - q2sort(right, (a - right)/sz + n, sz, cmp); -} -static int binsearch(const void *a, int n, int sz, const void *v, compare_fnt cmp) { - int l = 0, r = n - 1; - while (l <= r) { - int m = (l + r) / 2; - int x = cmp(a + m * sz, v); - if (x == 0) { - return m; - } - if (x < 0) { - l = m + 1; - } else if (x > 0) { - r = m - 1; - } - } - return -1; -} -#define HTOP(i) (((i) - 1) >> 1) -#define HLEFT(i) (((i) << 1) + 1) -#define HRIGHT(i) (((i) << 1) + 2) -static void heap_shift_down(void *a, int n, int sz, int i, compare_fnt cmp) { - for (;;) { - int lc = HLEFT(i), rc = HRIGHT(i), top = i; - if (lc < n && cmp(a + top * sz, a + lc * sz) < 0) { - top = lc; - } - if (rc < n && cmp(a + top * sz, a + rc * sz) < 0) { - top = rc; - } - if (top == i) { - break; - } - vswap(a + i * sz, a + top * sz, sz); - i = top; - } -} -static void heap_shift_up(void *a, int n, int sz, int i, compare_fnt cmp) { - int j = HTOP(i); - while (i > 0 && cmp(a + j * sz, a + i * sz) < 0) { - vswap(a + i * sz, a + j * sz, sz); - i = j; - j = HTOP(i); - } -} -static void heap_make(void *a, int n, int sz, compare_fnt cmp) { - for (int i = n / 2; i >= 0; --i) { - heap_shift_down(a, n, sz, i, cmp); - } -} -static int *first_perm(int n) { - int *a = malloc((n + 1) * sizeof(int)); - a[0] = n; - ++a; - for (int i = 0; i < n; ++i) { - a[i] = i; - } - return a; -} -static int next_perm(int *a) { - int n = a[-1], k = n -1; - while (k > 0) { - if (a[k - 1] < a[k]) { - break; - } - --k; - } - if (k == 0) { - return 0; - } - int i = k - 1, l = n - 1; - while (a[l] < a[i]) { - --l; - } - int tmp = a[l]; - a[l] = a[i]; - a[i] = tmp; - l = n - 1; - while (k < l) { - tmp = a[k]; - a[k] = a[l]; - a[l] = tmp; - ++k; - --l; - } - return 1; -} -static int has_next_perm(int *a) { - int n = a[-1], k = n -1; - while (k > 0) { - if (a[k - 1] < a[k]) { - return 1; - } - --k; - } - return 0; -} -static void free_perm(int *a) { - free(a - 1); -} -static int *first_comb(int k, int n) { - if (k > n || k < 1) { - return NULL; - } - int *a = malloc((k + 2) * sizeof(int)); - a[0] = k; - a[1] = n; - a += 2; - for (int i = 0; i < k; ++i) { - a[i] = i; - } - return a; -} -static int next_comb(int *a) { - int k = a[-2], n = a[-1], r = k - 1; - while (r >= 0) { - if (a[r] < n - k + r) { - ++a[r]; - for (int i = r + 1; i < k; ++i) { - a[i] = a[i - 1] + 1; - } - return 1; - } - --r; - } - return 0; -} -static int has_next_comb(int *a) { - int k = a[-2], n = a[-1], r = k - 1; - while (r >= 0) { - if (a[r] < n - k + r) { - return 1; - } - --r; - } - return 0; -} -static void free_comb(int *a) { - free(a - 2); -} -#endif - -/***** ./cont/slist.h *****/ -#ifndef CONT_SLIST_H_ -#define CONT_SLIST_H_ -#define SDEFN(sname,dtype) \ -struct sname##_node { \ - dtype data; \ - struct sname##_node *next; \ -}; \ -struct sname { \ - struct sname##_node *front; \ - struct sname##_node *back; \ - void (*fv)(dtype); \ - long size; \ -} -#define SDECL(sname,dtype,prefix) \ -SDEFN(sname, dtype); \ -struct sname *prefix##create(); \ -struct sname *sname(); \ -void prefix##clear(struct sname *list); \ -void prefix##free(struct sname *list); \ -struct sname *prefix##append(struct sname *list, dtype data); \ -struct sname *prefix##prepend(struct sname *list, dtype data); \ -struct sname *prefix##dfront(struct sname *list); \ -struct sname *prefix##push(struct sname *list, dtype elem); \ -dtype *prefix##top(struct sname *list); \ -struct sname *prefix##pop(struct sname *list); \ -struct sname *prefix##enque(struct sname *list, dtype elem); \ -dtype *prefix##peek(struct sname *list); \ -struct sname *prefix##deque(struct sname *list); \ -int prefix##empty(struct sname *list) -#define SIMPL(sname,dtype,prefix) \ -struct sname##_node *sname##_node(dtype data) { \ - struct sname##_node *tmp = malloc(sizeof(struct sname##_node)); \ - if (!tmp) { \ - FLOG("Lỗi cấp phát bộ nhớ tạo nút."); \ - return NULL; \ - } \ - tmp->data = data; \ - tmp->next = NULL; \ - return tmp; \ -} \ -struct sname *prefix##create() { \ - struct sname *tmp = malloc(sizeof(struct sname)); \ - if (!tmp) { \ - FLOG("Lỗi cấp phát bộ nhớ tạo danh sách."); \ - return NULL; \ - } \ - tmp->front = tmp->back = NULL; \ - tmp->fv = NULL; \ - tmp->size = 0; \ - return tmp; \ -} \ -struct sname *sname() { \ - return prefix##create(); \ -} \ -void prefix##clear(struct sname *list) { \ - while (!prefix##empty(list)) { \ - prefix##dfront(list); \ - } \ -} \ -void prefix##free(struct sname *list) { \ - prefix##clear(list); \ - free(list); \ -} \ -struct sname *prefix##append(struct sname *list, dtype data) {\ - struct sname##_node *node = sname##_node(data); \ - if (!node) { \ - FLOG("Lỗi tạo nút"); \ - return NULL; \ - } \ - if (list->front == NULL) { \ - list->front = list->back = node; \ - } else { \ - list->back->next = node; \ - list->back = node; \ - } \ - ++list->size; \ - return list; \ -} \ -struct sname *prefix##prepend(struct sname *list, dtype data) {\ - struct sname##_node *node = sname##_node(data); \ - if (!node) { \ - FLOG("Lỗi tạo nút"); \ - return NULL; \ - } \ - if (list->front == NULL) { \ - list->front = list->back = node; \ - } else { \ - node->next = list->front; \ - list->front = node; \ - } \ - ++list->size; \ - return list; \ -} \ -struct sname *prefix##dfront(struct sname *list) {\ - if (!list || prefix##empty(list)) { \ - FLOG("Xóa đầu danh sách không hợp lệ."); \ - return NULL; \ - } \ - if (list->front == list->back) { \ - list->back = NULL; \ - } \ - struct sname##_node *tmp = list->front; \ - list->front = tmp->next; \ - if (list->fv) { \ - list->fv(tmp->data); \ - } \ - free(tmp); \ - --list->size; \ - return list; \ -} \ -struct sname *prefix##push(struct sname *list, dtype elem) { \ - return prefix##prepend(list, elem); \ -} \ -dtype *prefix##top(struct sname *list) { \ - return &list->front->data; \ -} \ -struct sname *prefix##pop(struct sname *list) {\ - return prefix##dfront(list); \ -} \ -struct sname *prefix##enque(struct sname *list, dtype elem) { \ - return prefix##append(list, elem); \ -} \ -dtype *prefix##peek(struct sname *list) {\ - return &list->front->data; \ -} \ -struct sname *prefix##deque(struct sname *list) { \ - return prefix##dfront(list); \ -} \ -int prefix##empty(struct sname *list) { \ - return list->front == NULL || list->back == NULL; \ -} -#define SDECL_IMPL(sname,dtype,prefix) \ -SDECL(sname, dtype, prefix); \ -SIMPL(sname, dtype, prefix) -#endif - -/***** ./cont/tmap.h *****/ -#ifndef CONT_TMAP_H_ -#define CONT_TMAP_H_ -#include -enum tcolors { - RED = 0, - BLACK = 1 -}; -#define TNN(tname) tname ##_node -#define TDEFN(tname,ktype,vtype) \ -struct TNN(tname) { \ - ktype key; \ - vtype value; \ - enum tcolors color; \ - struct TNN(tname) *left; \ - struct TNN(tname) *right; \ - struct TNN(tname) *top; \ -}; \ -struct tname { \ - struct TNN(tname) *root; \ - compare_fnt cmp; \ - void (*fk)(ktype);\ - void (*fv)(vtype); \ - long size; \ -} -#define TCOLOR(n) ((n)? (n)->color: BLACK) -#define TIS_RED(n) ((n) && (n)->color == RED) -#define TIS_BLACK(n) (!(n) || (n)->color == BLACK) -#define TPAINT_BLACK(n) (n)->color = BLACK -#define TPAINT_RED(n) (n)->color = RED -#define TPAINT(n,c) (n)->color = (c) -#define TDECL(tname,ktype,vtype,prefix) \ -TDEFN(tname, ktype, vtype); \ -struct TNN(tname) *prefix##left_most(struct TNN(tname) *n); \ -struct TNN(tname) *prefix##right_most(struct TNN(tname) *n); \ -struct TNN(tname) *prefix##left_deepest(struct TNN(tname) *n); \ -struct TNN(tname) *prefix##next_lrn(struct TNN(tname) *n); \ -struct TNN(tname) *prefix##next_lnr(struct TNN(tname) *n); \ -struct TNN(tname) *prefix##prev_lnr(struct TNN(tname) *n); \ -struct TNN(tname) *prefix##first_lrn(struct tname *tm); \ -struct TNN(tname) *prefix##first_lnr(struct tname *tm); \ -struct TNN(tname) *prefix##last_lnr(struct tname *tm); \ -struct TNN(tname) *prefix##pval_node(void *pv); \ -struct tname *prefix##create(compare_fnt cmp); \ -struct tname *tname(compare_fnt cmp); \ -void prefix##clear(struct tname *tm); \ -void prefix##free(struct tname *tm); \ -vtype *prefix##put(struct tname *tm, ktype key, vtype value); \ -vtype *prefix##get(struct tname *tm, ktype key); \ -struct tname *prefix##remove(struct tname *tm, ktype key) -#define TIMPL(tname,ktype,vtype,prefix) \ -static inline void prefix##change(struct tname *tm, struct TNN(tname) *old_node, \ - struct TNN(tname) *new_node) { \ - struct TNN(tname) *top = old_node->top; \ - if (top) { \ - if (top->left == old_node) { \ - top->left = new_node; \ - } else { \ - top->right = new_node; \ - } \ - } else { \ - tm->root = new_node; \ - } \ - if (new_node) { \ - (new_node)->top = top; \ - } \ -} \ - \ -static inline void prefix##rotate_left(struct tname *tm, struct TNN(tname) *x) { \ - struct TNN(tname) *y = x->right; \ - x->right = y->left; \ - if (y->left != NULL) { \ - y->left->top = x; \ - } \ - prefix##change(tm, x, y); \ - y->left = x; \ - x->top = y; \ -}\ - \ -static inline void prefix##rotate_right(struct tname *tm, struct TNN(tname) *x) { \ - struct TNN(tname) *y = x->left; \ - x->left = y->right; \ - if (y->right != NULL) { \ - y->right->top = x; \ - } \ - prefix##change(tm, x, y); \ - y->right = x; \ - x->top = y; \ -}\ -static inline void prefix##put_fixup(struct tname *tm, struct TNN(tname) *n) {\ - \ - struct TNN(tname) *t = n->top, *tt = t->top; \ - \ - while (1) { \ - if (t == tt->left) { \ - \ - if (TIS_RED(tt->right)) { \ - \ - TPAINT_BLACK(t); \ - TPAINT_BLACK(tt->right); \ - TPAINT_RED(tt); \ - n = tt; \ - t = n->top; \ - if (t == NULL) { \ - \ - TPAINT_BLACK(n); \ - break; \ - } \ - tt = t->top; \ - if (TIS_BLACK(t)) { \ - \ - break; \ - } \ - } else { \ - \ - if (n == t->right) { \ - \ - prefix##rotate_## left(tm, t); \ - n = t; \ - t = n->top; \ - } \ - \ - TPAINT_BLACK(t); \ - TPAINT_RED(tt); \ - prefix##rotate_## right(tm, tt); \ - break; \ - } \ - } else { \ - \ - if (TIS_RED(tt->left)) { \ - TPAINT_BLACK(t); \ - TPAINT_BLACK(tt->left); \ - TPAINT_RED(tt); \ - n = tt; \ - t = n->top; \ - if (t == NULL) { \ - TPAINT_BLACK(n); \ - break; \ - } \ - tt = t->top; \ - if (TIS_BLACK(t)) { \ - break; \ - } \ - } else { \ - if (n == t->left) { \ - prefix##rotate_## right(tm, t); \ - n = t; \ - t = n->top; \ - tt = t->top; \ - } \ - TPAINT_BLACK(t); \ - TPAINT_RED(tt); \ - prefix##rotate_## left(tm, tt); \ - break; \ - } \ - } \ - } \ -}\ -static inline void prefix##delete_fixup(struct tname *tm, struct TNN(tname)* t) {\ - \ - struct TNN(tname) *n = NULL, *s, *cn, *dn; \ - \ - \ - while (1) { \ - s = t->right; \ - if (n != s) { \ - \ - if (TIS_RED(s)) { \ - \ - prefix##rotate_## left(tm, t); \ - TPAINT_RED(t); \ - TPAINT_BLACK(s); \ - s = t->right; \ - } \ - dn = s->right; \ - if (TIS_BLACK(dn)) { \ - cn = s->left; \ - if (TIS_BLACK(cn)) { \ - \ - TPAINT_RED(s); \ - if (TIS_RED(t)) { \ - TPAINT_BLACK(t); \ - break; \ - } \ - n = t; \ - t = n->top; \ - if (t) { \ - continue; \ - } else { \ - break; \ - } \ - } \ - \ - prefix##rotate_## right(tm, s); \ - \ - prefix##rotate_ ##left(tm, t); \ - TPAINT(cn, t->color); \ - TPAINT_BLACK(t); \ - break; \ - } \ - \ - dn = s->right; \ - prefix##rotate_ ##left(tm, t); \ - TPAINT(s, t->color); \ - TPAINT_BLACK(t); \ - TPAINT_BLACK(dn); \ - break; \ - } else { \ - \ - s = t->left; \ - if (TIS_RED(s)) { \ - prefix##rotate_## right(tm, t); \ - TPAINT_RED(t); \ - TPAINT_BLACK(s); \ - s = t->left; \ - } \ - dn = s->left; \ - if (TIS_BLACK(dn)) { \ - cn = s->right; \ - if (TIS_BLACK(cn)) { \ - TPAINT_RED(s); \ - if (TIS_RED(t)) { \ - TPAINT_BLACK(t); \ - } else { \ - n = t; \ - t = n->top; \ - if (t) { \ - continue; \ - } \ - } \ - break; \ - } \ - prefix##rotate_## left(tm, s); \ - s = t->left; \ - } \ - dn = s->left; \ - prefix##rotate_ ##right(tm, t); \ - TPAINT(s, t->color); \ - TPAINT_BLACK(t); \ - TPAINT_BLACK(dn); \ - break; \ - } \ - } \ -}\ -struct TNN(tname) *TNN(tname)(ktype key, vtype value) { \ - struct TNN(tname) *nn = malloc(sizeof(struct TNN(tname))); \ - nn->key = key; \ - nn->value = value; \ - TPAINT_RED(nn); \ - nn->left = nn->right = nn->top = NULL; \ - return nn; \ -} \ -struct TNN(tname) *prefix##left_most(struct TNN(tname) *n) { \ - while (n->left) { \ - n = n->left; \ - } \ - return n; \ -} \ -struct TNN(tname) *prefix##right_most(struct TNN(tname) *n) { \ - while (n->right) { \ - n = n->right; \ - } \ - return n; \ -} \ -struct TNN(tname) *prefix##left_deepest(struct TNN(tname) *n) { \ - for (;;) { \ - if (n->left) { \ - n = n->left; \ - } else if (n->right) { \ - n = n->right; \ - } else { \ - break; \ - } \ - } \ - return n; \ -} \ -struct TNN(tname) *prefix##next_lrn(struct TNN(tname) *n) { \ - struct TNN(tname) *top = n->top; \ - if (top && n == top->left && top->right) { \ - return prefix##left_deepest(top->right); \ - } \ - return top; \ -} \ -struct TNN(tname) *prefix##next_lnr(struct TNN(tname) *n) { \ - if (n->right) { \ - return prefix##left_most(n->right); \ - } \ - struct TNN(tname) *top = n->top; \ - while (top != NULL && n == top->right) { \ - n = top; \ - top = n->top; \ - } \ - return top; \ -} \ -struct TNN(tname) *prefix##prev_lnr(struct TNN(tname) *n) { \ - if (n->left) { \ - return prefix##right_most(n->left); \ - } \ - struct TNN(tname) *top = n->top; \ - while (top != NULL && n == top->left) { \ - n = top; \ - top = n->top; \ - } \ - return top; \ -} \ -struct TNN(tname) *prefix##first_lrn(struct tname *tm) { \ - return tm->root? prefix##left_deepest(tm->root): NULL; \ -} \ -struct TNN(tname) *prefix##first_lnr(struct tname *tm) { \ - return tm->root? prefix##left_most(tm->root): NULL; \ -} \ -struct TNN(tname) *prefix##last_lnr(struct tname *tm) { \ - return tm->root? prefix##right_most(tm->root): NULL; \ -} \ -struct TNN(tname) *prefix##pval_node(void *pv) { \ - return pv? pv - offsetof(struct TNN(tname), value): NULL; \ -} \ -struct tname *prefix##create(compare_fnt cmp) { \ - if (!cmp) { \ - FLOG("Không thể tạo bảng cây nếu không biết hàm so sánh."); \ - return NULL; \ - } \ - struct tname *tm = malloc(sizeof(struct tname)); \ - if (!tm) { \ - FLOG("Không thể cấp phát bộ nhớ."); \ - return NULL; \ - } \ - tm->root = NULL; \ - tm->cmp = cmp; \ - tm->fk = NULL; \ - tm->fv = NULL; \ - tm->size = 0; \ - return tm; \ -} \ -struct tname *tname(compare_fnt cmp) { \ - return prefix##create(cmp); \ -} \ -void prefix##clear(struct tname *tm) { \ - struct TNN(tname) *n = prefix##first_lrn(tm); \ - struct TNN(tname) *tmp = NULL; \ - while (n) { \ - if (tm->fk) { \ - tm->fk(n->key); \ - } \ - if (tm->fv) { \ - tm->fv(n->value); \ - } \ - tmp = n; \ - n = prefix##next_lrn(n); \ - free(tmp); \ - } \ - tm->size = 0; \ -} \ -void prefix##free(struct tname *tm) { \ - prefix##clear(tm); \ - free(tm); \ -} \ -static struct tname *prefix##delete(struct tname *tm, struct TNN(tname) *dn) { \ - struct TNN(tname) *node = dn; \ - struct TNN(tname) *child = node->right, \ - *tmp = node->left, \ - *top, *rebalance; \ - \ - if (!tmp) { \ - prefix##change(tm, node, child); \ - if (child) { \ - \ - TPAINT_BLACK(child); \ - rebalance = NULL; \ - } else { \ - \ - rebalance = TIS_BLACK(node)? node->top: NULL; \ - } \ - } else if (!child) { \ - \ - prefix##change(tm, node, tmp); \ - TPAINT_BLACK(tmp); \ - rebalance = NULL; \ - } else { \ - struct TNN(tname) *sss = child, *child2; \ - tmp = child->left; \ - if (!tmp) { \ - \ - top = sss; \ - child2 = sss->right; \ - } else { \ - \ - do { \ - sss = tmp; \ - tmp = tmp->left; \ - } while (tmp); \ - top = sss->top; \ - child2 = sss->right; \ - top->left = child2; \ - if (child2) { \ - child2->top = top; \ - } \ - sss->right = child; \ - child->top = sss; \ - } \ - \ - tmp = node->left; \ - sss->left = tmp; \ - tmp->top = sss; \ - if (child2) { \ - \ - TPAINT_BLACK(child2); \ - rebalance = NULL; \ - } else {\ - \ - rebalance = TIS_BLACK(sss) ? top: NULL; \ - }\ - prefix##change(tm, node, sss); \ - TPAINT(sss, node->color); \ - } \ - if (rebalance) { \ - \ - prefix##delete_fixup(tm, rebalance); \ - } \ - free(dn); \ - --(tm->size); \ - return tm; \ -} \ -vtype *prefix##put(struct tname *tm, ktype key, vtype value) { \ - struct TNN(tname) *nn = TNN(tname)(key, value), \ - *top = NULL, \ - *x = tm->root, \ - **loc = &tm->root; \ - int rl = 0; \ - while (x) { \ - rl = tm->cmp(&key, &x->key); \ - if (rl == 0) { \ - free(nn); \ - return &x->value; \ - } \ - top = x; \ - if (rl < 0) { \ - x = top->left; \ - loc = &top->left; \ - } else { \ - x = top->right; \ - loc = &top->right; \ - } \ - } \ - *loc = nn; \ - nn->top = top; \ - if (top == NULL) { \ - nn->color = BLACK; \ - } else if (TIS_RED(top)) { \ - \ - prefix##put_fixup(tm, nn); \ - } \ - ++tm->size; \ - return NULL; \ -} \ -vtype *prefix##get(struct tname *tm, ktype key) { \ - struct TNN(tname) *x = tm->root; \ - while (x) { \ - int rl = tm->cmp(&key, &x->key); \ - if (rl == 0) { \ - break; \ - } \ - x = rl < 0? x->left: x->right; \ - } \ - if (!x) { \ - return NULL; \ - } \ - return &x->value; \ -} \ -struct tname *prefix##remove(struct tname *tm, ktype key) { \ - struct TNN(tname) *n = prefix##pval_node(prefix##get(tm, key)); \ - if (!n) { \ - return NULL; \ - } \ - if (tm->fk) { \ - tm->fk(n->key); \ - } \ - if (tm->fv) { \ - tm->fv(n->value); \ - }\ - prefix##delete(tm, n);\ - return tm; \ -} -#define TDECL_IMPL(tname,keytype,valtype,prefix) \ -TDECL(tname, keytype, valtype, prefix); \ -TIMPL(tname, keytype, valtype, prefix) -#endif - -/***** ./cont/vector.h *****/ -#ifndef CONT_VECTOR_H_ -#define CONT_VECTOR_H_ -#define VDEFN(vecname,elemtype) \ - struct vecname { \ - elemtype *elems; \ - long size; \ - long cap; \ - double rio; \ - void (*fv)(elemtype); \ - } -#define VDECL(vecname,elemtype,prefix) \ -VDEFN(vecname, elemtype); \ -struct vecname *prefix##create(long sz); \ -struct vecname *vecname(long sz); \ -struct vecname *prefix##clear(struct vecname *v); \ -void prefix##free(struct vecname *po); \ -int prefix##empty(struct vecname *v); \ -struct vecname *prefix##reserve(struct vecname *v, long newcap); \ -struct vecname *prefix##resize(struct vecname *v, long newsz); \ -struct vecname *prefix##append(struct vecname *v, elemtype val); \ -struct vecname *prefix##remove(struct vecname *v, long idx); \ -struct vecname *prefix##insertb(struct vecname *v, elemtype elem, long pos); \ -struct vecname *prefix##fill(struct vecname *v, elemtype value); \ -struct vecname *prefix##push(struct vecname *v, elemtype elem); \ -struct vecname *prefix##pop(struct vecname *v); \ -elemtype *prefix##top(struct vecname *v) -#define VIMPL(vecname,elemtype,prefix) \ -struct vecname *prefix##create(long sz) { \ - if (sz < 0) { \ - FLOG("Tạo vec-tơ với kích thước không hợp lệ, sz = %ld", sz); \ - return NULL; \ - } \ - struct vecname *v = malloc(sizeof(struct vecname)); \ - v->fv = NULL; \ - v->size = sz; \ - v->cap = sz > 0? sz: 8; \ - v->rio = 2.0; \ - v->elems = calloc(v->cap, sizeof(elemtype)); \ - return v; \ -} \ -struct vecname *vecname(long sz) { \ - return prefix##create(sz); \ -} \ -struct vecname *prefix##clear(struct vecname *v) { \ - return prefix##resize(v, 0); \ -} \ -void prefix##free(struct vecname *v) { \ - prefix##clear(v); \ - free(v->elems); \ - free(v); \ -} \ -int prefix##empty(struct vecname *v) { \ - return v->size == 0; \ -} \ -struct vecname *prefix##reserve(struct vecname *v, long newcap) {\ - if (newcap < v->size) { \ - FLOG("Dự trữ với dung lượng (%ld) < kích thước (%ld)", newcap, v->size); \ - return NULL; \ - } \ - v->elems = realloc(v->elems, newcap * sizeof(elemtype)); \ - v->cap = newcap; \ - return v; \ -} \ -struct vecname *prefix##resize(struct vecname *v, long newsize) {\ - if (newsize > v->cap) { \ - prefix##reserve(v, newsize); \ - } else if (newsize < v->size && v->fv) { \ - for (long j_ = newsize; j_ < v->size; ++j_) { \ - v->fv(v->elems[j_]); \ - } \ - } \ - v->size = newsize; \ - return v; \ -} \ -struct vecname *prefix##append(struct vecname *v, elemtype val) {\ - if (v->size == v->cap) { \ - prefix##reserve(v, v->rio * v->size); \ - } \ - v->elems[v->size] = val; \ - ++v->size; \ - return v; \ -} \ -struct vecname *prefix##remove(struct vecname *v, long idx) { \ - long sz = v->size; \ - if (idx >= sz || idx < 0) { \ - FLOG("Xóa phần tử với chỉ số không hợp lệ sz = %ld, idx = %ld", sz, idx); \ - return NULL; \ - } \ - elemtype *arr = v->elems; \ - elemtype tmp = arr[idx]; \ - for (long i = idx; i < sz - 1; ++i) { \ - arr[i] = arr[i + 1]; \ - } \ - arr[sz - 1] = tmp; \ - return prefix##resize(v, sz - 1); \ -}\ -struct vecname *prefix##insertb(struct vecname *v, elemtype elem, long pos) { \ - prefix##resize(v, v->size + 1); \ - elemtype *arr = v->elems; \ - for (long i = v->size - 1; i > pos; --i) { \ - arr[i] = arr[i - 1]; \ - } \ - arr[pos] = elem; \ - return v; \ -} \ -struct vecname *prefix##fill(struct vecname *v, elemtype value) { \ - for (long i = 0; i < v->size; ++i) {\ - v->elems[i] = value; \ - } \ - return v; \ -} \ -struct vecname *prefix##push(struct vecname *v, elemtype elem) { \ - return prefix##append(v, elem); \ -}\ -struct vecname *prefix##pop(struct vecname *v) { \ - return prefix##remove(v, v->size - 1); \ -} \ -elemtype *prefix##top(struct vecname *v) { \ - return v->elems + (v->size - 1); \ -} -#define VDECL_IMPL(vecname,elemtype,prefix) \ -VDECL(vecname, elemtype, prefix); \ -VIMPL(vecname, elemtype, prefix) -#endif -#endif // CGEN_H_ +#include "cgen.h" #include VDECL_IMPL(dvect, double, dv) @@ -1017,13 +11,13 @@ int main() { scanf("%lf", &tmp); if (i < k) { dvappend(v, tmp); - heap_shift_up(v->elems, v->size, sizeof(v->elems[0]), v->size - 1, cmpi); + heap_shift_up(v->elems, v->size, sizeof(v->elems[0]), v->size - 1, cmpd); } else if (tmp < v->elems[0]) { v->elems[0] = tmp; - heap_shift_down(v->elems, v->size, sizeof(v->elems[0]), 0, cmpi); + heap_shift_down(v->elems, v->size, sizeof(v->elems[0]), 0, cmpd); } } - qsort(v->elems, v->size, sizeof(v->elems[0]), cmpi); + qsort(v->elems, v->size, sizeof(v->elems[0]), cmpd); for (int i = 0; i < v->size; ++i) { printf("%.2f\n", v->elems[i]); }