2023-04-01  阅读(4)
原文作者:LoveLion 原文地址:https://blog.csdn.net/lovelion/article/details/8624325

9.3 完整解决方案

Sunny软件公司开发人员决定使用适配器模式来重用算法库中的算法,其基本结构如图9-4所示:

202304012100311581.png

图9-4 算法库重用结构图

在图9-4中,ScoreOperation接口充当抽象目标,QuickSort和BinarySearch类充当适配者,OperationAdapter充当适配器。完整代码如下所示:

    //抽象成绩操作类:目标接口
    interface ScoreOperation {
    	public int[] sort(int array[]); //成绩排序
    	public int search(int array[],int key); //成绩查找
    }
    
    //快速排序类:适配者
    class QuickSort {
    	public int[] quickSort(int array[]) {
    		sort(array,0,array.length-1);
    		return array;
    	}
    
    	public void sort(int array[],int p, int r) {
    		int q=0;
    		if(p<r) {
    			q=partition(array,p,r);
    			sort(array,p,q-1);
                sort(array,q+1,r);
    		}
    	}
    
    	public int partition(int[] a, int p, int r) {
    		int x=a[r];
    		int j=p-1;
    		for (int i=p;i<=r-1;i++) {
    			if (a[i]<=x) {
    				j++;
    				swap(a,j,i);
    			}
    		}
    		swap(a,j+1,r);
    		return j+1;	
    	}
    
    	public void swap(int[] a, int i, int j) {   
            int t = a[i];   
            a[i] = a[j];   
            a[j] = t;   
    	}
    }
    
    //二分查找类:适配者
    class BinarySearch {
    	public int binarySearch(int array[],int key) {
    		int low = 0;
    		int high = array.length -1;
    		while(low <= high) {
    			int mid = (low + high) / 2;
    			int midVal = array[mid];
    			if(midVal < key) {  
    low = mid +1;  
    }
    			else if (midVal > key) {  
    high = mid -1;  
    }
    			else {  
    return 1; //找到元素返回1  
    }
    		}
    		return -1;  //未找到元素返回-1
    	}
    }
    
    //操作适配器:适配器
    class OperationAdapter implements ScoreOperation {
    	private QuickSort sortObj; //定义适配者QuickSort对象
    	private BinarySearch searchObj; //定义适配者BinarySearch对象
    
    	public OperationAdapter() {
    		sortObj = new QuickSort();
    		searchObj = new BinarySearch();
    	}
    
    	public int[] sort(int array[]) {  
    return sortObj.quickSort(array); //调用适配者类QuickSort的排序方法
    }
    
    	public int search(int array[],int key) {  
    return searchObj.binarySearch(array,key); //调用适配者类BinarySearch的查找方法
    }
    }

为了让系统具备良好的灵活性和可扩展性,我们引入了工具类XMLUtil和配置文件,其中,XMLUtil类的代码如下所示:

    import javax.xml.parsers.*;
    import org.w3c.dom.*;
    import org.xml.sax.SAXException;
    import java.io.*;
    class XMLUtil {
    //该方法用于从XML配置文件中提取具体类类名,并返回一个实例对象
    	public static Object getBean() {
    		try {
    			//创建文档对象
    			DocumentBuilderFactory dFactory = DocumentBuilderFactory.newInstance();
    			DocumentBuilder builder = dFactory.newDocumentBuilder();
    			Document doc;							
    			doc = builder.parse(new File("config.xml")); 
    		
    			//获取包含类名的文本节点
    			NodeList nl = doc.getElementsByTagName("className");
                Node classNode=nl.item(0).getFirstChild();
                String cName=classNode.getNodeValue();
                
                //通过类名生成实例对象并将其返回
                Class c=Class.forName(cName);
    	  	    Object obj=c.newInstance();
                return obj;
            }   
            catch(Exception e) {
               	e.printStackTrace();
               	return null;
           	}
    	}
    }

配置文件config.xml中存储了适配器类的类名,代码如下所示:

    <?xml version="1.0"?>
    <config>
    	<className>OperationAdapter</className>
    </config>

编写如下客户端测试代码:

    class Client {
    	public static void main(String args[]) {
    		ScoreOperation operation;  //针对抽象目标接口编程
    		operation = (ScoreOperation)XMLUtil.getBean(); //读取配置文件,反射生成对象
    		int scores[] = {84,76,50,69,90,91,88,96}; //定义成绩数组
    		int result[];
    		int score;
    		
    		System.out.println("成绩排序结果:");
    		result = operation.sort(scores);
    
            //遍历输出成绩
    		for(int i : scores) {
    			System.out.print(i + ",");
    		}
    		System.out.println();
    		
    		System.out.println("查找成绩90:");
    		score = operation.search(result,90);
    		if (score != -1) {
    			System.out.println("找到成绩90。");
    		}
    		else {
    			System.out.println("没有找到成绩90。");
    		}
    		
    		System.out.println("查找成绩92:");
    		score = operation.search(result,92);
    		if (score != -1) {
    			System.out.println("找到成绩92。");
    		}
    		else {
    			System.out.println("没有找到成绩92。");
    		}
    	}
    }

编译并运行程序,输出结果如下:

成绩排序结果:

50,69,76,84,88,90,91,96,

查找成绩90:

找到成绩90。

查找成绩92:

没有找到成绩92。

在本实例中使用了对象适配器模式,同时引入了配置文件,将适配器类的类名存储在配置文件中。如果需要使用其他排序算法类和查找算法类,可以增加一个新的适配器类,使用新的适配器来适配新的算法,原有代码无须修改。通过引入配置文件和反射机制,可以在不修改客户端代码的情况下使用新的适配器,无须修改源代码,符合“开闭原则”。


Java 面试宝典是大明哥全力打造的 Java 精品面试题,它是一份靠谱、强大、详细、经典的 Java 后端面试宝典。它不仅仅只是一道道面试题,而是一套完整的 Java 知识体系,一套你 Java 知识点的扫盲贴。

它的内容包括:

  • 大厂真题:Java 面试宝典里面的题目都是最近几年的高频的大厂面试真题。
  • 原创内容:Java 面试宝典内容全部都是大明哥原创,内容全面且通俗易懂,回答部分可以直接作为面试回答内容。
  • 持续更新:一次购买,永久有效。大明哥会持续更新 3+ 年,累计更新 1000+,宝典会不断迭代更新,保证最新、最全面。
  • 覆盖全面:本宝典累计更新 1000+,从 Java 入门到 Java 架构的高频面试题,实现 360° 全覆盖。
  • 不止面试:内容包含面试题解析、内容详解、知识扩展,它不仅仅只是一份面试题,更是一套完整的 Java 知识体系。
  • 宝典详情:https://www.yuque.com/chenssy/sike-java/xvlo920axlp7sf4k
  • 宝典总览:https://www.yuque.com/chenssy/sike-java/yogsehzntzgp4ly1
  • 宝典进展:https://www.yuque.com/chenssy/sike-java/en9ned7loo47z5aw

目前 Java 面试宝典累计更新 400+ 道,总字数 42w+。大明哥还在持续更新中,下图是大明哥在 2024-12 月份的更新情况:

想了解详情的小伙伴,扫描下面二维码加大明哥微信【daming091】咨询

同时,大明哥也整理一套目前市面最常见的热点面试题。微信搜[大明哥聊 Java]或扫描下方二维码关注大明哥的原创公众号[大明哥聊 Java] ,回复【面试题】 即可免费领取。

阅读全文