本文主要是介绍初学SSM框架感觉一团糟,看完它你就不蒙了!(Spring),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
建议耐心阅读,可能你收获不会很多,但是会让你对spring不是那么恐惧。
如题,很多小伙伴在初学三大框架时都会各种各样的问题,本文先将问题抛出再谈如何解决。旨在为小伙伴们学习框架做个铺垫。我觉得导致这种问题的原因有以下几点:
- 1 基础不牢,导致学习过程中一个点不懂,造成疑问的无限递归,从而厌学觉得自己不行了,不适合学Java了!
- 2 如果是初学,那么从之前的编码式开发转向配置式开发会让很多同学不适应。代码量减少了,但是随之而来的是一堆配置,还有一堆注解。记得头疼有没有?
- 3 在之前的学习中,曾听闻三大框架非常的厉害,但是学习到这部分知识时,却被告知我们十几天就能学完三大框架了。
第一个问题:基础不牢
首先说说第一个问题,不知道文字是否可以表达清楚,为了表达明确再配个图吧。我不知道学习spring时是什么样的,作为一个学过框架的过来人来说,我认为我可以代表一部分初学的人,那么我们的问题是啥?无论自学还是上培训班,老师说三大框架实际上是个半成品软件,好了大家都懂了。可以随着知识点的深入,很多人基础不牢的问题就都暴露出来了,一个知识点不会导致后面学习非常难受!如果忽略这些不懂,学习怎么使用,或许也是一种办法,但是大多数受不了这罪,死记硬背。
第二个问题:编码式开发转配置式开发不适应
首先,大家需要明确的一点是:**不是你一个人在蒙,是所有人都蒙!**经过前面两个阶段的学习,我们早就习惯了编码式开发,突然来到了新环境–配置式开发,难免会有不适应,这是非常正常的。就是有时候一个注解没加或者配置少了导致程序报错,又不知从何找起,这种抓狂的感觉,我曾体会,兄弟你呢?
针对这个问题,我的建议是对比学习,如果你不知道什么是对比学习,建议看看一些什么减肥公司,整型公司的广告,这一点他们做的非常好。比如学习一个注解时,如果你能想起来用配置的方式如何书写代码,那么你就不会那么蒙了。
“过度”认为框架非常的优秀,认为框架学习周期长
框架可以理解为半成品软件,确实是我们开发的利器,但是框架无非就是一些类的封装,整合。跟我们之前导入的任何jar包,有啥异同?不就是封装的类多一点吗,所以大家不要觉得框架学习周期太短了,实际上不是周期短而是你的心理作用。纵观市面上的培训视频,ssm学习周期大概就是在十二天左右。所以,要对ssm三大框架有个正确的认知。
抛出问题后,我们一起来探讨应该如何解决这些问题,关于后面两个问题已经给出了建议,如果还有问题,欢迎评论区留言交流学习。下面针对第一个问题来解决,对于前面基础不牢,应该自己挤出时间来补上。现在我将用一个案例引出Spring是什么,这样初学者可以暂时抛掉不懂的知识,先学习Spring的原理以及使用。
可能用到的名词
- 耦合:程序之间的依赖关系,在java中的耦合可以分为类之间的耦合与方法之间的耦合
- 解耦:降低程序之间的关系。
首先来看一下熟悉的原生的JDBC代码,我们刚接触jdbc都是这么写的吧,现在过了这么长时间大家能看出来代码有哪些问题吗?
public class JdbcDemo1 {public static void main(String[] args) throws SQLException, ClassNotFoundException {// 1.注册驱动
// DriverManager.registerDriver(new com.mysql.jdbc.Driver());Class.forName("com.mysql.jdbc.Driver");// 2.获取连接对象Connection conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/eesy","root","1234");// 3.获取预编译对象PreparedStatement preparedStatement = conn.prepareStatement("select * from account");// 4.执行sql,获取结果集ResultSet resultSet = preparedStatement.executeQuery();// 5.遍历结果集while (resultSet.next()){System.out.println(resultSet.getString("name"));}// 6.释放资源resultSet.close();preparedStatement.close();conn.close();}
}
程序间都是存在耦合的。在注册驱动注释掉的代码中使用到了new关键字,产生的耦合就是该类没有Driver类就无法编译通过,这个问题我们可以通过下面的代码来解决,这里使用类的全限定类名,通过反射来创建对象,解决了new的问题。
解决了问题后又产生了新问题,在编程中有开闭原则,对修改关闭,对扩展开放。假设现在要使用oracle数据库,那么字符串形式的类的全限定名是否就有硬编码的问题,我们要使用其他数据库就必须修改源码。这个问题该如何解决呢?
jdbc中存在的一个问题就是程序间的耦合,类之间互相依赖。接下来我们通过一个案例,然后通过分析案例存在问题,接着解决问题,最终达到让大家了解Spring框架的核心所在。
一个小案例
这里我们要模拟一个保存用户账户的操作,也就是在数据库中插入一条数据,为了突出重点,这里并没有真实向数据库中插入,而是用一句输出来模拟插入成功的操作。
由于自己也看了不少大佬的博客,但是以涉及代码便没有点开的欲望,我想过如何会让大家对代码没有那么恐惧?思考了很久,目前我的想法是,贴出左侧项目图,让大家有个整体的了解。
如图,dao是持久层,操作数据库相关的。service是业务层,ui对应的就是我们的表现层(也叫Web层)。这个factory包与下面的配置文件我们暂且不看。我们知道一个完整的流程是这样的:表现层调用业务层,业务层调用持久层,然后由持久层将数据插入到数据库中。下面贴图说明一下这段文字,帮助更好的理解三层之间的联系。
这个保存账户的功能我们实现了米有?实现了吧,但是不够完美,这里出现了与原生jdbc一样的问题。这个问题就是表现层依赖了业务层,业务层依赖了持久层。这句话相信很多初学者是似懂非懂的感觉,说简单点就是如果没有AccountServiceImpl这个业务层实现类,表现层的代码是不是编译都通过不了,同样的业务层跟持久层也存在这样的问题。总之,一出现new就会产生依赖。所以我们解决当下这个问题刻不容缓。
针对上面的问题我们又应该如何解决呢?从原生jdbc中我们似乎用到了全限定类名可以规避new对象的问题。同样的在这里我们可以使用通过配置文件来存储类的全限定类名,存储的方式我们使用key-value形式。然后通过反射,通过key就能创建key对应的对象。从上面的项目结构图中可以看到一个bean.properties和一个工厂类,下面贴出这两部分代码,然后在说明一下。
public class BeanFactory1 {// 定义一个Properties对象private static Properties properties;// 使用静态代码块为Properties赋值// 注意:使用static初始化一个成员变量,这个成员必须是静态的static {try {// 实例化对象properties = new Properties();// 获取properties文件的流对象InputStream asStream = BeanFactory1.class.getClassLoader().getResourceAsStream("bean.properties");properties.load(asStream);} catch (IOException e) {e.printStackTrace();}}/*** 根据bean的名称获取bean*/public static Object getBean(String beanName){Object bean = null;try {String beanPath = properties.getProperty(beanName);bean = Class.forName(beanPath).newInstance();} catch (Exception e) {e.printStackTrace();}return bean;}
}
首先是配置文件,我们编写了两个键值对,用来表示两个类的全限定类名。这个配置文件用在工厂类中,在工厂类中先用静态代码块获取配置文件中的信息,这样就能获取到配置文件中的key,从而通过key来创建自己想要的对象。(通过反射创建,具体看图)。如果你对反射不太了解可以看看这篇文章,可选读6即可理解这里如何通过反射创建对象。
反射:一个困扰我很久的知识点 - Evader1997的文章 - 知乎
https://zhuanlan.zhihu.com/p/158630818
通过上面的波操作,我们的代码变成了这样:
通过上图我们可以看出这里我们已经消除了表现层与业务层,业务层与持久层之间的一部分依赖。但是还是存在问题:在工厂类的倒数第四行,每创建一个bean都会调用默认构造函数创建对象,但是java存在垃圾回收机制,那么这个bean长时间不用就会被回收。这个时候我们又应当如何?
针对上面的问题,我的解决方案是:这个时候我们应该用一个容器来存放这些bean,避免被回收。另外放到容器中另一个好处是,这样的bean都是单例的,效率相对于多例的效率肯定是要高很多的。至于单例,多例以后随着学习的深入会涉及到,这里不必深究!所以我把上面的工厂类做了这样的更改。
public class BeanFactory {// 定义一个Properties对象private static Properties properties;// 定义一个Map,用于存放我们要创建的对象,我们把他称之为容器private static Map<String,Object> beans;// 使用静态代码块为Properties赋值// 注意:使用static初始化一个成员变量,这个成员必须是静态的static {try {// 实例化对象properties = new Properties();// 获取properties文件的流对象// 这个方法是获取什么的InputStream asStream = BeanFactory.class.getClassLoader().getResourceAsStream("bean.properties");properties.load(asStream);// 实例化容器beans = new HashMap<String, Object>();// 取出配置文件中的所有的keyEnumeration<Object> keys = properties.keys();// 遍历枚举while (keys.hasMoreElements()){// 取出每一个key// 这个方法和jdbc的有点像,看看是什么作用String key = keys.nextElement().toString();// 根据key获取valueString beanPath = properties.getProperty(key);// 反射创建对象Object value = Class.forName(beanPath).newInstance();// 把key和value存入容器中beans.put(key,value);}} catch (Exception e) {throw new ExceptionInInitializerError("初始化properties失败");}}/*** 根据bean的名称获取对象* beans容器是static静态块初始化好的,对properties文件中的所有类都创建了一个唯一的对象* 我么通过键取值就行了* @param beanName* @return*/public static Object getBean(String beanName){return beans.get(beanName);}
}
改进后的代码主要体现在了成员位置定义了一个Map类型的容器beans,在创建这个beans之后,在静态代码块中实例化该容器,并且取出配置文件中的所有的key,然后为其对应的全限定类名创建相应的对象,还是单例的,然后存放到Map容器中!至此,问题解决及优化已经完成了。
至此相信很多小伙伴对于spring存在的意义,以及实现原理有了一定的了解吧。在Spring这框架中,IOC应该是主体,实际上就是一个管理对象的大容器,至于AOP,DI我相信大家自己能够处理好,如问题较多,考虑再出一篇关于AOP的文章。
有的key,然后为其对应的全限定类名创建相应的对象,还是单例的,然后存放到Map容器中!至此,问题解决及优化已经完成了。
至此相信很多小伙伴对于spring存在的意义,以及实现原理有了一定的了解吧。在Spring这框架中,IOC应该是主体,实际上就是一个管理对象的大容器,至于AOP,DI我相信大家自己能够处理好,如问题较多,考虑再出一篇关于AOP的文章。
如果你看完了这篇文章,我真心感谢,如果有问题请私聊斧正,万分感谢,抱拳了老铁们!
如果你觉得这篇文章对你有帮助的话,建议阅读以下同系列文章!
初学SSM框架感觉一团糟,看完它你就不蒙了!(Mybatis)
这篇关于初学SSM框架感觉一团糟,看完它你就不蒙了!(Spring)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!