spring 注入线程
很多同学在进行编程学习时缺乏系统学习的资料。本页面基于spring 注入线程内容,从基础理论到综合实战,通过实用的知识类文章,标准的编程教程,丰富的视频课程,为您在spring 注入线程相关知识领域提供全面立体的资料补充。同时还包含 safari浏览器、samba、SAMP 的知识内容,欢迎查阅!
spring 注入线程相关知识
-
最详细的 Spring IOC 注入 (xml 注入 + 注解注入)@[toc](Spring IOC 容器的基本使用) 一、为什么要使用 Spring? 1.1 传统的 MVC 架构的程序 1.2 程序耦合性过高? 1.3 如何解耦? 1.4 Spring IOC 的依赖注入 二、Spring IOC 的依赖注入 2.1 使用构造函数完成依赖注入 2.1.1 标签的使用讲解 2.1.2 构造函数依赖注入的优缺点 2.1.3 使用构造函数完成依赖注入的实例 2.2 使用 setter 完成注入 2.2.1 使用 setter 完成依赖注入的功能 2.2.2 基于 setter 完成依赖注入的分析 2.3 复杂数据类型注入 2.3.1
-
Spring--依赖注入 or 方法注入 ?依赖注入 我们在 [Spring — 循环依赖]中谈到 Spring 的两种依赖注入方式 构造器注入 属性注入(setter注入也归属于此) @Service public class HelloService { /** * 属性注入 */ @Autowired private BeanFactory beanFactory; /** * 构造器注入 */ public HelloService(ApplicationContext applicationContext) { } /** * 属性注入 * */ @Autowired public void setEnvironment(Environment environment) { System.out.println(""); } } 关于构造
-
线程注入ThreadInject(dll)原理通过挂起线程(SuspendThread),设置线程(SetThreadContext)上下文中的eip(rip)方式注入。//ThreadInject.h#pragma once// ThreadInject 对话框class ThreadInject : public CDialogEx{ DECLARE_DYNAMIC(ThreadInject)public: ThreadInject(CWnd* pParent = NULL); // 标准构造函数 virtual ~ThreadInject();// 对话框数据 enum { IDD = IDD_DIALOG2 };protected: virtual void DoDataExchange(CDataExchange* pDX); // DDX/DDV 支持 &n
-
Spring 注入集合注入集合你已经看到了如何使用 value 属性来配置基本数据类型和在你的 bean 配置文件中使用<property>标签的 ref 属性来配置对象引用。这两种情况下处理奇异值传递给一个 bean。现在如果你想传递多个值,如 Java Collection 类型 List、Set、Map 和 Properties,应该怎么做呢。为了处理这种情况,Spring 提供了四种类型的集合的配置元素,如下所示:元素描述<list>它有助于连线,如注入一列值,允许重复。<set>它有助于连线一组值,但不能重复。<map>它可以用来注入名称-值对的集合,其中名称和值可以是任何类型。<props>它可以用来注入名称-值对的集合,其中名称和值都是字符串类型。你可以使用<list>或<set>来连接任何 java.util.Collection 的实现或数组。你会遇到两种情况(a)传递集合中直接的值(b)传递一个 bean 的引用作为集合的元素
spring 注入线程相关课程
spring 注入线程相关教程
- 2.2 指定线程的入口函数 下面通过一个具体的例子,说明通过指定线程的入口函数的方式使用多线程。import timeimport threadingdef thread_entry(begin, end): for i in range(begin, end): time.sleep(1) print(i)t0 = threading.Thread(target = thread_entry, args = (1, 4))t1 = threading.Thread(target = thread_entry, args = (101, 104))t0.start()t1.start()t0.join()t1.join()在第 9 行和第 10 行,通过调用 Thread 的构造函数创建了两个线程。在第 9 行,设定线程的入口函数为 thread_entry,传递给入口函数两个参数:1 和 4,新的线程将执行 thread_entry(1, 4),变量 t0 指向新创建的线程对象。在第 10 行,设定线程的入口函数为 thread_entry,传递给入口函数两个参数:101 和 104,新的线程将执行 thread_entry(101, 104),变量 t1 指向新创建的线程对象。在第 4 行到第 7 行,定义了线程入口函数,该函数的功能是打印在 [begin, end) 区间的整数,每打印一个整数,调用 time.sleep(1) 睡眠 1 秒钟。在第 11 行,调用 start 方法启动线程 t0,t0 开始执行 thread_entry(1, 4)。在第 12 行,调用 start 方法启动线程 t1,t1 开始执行 thread_entry(101, 104)。在第 13 行和第 14 行,调用 join 方法,等待线程 t0 和 t1 执行完毕。程序的运行结果如下:110121023103线程 t0 的输出结果为 1、2、3,线程 t1 的输出结果为 101、102、103。由于两者是并发执行的,所以结果交织在一起。
- 4.2 注入 OutputStream 在控制器的方法中注入 OutputStream 对象,只需要在方法中添加参数声明。如下实例:可使用 OutputStream 对象读取指定文件中的内容后直接响应给浏览器。@RequestMapping(value = "/testApi05")public void hello(OutputStream outputStream) throws IOException { Resource res = new ClassPathResource("/test.txt"); FileCopyUtils.copy(res.getInputStream(), outputStream);}test.txt 文件的内容是”this is a test’。文件直接放在项目的 src/main/java 目录下。在浏览器中输入请求路径 http://localhost:8888/sm-demo/testApi05 。你将在浏览器中看到:有句话叫做 “条条道路通罗马”,用在 Spring MVC 中真的是合适,依靠 Spring 强大的注入功能,只要原生开发中能有的对象基本上都能注入进去。
- 3. 线程上下文 current_session_context_class 可配置值除 thread 外还有 jta、managed 等,简单描述下:当使用本地 Jdbc 事务时选择 Thread。当使用全局 jta 事务时选择 jta。当使用 session 管理机制时选择 managed;如和 Spring 一起整合使用时,使用 Spring 的事务管理机制。主要聊聊 thread 上下文是如何实现保存 Session,回顾一下上一节课程 HibernateSessionFactory 类中的代码片段:private static final ThreadLocal<Session> threadLocal = new ThreadLocal<Session>();public static Session getSession() throws HibernateException { Session session = (Session)threadLocal.get(); aif(session == null || !session.isOpen()) { session = (sessionFactory!= null) ? sessionFactory.openSession():null; threadLocal.set(session); } return session;}实现的关键就在于 ThreadLocal 这个类,ThreadLocal 是 Java SE 原生 API,此类实例化对象本质就是一个 Map 集合,与 Map 保存数据时不同,key 由线程对象充当。使用此对象可以为每一个线程保存只属于当前线程的数据。HibernateSessionFactory 中重构过的 getSession() 方法解析如下:以当前线程对象为 key 查询 threadLocal 集合中是否存在 Session 对象,如有直接返回;Session session = (Session) threadLocal.get();return session;如果没有,则创建 Session 对象,用当前线程作为 key 保存 Session 对象到 threadLocal 对象中。if(session == null || !session.isOpen()) { session = (sessionFactory!= null) ? sessionFactory.openSession():null; threadLocal.set(session);}如上面代码所述,只要线程生命周期没走到尽头,与其关联的 Session 对象就能重复使用。并且每一个线程中使用的是与本线程相关联的 Session,避免了多线程环境下 Session 变成临界资源,避开线程安全隐患。
- 4. Spring MVC 的特性 要了解 Spring MVC 的功能特性,就需要从 WEB 应用开发的源头说起。WEB 应用程序的主流开发技术有 3 种:servlet;php;.net。Servlet 是基于 Java 语言的动态 web开发技术,Servlet 指的是 J2EE 中所提出来的企业级服务器开发规范。原生 Servlet 构建的 MVC 开发模式有几个缺点:Servlet 本身采用单例设计模式,生命周期由服务器维护,存在线程安全隐患的问题;理论上讲,每一次不同的 http 请求需要一个 Servlet 组件来响应,当请求类型比较多时,Servlet 就会相应增加,也意味着每一个 Servlet 都可以成为进入服务器的入口。想想你家里对外开了好多扇门,小偷终能找到破绽;随着 Servlet 数量的增加,对服务器的存储空间也会产生压力;使用 Servlet 响应用户请求时,每一个响应逻辑都需要开发者不厌其烦地做些重复的事情,如,解析请求包中的数据、构建响应包、设置页面跳转等等。这些问题,在 Spring MVC 中都得到了很好的解决。使用 Spring MVC 时,就只有一个门可以进入应用程序,这个门叫前端(中央)控制器,所有请求统一经过这个前端控制器分流到具体的内部响应组件;Spring MVC 是 Spring 家族中的一员,有句话叫做“近水楼台先得月”。使用 Spring MVC 时所需要的 WEB 组件也好、其它的逻辑组件也好,都经由 Spring IOC 创建,Spring IOC 容器对组件的生命周期可进行伸缩性设置管理,可根据组件特性保证其线程安全性;Spring MVC 和 Spring 有直属血缘的关系,两者完美结合,使得程序的安全性和稳定性有一定的保证;Spring MVC 利用 Spring 的自动注入功能,能轻松地装配好各组件之间的依赖,开发者只需要关注编写自己的业务逻辑便可,和所有框架理念一样,解放双手,释放大脑。Spring MVC 设计的初衷,就是要做成一款轻量级框架,其内在的原力让我们一起在后续课程中慢慢释放。
- 6.2 线程唤醒 思考:当前线程休眠了,那么什么时候进行唤醒呢?源码分析如下所示:@Overridepublic Promise<V> setSuccess(V result) { //1.setSuccess0 赋值操作 if (setSuccess0(result)) { //2.通知执行监听器 notifyListeners(); return this; } throw new IllegalStateException("complete already: " + this);}private boolean setSuccess0(V result) { //继续进入方法 return setValue0(result == null ? SUCCESS : result);}private boolean setValue0(Object objResult) { if (RESULT_UPDATER.compareAndSet(this, null, objResult) || RESULT_UPDATER.compareAndSet(this, UNCANCELLABLE, objResult)) { //继续进入方法 checkNotifyWaiters(); return true; } return false;}private synchronized void checkNotifyWaiters() { if (waiters > 0) { //核心:唤醒之前休眠的线程 notifyAll(); }}源码分析总结:堵塞的核心是通过 Object.wait () 方法进行休眠当前线程,普通的 Java 多线程知识;执行完成之后给不同状态(setSuccess、setFailure)赋值的时候唤醒休眠的线程;唤醒线程之后调用监听器的方法 l.operationComplete(future);
- 2. 依赖注入案例 2.1概念介绍知识回顾对于依赖注入,我们在第一章第一节已经介绍过,我们回顾一下概念解释上面是我们之前对于依赖注入的一个通俗解释。那么这里再着重强调一下 IOC 控制反转与 DI 依赖注入的关系:IOC 控制反转是将对象实例化的动作交由了 Spring 框架, 它的作用是降低了程序的耦合,不需要我们手动的创建对象,但是程序的耦合性还是存在。对象中肯定会有一些其余对象的引用,那么这种引用就称呼为对象的依赖,而 DI 依赖注入其实 是 IOC 设计思想的一种表现形式。对于 这种属性依赖,我们无需手动赋予,也是讲赋值的动作交给 Spring ,那么这种操作就是 依赖注入。依赖注入方式:第一种方式是通过 xml 配置的方式实现;第二种方式是在属性或者方法上使用注解的方式实现。那么,本章节先带大家体验下 xml 方式实现依赖注入。2.2 工程实现:搭建动作介绍创建一个 maven 工程导入Spring 使用的依赖编写业务层的 Service 和持久层的 Dao java 类编写 Spring 的配置文件创建工程 导入依赖 省略可以参考之前创建过的IOC工程java 代码创建 Servcie 的接口和接口的实现类,代码如下://接口代码public interface UserService { public void deleteById(Integer id);}//实现类代码public class UserServiceImpl implements UserService { private UserDao userDao; public UserDao getUserDao() { return userDao; } public void setUserDao(UserDao userDao) { this.userDao = userDao; } public void deleteById(Integer id) { System.out.println("删除的方法执行"); }}UserDao 接口和实现类代码://dao接口代码public interface UserDao {}//dao实现类代码public class UserDaoImpl implements UserDao {}代码解释: dao的接口和实现类中并没有方法,只是为了测试 作为service中的属性依赖,可以实现由 Spring 完成动态注入。重点来了:spring 的核心配置文件:配置解释:在上面的配置文件中:bean 标签是描述一个被实例化的类 而 property 则表示一类中的属性property 标签中的属性 name 一般我们写成类中的属性名称, 实际上,起决定作用的并不是属性名,下面示例再展示ref 表示当前的属性 是一个引用对象,而引用的是谁呢? ref 中的值 必须是在容器中已经实例化的一个引用对象的唯一标识。value 当前的属性可以直接赋值,所以通过 value 中,填写要赋予的数值即可测试结果代码解释可以看到 我们得到了 service 中的类属性 Userdao 的实例,并且也 得到了 字符串属性 userName的值 zs2.3 property注入属性的解释刚刚我们在上面的示例中 展示了xml依赖属性的注入,也是比较好理解。这里我们强调一下使用的注意事项:如果是 property 属性标签实现属性注入,那么类中必须由配置在 property 标签中 name 属性的 set 方法下面我们测试一下set方法的改变:先讲 service 中 dao 的 set 方法改造如下:public void setDao(UserDao userDao) { System.out.println("执行了set方法 给dao属性赋值"); this.userDao = userDao;}这时候代码中的set方法变成了 setDao 配置文件不变,依然是<property name="userDao" ref="userDao"></property>我们看看会产生什么问题Caused by: org.springframework.beans.NotWritablePropertyException: Invalid property 'userDao' of bean class [com.wyan.service.UserServiceImpl]: Bean property 'userDao' is not writable or has an invalid setter method. Does the parameter type of the setter match the return type of the getter? at org.springframework.beans.BeanWrapperImpl.createNotWritablePropertyException(BeanWrapperImpl.java:247) at org.springframework.beans.AbstractNestablePropertyAccessor.processLocalProperty(AbstractNestablePropertyAccessor.java:426) at org.springframework.beans.AbstractNestablePropertyAccessor.setPropertyValue(AbstractNestablePropertyAccessor.java:278) at org.springframework.beans.AbstractNestablePropertyAccessor.setPropertyValue(AbstractNestablePropertyAccessor.java:266) at org.springframework.beans.AbstractPropertyAccessor.setPropertyValues(AbstractPropertyAccessor.java:97) at org.springframework.beans.AbstractPropertyAccessor.setPropertyValues(AbstractPropertyAccessor.java:77) at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.applyPropertyValues(AbstractAutowireCapableBeanFactory.java:1646)可以看到异常的堆栈信息 无效的 userDao 属性, userDao 不可以 或者 没有有效的 setter 方法提供。更改xml文件中的 property 标签的 name 属性 为 dao<property name="dao" ref="userDao"></property>测试结果如下:所以我们说 property 中的 name 属性不一定要跟 Java类中的属性名保持一致 而是必须跟 setter 方法的名称一致
spring 注入线程相关搜索
-
s line
safari浏览器
samba
SAMP
samplerate
sandbox
sanitize
saper
sas
sass
save
smarty模板
smil
smtp
snapshot
snd
snmptrap
soap
soapclient
soap协议