一、背景介绍
我创建了两个Springboot工程,A作为公共工程,打包成Jar包供其他工程使用;B使用A提供的Jar包。A工程中的SceneCaseDebugAPI类的代码如下:
import com.alibaba.fastjson.JSONArray;
import com.alibaba.fastjson.JSONObject;
import com.jayway.jsonpath.JsonPath;
import com.umetrip.qa.model.*;
import com.umetrip.qa.service.GatewayCaseService;
import com.umetrip.qa.service.SceneCaseService;
import com.umetrip.qa.service.ScenePublicParameterService;
import com.umetrip.qa.service.SceneStepsService;
import com.umetrip.qa.tester.singlecase.SingleCaseDebugAPI;
import com.umetrip.qa.utils.JsonStringUtils;
import com.umetrip.qa.utils.OkhttpUtil;
import com.umetrip.qa.utils.PrettyTimeTool;
import com.umetrip.qa.utils.request.ApiRequestCommonUtil;
import com.umetrip.qa.utils.request.RequestHttp_error;
import org.jsoup.helper.StringUtil;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;
import java.util.regex.Pattern;
/**
* @description: 普通java类要使用 @Autowired注解 的话,需要使用@Component注解申明该类是spring组件
* @time: 2021-02-23 11:12
*/
@Component
public class SceneCaseDebugAPI {
@Autowired
static SceneCaseService sceneCaseService;
@Autowired
static SceneStepsService sceneStepsService;
@Autowired
static ScenePublicParameterService scenePublicParameterService;
@Autowired
static GatewayCaseService gatewayCaseService;
@Autowired
static SingleCaseDebugAPI singleCaseDebugAPI;
public static ApiResult sceneCaseCommonRequest(JSONObject sceneCaseJson){
ApiResult apiResult = new ApiResult();
//步骤1:从DB中查询出场景用例的详细信息
System.out.println("sceneCase: " + JSONObject.toJSONString(sceneCaseJson));
SceneCase sceneCase = JSONObject.parseObject(JSONObject.toJSONString(sceneCaseJson), SceneCase.class);
String singleSceneCaseRedisKey = "SceneCase" + "_" + sceneCase.getId() + "_" + sceneCase.getSceneName() + "_" + "Result";
List<SceneSteps> sceneSteps = sceneStepsService.findAllSceneStepsBySceneCaseId(sceneCase.getId());
// 清空redis中场景用例的用例步骤信息
// baseRedisDao.remove(singleSceneCaseRedisKey);
//设置一个数组存储场景用例步骤的结果
List<String> stepsResult = new ArrayList<>();
try{
for(SceneSteps step : sceneSteps){
System.out.println("当前正在处理的场景用例步骤信息为:" + JSONObject.toJSONString(step));
JSONObject stepJson = JSONObject.parseObject(JSONObject.toJSONString(step), JSONObject.class);
String sceneStepType = step.getSceneStepType();
if(sceneStepType.equals("HTTP用例")){
ResMsg resMsg = executeHttpCaseStep(sceneCase.getId(), step);
stepJson = JSONObject.parseObject(JSONObject.toJSONString(resMsg), JSONObject.class);
stepsResult.add(resMsg.getResult());
}else if(sceneStepType.equals("等待")){
Thread.sleep(step.getSceneStepWaitingtime());
}else if(sceneStepType.equals("GET请求")){
ResMsg resMsg = executeGetRequestStep(sceneCase.getId(), step);
stepJson = JSONObject.parseObject(JSONObject.toJSONString(resMsg), JSONObject.class);
stepsResult.add(resMsg.getResult());
}else if(sceneStepType.equals("条件")){
stepJson.put("debugCaseStartTime", new Date());
boolean result = executeConditionStep(sceneCase.getId(), step);
if(result){
stepJson.put("sceneCaseStepResult", "T");
stepsResult.add("T");
}else {
stepJson.put("sceneCaseStepResult", "F");
stepsResult.add("F");
}
}
PrettyTimeTool.createNewInterfaceCaseDebugLog(stepJson.getDate("debugCaseStartTime"), "SingleInterfaceCase", sceneCase.getId(), stepJson);
}
}catch (Exception e){
e.printStackTrace();
apiResult.setCode(400);
apiResult.setResult("场景用例执行失败");
}
// 还需要增加场景用例结果判断逻辑,只要有一个失败步骤失败,场景用例结果即为失败;否则为成功
if(stepsResult.contains("F")){
apiResult.setCode(400);
apiResult.setResult("场景用例执行失败");
}else {
apiResult.setCode(200);
apiResult.setResult("场景用例执行成功");
}
return apiResult;
}
public static ResMsg executeGetRequestStep(Integer sceneCaseId, SceneSteps step){
System.out.println("执行Get请求步骤");
ResMsg resMsg = new ResMsg();
ApiRequestCommonUtil apiRequestCommonUtil = new ApiRequestCommonUtil();
String newUrl = step.getSceneStepGetUrl();
JSONArray params = JSONArray.parseArray(step.getSceneStepGetParams());
JSONArray header = JSONArray.parseArray(step.getSceneStepGetHeader());
System.out.println("请求参数的数据:" + params.size() + " 请求头的数据:" + header.size());
for(int i = 0; i < params.size(); i++){
JSONObject paramsTableOneRow = params.getJSONObject(i);
String key = paramsTableOneRow.getString("key");
String val = paramsTableOneRow.getString("value");
if(val.contains("${")){
String dbParam = val.substring(2, val.length() - 1);
System.out.println("1 dbParam: " + dbParam);
// 从场景内公共参数查找变量,当查找不到时,需要去全局公共参数中查找(目前尚未实现)
ScenePublicParameter scenePublicParameter = apiRequestCommonUtil.findPublicParamValue(sceneCaseId, dbParam);
val = scenePublicParameter.getSceneParameterVal();
}
if (i == 0) {
System.out.println("2 newUrl: " + newUrl + " key:" + key + " val:" + val);
newUrl = newUrl + "?" + key + "=" + val;
} else {
System.out.println("3 newUrl: " + newUrl + " key:" + key + " val:" + val);
newUrl = newUrl + "&" + key + "=" + val;
}
}
System.out.println("处理后的url:" + newUrl);
// 处理header
if(header.size() != 0){
for(int j = 0; j < header.size(); j++){
JSONObject headerTableOneRow= header.getJSONObject(j);
String key = headerTableOneRow.getString("key");
String val = headerTableOneRow.getString("value");
if(val.contains("${")){
String dbParam = val.substring(2, val.length() - 1);
System.out.println("1 dbParam: " + dbParam);
ScenePublicParameter scenePublicParameter = apiRequestCommonUtil.findPublicParamValue(sceneCaseId, dbParam);
val = scenePublicParameter.getSceneParameterVal();
}
headerTableOneRow.put("value", val);
}
}
System.out.println("header str: " + JSONArray.toJSONString(header));
long startTime = System.currentTimeMillis();
Date deBugCaseStartTime = new Date();
// 请求url和header已经准备完毕,真正发送GET请求
String getResult = OkhttpUtil.dynamicGet(newUrl, header);
long endTime = System.currentTimeMillis();
long consumetime = endTime - startTime;
// 判断是否需要将响应体中数据存储至场景内的公共参数表
if(!StringUtil.isBlank(step.getSceneStepExtractJsonPath())){
String[] jsonPathList = step.getSceneStepExtractJsonPath().split("\r|\n");
for(int i = 0; i<jsonPathList.length; i++){
getResponseBodyJsonPathVal(sceneCaseId, getResult, jsonPathList[i]);
}
}
//需要加断言处理
System.out.println("Get请求的返回结果:" + getResult);
RequestHttp_error requestHttp = new RequestHttp_error();
requestHttp.assertResponse(getResult, step.getSceneStepGetExpectResponse(), resMsg);
resMsg.setConsumetime(consumetime);
resMsg.setInterfaceType("SceneCase");
resMsg.setRequestBodyParameters(JSONObject.toJSONString(params));
resMsg.setResponsebody(getResult);
resMsg.setUrl(newUrl);
resMsg.setHttpHeaders(JSONObject.toJSONString(header));
resMsg.setTestpoint(step.getSceneStepName());
resMsg.setDebugCaseStartTime(deBugCaseStartTime);
return resMsg;
}
public static boolean executeConditionStep(Integer sceneCaseId, SceneSteps step){
System.out.println("执行条件步骤");
boolean res = false;
ApiRequestCommonUtil apiRequestCommonUtil = new ApiRequestCommonUtil();
String firstExpression = step.getSceneFirstExpression();
String secondExpression = step.getSceneSecondExpression();
String judgeCondition = step.getSceneExpressionJudgeCondition();
if(secondExpression.contains("${")){
String dbParam = secondExpression.substring(2, secondExpression.length() - 1);
System.out.println("1 dbParam: " + dbParam);
ScenePublicParameter scenePublicParameter = apiRequestCommonUtil.findPublicParamValue(sceneCaseId, dbParam);
secondExpression = scenePublicParameter.getSceneParameterVal();
}
switch (judgeCondition){
case "等于":
res = firstExpression.equals(secondExpression);
break;
case "不等于":
res = !firstExpression.equals(secondExpression);
break;
case "包含":
res = firstExpression.contains(secondExpression);
break;
case "不包含":
res = !firstExpression.contains(secondExpression);
break;
case "大于":
res = Integer.parseInt(firstExpression) > Integer.parseInt(secondExpression);
break;
case "小于":
res = Integer.parseInt(firstExpression) < Integer.parseInt(secondExpression);
break;
}
System.out.println("条件步骤的运行结果:" + res);
return res;
}
public static ResMsg executeHttpCaseStep(Integer sceneCaseId, SceneSteps step){
System.out.println("执行Http用例步骤");
Integer gatewaySingleCaseId = step.getSceneGatewaySingleCaseId();
GatewayCase gatewayCase = gatewayCaseService.findByPrimaryKey(gatewaySingleCaseId);
System.out.println("gateway case rparams before:" + gatewayCase.getRparams());
JSONObject jsonObjectDB = JSONObject.parseObject(JsonStringUtils.StringInDBtoJSONString(gatewayCase.getRparams()));
ApiRequestCommonUtil.processRequestParameterization(sceneCaseId, jsonObjectDB);
System.out.println("gateway case rparams after:" + JsonStringUtils.JSONStringtoStringInDB(JSONObject.toJSONString(jsonObjectDB)));
gatewayCase.setRparams(JsonStringUtils.JSONStringtoStringInDB(JSONObject.toJSONString(jsonObjectDB)));
JSONObject jsonObject = JSONObject.parseObject(JSONObject.toJSONString(gatewayCase), JSONObject.class);
jsonObject.put("env", "gray");
jsonObject.put("interfaceType", "GateWay");
Date deBugCaseStartTime = new Date();
ResMsg resMsg = singleCaseDebugAPI.singleCaseDebug(jsonObject);
resMsg.setDebugCaseStartTime(deBugCaseStartTime);
// jsonpath
if(!StringUtil.isBlank(gatewayCase.getExtractJsonPath())){
String[] jsonPathList = gatewayCase.getExtractJsonPath().split("\r|\n");
System.out.println("executeHttpCaseStep jsonPathList length:" + jsonPathList.length);
for(int i = 0; i<jsonPathList.length; i++){
getResponseBodyJsonPathVal(sceneCaseId, resMsg.getResponsebody(), jsonPathList[i]);
}
}
System.out.println("Http用例执行的结果:" + JSONObject.toJSONString(resMsg));
return resMsg;
}
public static void getResponseBodyJsonPathVal(Integer sceneCaseId, String responseBody, String jsonPathKey){
System.out.println("jsonPathKey: " + jsonPathKey);
try{
Object actualValue = JsonPath.read(responseBody, jsonPathKey);
System.out.println("actualValue: " + actualValue.toString());
// 存入数据库的字段,以jsonpath分隔出的最后一个为准
String[] paramList = jsonPathKey.split("\\.");
System.out.println("paramList length: " + paramList.length);
String param = paramList[paramList.length - 1];
System.out.println("param: " + param);
boolean isNumber = isNumber(actualValue.toString());
boolean isBoolean = isBoolean(actualValue.toString());
ScenePublicParameter scenePublicParameterDB = scenePublicParameterService.findPublicParamValueBySceneCaseIdAndName(sceneCaseId, param);
if(scenePublicParameterDB == null){
System.out.println("新增~~");
ScenePublicParameter scenePublicParameter = new ScenePublicParameter();
scenePublicParameter.setSceneCaseId(sceneCaseId);
scenePublicParameter.setSceneParameterName(param);
if(isNumber){
// 这种类型怎么办
scenePublicParameter.setSceneParameterType("整数");
scenePublicParameter.setSceneParameterVal(String.valueOf(actualValue));
} else if(isBoolean){
scenePublicParameter.setSceneParameterType("布尔");
scenePublicParameter.setSceneParameterVal(String.valueOf(actualValue));
} else {
scenePublicParameter.setSceneParameterType("字符串");
scenePublicParameter.setSceneParameterVal((String)actualValue);
}
scenePublicParameterService.insert(scenePublicParameter);
}else {
System.out.println("更新~~");
if(isNumber){
scenePublicParameterDB.setSceneParameterVal(String.valueOf(actualValue));
} else if(isBoolean){
scenePublicParameterDB.setSceneParameterVal(String.valueOf(actualValue));
} else {
scenePublicParameterDB.setSceneParameterVal((String)actualValue);
}
scenePublicParameterService.update(scenePublicParameterDB);
}
}catch (Exception e){
e.printStackTrace();
}
}
public static boolean isNumber(String str) {
//采用正则表达式的方式来判断一个字符串是否为数字,这种方式判断面比较全
//可以判断正负、整数小数
boolean isInt = Pattern.compile("^-?[1-9]\\d*$").matcher(str).find();
boolean isDouble = Pattern.compile("^-?([1-9]\\d*\\.\\d*|0\\.\\d*[1-9]\\d*|0?\\.0+|0)$").matcher(str).find();
return isInt || isDouble;
}
public static boolean isBoolean(String str){
return str.equals("true") || str.equals("false");
}
}
其中的sceneCaseService、sceneStepsService…等五个对象是使用@Autowired从Spring容器中获取的,并都声明成了静态变量。但是在使用上述变量时发现,上述五个变量会报空指针错误
二、问题分析
我们都知道静态变量,静态方法都是属于类的;普通变量、普通方法都是属于实例化对象的(即new处理的对象);Spring依赖注入是在容器中实例化对象;
当我们使用静态变量或静态方法时,不需要new出来一个类的实例化对象,所以使用@Autowired修饰一个静态变量时,该静态变量并没有真正实例化成一个对象,因此该静态变量为null
三、解决方法
3.1 将@Autowired加到构造方法上
@Component
public class RuleFunctions {
@Autowired
private static RuleEntityItemInfoBiz ruleEntityItemInfoBiz;
@Autowired
public RuleFunctions(RuleEntityItemInfoBiz ruleEntityItemInfoBiz) {
RuleFunctions.ruleEntityItemInfoBiz = ruleEntityItemInfoBiz;
}
public static double calculateCurrentMonthIncomeTax(String fileId, String salaryMonth, String taxPlanId){
//此处省略方法实现逻辑
}
}
3.2 使用set方法
@Component
public class RuleFunctions {
@Autowired
private static RuleEntityItemInfoBiz ruleEntityItemInfoBiz;
@Autowired
public void setRuleEntityItemInfoBiz(RuleEntityItemInfoBiz ruleEntityItemInfoBiz) {
RuleFunctions.ruleEntityItemInfoBiz = ruleEntityItemInfoBiz;
}
public static double calculateCurrentMonthIncomeTax(String fileId, String salaryMonth, String taxPlanId){
//此处省略方法实现逻辑
}
}
3.3 使用@PostConstruct注解
3.3.1 @PostConstruct注解
@PostConstruct注解位于javax.annotation包下,可以用来修饰一个非静态的void返回的方法。
被@PostConstruct注解修饰的方法的执行时机是:在服务器加载Servlet的时候运行,并且只会被服务器加载一次;在构造函数之后,init()方法之前执行
被@PostConstruct注解修饰的方法参数规范:除了拦截器这种特殊情况以外,其他情况都不允许有参数,否则Spring框架会报IllegalStateException;而且返回值要是void,但实际也可以有返回值,至少不会报错,只会忽略
Spring框架中使用到@PostConstruct注解,该注解修饰的方法在整个Bean初始化过程中的执行顺序如下:
Constructor(构造方法) -> @Autowired(依赖注入) -> @PostConstruct(注释的方法)
3.3.2 实际例子
@Component
public class RuleFunctions {
@Autowired
private RuleEntityItemInfoBiz ruleEntityItemInfoBiz;
private static RuleEntityItemInfoBiz staticRuleEntityItemInfoBiz;
/**
*注释用于在完成依赖注入以后执行任何初始化之后需要执行的方法。必须在类投入使用之前调用此方法。
*/
@PostConstruct
public void beforeInit() {
staticRuleEntityItemInfoBiz = ruleEntityItemInfoBiz;
}
public static double calculateCurrentMonthIncomeTax(String fileId, String salaryMonth, String taxPlanId){
//此处省略方法实现逻辑
}
}
参考博文:spingboot使用@Resource注入静态变量报空指针的问题解决