java调用python脚本失败_java调用python的惨痛史(无法获取环境变量)

环境:java,was,python2.6,红帽linux,oracle,python用cx_Oracle

事情是这样的,有个需求,需要对数据库进行处理,简单说就是把数据取出来,用python使用外部传参做一个处理,再写回到数据库。如果使用java直接搞的话需要绕很大一个圈,比较麻烦,用python的话就很简单了,于是就有了下面的故事,希望和我遇到同样问题的同学能对你们有一些帮助,少绕一些圈子,java和python功能开发都很顺利,于是到了java调用python环节

大概把需要用到的功能写两个脚本,一个是用于连接数据库的,一个是用来实现功能的,方便后面说明

dbconn.py 这个用来连接数据库

#!/usr/bin/python#coding: UTF-8

importcx_Oracledefora_conn():

user_name= 'user'passwd= 'pass'host= 'localhost/orcl'conn=cx_Oracle()return conn

case.py 这个用来加工数据

'''接收传入参数

使用传入参数作为where条件到oracle中进行查找

对数据进行处理

写回到oracle'''

#!/usr/bin/python#coding: UTF-8

importosimportsys#尝试获取环境变量my_home

my_home = os.getenv('my_home')if notmy_home:

my_home= '/home/user_name'

#cx_Oracle脚本是放在pypk目录里的,因此需要增加这个地方

sys.path.append('%s/pypk' %my_home)importdbconn#接收参数,为了防止后面因为手误将此变量变更这里用global声明下全局,可有可无

globalout_into

out_into= sys.argv[1]#获取数据库数据

defget_data():print 'get data'conn=dbconn.ora_conn()

conn_cur=conn.cursor()

sql= 'select col1,col2 from tablename where col1 = %r' %out_into

data_values=conn_cur.execute(sql)

result=list(data_values)

conn_cur.close()

conn.close()returnresult#处理,为了方便,假设就是在col1前面加a和col2前面加b,然后拼在一起用;分隔

defmanage_data(values):print 'manage data'col1= 'a' +str(values[0])

col2= 'b' + str(values[1])

col_return= col1 + ';' +col2returncol_return#写入数据库d

definto_data(into_values):print 'insert into data'conn=dbconn.ora_conn()

conn_cur=conn.cursor()

sql= 'insert into result_table(col) values(%r)' %into_values

conn_cur.execute(sql)

conn.commit()

conn_cur.close()

conn.close()defmain():print 'run start'datas=get_data()for line indatas:

ok_values=manage_data(line)

into_data(ok_values)if __name__ == '__main__':

main()

exit()

因为不知道java调用效果怎样,先写了个简单的python脚本用java调用测试了下

test.py 这个只是用来测试功能,所以只有一行代码

print 'hello'

测试成功,这里因为这边没有环境,就不截图了,大家凑合看吧。又加入两行,测试了一下传参

test.py

print 'hello'a= sys.argv[1]print a

一样成功了,信心满满的上正式脚本来测试,正常打印了脚本中的内容

run start

get data

manage data

insert into data

去数据库里面查,结果目标表result_table没有数据。我擦,这什么鬼,没报错也没有处理,完全没有头绪,尝试加上try看看有没有什么帮助。于是case.py的main函数就变成了下面这样

defmain():print 'run start'

try:

datas=get_data()for line indatas:

ok_values=manage_data(line)

into_data(ok_values)exceptException, e:print e

这次果然有结果了,cx_Oracle连接失败,缺少libclntsh.so.11.1,这里着重说明下,网上都是win调用cx_Oracle的,需要拷贝什么dll,这个在linux中是没用的,而且就算是win也不是像上面说的那样把一堆dll拷贝到脚本当前目录里面,这样只是治标不治本。

正确的方法是加三个环境变量

export ORACLE_HOME=oracle_install_path #第一个当然是ORACLE_HOME

export PATH=$PATH:$ORACLE_HOME/bin #第二个是bin

export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:$ORACLE_HOME/lib #第三个是lib

我不想留坑,建议这三个变量加到~/.bashrc里面,不过这个并没有解决根本问题,后面会说明,请各位继续往下看

