更新时间:2024-11-12 GMT+08:00
常用示例
ecpg常用示例代码
#include <locale.h> #include <string.h> #include <stdlib.h> exec sql whenever sqlerror sqlprint; exec sql include sqlca; int main(void) { EXEC SQL BEGIN DECLARE SECTION; char *temp_str = (char *)malloc(11); EXEC SQL END DECLARE SECTION; ECPGdebug(1, stderr); /* 提前创建好testdb库 */ exec sql connect to testdb; /* 打开自动提交,以下执行exec sql时不用手动commit */ exec sql set autocommit = on; exec sql drop table if exists test_t; /* 建表,插入数据 */ exec sql create table test_t(f float, i int, a int[10], mstr char(10)); exec sql insert into test_t(f, i, a, mstr) values(1.01,1,'{0,1,2,3,4,5,6,7,8,9}', 'China'); /* 关闭自动提交,以下插入数据的sql语句需要手动commit才能提交 */ exec sql set autocommit = off; exec sql insert into test_t(f, i, a, mstr) values(2.01,2,'{0,1,2,3,4,5,6,7,8,9}', 'USA'); exec sql commit; exec sql insert into test_t(f, i, a, mstr) values(3.01,3,'{0,1,2,3,4,5,6,7,8,9}', 'AUS'); exec sql insert into test_t(f, i, a, mstr) values(4.01,4,'{0,1,2,3,4,5,6,7,8,9}', 'JAP'); exec sql commit; EXEC SQL BEGIN DECLARE SECTION; int a[10] = {9,8,7,6,5,4,3,2,1,0}; int id = 6; EXEC SQL END DECLARE SECTION; /* 从宿主变量取数据插入到表中,宿主变量的类型与表定义的类型一致 */ strcpy(temp_str, "RUS"); exec sql insert into test_t(f, i, a, mstr) values(5.01,5,:a,:temp_str); exec sql commit; exec sql set autocommit = on; exec sql begin; exec sql insert into test_t(f, i, a, mstr) values(6.01,:id,:a,'SIG'); exec sql commit; exec sql set autocommit = off; exec sql begin declare section; float ff; char tmp_text[25] = "klmnopqrst"; exec sql end declare section; exec sql set autocommit = on; exec sql begin work; printf("Found ff=%f tmp_text=%10.10s\n", ff, tmp_text); /* 条件查询语句示例 */ exec sql select f, mstr into :ff,:tmp_text from test_t where f > (select f from test_t where i = 4 or i < 0) order by a limit 1; printf("Found ff=%f tmp_text=%10.10s\n", ff, tmp_text); exec sql select f, mstr into :ff,:tmp_text from test_t where mstr = 'JAP' order by i; printf("Found ff=%f tmp_text=%10.10s\n", ff, tmp_text); exec sql select f, mstr into :ff,:tmp_text from test_t order by i DESC limit 1; printf("Found ff=%f tmp_text=%10.10s\n", ff, tmp_text); exec sql select f, mstr into :ff,:tmp_text from test_t order by mstr limit 1; printf("Found ff=%f tmp_text=%10.10s\n", ff, tmp_text); exec sql select count(f), a into :ff,:tmp_text from test_t where i > 2 group by a limit 1; printf("Found ff=%f tmp_text=%20.30s\n", ff, tmp_text); exec sql select count(f), a into :ff,:tmp_text from test_t where i > 3 group by a order by a limit 1; printf("Found ff=%f tmp_text=%20.30s\n", ff, tmp_text); exec sql select sum(f), a into :ff,:tmp_text from test_t where i > 2 group by a order by a limit 1; printf("Found ff=%f tmp_text=%20.30s\n", ff, tmp_text); exec sql select distinct a into :tmp_text from test_t order by a limit 1; exec sql drop table test_t; exec sql commit; /* 释放连接,释放为宿主变量分配的内存 */ exec sql disconnect; free(temp_str); return 0; }
pgtypes库函数示例代码
示例一:使用库函数对时间和日期类型进行不同操作。具体使用方式请参见使用库函数章节。
#include <stdio.h> #include <string.h> #include <stdlib.h> #include <limits.h> #include <pgtypes_date.h> #include <pgtypes_timestamp.h> char *dates[] = { "19990108foobar", "19990108 foobar", "1999-01-08 foobar", "January 8, 1999", "1999-01-08", "1/8/1999", "1/18/1999", "01/02/03", "1999-Jan-08", "Jan-08-1999", "08-Jan-1999", "99-Jan-08", "08-Jan-99", "08-Jan-06", "Jan-08-99", "19990108", "990108", "1999.008", "J2451187", "January 8, 99 BC", NULL }; /* 不可与libc的“times”冲突 */ static char *times[] = { "0:04", "1:59 PDT", "13:24:40 -8:00", "13:24:40.495+3", NULL }; char *intervals[] = { "1 minute", "1 12:59:10", "2 day 12 hour 59 minute 10 second", "1 days 12 hrs 59 mins 10 secs", "1 days 1 hours 1 minutes 1 seconds", "1 year 59 mins", "1 year 59 mins foobar", NULL }; int main(void) { exec sql begin declare section; date date1; timestamp ts1, ts2; char *text; interval *i1; date *dc; exec sql end declare section; int i, j; char *endptr; ECPGdebug(1, stderr); /* 从文本中解析一个时间戳并将日期转换成字符串 */ ts1 = PGTYPEStimestamp_from_asc("2003-12-04 17:34:29", NULL); text = PGTYPEStimestamp_to_asc(ts1); printf("timestamp: %s\n", text); free(text); /* 从时间戳中抽取日期部分 */ date1 = PGTYPESdate_from_timestamp(ts1); dc = PGTYPESdate_new(); *dc = date1; /* 返回一个日期变量的文本表达 */ text = PGTYPESdate_to_asc(*dc); printf("Date of timestamp: %s\n", text); free(text); PGTYPESdate_free(dc); for (i = 0; dates[i]; i++) { bool err = false; /* 从日期的文本表达解析一个日期 */ date1 = PGTYPESdate_from_asc(dates[i], &endptr); if (date1 == INT_MIN) { err = true; } /* 返回一个日期变量的文本表达 */ text = PGTYPESdate_to_asc(date1); printf("Date[%d]: %s (%c - %c)\n", i, err ? "-" : text, endptr ? 'N' : 'Y', err ? 'T' : 'F'); free(text); if (!err) { for (j = 0; times[j]; j++) { int length = strlen(dates[i])+ 1+ strlen(times[j])+ 1; char* t = (char *)malloc(length); sprintf(t, "%s %s", dates[i], times[j]); /* 从文本中解析一个时间戳并将日期转换成字符串 */ ts1 = PGTYPEStimestamp_from_asc(t, NULL); text = PGTYPEStimestamp_to_asc(ts1); if (i != 19 || j != 3) printf("TS[%d,%d]: %s\n",i, j, errno ? "-" : text); free(text); free(t); } } } /* 从文本中解析一个时间戳 */ ts1 = PGTYPEStimestamp_from_asc("2004-04-04 23:23:23", NULL); for (i = 0; intervals[i]; i++) { interval *ic; /* 从文本中解析一个区间 */ i1 = PGTYPESinterval_from_asc(intervals[i], &endptr); if (*endptr) printf("endptr set to %s\n", endptr); if (!i1) { printf("Error parsing interval %d\n", i); continue; } /* 把一个区间变量加到时间戳变量上 */ j = PGTYPEStimestamp_add_interval(&ts1, i1, &ts2); if (j < 0) continue; /* 将一个区间类型变量转换成文本格式 */ text = PGTYPESinterval_to_asc(i1); printf("interval[%d]: %s\n", i, text ? text : "-"); free(text); /* 返回一个已分配区间变量的指针 */ ic = PGTYPESinterval_new(); /* 复制一个区间类型的变量 */ PGTYPESinterval_copy(i1, ic); /* 将一个区间类型变量转换成文本格式 */ text = PGTYPESinterval_to_asc(i1); printf("interval_copy[%d]: %s\n", i, text ? text : "-"); free(text); /* 释放已经分配区间变量的内存 */ PGTYPESinterval_free(ic); PGTYPESinterval_free(i1); } return (0); }
示例二:使用pgtypes库函数对numeric类型进行不同操作。
#include <stdio.h> #include <stdlib.h> #include <pgtypes_numeric.h> #include <pgtypes_error.h> #include <decimal.h> char* nums[] = { "2E394", "-2", ".794", "3.44", "592.49E21", "-32.84e4", "2E-394", ".1E-2", "+.0", "-592.49E-07", "+32.84e-4", ".500001", "-.5000001", "1234567890123456789012345678.91", /* 30个数位应转换为十进制 */ "1234567890123456789012345678.921", /* 31个数位的数字不应转为十进制 */ "not a number", NULL }; static void check_errno(void); int main(void) { char *text="error\n"; char *endptr; numeric *num, *nin; decimal *dec; long l; int i, j, k, q, r, count = 0; double d; numeric **numarr = (numeric **) calloc(1, sizeof(numeric)); ECPGdebug(1, stderr); for (i = 0; nums[i]; i++) { /* 返回由malloc分配的字符串的指针,它包含numeric类型nums[i]的字符串表达 */ num = PGTYPESnumeric_from_asc(nums[i], &endptr); if (!num) check_errno(); if (endptr != NULL) { printf("endptr of %d is not NULL\n", i); if (*endptr != '\0') printf("*endptr of %d is not \\0\n", i); } if (!num) continue; numarr = (numeric **)realloc(numarr, sizeof(numeric *) * (count + 1)); numarr[count++] = num; /* 返回由malloc分配的字符串的指针,它包含numeric类型num的字符串表达 */ text = PGTYPESnumeric_to_asc(num, -1); if (!text) check_errno(); printf("num[%d,1]: %s\n", i, text); free(text); text = PGTYPESnumeric_to_asc(num, 0); if (!text) check_errno(); printf("num[%d,2]: %s\n", i, text); free(text); text = PGTYPESnumeric_to_asc(num, 1); if (!text) check_errno(); printf("num[%d,3]: %s\n", i, text); free(text); text = PGTYPESnumeric_to_asc(num, 2); if (!text) check_errno(); printf("num[%d,4]: %s\n", i, text); free(text); /* 请求一个指向新分配的numeric变量的指针 */ nin = PGTYPESnumeric_new(); text = PGTYPESnumeric_to_asc(nin, 2); if (!text) check_errno(); printf("num[%d,5]: %s\n", i, text); free(text); /* 将一个numeric类型的变量转换为长整型 */ r = PGTYPESnumeric_to_long(num, &l); if (r) check_errno(); printf("num[%d,6]: %ld (r: %d)\n", i, r?0L:l, r); if (r == 0) { /* 把一个长整型变量转换为一个numeric变量 */ r = PGTYPESnumeric_from_long(l, nin); if (r) check_errno(); /* 返回由malloc分配的字符串的指针,它包含numeric类型nin的字符串表达 */ text = PGTYPESnumeric_to_asc(nin, 2); /* 比较两个numeric变量 */ q = PGTYPESnumeric_cmp(num, nin); printf("num[%d,7]: %s (r: %d - cmp: %d)\n", i, text, r, q); free(text); } /* 将一个numeric类型的变量转换成整数 */ r = PGTYPESnumeric_to_int(num, &k); if (r) check_errno(); printf("num[%d,8]: %d (r: %d)\n", i, r?0:k, r); if (r == 0) { /* 把一个整数变量转换成一个numeric变量 */ r = PGTYPESnumeric_from_int(k, nin); if (r) check_errno(); /* 返回由malloc分配的字符串的指针,它包含numeric类型nin的字符串表达 */ text = PGTYPESnumeric_to_asc(nin, 2); q = PGTYPESnumeric_cmp(num, nin); printf("num[%d,9]: %s (r: %d - cmp: %d)\n", i, text, r, q); free(text); } if (i != 6) { /* 将一个numeric类型的变量转换成双精度类型 */ r = PGTYPESnumeric_to_double(num, &d); if (r) check_errno(); printf("num[%d,10]: %g (r: %d)\n", i, r?0.0:d, r); } /* 请求一个指向新分配的numeric变量的指针*/ dec = PGTYPESdecimal_new(); /* 将一个decimal类型的变量转换成numeric */ r = PGTYPESnumeric_to_decimal(num, dec); if (r) check_errno(); printf("num[%d,11]: - (r: %d)\n", i, r); if (r == 0) { /* 将一个decimal类型的变量转换成numeric */ r = PGTYPESnumeric_from_decimal(dec, nin); if (r) check_errno(); /* 返回由malloc分配的字符串的指针,它包含numeric类型nin的字符串表达 */ text = PGTYPESnumeric_to_asc(nin, 2); /* 比较两个numeric变量 */ q = PGTYPESnumeric_cmp(num, nin); printf("num[%d,12]: %s (r: %d - cmp: %d)\n", i, text, r, q); free(text); } /* 释放numeric变量的内存 */ PGTYPESdecimal_free(dec); PGTYPESnumeric_free(nin); printf("\n"); } for (i = 0; i < count; i++) { for (j = 0; j < count; j++) { /* 请求一个指向新分配的numeric变量的指针 */ numeric* a = PGTYPESnumeric_new(); numeric* s = PGTYPESnumeric_new(); numeric* m = PGTYPESnumeric_new(); numeric* d = PGTYPESnumeric_new(); /* 把两个numeric变量相加放到第三个numeric变量中 */ r = PGTYPESnumeric_add(numarr[i], numarr[j], a); if (r) { check_errno(); printf("r: %d\n", r); } else { /* 返回由malloc分配的字符串的指针,它包含numeric类型a的字符串表达 */ text = PGTYPESnumeric_to_asc(a, 10); printf("num[a,%d,%d]: %s\n", i, j, text); free(text); } /* 把两个numeric变量相减并且把结果返回到第三个numeric变量 */ r = PGTYPESnumeric_sub(numarr[i], numarr[j], s); if (r) { check_errno(); printf("r: %d\n", r); } else { /* 返回由malloc分配的字符串的指针,它包含numeric类型s的字符串表达 */ text = PGTYPESnumeric_to_asc(s, 10); printf("num[s,%d,%d]: %s\n", i, j, text); free(text); } /* 把两个numeric变量相乘并且把结果返回到第三个numeric变量 */ r = PGTYPESnumeric_mul(numarr[i], numarr[j], m); if (r) { check_errno(); printf("r: %d\n", r); } else { /* 返回由malloc分配的字符串的指针,它包含numeric类型m的字符串表达 */ text = PGTYPESnumeric_to_asc(m, 10); printf("num[m,%d,%d]: %s\n", i, j, text); free(text); } /* 把两个numeric变量相除并且把结果返回到第三个numeric变量 */ r = PGTYPESnumeric_div(numarr[i], numarr[j], d); if (r) { check_errno(); printf("r: %d\n", r); } else { /* 返回由malloc分配的字符串的指针,它包含numeric类型d的字符串表达 */ text = PGTYPESnumeric_to_asc(d, 10); printf("num[d,%d,%d]: %s\n", i, j, text); free(text); } /* 释放一个numeric变量的内存 */ PGTYPESnumeric_free(a); PGTYPESnumeric_free(s); PGTYPESnumeric_free(m); PGTYPESnumeric_free(d); } } for (i = 0; i < count; i++) { /* 返回由malloc分配的字符串的指针,它包含numeric类型numarr[i]的字符串表达 */ text = PGTYPESnumeric_to_asc(numarr[i], -1); printf("%d: %s\n", i, text); free(text); /* 释放内存 */ PGTYPESnumeric_free(numarr[i]); } free(numarr); return (0); } /* 错误处理 */ static void check_errno(void) { switch(errno) { case 0: printf("(no errno set) - "); break; case PGTYPES_NUM_OVERFLOW: printf("(errno == PGTYPES_NUM_OVERFLOW) - "); break; case PGTYPES_NUM_UNDERFLOW: printf("(errno == PGTYPES_NUM_UNDERFLOW) - "); break; case PGTYPES_NUM_BAD_NUMERIC: printf("(errno == PGTYPES_NUM_BAD_NUMERIC) - "); break; case PGTYPES_NUM_DIVIDE_ZERO: printf("(errno == PGTYPES_NUM_DIVIDE_ZERO) - "); break; default: printf("(unknown errno (%d))\n", errno); printf("(libc: (%s)) ", strerror(errno)); break; } }
父主题: 基于ecpg开发