【第一步】在Windows下编辑一个.pc程序(Pro*C源程序,作者用到:C:\proctest\exam1.pc),其内容如下:

  1. #include <stdio.h>
  2. #include <string.h>
  3. EXEC SQL INCLUDE SQLCA;
  4. int main()
  5. {
  6. /*declare variables*/
  7. EXEC SQL BEGIN DECLARE SECTION;
  8. VARCHAR usr[20], pass[20], serv[20];
  9. char name[8];
  10. int age;
  11. EXEC SQL END DECLARE SECTION;
  12. /*connect DB*/
  13. strcpy(usr.arr,"any3");
  14. usr.len=(unsigned short)strlen((char *)usr.arr);
  15. strcpy(pass.arr,"any3");
  16. pass.len=(unsigned short)strlen((char *)pass.arr);
  17. strcpy(serv.arr,"ORCL");
  18. serv.len=(unsigned short)strlen((char *)serv.arr);
  19. /*Connecting DB Command*/
  20. EXEC SQL CONNECT :usr IDENTIFIED BY :pass USING :serv;
  21. printf("Connect!\n");
  22. /*Input*/
  23. printf("Input Member Age:");
  24. scanf("%d", &age);
  25. /*Exec SQL*/
  26. EXEC SQL SELECT age, name into :age, :name from family_lu where age=:age;
  27. printf("Name=%s    ,Age=%d\n", name, age);
  28. /*commit and disconnect DB*/
  29. EXEC SQL COMMIT WORK RELEASE;
  30. printf("Disconnect!\n");
  31. return 1;
  32. }

作者机器上使用到的oracle实例ORCL,并以用户名/密码(any3/any3)登陆。

为示例,作者建立测试数据表 family_lu ,其结构明细为:

CREATE TABLE family_lu
(
       age   number,
       name  varchar2(20),
       reltn varchar2(20)          
);

插入数据:

insert into family_lu values(0, 'anan', 'child');

【第二步】在Windows下使用Oracle数据库的...\bin\Proc.exe预编译该.pc文件,生成对应的.c源程序。

【建议一】

注意到安装Oracle时已经将“D:\oracle\product\10.2.0\db_1\bin;”自动添加到“系统环境变量->系统变量->Path”中,因此,在“开始->运行->cmd”回车后,直接进入到.pc文件所在路径,输入proc命令。

如本人Win7系统下为:

C:\Users\Administrator>cd ../../proctest

然后输入命令:proc exam1.pc

【第三步】使用Windows下的gcc编译器对上述.c源文件进行编译,并生成windows下的可执行文件。

注意到这里是最容易报错(非犯错)的地方!!!

【建议二】使用最新的gcc编译器。

对于较低版本的gcc,对于这一段源程序

usr.len=(unsigned short)strlen((char *)usr.arr);

编译会报错:

invalid conversion from `const char*' to `char*'

而升级到最新编译器则不存在此问题

——当然也可能是编码时语法不严谨导致,但无论如何,最新的编译器可用性更好,并且向下兼容。

作者机器上安装了JFE and GCC,它自动被添加到了“系统环境变量->系统变量->Path”中,但其版本较低。

这里手动将“D:\Program Files\CodeBlocks\MinGW\bin;”添加到“系统环境变量->系统变量->Path”的最前方(注意MinGW已经安装在本机上,为作者安装CodeBlocks时自带),即对于同名的gcc编译器,希望调用Path中靠前的那一个。MinGW的GCC编译器版本为4.7.1。

【建议三】使用正确的库文件

对于.pc源程序中有SQL(DB连接、数据查询等)相关代码的,在Pro*C将.pc转化为.c时,会自动在.c文件中声明并引用

extern void sqlcxt (void **, unsigned int *,  struct sqlexd *, const struct sqlcxp *);

extern void sqlcx2t(void **, unsigned int *, struct sqlexd *, const struct sqlcxp *);
extern void sqlbuft(void **, char *);
extern void sqlgs2t(void **, char *);
extern void sqlorat(void **, unsigned int *, void *);

等相关函数,因为在简单的Pro*C编程示例中,在.pc预编译成.c文件后,多数情况下都只使用了sqlcxt函数(而未使用sqlbuft等),因此,gcc在编译.c时将报错:

