假设我们拥有三个不同的代码源:product1, product2, product3;在这三个分别代表不同版本但是代码都是类似,或者说只是部分类的内部实现功能有所差别,把代码复制出三份到不同的版本下这个是我们不想看到的,网上很多处理方式是将公共代码抽取出来,像如下打包:
sourceSets {
product2 {
java.srcDirs += ['src/product2/java']
java.srcDirs += ['src/product1/java']
}
}
这种方式可以将代码打在一起,但是有重复的类就会出现冲突了,为了解决冲突问题,我们可以这样打:
sourceSets {
product1 {
java.srcDirs += ['src/product1/java']
}
product2 {
java.srcDirs += ['src/product2/java']
java.srcDirs += getJavaSource('product2','src/product2/java')
}
product3 {
java.srcDirs += ['src/product3/java']
java.srcDirs += getJavaSource('product3','src/product3/java')
}
}
在此product1我们把它称为公共代码源并且也是一个独立的版本,单独打product1可以打出一个包,也是里面代码功能是完整的,而此时product2、product3放的是可同名类的差异代码,代表不同的版本,接下来我们要重点看getJavaSource('product2','src/product2/java')的实现方式:
def getJavaSource(product, productPath){
def product_list = get_product_file_list(productPath)//1
def main_list = get_file_list(product_list,product)//2
def list = []
list.addAll(main_list)//3
return list
}
接下来我们来看1处,具体实现如下:
def get_product_file_list(dirPath){
def _dir =new File(dirPath)
def fileList = []
def _file_folder_queen =new Stack()
_file_folder_queen.push(_dir)
while (!_file_folder_queen.isEmpty()){
for(File f : _file_folder_queen.pop().listFiles()){
if(f.isFile()){
fileList.add(f.getPath().toString())
}else if(f.isDirectory() && f.listFiles().length >0){
_file_folder_queen.add(f)
}
}
}
return fileList
}
在这个1处的实现就是将路径下的类全部遍历出来,然后将这些类的具体路径存放到一个list表中返回,注意的是这里也包含里各个文件夹的路径。
接下来看看2处的实现:
def get_file_list(product_java_file_array,product){
def _dir =new File('src/product1/java')
def fileList = []
def _file_folder_queen =new Stack()
_file_folder_queen.push(_dir)
while (!_file_folder_queen.isEmpty()){
for(File f : _file_folder_queen.pop().listFiles()){
if(f.isFile()){
def flag =false
def javaPath = f.getPath().toString()
javaPath = javaPath.replaceFirst("product1",product)
for(String path : product_java_file_array){
if(path == javaPath){
flag =true
break
}
}
if(!flag){
fileList.add(f.getPath().toString())
}
}else if(f.isDirectory() && f.listFiles().length >0){
_file_folder_queen.add(f)
}
}
}
return fileList
}
在2处实现是把具有公共类的product1中的类路径与1处的路径进行对比判断,把相同的路径下的类去除然后存在list里面返回。
最后是3处把2处list存起来并返回,到此处算是完成了替换,但是需要注意的是product1,product2,product3路径在存各个list时确保相同,不然会出现问题