spring-batch DefaultJobParametersConverter 源码
spring-batch DefaultJobParametersConverter 代码
文件路径:/spring-batch-core/src/main/java/org/springframework/batch/core/converter/DefaultJobParametersConverter.java
/*
* Copyright 2006-2022 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.springframework.batch.core.converter;
import org.springframework.batch.core.JobInstance;
import org.springframework.batch.core.JobParameter;
import org.springframework.batch.core.JobParameter.ParameterType;
import org.springframework.batch.core.JobParameters;
import org.springframework.batch.core.JobParametersBuilder;
import org.springframework.lang.Nullable;
import org.springframework.util.StringUtils;
import java.text.DateFormat;
import java.text.DecimalFormat;
import java.text.NumberFormat;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Iterator;
import java.util.Locale;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Properties;
/**
* Converter for {@link JobParameters} instances that uses a simple naming convention for
* property keys. Key names that are prefixed with a {@code -} are considered
* non-identifying and do not contribute to the identity of a {@link JobInstance}. Key
* names ending with "(<type>)" (where type is one of string, date, long) are
* converted to the corresponding type. The default type is string. Consider the following
* example:
*
* <pre>
* schedule.date(date)=2007/12/11
* department.id(long)=2345
* </pre>
*
* The literal values are converted to the correct type by using the default Spring
* strategies, augmented if necessary by any custom editors that have been provided.
*
* <br>
*
* If you need to be able to parse and format local-specific dates and numbers, you can
* inject formatters ({@link #setDateFormat(DateFormat)} and
* {@link #setNumberFormat(NumberFormat)}).
*
* @author Dave Syer
* @author Michael Minella
* @author Mahmoud Ben Hassine
*
*/
public class DefaultJobParametersConverter implements JobParametersConverter {
/**
* Parameter key suffix representing the date type.
*/
public static final String DATE_TYPE = "(date)";
/**
* Parameter key suffix representing the string type.
*/
public static final String STRING_TYPE = "(string)";
/**
* Parameter key suffix representing the long type.
*/
public static final String LONG_TYPE = "(long)";
/**
* Parameter key suffix representing the double type.
*/
public static final String DOUBLE_TYPE = "(double)";
private static final String NON_IDENTIFYING_FLAG = "-";
private static final String IDENTIFYING_FLAG = "+";
private static NumberFormat DEFAULT_NUMBER_FORMAT = NumberFormat.getInstance(Locale.US);
private DateFormat dateFormat = new SimpleDateFormat("yyyy/MM/dd");
private NumberFormat numberFormat = DEFAULT_NUMBER_FORMAT;
private final NumberFormat longNumberFormat = new DecimalFormat("#");
/**
* Check for a suffix on keys and use those to decide how to convert the value.
* @throws IllegalArgumentException if a number or date is passed in that cannot be
* parsed or cast to the correct type.
*
* @see org.springframework.batch.core.converter.JobParametersConverter#getJobParameters(java.util.Properties)
*/
@Override
public JobParameters getJobParameters(@Nullable Properties props) {
if (props == null || props.isEmpty()) {
return new JobParameters();
}
JobParametersBuilder propertiesBuilder = new JobParametersBuilder();
for (Iterator<Entry<Object, Object>> it = props.entrySet().iterator(); it.hasNext();) {
Entry<Object, Object> entry = it.next();
String key = (String) entry.getKey();
String value = (String) entry.getValue();
boolean identifying = isIdentifyingKey(key);
if (!identifying) {
key = key.replaceFirst(NON_IDENTIFYING_FLAG, "");
}
else if (identifying && key.startsWith(IDENTIFYING_FLAG)) {
key = key.replaceFirst("\\" + IDENTIFYING_FLAG, "");
}
if (key.endsWith(DATE_TYPE)) {
Date date;
synchronized (dateFormat) {
try {
date = dateFormat.parse(value);
}
catch (ParseException ex) {
String suffix = (dateFormat instanceof SimpleDateFormat)
? ", use " + ((SimpleDateFormat) dateFormat).toPattern() : "";
throw new IllegalArgumentException("Date format is invalid: [" + value + "]" + suffix);
}
}
propertiesBuilder.addDate(StringUtils.replace(key, DATE_TYPE, ""), date, identifying);
}
else if (key.endsWith(LONG_TYPE)) {
Long result;
try {
result = (Long) parseNumber(value);
}
catch (ClassCastException ex) {
throw new IllegalArgumentException("Number format is invalid for long value: [" + value
+ "], use a format with no decimal places");
}
propertiesBuilder.addLong(StringUtils.replace(key, LONG_TYPE, ""), result, identifying);
}
else if (key.endsWith(DOUBLE_TYPE)) {
Double result = parseNumber(value).doubleValue();
propertiesBuilder.addDouble(StringUtils.replace(key, DOUBLE_TYPE, ""), result, identifying);
}
else if (StringUtils.endsWithIgnoreCase(key, STRING_TYPE)) {
propertiesBuilder.addString(StringUtils.replace(key, STRING_TYPE, ""), value, identifying);
}
else {
propertiesBuilder.addString(key, value, identifying);
}
}
return propertiesBuilder.toJobParameters();
}
private boolean isIdentifyingKey(String key) {
boolean identifying = true;
if (key.startsWith(NON_IDENTIFYING_FLAG)) {
identifying = false;
}
return identifying;
}
/**
* Delegate to {@link NumberFormat} to parse the value.
*/
private Number parseNumber(String value) {
synchronized (numberFormat) {
try {
return numberFormat.parse(value);
}
catch (ParseException ex) {
String suffix = (numberFormat instanceof DecimalFormat)
? ", use " + ((DecimalFormat) numberFormat).toPattern() : "";
throw new IllegalArgumentException("Number format is invalid: [" + value + "], use " + suffix);
}
}
}
/**
* Use the same suffixes to create properties (omitting the string suffix because it
* is the default). Non-identifying parameters are prefixed with the
* {@link #NON_IDENTIFYING_FLAG}. However, since parameters are identifying by
* default, they are <em>not</em> prefixed with the {@link #IDENTIFYING_FLAG}.
*
* @see org.springframework.batch.core.converter.JobParametersConverter#getProperties(org.springframework.batch.core.JobParameters)
*/
@Override
public Properties getProperties(@Nullable JobParameters params) {
if (params == null || params.isEmpty()) {
return new Properties();
}
Map<String, JobParameter> parameters = params.getParameters();
Properties result = new Properties();
for (Entry<String, JobParameter> entry : parameters.entrySet()) {
String key = entry.getKey();
JobParameter jobParameter = entry.getValue();
Object value = jobParameter.getValue();
if (value != null) {
key = (!jobParameter.isIdentifying() ? NON_IDENTIFYING_FLAG : "") + key;
if (jobParameter.getType() == ParameterType.DATE) {
synchronized (dateFormat) {
result.setProperty(key + DATE_TYPE, dateFormat.format(value));
}
}
else if (jobParameter.getType() == ParameterType.LONG) {
synchronized (longNumberFormat) {
result.setProperty(key + LONG_TYPE, longNumberFormat.format(value));
}
}
else if (jobParameter.getType() == ParameterType.DOUBLE) {
result.setProperty(key + DOUBLE_TYPE, decimalFormat((Double) value));
}
else {
result.setProperty(key, "" + value);
}
}
}
return result;
}
/**
* Makes a best guess at converting a double to a string representation of a decimal
* format.
* @param value A decimal value.
* @return a best guess at the desired format.
*/
private String decimalFormat(double value) {
if (numberFormat != DEFAULT_NUMBER_FORMAT) {
synchronized (numberFormat) {
return numberFormat.format(value);
}
}
return Double.toString(value);
}
/**
* Public setter for injecting a date format.
* @param dateFormat A {@link DateFormat}, defaults to "yyyy/MM/dd".
*/
public void setDateFormat(DateFormat dateFormat) {
this.dateFormat = dateFormat;
}
/**
* Public setter for the {@link NumberFormat}. Used to parse longs and doubles, so
* must not contain decimal place (for example, use "#" or "#,###" but not "#.##").
* @param numberFormat the {@link NumberFormat} to set
*/
public void setNumberFormat(NumberFormat numberFormat) {
this.numberFormat = numberFormat;
}
}
相关信息
相关文章
0
赞
热门推荐
-
2、 - 优质文章
-
3、 gate.io
-
8、 golang
-
9、 openharmony
-
10、 Vue中input框自动聚焦