restlet和ajax,记录Spring Boot小项目的一些坑

现有一个用spring boot的后台项目,项目开发rest端口与前端交互,使用Jython调用本地python代码。项目基于IDEA开发,部署在windows系统中。

第一个坑:跨域请求

前端使用ajax请求后台接口,后台返回json数据。后台独立测试(curl、restlet)没问题,前端出现报错跨域问题。

首先我们试图在ajax请求中增加dataType并设定为jsonp,结果没有报跨域问题,返回直接进入了error,状态码200,一些博客认为是后台数据不够严谨,不是严格的json格式,然后并不是。

$.ajax({ //请求方式

type: "GET",

contentType:"application/json;charset=UTF-8",//请求的媒体类型

url: "http://127.0.0.1:8088/search?carId=" + carId, //请求地址

dataType: "jsonp",

success:function(result) {

...

最终解决问题的办法是通过后台来支持跨域,这里还有一个小坑,在springboot1和springboot2中支持跨域的处理方式不相同。

在springboot2之下,我们可以通过如下两步来增加跨域支持。

1、写一个配置类,实现增加跨域映射

importorg.springframework.context.annotation.Configuration;importorg.springframework.web.servlet.config.annotation.CorsRegistry;importorg.springframework.web.servlet.config.annotation.EnableWebMvc;importorg.springframework.web.servlet.config.annotation.WebMvcConfigurer;

@Configuration

@EnableWebMvcpublic class CorsConfig implementsWebMvcConfigurer {public voidaddCorsMappings(CorsRegistry registry){

registry.addMapping("/**")//设置允许跨域请求的域名

.allowedOrigins("*")//是否允许证书 不再默认开启

.allowCredentials(true)//设置允许的方法

.allowedMethods("*")//跨域允许时间

.maxAge(3600);

}

}

2、利用注解配置Controller

@CrossOrigin(maxAge = 3600)

@RestController

@RequestMapping("/")public classIndexController {privateAlgorithmModel algorithmModel;privateFileModel fileModel;

@CrossOrigin(origins= "http://localhost:63342")

@GetMapping(value= "/search")publicResult search(@RequestParam String carId)

{

List routes = null;//check if file exists

if(fileModel.fileExists(carId)){

routes=fileModel.genRoutes(carId);

}else{if(algorithmModel.call(carId) != null){

routes=fileModel.genRoutes(carId);

}else{return Result.fail("1");

}

}return Result.success("0", routes);

}

}

PS:注解的位置比较灵活

可以直接注解整个类

可以将部分注解类,然后将另一部分注解方法上

或者直接注解方法

第二个坑:打包运行时Jython下的PythonInterpreter报错

因为要调用外部的python代码,我使用了Jython。将PythonInterpreter设置为bean,并组装进其他的bean中。

@BeanpublicPythonInterpreter getPythonInterpreter() {

PythonInterpreter pyInterpreter= newPythonInterpreter();returnpyInterpreter;

}

在idea中运行的时候没有任何问题,打包运行时报错如下:

...

Error starting ApplicationContext. To display the conditions report re-run your application with ‘debug‘enabled.2019-06-24 16:56:04.915 ERROR 51338 ---[ main] o.s.boot.SpringApplication : Application run failed

org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name‘indexController‘ defined in URL [jar:file:/Users/zhengshuangxi/Desktop/car/vehicle-web-0.0.2-SNAPSHOT.jar!/BOOT-INF/classes!/cn/xidian/sxzheng/vehicleweb/controller/IndexController.class]: Unsatisfied dependency expressed through constructor parameter 0;

nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name‘getAlgorithmModel‘ defined in class path resource [cn/xidian/sxzheng/vehicleweb/config/MyConfig.class]: Bean instantiation via factory method failed;

nested exception is org.springframework.beans.BeanInstantiationException: Failed to instantiate [cn.xidian.sxzheng.vehicleweb.model.AlgorithmModel]: Factory method‘getAlgorithmModel‘threw exception;

nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name‘getPythonInterpreter‘ defined in class path resource [cn/xidian/sxzheng/vehicleweb/config/MyConfig.class]: Bean instantiation via factory method failed;

nested exception is org.springframework.beans.BeanInstantiationException: Failed to instantiate [org.python.util.PythonInterpreter]: Factory method‘getPythonInterpreter‘threw exception;

nested exception is ImportError: Cannot import site module and its dependencies: No module named site

...

根据报错信息可以得知 getPythonInterpreter 方法报错,报错信息为 "Cannot import site module and its dependencies: No module named site" ,这里的原因是没有site模块,解决办法就是修改getPythonInterpreter方法

@BeanpublicPythonInterpreter getPythonInterpreter() {

Properties props= newProperties();

props.put("python.home", "../jython-2.7.0");

props.put("python.console.encoding", "UTF-8");

props.put("python.security.respectJavaAccessibility", "false");

props.put("python.import.site", "false");

Properties preprops=System.getProperties();

PythonInterpreter.initialize(preprops, props,new String[0]);

PythonInterpreter pyInterpreter= newPythonInterpreter();returnpyInterpreter;

}

修改代码处理之后,site的问题没有了,但是后来又报错“os模块找不见”。这些问题的原因主要是因为在java中使用Jython运行python程序时候,它的搜索路径需要手动设置。而且Jython有个致命的弱点,就是它支持的库太少了,像一些科学计算库numpy等都比较麻烦,因此我后来改用Runtime执行python文件,同时我也强烈推荐大家使用Runtime执行外部程序,并通过io流收集结果。

publicString call(String carId){

Process process;

String command= "python3 " + pythonFilePath + " " +carId;

System.out.println(command);

String result= "";try{

process=Runtime.getRuntime().exec(command);

BufferedReader in= new BufferedReader(newInputStreamReader(process.getInputStream()));

String line= null;while((line = in.readLine()) != null){

result=result.concat(line);

}

in.close();

process.waitFor();

}catch(IOException e){

e.printStackTrace();

}catch(InterruptedException e){

e.printStackTrace();

}returnresult;

}

第三个坑:读取外部属性文件

因为后台开发完成之后需要打包运行在别的机器上,因此关于python程序位置的设置,一些外部的参数设置就必须通过外部配置文件来完成,写入工程里面的application.properties不能进行修改,所以必须要实现可以读取外部属性配置文件。每次修改属性文件后重新启动程序就可以生效。

像图中的application.properties文件最终是打包到jar中的,无法进行修改。我们需要类似右边的情况,在config中对配置文件进行随时修改,然后可以通过重新运行jar文件来使得修改生效。

8dbbc3af9c0f53b69456db2f7db60592.png

bfa47bcd05e4ac1358b9f2fd7bfd38bd.png

打包部署后,我们可以直接在jar的当前目录下新建config文件夹,并新建application.properties文件,外部文件的优先级大于内部配置文件,因此我们可以手动修改application.properties来实现配置参数在生产环境中的修改。

原文:https://www.cnblogs.com/zhengshuangxi/p/11079420.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值