我现在从事的工作是做银行外包的软件项目(HCE、ApplePay)的后台业务处理,作为公司外派人员,在银行上班。主要开发环境就是在Red Hat的linux服务器上用C语言进行二次或者三次的开发来实现相关业务逻辑或者操作IBM 的DB2数据库。感觉工作需掌握以下技能就差不多可以干了:Linux系统的熟练使用、vi、shell脚本,扎实的C语言基础,基本的数据结构与算法(不一定用的到,但有助于掌握C语言),Linux下C应用开发相关知识(文件操作、信号、进程、线程、socket),数据库SQL语言的熟练使用。当然,还有相关工具的使用及一些业务知识的了解,这可以在以后工作中不断熟练。
好啦,不多说啦,自己在博客“Linux下C应用编程”这一类别下把以前自己参加培训的笔记重新学习并整理到了博客上。在最后这里,就实际工作中自己遇到的比较好的代码和技术总结一下。当然,这也是一个不断完善和积累的过程,什么时候碰见了,就在此记一笔。广大的猿类朋友们若不经意间看见了这一系列的文章,如果有什么好的想法,也欢迎交流。
/*****************************************************************************
** 函数名称: GetDate
** 功能描述: 取当前系统日期
** 当前版本: 1.0.0.0
** 作 者:
** 修 改:
** 输入参数:
** 输出参数: char * psDate -- 系统日期, 格式为yyyymmdd
** 返回结果:int
0 ---> 成功
****************************************************************************/
int GetDate(char * psDate)
{
time_t nSeconds;
struct tm * pTM;
time(&nSeconds);
pTM = localtime(&nSeconds);
/* 系统日期, 格式:YYYYMMDD */
sprintf( psDate,"%04d%02d%02d",
pTM->tm_year + 1900, pTM->tm_mon + 1,pTM->tm_mday );
return 0;
}
/*****************************************************************************
** 函数名称: GetTime
** 功能描述: 取当前系统时间
** 当前版本: 1.0.0.0
** 作 者:
** 修 改:
** 输入参数:
** 输出参数: char * psTime -- 系统时间, 格式为HHMMSS
** 返回结果:int
0 ---> 成功
****************************************************************************/
int GetTime(char * psTime)
{
time_t nSeconds;
struct tm * pTM;
time(&nSeconds);
pTM = localtime(&nSeconds);
/* 系统时间, 格式:HHMMSS */
sprintf( psTime,"%02d%02d%02d",
pTM->tm_hour,pTM->tm_min, pTM->tm_sec);
return 0;
}
/*****************************************************************************
** 函数名称: GetDateTime
** 功能描述: 取当前系统日期和时间
** 当前版本: 1.0.0.0
** 作 者:
** 修 改:
** 输入参数:
** 输出参数: char * psDateTime -- 系统日期时间, 格式为yyyymmddHHMMSS
** 返回结果:int
0 ---> 成功
****************************************************************************/
int GetDateTime(char * psDateTime)
{
time_t nSeconds;
struct tm * pTM;
time(&nSeconds);
pTM = localtime(&nSeconds);
/* 系统日期和时间, 格式:yyyymmddHHMMSS */
sprintf( psDateTime,"%04d%02d%02d%02d%02d%02d",
pTM->tm_year + 1900, pTM->tm_mon + 1,pTM->tm_mday,
pTM->tm_hour,pTM->tm_min, pTM->tm_sec );
return 0;
}
调用的时候定义一个char数组,大小为日期的长度大小加1,然后直接调用上面的函数,参数为数组名即可。
当然,还有其他许多关于日期、时间操作的函数,比如不同日期、时间格式间的转换等。
Linux下C语言编程中有几个很实用的调试宏
__LINE__ __FILE__ __FUNCTION__ __TIME__ __DATA__
这几个预定义宏是属于ANSI标准的,内置于编译器,全局性的变量,可以方便地实现代码跟踪调试,不是在哪个头文件中包含的,见下例:
#include
int main()
{
printf("The file is %s.\n",__FILE__);
printf( "The date is %s.\n", __DATE__ );
printf( "The time is %s.\n", __TIME__ );
printf( "This is line %d.\n", __LINE__ );
printf( "This function is %s.\n", __FUNCTION__ );
return 0;
}
运行结果:
The file is macro.c.
The date is Aug 24 2012.
The time is 23:13:26.
This is line 8.
This function is main.
__LINE__ 及__FILE__宏指示
“#line 行数 文件名”指令可以改变它的值,简单的讲,编译时,它们包含程序的当前行数和文件名。
DATE 宏指令含有形式为月/日/年的串,表示源文件被翻译到代码时的日期。
TIME 宏指令包含程序编译的时间。时间用字符串表示,其形式为时:分:秒
#define TRACE_NONE 0
#define TRACE_FATAL 1
#define TRACE_ERROR 2
#define TRACE_WARNING 3
#define TRACE_INFO 4
#define TRACE_DEBUG 5
#define TRACE_LEN_MAX 64
extern int *TraceLevel;
extern char TraceName[TRACE_LEN_MAX + 1];
#define Log(A, format,args...) \
((TraceLevel == NULL || TraceName == NULL || *TraceLevel < (A)) ? 0 : LogMsg(A, __FILE__, __LINE__, format, ##args))
#define LogFatal(format,args...) \
Log(TRACE_FATAL, format, ##args)
#define LogError(format,args...) \
Log(TRACE_ERROR, format, ##args)
#define LogWarning(format,args...) \
Log(TRACE_WARNING, format, ##args)
#define LogInfo(format,args...) \
Log(TRACE_INFO, format, ##args)
#define LogDebug(format,args...) \
Log(TRACE_DEBUG, format, ##args)
int LogMsg(int level, const char *filename,
int line, const char *fmt, ...)
{
va_list ap;
FILE *fp;
char sLogFile[128 + 1];
char sCurrTime[6 + 1];
struct timeb tTimeB;
char sMilliTM[4];
memset(sLogFile, 0, sizeof(sLogFile));
LogFile(sLogFile);
GetTime_HHMMSS(sCurrTime);
memset(&tTimeB, 0, sizeof(tTimeB));
ftime(&tTimeB);
snprintf(sMilliTM, sizeof(sMilliTM), "%03d", tTimeB.millitm);
fp = fopen(sLogFile, "a+");
if (fp != (FILE*)NULL) {
fprintf(fp, "[%08d][%.6s:%.3s][%16s][%04d][%7s]",
getpid(), sCurrTime, sMilliTM, filename, line, g_LevelDsp[level]);
va_start(ap, fmt);
vfprintf(fp, fmt, ap);
va_end(ap);
fprintf(fp, "\n");
fflush(fp);
fclose(fp);
}
return 0;
}
再在后台进程中设置TraceLevel和TraceName即可。
在跑一个贷记卡的主卡注销文件通知和下载的批量时,由于贷记卡的系统暂时有问题,需要把我们业务中的这两个批停掉。而这两个批是读取数据库中一个表的字段值来判断是否执行的,所以停掉这两个批,就需要修改表中字段值,修改脚本如下:
##########################################################
#修改表TBL_BAT_TASK_CTL的USE_FLAG字段,启动或停止贷记卡销卡的批处理
#useage:执行脚本时加-n参数就是关闭批处理
# 执行脚本时加-y参数就是打开批处理
##########################################################
#!/bin/bash
if test "$1" = "-n"
then
db2 connect to $DBLINK
db2 "update TBL_BAT_TASK_CTL set USE_FLAG='N' where BAT_ID='0024' or BAT_ID='0025'"
db2 terminate
# exit 0
fi
if test "$1" = "-y"
then
db2 connect to $DBLINK
db2 "update TBL_BAT_TASK_CTL set USE_FLAG='Y' where BAT_ID='0024' or BAT_ID='0025'"
db2 terminate
fi
在Linux服务器上跑一个应用系统,则要搭建应用系统运行的环境,主要分两个方面:
一、新建用户,该用户下应用系统的一些环境变量的设置,这个需要修改家目录下的隐藏文件.bash_profile和.bashrc,在.bash_profile中去执行一些应用系统的环境配置脚本。
二、有关数据库的设置
1、首先要安装ibm db2数据库客户端。
2、建数据库通信节点(catlog)
由于数据库服务器和应用系统服务器不在同一台机器上,所以要建编目节点,把远程服务器映射到本地。
db2 catlog tcpip node P570 remote 172.10.10.10 server 5000
db2 catlog db REMOTEDB at node P570
db2 terminate
3、由于存在开发环境、编译环境、测试环境和生产环境四个环境,不同环境用的的数据库也不一样,所以需要设置一下连接数据库的环境变量。
export “DBLINK=tsmpdb user xxx using xxx”
(由于行内用的就是db2数据库,所以上述例子中就是db2的,还有一些命令是网上找的,实际的用相应的值替换即可。)
1、连接数据库
db2 connect to $DBLINK
2、db2 list tables
3、db2 describe table 表名
4、查询
db2 “select * from 表名”
db2 “select * from TBL_HCE_CARD_APPLY where TXN_DATE between ‘xxx’ and ‘xxx’”
db2 “select * from TBL_BLACK_LIST_INFO where BLACK_INVALID_TIME >= (select to_char(current timestamp,’yyyymmddhh24miss’) from sysibm.dual)”
5、更新修改
db2 “update TBL_BAT_TASK_CTL set USE_FLAG=’N’ where BAT_ID = ‘0024’ or BAT_ID=’0025’”