我的代码有一个奇怪的问题。
我已经实现了template method pattern
来提供相同alghoritm的不同实现。
所以我创建了抽象类:
@Service
public abstract class ExcelRdi_Iupr_Sl {
@Autowired
private Environment env;
private static String PROPERTY_NAME_FILESYSTEM_EXCELPATH = "temporary.excelpath";
private XSSFWorkbook workbook;
private XSSFCellStyle unlockedNumericStyle;
private XSSFSheet sheet;
/**
* Create the excel file with the RDI,IUPR or SL
* @param car
* @param fileName
* @return
* @throws Exception
*/
public final String retrieve(Car car, String fileName) throws Exception{
writeInitialize();
Map<Integer,Integer> defMap = firstTwoRow(car, sheet);
elaboration(car, sheet, unlockedNumericStyle, defMap);
return writeFile(car, fileName);
}
/**
* Import the excel file with the RDI,IUPR or SL
* @param file
* @param car
* @throws Exception
*/
public final int update(MultipartFile file, Car car) throws Exception{
String filePath = saveFile(file, car);
readInitialize(filePath);
return updateRdi(car, sheet);
}
/**
* Save the imported file into file system inside a temporary folder
* @param file
* @param car
* @return
* @throws Exception
*/
private String saveFile(MultipartFile multipartFile, Car car) throws Exception {
String path = env.getRequiredProperty(PROPERTY_NAME_FILESYSTEM_EXCELPATH) + File.separator + String.valueOf(System.currentTimeMillis());
File file = new File (path+ File.separator + multipartFile.getOriginalFilename());
file.getParentFile().mkdirs();
multipartFile.transferTo(file);
return file.getAbsolutePath();
}
/**
* Initialize the object for the creating procedure
*/
protected void writeInitialize(){
//initialize class variable
}
/**
* Initialize the object for the reading procedure
* @param filePath
* @throws Exception
*/
protected void readInitialize(String filePath) throws Exception{
//read ile and initialize class variable
}
/**
* Write the created file inside the file system
* @param car
* @param fileName
* @return
* @throws Exception
*/
private String writeFile(Car car, String fileName) throws Exception{
//write on file
}
/**
* This excel creating method has to be implemented by the concrete class
* @param car
* @param sheet
* @throws Exception
*/
protected abstract Map<Integer,Integer> firstTwoRow(Car car, XSSFSheet sheet) throws Exception;
/**
* This excel creating method has to be implemented by the concrete class
* @param car
* @param sheet
* @param defMap
* @throws Exception
*/
protected abstract void elaboration(Car car, XSSFSheet sheet, XSSFCellStyle unlockedNumericStyle, Map<Integer,Integer> defMap);
/**
* This excel import method has to be implemented by the concrete class
* @param car
* @param sheet
* @return number of elaborated row
* @throws Exception
*/
protected abstract int updateRdi(Car car, XSSFSheet sheet) throws Exception;
}
扩展上述类的其中一个类具有此实现:
@Component
public class ExcelRdi extends ExcelRdi_Iupr_Sl {
@Autowired
private RdiServices rdiServices;
@Autowired
private AcquisitionServices acquisitionServices;
static final Logger LOG = LoggerFactory.getLogger(ExcelRdi.class);
/**
* Create the first two row of excel file
*/
@Override
protected Map<Integer,Integer> firstTwoRow(Car car, XSSFSheet sheet) throws Exception {
//code
}
/**
* Create the row with RDI for excel file
*/
@Override
protected void elaboration(Car car, XSSFSheet sheet, XSSFCellStyle unlockedNumericStyle, Map<Integer,Integer> defRdiMap) {
//code
}
/**
* Import the Excel file
* @throws Exception
*/
@Override
@Transactional(rollbackFor= Exception.class)
protected int updateRdi(Car car, XSSFSheet sheet) throws Exception{
//COde
}
}
因此,通过控制器收到请求的服务我有一个简单的开关:
@Autowired
private ExcelRdi excelRdi;
@Autowired
private ExcelIupr excelIupr;
@Autowired
private ExcelSl excelSl;
@Override
public String getExcel(Car car, String type) throws Exception {
String fileName = type + "$"+car.getFleet().getFleetName().getFleetName() +"$"+ car.getFleet().getApplication()+"$"+ car.getCarType().getIdCarType()+car.getId() +"$"+car.getIdCar()+".xlsx";
switch (type){
case "RDI": return excelRdi.retrieve(car, fileName);
case "IUPR": return excelIupr.retrieve(car, fileName);
case "SL": return excelSl.retrieve(car, fileName);
default: throw new FileFormatException("You can create only RDI, SL or IUPR file!");
}
}
@Override
public int importExcel(MultipartFile file, Car car, String type) throws Exception {
//Check if the file has the correct name before import
String fileName = type + "$"+car.getFleet().getFleetName().getFleetName() +"$"+ car.getFleet().getApplication()+"$"+ car.getCarType().getIdCarType()+car.getId() +"$"+car.getIdCar()+".xlsx";
if (!file.getOriginalFilename().equals(fileName))
throw new FileFormatException("The file is not for this car");
switch (type){
case "RDI": return excelRdi.update(file, car);
case "IUPR": return excelIupr.update(file, car);
case "SL": return excelSl.update(file, car);
default: throw new FileFormatException("You can create only RDI, SL or IUPR file!");
}
}
所以这就是问题:
protected
updateRdi
工作正常,@Transactional
注释需要公开方法,否则无法正常工作public
updateRdi
nullPointerException
sheet
firstTwoRow
env
$("body").tooltip({ selector: "[data-toggle=tooltip]" });
对象placement: "top"
,show.bs.tooltip
变量相同。
你知道为什么我有这种行为以及如何解决这个问题吗?非常感谢答案 0 :(得分:2)
问题在于,您的receive
和update
方法已标记为final
。此外,如果您观察日志记录,您将看到一条警告,指出final
方法无法代理。
由于没有接口这一事实,将会创建一个基于类的代理。此代理将向所有public
方法添加行为(例如,如@Transactional
)。它通过覆盖实际方法,应用建议并在非代理实例(驻留在代理内部)上调用方法来完成此操作。
然而,由于final
性质,这种情况不会发生,并且将在代理上调用该方法,而不是代理中的实际对象。由于代理不是Spring bean,所有@Autowired
字段都是null
,因此NullPointerException
。
如果要在update
方法中放置断点,可以观察到这种行为。您将在调试器中看到所有实例字段都是null
,并且类名是{{1} 1}}。还会有一个字段YOurClass$CgLibProxy
(如果我没记错的话),其中包含已注入依赖项的实际Spring bean。
问题的解决方案实际上很简单,只需从您的方法中删除target
即可。