Skip to content

Commit 8b0e52f

Browse files
authored
Merge pull request #416 from dinghuang/feature-handle-table
[ADD] spring optimize
2 parents bdfc60d + 021ebe4 commit 8b0e52f

File tree

11 files changed

+408
-2
lines changed

11 files changed

+408
-2
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
/*
2+
* Copyright (c) 2011-2019, hubin ([email protected]).
3+
* <p>
4+
* Licensed under the Apache License, Version 2.0 (the "License"); you may not
5+
* use this file except in compliance with the License. You may obtain a copy of
6+
* the License at
7+
* <p>
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
* <p>
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
12+
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
13+
* License for the specific language governing permissions and limitations under
14+
* the License.
15+
*/
16+
package com.codingapi.txlcn.tc.annotation;
17+
18+
import java.lang.annotation.*;
19+
20+
/**
21+
22+
* @since 2019/7/29
23+
*/
24+
@Documented
25+
@Retention(RetentionPolicy.RUNTIME)
26+
@Target(ElementType.FIELD)
27+
public @interface TableId {
28+
29+
/**
30+
* 字段值(驼峰命名方式,该值可无)
31+
*/
32+
String value() default "";
33+
34+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
/*
2+
* Copyright (c) 2011-2019, hubin ([email protected]).
3+
* <p>
4+
* Licensed under the Apache License, Version 2.0 (the "License"); you may not
5+
* use this file except in compliance with the License. You may obtain a copy of
6+
* the License at
7+
* <p>
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
* <p>
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
12+
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
13+
* License for the specific language governing permissions and limitations under
14+
* the License.
15+
*/
16+
package com.codingapi.txlcn.tc.annotation;
17+
18+
import java.lang.annotation.*;
19+
20+
/**
21+
22+
* @since 2019/7/29
23+
*/
24+
@Documented
25+
@Retention(RetentionPolicy.RUNTIME)
26+
@Target(ElementType.TYPE)
27+
public @interface TableName {
28+
29+
/**
30+
* 实体对应的表名
31+
*/
32+
String value() default "";
33+
}

txlcn-tc/src/main/java/com/codingapi/txlcn/tc/core/transaction/txc/analy/PrimaryKeyListVisitor.java

+7-1
Original file line numberDiff line numberDiff line change
@@ -76,8 +76,14 @@ private Map<String, Object> newKeyValues(List<Expression> expressions) {
7676
Map<String, Object> keyValues = new HashMap<>();
7777
for (int i = 0; i < columns.size(); i++) {
7878
columns.get(i).setTable(table);
79+
//解决Long类型字段作为主键空指针异常
7980
if (primaryKeys.contains(columns.get(i).getFullyQualifiedName())) {
80-
Object expression = expressions.get(i).getASTNode().jjtGetValue();
81+
Object expression = null;
82+
if (expressions.get(i).getASTNode() != null) {
83+
expressions.get(i).getASTNode().jjtGetValue();
84+
} else {
85+
expression = expressions.get(i);
86+
}
8187
keyValues.put(columns.get(i).getFullyQualifiedName(),
8288
Reflection.invokeN(expression.getClass(), "getValue", expression, new Object[0]));
8389
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
package com.codingapi.txlcn.tc.core.transaction.txc.analy;
2+
3+
import com.codingapi.txlcn.tc.core.transaction.txc.analy.def.PrimaryKeysProvider;
4+
import com.codingapi.txlcn.tc.core.transaction.txc.analy.util.AnnotationUtils;
5+
import com.codingapi.txlcn.tc.core.transaction.txc.analy.util.ClassUtils;
6+
import org.springframework.beans.factory.annotation.Value;
7+
import org.springframework.stereotype.Component;
8+
9+
import java.util.List;
10+
import java.util.Map;
11+
12+
/**
13+
14+
* @since 2019/7/28
15+
*/
16+
@Component
17+
public class PrimaryKeyListVisitorHandler implements PrimaryKeysProvider {
18+
19+
@Value("${tx-lcn.primary-key-package}")
20+
private String primaryKeyPackage;
21+
22+
@Override
23+
public Map<String, List<String>> provide() {
24+
if (primaryKeyPackage != null) {
25+
//扫描所有注解,把主键拿进来
26+
// 获取特定包下所有的类(包括接口和类)
27+
List<Class<?>> clsList = ClassUtils.getClasses(primaryKeyPackage);
28+
//输出所有使用了特定注解的类的注解值
29+
return AnnotationUtils.getPrimaryKeyList(clsList);
30+
} else {
31+
return null;
32+
}
33+
}
34+
35+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,66 @@
1+
package com.codingapi.txlcn.tc.core.transaction.txc.analy.util;
2+
3+
import com.codingapi.txlcn.tc.annotation.TableId;
4+
import com.codingapi.txlcn.tc.annotation.TableName;
5+
6+
import java.lang.reflect.Field;
7+
import java.util.Collections;
8+
import java.util.HashMap;
9+
import java.util.List;
10+
import java.util.Map;
11+
import java.util.regex.Matcher;
12+
import java.util.regex.Pattern;
13+
14+
/**
15+
16+
* @since 2019/7/29
17+
*/
18+
public class AnnotationUtils {
19+
20+
private static Pattern humpPattern = Pattern.compile("[A-Z]");
21+
22+
public static Map<String, List<String>> getPrimaryKeyList(List<Class<?>> clsList) {
23+
if (clsList != null && clsList.size() > 0) {
24+
Map<String, List<String>> map = new HashMap<>(clsList.size());
25+
for (Class<?> cls : clsList) {
26+
//获取类中的所有的方法
27+
boolean classHasAnnotation = cls.isAnnotationPresent(TableName.class);
28+
if (classHasAnnotation) {
29+
TableName tableName = cls.getAnnotation(TableName.class);
30+
if (tableName.value().length() != 0) {
31+
Field[] fields = cls.getDeclaredFields();
32+
String tableIdVale = null;
33+
for (Field field : fields) {
34+
boolean fieldHasAnnotation = field.isAnnotationPresent(TableId.class);
35+
if (fieldHasAnnotation) {
36+
TableId tableId = field.getAnnotation(TableId.class);
37+
//输出注解属性
38+
if (tableId.value().length() != 0) {
39+
tableIdVale = tableId.value();
40+
} else {
41+
//转驼峰
42+
tableIdVale = humpToLine(field.getName());
43+
}
44+
break;
45+
}
46+
}
47+
map.put(tableName.value(), Collections.singletonList(tableIdVale));
48+
}
49+
}
50+
}
51+
return map;
52+
} else {
53+
return null;
54+
}
55+
}
56+
57+
public static String humpToLine(String str) {
58+
Matcher matcher = humpPattern.matcher(str);
59+
StringBuffer sb = new StringBuffer();
60+
while (matcher.find()) {
61+
matcher.appendReplacement(sb, "_" + matcher.group(0).toLowerCase());
62+
}
63+
matcher.appendTail(sb);
64+
return sb.toString();
65+
}
66+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,116 @@
1+
package com.codingapi.txlcn.tc.core.transaction.txc.analy.util;
2+
3+
import org.slf4j.Logger;
4+
import org.slf4j.LoggerFactory;
5+
6+
import java.io.File;
7+
import java.io.IOException;
8+
import java.net.JarURLConnection;
9+
import java.net.URL;
10+
import java.net.URLDecoder;
11+
import java.util.ArrayList;
12+
import java.util.Enumeration;
13+
import java.util.List;
14+
import java.util.jar.JarEntry;
15+
import java.util.jar.JarFile;
16+
17+
/**
18+
19+
* @since 2019/7/29
20+
*/
21+
public class ClassUtils {
22+
23+
private static final Logger LOGGER = LoggerFactory.getLogger(ClassUtils.class);
24+
25+
private static final String FILE = "file";
26+
private static final String JAR = "jar";
27+
private static final String CLASS_SIGN = ".class";
28+
29+
/**
30+
* 从包package中获取所有的Class
31+
*
32+
* @param packageName packageName
33+
* @return List
34+
*/
35+
public static List<Class<?>> getClasses(String packageName) {
36+
List<Class<?>> classes = new ArrayList<>();
37+
boolean recursive = true;
38+
String packageDirName = packageName.replace('.', '/');
39+
Enumeration<URL> dirs;
40+
try {
41+
dirs = Thread.currentThread().getContextClassLoader().getResources(packageDirName);
42+
while (dirs.hasMoreElements()) {
43+
URL url = dirs.nextElement();
44+
String protocol = url.getProtocol();
45+
if (FILE.equals(protocol)) {
46+
String filePath = URLDecoder.decode(url.getFile(), "UTF-8");
47+
findAndAddClassesInPackageByFile(packageName, filePath, recursive, classes);
48+
} else if (JAR.equals(protocol)) {
49+
JarFile jar;
50+
try {
51+
jar = ((JarURLConnection) url.openConnection()).getJarFile();
52+
Enumeration<JarEntry> entries = jar.entries();
53+
while (entries.hasMoreElements()) {
54+
JarEntry entry = entries.nextElement();
55+
String name = entry.getName();
56+
if (name.charAt(0) == '/') {
57+
name = name.substring(1);
58+
}
59+
if (name.startsWith(packageDirName)) {
60+
int idx = name.lastIndexOf('/');
61+
if (idx != -1) {
62+
packageName = name.substring(0, idx).replace('/', '.');
63+
}
64+
if (idx != -1) {
65+
if (name.endsWith(CLASS_SIGN) && !entry.isDirectory()) {
66+
// 去掉后面的".class" 获取真正的类名
67+
String className = name.substring(packageName.length() + 1, name.length() - 6);
68+
try {
69+
classes.add(Class.forName(packageName + '.' + className));
70+
} catch (ClassNotFoundException e) {
71+
LOGGER.error(e.getMessage());
72+
}
73+
}
74+
}
75+
}
76+
}
77+
} catch (IOException e) {
78+
LOGGER.error(e.getMessage());
79+
}
80+
}
81+
}
82+
} catch (IOException e) {
83+
LOGGER.error(e.getMessage());
84+
}
85+
return classes;
86+
}
87+
88+
/**
89+
* 以文件的形式来获取包下的所有Class
90+
*
91+
* @param packageName packageName
92+
* @param packagePath packagePath
93+
* @param recursive recursive
94+
* @param classes classes
95+
*/
96+
private static void findAndAddClassesInPackageByFile(String packageName, String packagePath, final boolean recursive, List<Class<?>> classes) {
97+
File dir = new File(packagePath);
98+
if (!dir.exists() || !dir.isDirectory()) {
99+
return;
100+
}
101+
File[] dirFiles = dir.listFiles(file -> (recursive && file.isDirectory()) || (file.getName().endsWith(CLASS_SIGN)));
102+
assert dirFiles != null;
103+
for (File file : dirFiles) {
104+
if (file.isDirectory()) {
105+
findAndAddClassesInPackageByFile(packageName + "." + file.getName(), file.getAbsolutePath(), recursive, classes);
106+
} else {
107+
String className = file.getName().substring(0, file.getName().length() - 6);
108+
try {
109+
classes.add(Class.forName(packageName + '.' + className));
110+
} catch (ClassNotFoundException e) {
111+
LOGGER.error(e.getMessage());
112+
}
113+
}
114+
}
115+
}
116+
}

txlcn-tracing/src/main/java/com/codingapi/txlcn/tracing/TracingAutoConfiguration.java

+6
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,8 @@
1515
*/
1616
package com.codingapi.txlcn.tracing;
1717

18+
import com.codingapi.txlcn.tracing.spring.TracingSpringContextUtils;
19+
import org.springframework.context.annotation.Bean;
1820
import org.springframework.context.annotation.ComponentScan;
1921
import org.springframework.context.annotation.Configuration;
2022

@@ -28,4 +30,8 @@
2830
@ComponentScan
2931
public class TracingAutoConfiguration {
3032

33+
@Bean
34+
public TracingSpringContextUtils tracingSpringContextUtils() {
35+
return new TracingSpringContextUtils();
36+
}
3137
}

txlcn-tracing/src/main/java/com/codingapi/txlcn/tracing/http/ribbon/TxlcnZoneAvoidanceRule.java

+15-1
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,8 @@
1717

1818
import com.alibaba.fastjson.JSONObject;
1919
import com.codingapi.txlcn.tracing.TracingContext;
20+
import com.codingapi.txlcn.tracing.spring.SpringConfig;
21+
import com.codingapi.txlcn.tracing.spring.TracingSpringContextUtils;
2022
import com.netflix.loadbalancer.Server;
2123
import com.netflix.loadbalancer.ZoneAvoidanceRule;
2224
import lombok.extern.slf4j.Slf4j;
@@ -57,7 +59,19 @@ public Server choose(Object key) {
5759

5860
// 1. 自己加入此事务组调用链
5961
assert Objects.nonNull(registration);
60-
TracingContext.tracing().addApp(registration.getServiceId(), registration.getHost() + ":" + registration.getPort());
62+
//兼容springBoot 2.0以下版本
63+
Boolean exitHost;
64+
try {
65+
exitHost = registration.getHost() != null;
66+
} catch (NoSuchMethodError noSuchMethodError) {
67+
exitHost = false;
68+
}
69+
if (!exitHost) {
70+
SpringConfig springConfig = (SpringConfig) TracingSpringContextUtils.getContext().getBean("springConfig");
71+
TracingContext.tracing().addApp(registration.getServiceId(), springConfig.getHost() + ":" + springConfig.getPort());
72+
} else {
73+
TracingContext.tracing().addApp(registration.getServiceId(), registration.getHost() + ":" + registration.getPort());
74+
}
6175

6276
// 2. 获取所有要访问服务的实例
6377
List<Server> servers = getLoadBalancer().getAllServers();

0 commit comments

Comments
 (0)