VIN码国际通用,一共17位,包含数字0~9,和大部分英文字母(不包括I、O、Q)。第9位为校验位,其值的计算方法:对其他位置求校验和,然后除以11得到余数,如果余数为10则由X
表示,这个余数就是第九位的值。
求校验和的算法:每个位置的数字对应值和位置权重相乘,再累加。
VIN码字符的数字对应值:
0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | |||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | |||||||||||||
A | B | C | D | E | F | G | H | J | K | L | M | N | P | R | S | T | U | V | W | X | Y | Z |
1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 1 | 2 | 3 | 4 | 5 | 7 | 9 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 |
VIN码位置对应权重:
1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 |
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
8 | 7 | 6 | 5 | 4 | 3 | 2 | 10 | 0 | 9 | 8 | 7 | 6 | 5 | 4 | 3 | 2 |
Java版本判断校验和是否正确算法:
import java.util.HashMap;
import java.util.Map;
public class VinUtil {
/**
* 字符权重表
*/
private static final Map<Character, Integer> CHAR_WEIGHTS = new HashMap<>();
/**
* 位置权重表
*/
private static final int[] POS_WEIGHTS = new int[]{8, 7, 6, 5, 4, 3, 2, 10, 0, 9, 8, 7, 6, 5, 4, 3, 2};
static {
for (int i = 0; i < 10; i++) {
CHAR_WEIGHTS.put(String.valueOf(i).charAt(0), i);
}
CHAR_WEIGHTS.put('A', 1);
CHAR_WEIGHTS.put('B', 2);
CHAR_WEIGHTS.put('C', 3);
CHAR_WEIGHTS.put('D', 4);
CHAR_WEIGHTS.put('E', 5);
CHAR_WEIGHTS.put('F', 6);
CHAR_WEIGHTS.put('G', 7);
CHAR_WEIGHTS.put('H', 8);
CHAR_WEIGHTS.put('J', 1);
CHAR_WEIGHTS.put('K', 2);
CHAR_WEIGHTS.put('L', 3);
CHAR_WEIGHTS.put('M', 4);
CHAR_WEIGHTS.put('N', 5);
CHAR_WEIGHTS.put('P', 7);
CHAR_WEIGHTS.put('R', 9);
CHAR_WEIGHTS.put('S', 2);
CHAR_WEIGHTS.put('T', 3);
CHAR_WEIGHTS.put('U', 4);
CHAR_WEIGHTS.put('V', 5);
CHAR_WEIGHTS.put('W', 6);
CHAR_WEIGHTS.put('X', 7);
CHAR_WEIGHTS.put('Y', 8);
CHAR_WEIGHTS.put('Z', 9);
}
public static boolean isValidVin(String vin) {
if (null == vin) {
return false;
} else if (vin.trim().length() == 17) {
vin = vin.toUpperCase();
int sum = 0;
int checkSum = 0;
for (int i = 0; i < vin.length(); i++) {
char code = vin.charAt(i);
Integer cw = CHAR_WEIGHTS.get(code);
if (cw == null) {
return false;
}
int pw = POS_WEIGHTS[i];
sum += cw * pw;
if (i == 8) {
// 获取校验位的值
if (code == 'X') {
checkSum = 10;
} else if (code >= '0' && code <= '9') {
checkSum = Integer.valueOf(String.valueOf(code));
} else {
return false;
}
}
}
return checkSum == sum % 11;
}
return false;
}
}