加入环境变量以后又执行了下,报错connect to Oracle Database,尼玛犬加上了啊,于是在本地linux服务器执行了下,结果。。。。。。正常执行,完全问题,这次真是见鬼了啊。某楠说的好,真相只有一个,特么就不信了,我倒要看看是出什么问题了,根据以往经验,环境变量加失败的话可能是使用用户和加环境变量的用户不一致,于是在case.py里面加上一段

importgetpass

now_user= getpass.getuser() #用来获取当前操作用户

print now_user

结果是root,对啊,是root啊,而且我也是SB了,.bashrc这个不是应该全局的么,就是全部用户通用的环境变量啊。算了,死马当活马医,倒要看看环境变量里都是什么玩意,在case.py里面再加一行

os.system('env') #调用linux的外部命令,查看全部环境变量

然后,意外出现了,打印出来的环境变量里面根本就没有加入的三个环境变量,而且大部分都不是linux里面的环境变量,好多都是was用的,难道是was有自己的环境变量么,问了下同事,果然坑啊,真的有自己的环境变量配置。这下问题找到了,那么解决办法就是把这三个加到was自有的环境变量里面就好了嘛,于是。。。。。。在was里加入了ORACLE_HOME、PATH、LD_LIBRARY_PATH这三个环境变量。

重新测试下,尼玛刚才加入的居然不在里面,这就尴尬了,怎么办。。。怎么办。。。在分针从12绕到11的时候,终于,办法有了,把环境变量作为外部命令加进去不就行了,继续在case.py中加入三行

os.system(export ORACLE_HOME=oracle_install_path)

os.system(export PATH=$PATH:$ORACLE_HOME/bin)

os.system(export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:$ORACLE_HOME/lib)

这时候我都快成福尔摩斯了,感觉已经超越某楠了有木有,但是事实仍然给了我当头一棒,失败了,打印出来的环境变量里这三个依然不存在的,原因是这样加入的环境变量不会直接有效的,需要注销才可以生效,简直崩溃了,然后分针又默默的从12走到了11。突然灵感一现,如果用python自有的加环境变量的方法能不能绕过注销这个步骤呢,说弄就弄,于是将case.py中的

os.system(export ORACLE_HOME=oracle_install_path)

os.system(export PATH=$PATH:$ORACLE_HOME/bin)

os.system(export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:$ORACLE_HOME/lib)

替换成下面的

oracle_home_env = [['ORACLE_HOME','oracle_install_path'],['PATH','$PATH:$ORACLE_HOME/bin'],['LD_LIBRARY_PATH','$LD_LIBRARY_PATH:$ORACLE_HOME/lib']]for env_line inoracle_home_env:

os.environ[env_line[0]]= env_line[1]

好吧,我承认,这个办法不是我想出来的,能相出这种加环境变量方法的人真是大神啊

继续我们的故事,然后,奇迹出现了,这三个环境变量终于在打印的结果中出现了,但是依然报错连接不上数据库,不过这就好办了,既然这个环境变量指向的linux,那只要用linux的方法调用脚本就好了嘛。说弄就弄,新加了一个脚本,用来调用case.py,只有几行

call_script.py 用来使用外部命令调用case.py

#!/usr/bin/python#coding: UTF-8

importos

out_into= sys.argv[1]

my_home= os.getenv('my_home')if notmy_home:

my_home= '/home/user_name'os.system('python %s/case.py %s' %(my_home,out_into))

到这里故事就结束了,不过就像大部分剧本一样,总要有彩蛋嘛哈哈

以为这样就结束了么?当然不是,有没有看到my_home,这个也是一个环境变量!这里没有报错的原因是因为在下面加入了一个if,如果没有获取到的话默认/home/user_name,但是路径换了呢?这一路升级的,感觉小棍子都能升级到金箍棒了,这点小事还搞不定么,好了,不说了,我要去吃点什么庆祝了。

春节了,一年了,终于等到这一天了,能看到这个文也算是缘分,在这里就祝大家在新的一年里 多多赚钱、少一些坑、身体健健康康、立的flag都能实现!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值