undefined reference to `sqlcxt(void**, unsigned*, sqlexd*, sqlcxp const*)'

这是由于gcc在编译时未加入Pro*C指定的库文件(静态库.lib或动态连接库.dll),该库包括了sqlcxt的定义。

由于Pro*C广泛应用于Linux+Oracle环境,因此在多数博客文章或论坛发帖中,常见到:

////////////引用部分开始////////////

需要用到$ORACLE_HOME/lib/libclntsh.so
故需加上 -L $ORACLE_HOME/lib -l clntsh
更正后的命令为:
gcc -o test test.c -I /home/oracle/oracle/product/10.2.0/db_1/precomp/public -L $ORACLE_HOME/lib -l clntsh

////////////引用部分结束////////////LINK: http://blog.chinaunix.net/uid-261392-id-2138943.html

在百度中搜索关键字“windows libclntsh”,非常好运地找到了《OCI Instant Client》 LINK:http://www.cnblogs.com/freewater/archive/2011/08/09/2132801.html

作者找到Linux和Unix下的libclntsh.so.11.1(.so是Linux等下的动态连接库),正好对应于Windows下的oci.dll。

最后,在《gcc编译dll和调用dll》一文中

LINK:http://blog.csdn.net/denglei265/article/details/3889470

找到调用.dll库文件生成windows下可执行文件的(简单)方法,如作者使用命令:

C:\proctest>gcc exam1.c D:\oracle\product\10.2.0\db_1\BIN\oci.dll -o exam1

成功生成可执行文件,运行之得到效果:

C:\proctest>exam1
Connect!
Input Member Age:0
Name=anan       ,Age=0
Disconnect!

而这恰好是本文Pro*C程序示例希望实现的效果!

Pro*C编程研究一:从.pc到.exe的更多相关文章

  1. 近中期3D编程研究目标

    近几年一直在用业余时间研究3D编程,研究的中期目标是建立一个实用的开源3D编程框架.3D编程技术最直接的应用是开发游戏,所以3D编程框架也就是3D游戏开发框架.在我看来,游戏是否好玩的关键是能否为玩家 ...

  2. USB编程研究之二(常见设备类型的GUID)

    在USB编程之前要事先了解一下GUID的概念. 应用其他网页中的定义: 全球唯一标识符 (GUID) 是一个字母数字标识符,用于指示产品的唯一性安装.在许多流行软件应用程序(例如 Web 浏览器和媒体 ...

  3. macbook pro retina 编程字体推荐

    使用VS2010.VS2012.Qt Creator编译工具首推等宽字体,等宽字体中consolas. 首先大家都知道等宽对于编码来说的直观性不言而喻,其次retina屏幕的特殊性,整天用特别小的字体 ...

  4. Autofac的AOP面向切面编程研究

    *:first-child { margin-top: 0 !important; } .markdown-body>*:last-child { margin-bottom: 0 !impor ...

  5. C#异步编程研究学习(一)

    可以使用Func<T>或者Action<T>简单实现如: Func<string, string,string,string, int> func = new Fu ...

  6. GDI+与图形编程研究

    GDI+的基本概念 GDI+的常用对象,包括Graphics.Font.Brush.Pen等对象的创建和使用 常用图形的绘制 Color结构.Point结构和Rectangle结构 1.GDI+的概念 ...

  7. 关于用C-free进行C语言编程在电脑中生成的.exe和.o文件

    在使用C-free进行C语言编程时,程序运行后会自动在电脑文件中生成以.exe和.o为后缀名的文件: 1,生成的.exe文件为系统自动打包完成的应用程序,该程序可直接在其他无C-free环境的电脑上运 ...

  8. IDA,IDA PRO 产品介绍

    IDA理念这是我们在开发产品时竭尽全力遵循的理念--在此过程中,我们相信我们将开发出能够为您带来所需的可靠性.便利性和易用性的软件.没有什么能打败人脑因为我们知道一秒钟的洞察力仍然胜过百年的处理时间, ...

  9. 十年学会编程 著者: Peter Norvig 翻译: Dai Yuwen

    为何人人都这么着急? 信步走进任何一家书店,你会看到名为<如何在7天内学会Java>的书,还有各 种各样类似的书: 在几天内或几小时内学会Visual Basic, Windows, In ...

随机推荐

  1. 【数字图像处理】帧差法与Kirsch边缘检测实现运动目标识别与分割

    本文链接:https://blog.csdn.net/qq_18234121/article/details/82763385 作者:冻人的蓝鲸梁思成 视频分割算法可以从时域和空域两个角度考虑.时域分 ...

  2. c# Format() 方法

  3. 全文检索引擎在Django中的使用

    Haystack 1.什么是Haystack Haystack是django的开源全文搜索框架(全文检索不同于特定字段的模糊查询,使用全文检索的效率更高 ),该框架支持Solr,Elasticsear ...

  4. python基础应用---格式化输出

    python的格式化输出,原来不是很理解,现在有点了解了,为此特意写一个博客来记录一下,以便自己会忘记了,随时查看, 程序主体 #格式化输出之一 name = input("pls inpu ...

  5. JQuery 遍历 操作数组 map、grep、filter 的区别

    filter() 方法将匹配元素集合缩减为匹配指定选择器的元素.例如:改变所有 div 的颜色,然后向类名为 "middle" 的类添加边框:$("div"). ...

  6. 0011SpringBoot的@EnableWebMvc全面接管SpringMVC的自动配置(源码)

    所谓的@EnableWebMvc全面接管SpringMVC的自动配置,是指@EnableWebMvc注解会使SpringMVC的自动配置失效,原理如下: 1.查看@EnableWebMvc的源码,如下 ...

  7. Selenium(四)使用xpath定位元素

    1.什么是xpath: 2.xpath的节点类型 3.xpath的表达式 4.开始定位 浏览器打开本地文件:   (python3.7的打开语法) 查找根节点: (绝对路径)查找子节点: 查找type ...

  8. 谷歌网页性能分析工具 Lighthouse 的安装及使用

    github地址:https://github.com/GoogleChrome/lighthouse 一.如果可以翻墙的话可以从 chrome 扩展插件里直接安装. 二.下面是另一种使用方法:基于 ...

  9. stm32中阻塞模式和非阻塞模式 in blocking mode 与 in non-blocking mode区别

    阻塞模式和非阻塞模式...... 我的理解是:阻塞模式就像是一个延时函数,当这个函数没处理完那么,所有的按照流程需要执行的代码都不会被执行,要等到这个延时完成,类似 平时看书上写的LED灯闪烁,用的d ...

  10. c++对c的扩展----引用类型

    变量是一个内存的别名,程序通过变量名使用内存空间,当然一个内存空间可以起多个别名么? 答案:可以,这就是c++中引用的由来,引用就是给变量起别名 引用是c++的概念!!!况且声明引用的符号&十 ...