Spring不是服務於開發web項目的功能,或業務。而是服務於項目的開發,方便各層間的解耦調用,方便對類的批量管理,是提升軟件開發效率,下降後期維護成本的框架。
Spring的核心思想是IOC(控制反轉),AOP(切面編程)兩點。
IOC:即再也不須要程序員去顯式地`new`一個對象,而是把Spring框架把框架建立出的對象拿來用。由於是spring框架建立的對象,對象都在spring框架對象中保存,亦稱爲spring容器,這樣spring就知道當前項目中都建立了哪些對象,這個對象歸屬於那一層,該如何管理。想使用spring的其餘功能第一點就是要用spring的對象,也稱爲將控制權交給spring管理。
AOP:對某種路徑下的全部類,或有共同特性的類或方法統一管理,在原任務執行的先後,加入新功能。作出監控,初始化,整理,銷燬等一系列統一的伴隨動做。
若是你從事Java編程有一段時間了, 那麼你或許會發現(可能你也實際使用過) 不少框架經過強迫應用繼承它們的類或實現它們的接口從而致使應用與框架綁死。這種侵入式的編程方式在早期版本的Struts以及無數其餘的Java規範和框架中都能看到。Spring竭力避免因自身的API而弄亂你的應用代碼。Spring不會強迫你實現Spring規範的接口或繼承Spring規範的類,相反,在基於Spring構建的應用中,它的類一般沒有任何痕跡代表你使用了Spring。 最壞的場景是, 一個類或許會使用Spring註解, 但它依舊是POJO。
任何一個有實際意義的應用(確定比Hello World示例更復雜) 都會由兩個或者更多的類組成, 這些類相互之間進行協做來完成特定的業務邏輯。 按照傳統的作法, 每一個對象負責管理與本身相互協做的對象(即它所依賴的對象) 的引用, 這將會致使高度耦合和難以測試的代碼。java
首先建立的Maven Poject,詳細包結構以下程序員
其中AOP會在下一篇進行講解;web
Controller_.javaspring
/** * @see HttpServlet#doGet(HttpServletRequest request, HttpServletResponse response) */ protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { // TODO Auto-generated method stub response.getWriter().append("Served at: ").append(request.getContextPath()); ClassPathXmlApplicationContext ctx=new ClassPathXmlApplicationContext("/ApplicationContext.xml"); Service_ s = (Service_) ctx.getBean("service_Impl1_new"); System.out.println(s); s.show(); }
因爲Spring沒法單獨演示,因此Controller_.java是建立的是一個Servlet,直接調用doPost或者doGet方法,進行Service的實現,輸出Service_對象s,執行show方法。express
public interface Service_ { public void show(); }
建立一個Service接口,用來實現Spring。編程
Service_Impl1.javaapp
public class Service_Impl1 implements Service_{ public Service_Impl1() { // TODO Auto-generated constructor stub System.out.println("service1-無參構造方法"); } @Override public void show() { // TODO Auto-generated method stub System.out.println("Service_Impl1"); } }
重寫Service_的show方法輸出實現了Service_Impl1,編寫無參構造方法。框架
ApplicationContext.xmlide
<!-- 默認構造方法 -->
<bean id="service_Impl1" class="com.zy.spring.service.serviceimpl.Service_Impl1"></bean>
只須要設置id與class,class對應Service_Impl1,id則是Controller_.java調用的getBean中的參數。運行結果見自定義構造方法注入bean測試
Service_Impl2.java
public class Service_Impl2 implements Service_{ public Service_Impl2(int a) { // TODO Auto-generated constructor stub System.out.println("service2-自定義構造參數:"+a); } @Override public void show() { // TODO Auto-generated method stub System.out.println("Service_Impl2"); } }
ApplicationContext.xml
<!-- 自定義構造方法 -->
<bean id="service_Impl2" class="com.zy.spring.service.serviceimpl.Service_Impl2">
<constructor-arg index="0" value="1024"></constructor-arg>
</bean>
<constructor-arg index="0" value="1024"></constructor-arg>這是構造方法中參數的設置,index顧名思義就是索引的意思,其中a參數是第0個,value是參數的值。
Service_Impl3.java
public class Service_Impl3 implements Service_{ public Service_Impl3() { // TODO Auto-generated constructor stub System.out.println("service3-懶加載 單實例"); } @Override public void show() { // TODO Auto-generated method stub System.out.println("Service_Impl3"); } }
ApplicationContext.xml
<!-- 單實例 懶加載 -->
<bean id="service_Impl3" class="com.zy.spring.service.serviceimpl.Service_Impl3" lazy-init="true" scope="singleton"></bean>
lazy-init="true" 設置懶加載,也就是調用的時候纔會加載bean,不會自動加載;scope="singleton" 做用域標籤,單實例也就是隻建立一個實例。
Service_Impl4.java
public class Service_Impl4 implements Service_{ Service_ s3; public Service_ getS3() { return s3; } public void setS3(Service_ s3) { this.s3 = s3; } public Service_Impl4() { // TODO Auto-generated constructor stub System.out.println("service4-參數引用bean"); } @Override public void show() { // TODO Auto-generated method stub System.out.println("Service_Impl4"); } }
ApplicationContext.xml
<!-- 參數引用bean --> <bean id="service_Impl4" class="com.zy.spring.service.serviceimpl.Service_Impl4"> <property name="s3" ref="service_Impl3"></property> </bean>
<property name="s3" ref="service_Impl3"></property> 參數標籤,name是Service_Impl4中的參數s3,ref連接要引用的bean。
Service_Impl5.java
public class Service_Impl5 implements Service_{ String name; ArrayList<String> list; HashMap<String, String> map; HashSet<Integer> set; @Override public int hashCode() { final int prime = 31; int result = 1; result = prime * result + ((list == null) ? 0 : list.hashCode()); result = prime * result + ((map == null) ? 0 : map.hashCode()); result = prime * result + ((name == null) ? 0 : name.hashCode()); result = prime * result + ((set == null) ? 0 : set.hashCode()); return result; } @Override public boolean equals(Object obj) { if (this == obj) return true; if (obj == null) return false; if (getClass() != obj.getClass()) return false; Service_Impl5 other = (Service_Impl5) obj; if (list == null) { if (other.list != null) return false; } else if (!list.equals(other.list)) return false; if (map == null) { if (other.map != null) return false; } else if (!map.equals(other.map)) return false; if (name == null) { if (other.name != null) return false; } else if (!name.equals(other.name)) return false; if (set == null) { if (other.set != null) return false; } else if (!set.equals(other.set)) return false; return true; } @Override public String toString() { return "Service_Impl5 [name=" + name + ", list=" + list + ", map=" + map + ", set=" + set + "]"; } public String getName() { return name; } public void setName(String name) { this.name = name; } public ArrayList<String> getList() { return list; } public void setList(ArrayList<String> list) { this.list = list; } public HashMap<String, String> getMap() { return map; } public void setMap(HashMap<String, String> map) { this.map = map; } public HashSet<Integer> getSet() { return set; } public void setSet(HashSet<Integer> set) { this.set = set; } public Service_Impl5() { // TODO Auto-generated constructor stub System.out.println("service5-初始化屬性"); } @Override public void show() { // TODO Auto-generated method stub System.out.println("Service_Impl5"); } }
其中初始化參數有list,map,set以及普通參數,重寫了hashCode和equals方法,詳見HashMap內存泄漏;重寫toString方法用來輸出初始化屬性。
ApplicationContext.xml
<!-- 初始化屬性 -->
<bean id="service_Impl5" class="com.zy.spring.service.serviceimpl.Service_Impl5">
<property name="name" value="zy"></property>
<property name="map">
<map>
<entry key="AAA" value="aaa"></entry>
<entry key="BBB" value="bbb"></entry>
</map>
</property>
<property name="list">
<list>
<value type="java.lang.String">QQQ</value>
<value type="java.lang.String">WWW</value>
</list>
</property>
<property name="set">
<set>
<value type="java.lang.Integer">111</value>
<value type="java.lang.Integer">222</value>
</set>
</property>
</bean>
其中map標籤內使用<entry key="AAA" value="aaa"></entry>進行賦值。其餘的正常使用property和value進行賦值。
Service_Impl6.java
public class Service_Impl6 implements Service_{ String s5_toString; public String getS5_toString() { return s5_toString; } public void setS5_toString(String s5_toString) { this.s5_toString = s5_toString; } public Service_Impl6() { // TODO Auto-generated constructor stub System.out.println("service6-調用方法返回值"); } @Override public void show() { // TODO Auto-generated method stub System.out.println("Service_Impl6 返回值"+s5_toString); } }
其中調用了Service_Impl5的toString方法而且進行了輸出。
ApplicationContext.xml
<!-- 調用方法返回值 --> <bean id="service_Impl6" class="com.zy.spring.service.serviceimpl.Service_Impl6"> <property name="s5_toString"> <bean class="org.springframework.beans.factory.config.MethodInvokingFactoryBean"> <property name="targetObject" ref="service_Impl5"></property> <property name="targetMethod" value="toString"></property> </bean> </property> </bean>
<bean class="org.springframework.beans.factory.config.MethodInvokingFactoryBean">固定用來聲明調用方法返回值。
targetObject——目標的bean
targetMethod——目標的方法
Service_Impl7.java
public class Service_Impl7 implements Service_{ public static Service_ StaticFactory(int num) { switch (num) { case 1: return new Service_Impl1(); case 2: return new Service_Impl2(100); case 3: return new Service_Impl3(); case 4: return new Service_Impl4(); case 5: return new Service_Impl5(); default: return new Service_Impl6(); } } public Service_Impl7() { // TODO Auto-generated constructor stub System.out.println("service7-靜態工廠"); } @Override public void show() { // TODO Auto-generated method stub System.out.println("Service_Impl7"); } }
工廠在實現類中使用了switch語句進行模擬,靜態工廠在方法前加上static關鍵字,分別調用上面的其餘實現類方法。
ApplicationContext.xml
<!-- 靜態工廠 -->
<bean id="service_Impl7" class="com.zy.spring.service.serviceimpl.Service_Impl7" factory-method="StaticFactory" >
<constructor-arg name="num" value="2"></constructor-arg>
</bean>
使用構造方法注入的方法來賦值<constructor-arg name="num" value="2"></constructor-arg> ;factory-method="StaticFactory" ( factory-method工廠的方法名)
Service_Impl8.java
public class Service_Impl8 implements Service_{ public Service_ factory1(int num) { switch (num) { case 1: return new Service_Impl1(); case 2: return new Service_Impl2(100); case 3: return new Service_Impl3(); case 4: return new Service_Impl4(); case 5: return new Service_Impl5(); default: return new Service_Impl6(); } } public Service_Impl8() { // TODO Auto-generated constructor stub System.out.println("service8-實例工廠"); } @Override public void show() { // TODO Auto-generated method stub System.out.println("Service_Impl8"); } }
ApplicationContext.xml
<!-- 實例工廠 -->
<bean id="service_Impl8" class="com.zy.spring.service.serviceimpl.Service_Impl8" >
</bean>
<bean id="service_Impl8_new" factory-bean="service_Impl8" factory-method="factory1">
<constructor-arg name="num" value="2"></constructor-arg>
</bean>
建立實例工廠bean,首先建立一個實例工廠的bean,而後再建立一個工廠方法的bean去調用工廠的bean。
調用的時候要調用工廠方法的bean,這裏就要調用service_Impl8_new
@Service:用於標註業務層組件
@Controller:用於標註控制層組件(如struts中的action)
@Repository:用於標註數據訪問組件,即DAO組件
@Component(value="*"):泛指組件,當組件很差歸類的時候,咱們可使用這個註解進行標註
Service_Impl9.java
@Service public class Service_Impl9 implements Service_{ public Service_Impl9() { // TODO Auto-generated constructor stub System.out.println("service9-註解注入bean"); } @Override public void show() { // TODO Auto-generated method stub System.out.println("Service_Impl9"); } }
@Service進行bean的聲明(註解只能聲明無參構造方法),使用註解默認聲明的bean是類名的首字母小寫,這裏聲明的bean的id應該是service_Impl9。
ApplicationContext.xml
<!-- 註解掃描IOC根目錄 -->
<context:component-scan base-package="com.zy.spring">
<context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/><!-- 掃描不包括controller -->
</context:component-scan>
使用註解須要加上註解掃描,其中base-package是掃描的目錄,通常使用的是項目的根目錄,之後使用SpringMVC的話,就不用掃描Controller。
@Resource(name="*" type="*")bean寫入@Autowired/@Qualifier@inject/@named