死磕源码系列【springboot之ServletContextInitializer接口源码解析】-程序员宅基地

技术标签: filter  【spring系列】  servlet  registration  initializer  

springboot提供在Servlet 3.0+环境中用于编码方式配置ServletContext的接口,此接口(ServletContextInitializer)主要被RegistrationBean抽象类实现用于往ServletContext容器中注册Servlet、Filter或者Listener。这些实现了此接口的Bean的生命周期将会交给Spring容器管理,而不会交给Servlet容器。

1.ServletContextInitializer接口源码:
@FunctionalInterface
public interface ServletContextInitializer {
    

	/**
	 * 使用初始化所需要的任何servlet、Filter、Listener上下文参数及所需要的参数进行初始化ServletContext
	 * @param servletContext 将要初始化的上下文
	 * @throws ServletException 抛出发生的异常信息
	 */
	void onStartup(ServletContext servletContext) throws ServletException;

}
2.RegistrationBean是基于Servlet3.0+的注册bean的基类,此类是一个抽象类,里面的很多方法都是抽象方法,具体由其子类来实现
public abstract class RegistrationBean implements ServletContextInitializer, Ordered {
    

	private static final Log logger = LogFactory.getLog(RegistrationBean.class);
	//注册bean的优先级
	private int order = Ordered.LOWEST_PRECEDENCE;
	//指示注册是否已经启用的标记
	private boolean enabled = true;

	@Override
	public final void onStartup(ServletContext servletContext) throws ServletException {
    
    //获取注册bean的描述信息
		String description = getDescription();
    //判定是否开启注册功能,否则打印info日志并且直接返回
		if (!isEnabled()) {
    
			logger.info(StringUtils.capitalize(description) + " was not registered (disabled)");
			return;
		}
    //调用抽象的注册方法
		register(description, servletContext);
	}

	/**
	 * 返回注册的描述说明
	 */
	protected abstract String getDescription();

	/**
	 * 在servlet上下文中注册这个bean.
	 * @param description 正在注册项的描述
	 * @param servletContext the servlet context
	 */
	protected abstract void register(String description, ServletContext servletContext);

	/**
	 * 指示注册是否已经启用的标记
	 */
	public void setEnabled(boolean enabled) {
    
		this.enabled = enabled;
	}

	/**
	 * 返回注册是否一起用的标记boolean值
	 */
	public boolean isEnabled() {
    
		return this.enabled;
	}

	/**
	 * 设置注册bean的优先级顺序
	 */
	public void setOrder(int order) {
    
		this.order = order;
	}

	/**
	 * 获取注册bean的优先级顺序
	 */
	@Override
	public int getOrder() {
    
		return this.order;
	}

}
3.DynamicRegistrationBean是一个抽象类,继承了RegistrationBean抽象类,是基于Servlet3.0+的注册bean的基类
public abstract class DynamicRegistrationBean<D extends Registration.Dynamic> extends RegistrationBean {
    

	private static final Log logger = LogFactory.getLog(RegistrationBean.class);
	//注册的名称,如果没有指定,将使用bean的名称
	private String name;
  //是否支持异步注册
	private boolean asyncSupported = true;

	private Map<String, String> initParameters = new LinkedHashMap<>();

	/**
	 * 设置注册的名称,如果没有指定,将使用bean的名称
	 */
	public void setName(String name) {
    
		Assert.hasLength(name, "Name must not be empty");
		this.name = name;
	}

	/**
	 * 如果此操作支持异步注册,则支持异步集,如果未指定,则默认为true
	 */
	public void setAsyncSupported(boolean asyncSupported) {
    
		this.asyncSupported = asyncSupported;
	}

	/**
	 * 判定当前注册是否支持异步注册
	 */
	public boolean isAsyncSupported() {
    
		return this.asyncSupported;
	}

	/**
	 * 为此注册设置init参数,调用此方法将替换任何现有的init参数
	 */
	public void setInitParameters(Map<String, String> initParameters) {
    
		Assert.notNull(initParameters, "InitParameters must not be null");
		this.initParameters = new LinkedHashMap<>(initParameters);
	}

	/**
	 * 返回注册的初始化参数
	 */
	public Map<String, String> getInitParameters() {
    
		return this.initParameters;
	}

	/**
	 * 添加一个init参数,用相同的名称替换任何现有的参数
	 * @param name 初始化参数名
	 * @param value 初始化参数值
	 */
	public void addInitParameter(String name, String value) {
    
		Assert.notNull(name, "Name must not be null");
		this.initParameters.put(name, value);
	}
	//注册方法的具体实现
	@Override
	protected final void register(String description, ServletContext servletContext) {
    
    //调用具体的注册方法
		D registration = addRegistration(description, servletContext);
		if (registration == null) {
    
			logger.info(StringUtils.capitalize(description) + " was not registered (possibly already registered?)");
			return;
		}
    //配置注册结果
		configure(registration);
	}
	//注册具体动作抽象方法
	protected abstract D addRegistration(String description, ServletContext servletContext);
  //注册结果及设置参数
	protected void configure(D registration) {
    
		registration.setAsyncSupported(this.asyncSupported);
		if (!this.initParameters.isEmpty()) {
    
			registration.setInitParameters(this.initParameters);
		}
	}

	/**
	 * 推断此注册的名称,将返回用户指定的名称或回退到基于约定的命名
	 */
	protected final String getOrDeduceName(Object value) {
    
		return (this.name != null) ? this.name : Conventions.getVariableName(value);
	}

}
4.ServletRegistrationBean是基于servlet3.0+容器注册Servlet,类似于ServletContext#addServlet(String, Servlet)注册方法,但是提供了注册为spring bean的友好设计
public class ServletRegistrationBean<T extends Servlet> extends DynamicRegistrationBean<ServletRegistration.Dynamic> {
    
	//默认路径匹配
	private static final String[] DEFAULT_MAPPINGS = {
     "/*" };
  //将要注册的servlet
	private T servlet;
	//映射的URL路由模式集合
	private Set<String> urlMappings = new LinkedHashSet<>();
	//如果省略了URL映射,则应将其替换为"/*"
	private boolean alwaysMapUrl = true;
	//启动优先级
	private int loadOnStartup = -1;
	//要设置的配置
	private MultipartConfigElement multipartConfig;

	/**
	 * 创建一个ServletRegistrationBean实例对象
	 */
	public ServletRegistrationBean() {
    
	}

	/**
	 * 创建一个ServletRegistrationBean实例对象,并且制定servlet和URL映射参数
	 */
	public ServletRegistrationBean(T servlet, String... urlMappings) {
    
		this(servlet, true, urlMappings);
	}

	/**
	 * 创建一个ServletRegistrationBean实例对象,并且制定servlet、如果省略URL则使用"/*"替换,和URL映射参数
	 * @param servlet the servlet being mapped
	 * @param alwaysMapUrl if omitted URL mappings should be replaced with '/*'
	 * @param urlMappings the URLs being mapped
	 */
	public ServletRegistrationBean(T servlet, boolean alwaysMapUrl, String... urlMappings) {
    
		Assert.notNull(servlet, "Servlet must not be null");
		Assert.notNull(urlMappings, "UrlMappings must not be null");
		this.servlet = servlet;
		this.alwaysMapUrl = alwaysMapUrl;
		this.urlMappings.addAll(Arrays.asList(urlMappings));
	}

	/**
	 * Sets the servlet to be registered.
	 * @param servlet the servlet
	 */
	public void setServlet(T servlet) {
    
		Assert.notNull(servlet, "Servlet must not be null");
		this.servlet = servlet;
	}

	/**
	 * 返回正在注册的servlet
	 */
	public T getServlet() {
    
		return this.servlet;
	}

	/**
	 * 设置servlet的URL映射,如果未指定,映射将默认为"/",这将替换以前指定的任何映射
	 */
	public void setUrlMappings(Collection<String> urlMappings) {
    
		Assert.notNull(urlMappings, "UrlMappings must not be null");
		this.urlMappings = new LinkedHashSet<>(urlMappings);
	}

	/**
	 * 返回注册的servlet的映射集合
	 * @return the urlMappings
	 */
	public Collection<String> getUrlMappings() {
    
		return this.urlMappings;
	}

	/**
	 * 添加注册的servlet的url映射
	 * @param urlMappings the mappings to add
	 * @see #setUrlMappings(Collection)
	 */
	public void addUrlMappings(String... urlMappings) {
    
		Assert.notNull(urlMappings, "UrlMappings must not be null");
		this.urlMappings.addAll(Arrays.asList(urlMappings));
	}

	/**
	 * 设置loadOnStartup方法的优先级
	 */
	public void setLoadOnStartup(int loadOnStartup) {
    
		this.loadOnStartup = loadOnStartup;
	}

	/**
	 * 设置注册servlet的配置
	 */
	public void setMultipartConfig(MultipartConfigElement multipartConfig) {
    
		this.multipartConfig = multipartConfig;
	}

	/**
	 * 获取servlet的配置
	 */
	public MultipartConfigElement getMultipartConfig() {
    
		return this.multipartConfig;
	}
	//获取注册servlet的描述
	@Override
	protected String getDescription() {
    
		Assert.notNull(this.servlet, "Servlet must not be null");
		return "servlet " + getServletName();
	}
	//核心,向ServletContext注册servlet对象
	@Override
	protected ServletRegistration.Dynamic addRegistration(String description, ServletContext servletContext) {
    
		String name = getServletName();
		return servletContext.addServlet(name, this.servlet);
	}

	/**
	 * 配置注册配置
	 */
	@Override
	protected void configure(ServletRegistration.Dynamic registration) {
    
		super.configure(registration);
		String[] urlMapping = StringUtils.toStringArray(this.urlMappings);
		if (urlMapping.length == 0 && this.alwaysMapUrl) {
    
			urlMapping = DEFAULT_MAPPINGS;
		}
		if (!ObjectUtils.isEmpty(urlMapping)) {
    
			registration.addMapping(urlMapping);
		}
		registration.setLoadOnStartup(this.loadOnStartup);
		if (this.multipartConfig != null) {
    
			registration.setMultipartConfig(this.multipartConfig);
		}
	}

	/**
	 * 获取将被注册的servlet的名字
	 * @return the servlet name
	 */
	public String getServletName() {
    
		return getOrDeduceName(this.servlet);
	}

	@Override
	public String toString() {
    
		return getServletName() + " urls=" + getUrlMappings();
	}

}

5.AbstractFilterRegistrationBean是基于Servlet3.0+容器注册Filter的抽象基类
public abstract class AbstractFilterRegistrationBean<T extends Filter> extends DynamicRegistrationBean<Dynamic> {
    
	//默认的URL映射路径
	private static final String[] DEFAULT_URL_MAPPINGS = {
     "/*" };
	
	private Set<ServletRegistrationBean<?>> servletRegistrationBeans = new LinkedHashSet<>();

	private Set<String> servletNames = new LinkedHashSet<>();

	private Set<String> urlPatterns = new LinkedHashSet<>();

	private EnumSet<DispatcherType> dispatcherTypes;

	private boolean matchAfter = false;

	/**
	 * 创建AbstractFilterRegistrationBean实例对象,参数为ServletRegistrationBean集合
	 */
	AbstractFilterRegistrationBean(ServletRegistrationBean<?>... servletRegistrationBeans) {
    
		Assert.notNull(servletRegistrationBeans, "ServletRegistrationBeans must not be null");
		Collections.addAll(this.servletRegistrationBeans, servletRegistrationBeans);
	}

	/**
	 * 设置ServletRegistrationBean,过滤器将会针对其进行注册
	 */
	public void setServletRegistrationBeans(Collection<? extends ServletRegistrationBean<?>> servletRegistrationBeans) {
    
		Assert.notNull(servletRegistrationBeans, "ServletRegistrationBeans must not be null");
		this.servletRegistrationBeans = new LinkedHashSet<>(servletRegistrationBeans);
	}

	/**
	 * 返回Filter简要根据ServletRegistrationBean集合对象进行注册的ServletRegistrationBean集合
	 */
	public Collection<ServletRegistrationBean<?>> getServletRegistrationBeans() {
    
		return this.servletRegistrationBeans;
	}

	/**
	 * 添加ServletRegistrationBean对象
	 */
	public void addServletRegistrationBeans(ServletRegistrationBean<?>... servletRegistrationBeans) {
    
		Assert.notNull(servletRegistrationBeans, "ServletRegistrationBeans must not be null");
		Collections.addAll(this.servletRegistrationBeans, servletRegistrationBeans);
	}

	/**
	 * 设置过滤器将注册的servlet名称,这将替换以前指定的任何servlet名称
	 */
	public void setServletNames(Collection<String> servletNames) {
    
		Assert.notNull(servletNames, "ServletNames must not be null");
		this.servletNames = new LinkedHashSet<>(servletNames);
	}

	/**
	 * 返回用于注册过滤器的servlet名称的可变集合
	 */
	public Collection<String> getServletNames() {
    
		return this.servletNames;
	}

	/**
	 * 为Filter添加servlet名称
	 */
	public void addServletNames(String... servletNames) {
    
		Assert.notNull(servletNames, "ServletNames must not be null");
		this.servletNames.addAll(Arrays.asList(servletNames));
	}

	/**
	 * 设置将根据其注册Filter的URL模式,这将替换以前指定的任何URL模式
	 */
	public void setUrlPatterns(Collection<String> urlPatterns) {
    
		Assert.notNull(urlPatterns, "UrlPatterns must not be null");
		this.urlPatterns = new LinkedHashSet<>(urlPatterns);
	}

	/**
	 * 返回一个URL模式的可变集合,如Servlet规范中定义的那样,过滤器针对这些模式进行注册
	 */
	public Collection<String> getUrlPatterns() {
    
		return this.urlPatterns;
	}

	/**
	 * 添加URL模式,如Servlet规范中所定义的,过滤器将针对这些模式进行注册
	 */
	public void addUrlPatterns(String... urlPatterns) {
    
		Assert.notNull(urlPatterns, "UrlPatterns must not be null");
		Collections.addAll(this.urlPatterns, urlPatterns);
	}

	/**
	 * 
	 * @param first the first dispatcher type
	 * @param rest additional dispatcher types
	 */
	public void setDispatcherTypes(DispatcherType first, DispatcherType... rest) {
    
		this.dispatcherTypes = EnumSet.of(first, rest);
	}

	/**
	 * Sets the dispatcher types that should be used with the registration. If not
	 * specified the types will be deduced based on the value of
	 * {@link #isAsyncSupported()}.
	 * @param dispatcherTypes the dispatcher types
	 */
	public void setDispatcherTypes(EnumSet<DispatcherType> dispatcherTypes) {
    
		this.dispatcherTypes = dispatcherTypes;
	}

	/**
	 * Set if the filter mappings should be matched after any declared filter mappings of
	 * the ServletContext. Defaults to {@code false} indicating the filters are supposed
	 * to be matched before any declared filter mappings of the ServletContext.
	 * @param matchAfter if filter mappings are matched after
	 */
	public void setMatchAfter(boolean matchAfter) {
    
		this.matchAfter = matchAfter;
	}

	/**
	 * Return if filter mappings should be matched after any declared Filter mappings of
	 * the ServletContext.
	 * @return if filter mappings are matched after
	 */
	public boolean isMatchAfter() {
    
		return this.matchAfter;
	}

	@Override
	protected String getDescription() {
    
		Filter filter = getFilter();
		Assert.notNull(filter, "Filter must not be null");
		return "filter " + getOrDeduceName(filter);
	}
	//注册过滤器
	@Override
	protected Dynamic addRegistration(String description, ServletContext servletContext) {
    
		Filter filter = getFilter();
		return servletContext.addFilter(getOrDeduceName(filter), filter);
	}

	/**
	 * 配置过滤器的配置
	 */
	@Override
	protected void configure(FilterRegistration.Dynamic registration) {
    
		super.configure(registration);
		EnumSet<DispatcherType> dispatcherTypes = this.dispatcherTypes;
		if (dispatcherTypes == null) {
    
			T filter = getFilter();
			if (ClassUtils.isPresent("org.springframework.web.filter.OncePerRequestFilter",
					filter.getClass().getClassLoader()) && filter instanceof OncePerRequestFilter) {
    
				dispatcherTypes = EnumSet.allOf(DispatcherType.class);
			}
			else {
    
				dispatcherTypes = EnumSet.of(DispatcherType.REQUEST);
			}
		}
		Set<String> servletNames = new LinkedHashSet<>();
		for (ServletRegistrationBean<?> servletRegistrationBean : this.servletRegistrationBeans) {
    
			servletNames.add(servletRegistrationBean.getServletName());
		}
		servletNames.addAll(this.servletNames);
		if (servletNames.isEmpty() && this.urlPatterns.isEmpty()) {
    
			registration.addMappingForUrlPatterns(dispatcherTypes, this.matchAfter, DEFAULT_URL_MAPPINGS);
		}
		else {
    
			if (!servletNames.isEmpty()) {
    
				registration.addMappingForServletNames(dispatcherTypes, this.matchAfter,
						StringUtils.toStringArray(servletNames));
			}
			if (!this.urlPatterns.isEmpty()) {
    
				registration.addMappingForUrlPatterns(dispatcherTypes, this.matchAfter,
						StringUtils.toStringArray(this.urlPatterns));
			}
		}
	}

	/**
	 * Return the {@link Filter} to be registered.
	 * @return the filter
	 */
	public abstract T getFilter();

	@Override
	public String toString() {
    
		StringBuilder builder = new StringBuilder(getOrDeduceName(this));
		if (this.servletNames.isEmpty() && this.urlPatterns.isEmpty()) {
    
			builder.append(" urls=").append(Arrays.toString(DEFAULT_URL_MAPPINGS));
		}
		else {
    
			if (!this.servletNames.isEmpty()) {
    
				builder.append(" servlets=").append(this.servletNames);
			}
			if (!this.urlPatterns.isEmpty()) {
    
				builder.append(" urls=").append(this.urlPatterns);
			}
		}
		builder.append(" order=").append(getOrder());
		return builder.toString();
	}

}

6.FilterRegistrationBean是基于Servlet3.0+容器注册Filter,类似于ServletContext#addFilter(String, Filter)注册过滤器的特性,但是提供了注册为spring bean的友好特性
public class FilterRegistrationBean<T extends Filter> extends AbstractFilterRegistrationBean<T> {
    
	//将要注册的Filter对象
	private T filter;

	/**
	 * 创建一个FilterRegistrationBean实例对象
	 */
	public FilterRegistrationBean() {
    
	}

	/**
	 * 创建一个新的FilterRegistrationBean实例,使用指定的Filter和ServletRegistrationBean作为参数
	 * @param filter the filter to register
	 * @param servletRegistrationBeans associate {@link ServletRegistrationBean}s
	 */
	public FilterRegistrationBean(T filter, ServletRegistrationBean<?>... servletRegistrationBeans) {
    
		super(servletRegistrationBeans);
		Assert.notNull(filter, "Filter must not be null");
		this.filter = filter;
	}
	//获取将要注册的获取器
	@Override
	public T getFilter() {
    
		return this.filter;
	}

	/**
	 * 设置将要被注册的过滤器
	 */
	public void setFilter(T filter) {
    
		Assert.notNull(filter, "Filter must not be null");
		this.filter = filter;
	}

}

GitHub地址:https://github.com/mingyang66/spring-parent

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/yaomingyang/article/details/110965562

智能推荐

51单片机的中断系统_51单片机中断篇-程序员宅基地

文章浏览阅读3.3k次,点赞7次,收藏39次。CPU 执行现行程序的过程中,出现某些急需处理的异常情况或特殊请求,CPU暂时中止现行程序,而转去对异常情况或特殊请求进行处理,处理完毕后再返回现行程序断点处,继续执行原程序。void 函数名(void) interrupt n using m {中断函数内容 //尽量精简 }编译器会把该函数转化为中断函数,表示中断源编号为n,中断源对应一个中断入口地址,而中断入口地址的内容为跳转指令,转入本函数。using m用于指定本函数内部使用的工作寄存器组,m取值为0~3。该修饰符可省略,由编译器自动分配。_51单片机中断篇

oracle项目经验求职,网络工程师简历中的项目经验怎么写-程序员宅基地

文章浏览阅读396次。项目经验(案例一)项目时间:2009-10 - 2009-12项目名称:中驰别克信息化管理整改完善项目描述:项目介绍一,建立中驰别克硬件档案(PC,服务器,网络设备,办公设备等)二,建立中驰别克软件档案(每台PC安装的软件,财务,HR,OA,专用系统等)三,能过建立的档案对中驰别克信息化办公环境优化(合理使用ADSL宽带资源,对域进行调整,对文件服务器进行优化,对共享打印机进行调整)四,优化完成后..._网络工程师项目经历

LVS四层负载均衡集群-程序员宅基地

文章浏览阅读1k次,点赞31次,收藏30次。LVS:Linux Virtual Server,负载调度器,内核集成, 阿里的四层SLB(Server Load Balance)是基于LVS+keepalived实现。NATTUNDR优点端口转换WAN性能最好缺点性能瓶颈服务器支持隧道模式不支持跨网段真实服务器要求anyTunneling支持网络private(私网)LAN/WAN(私网/公网)LAN(私网)真实服务器数量High (100)High (100)真实服务器网关lvs内网地址。

「技术综述」一文道尽传统图像降噪方法_噪声很大的图片可以降噪吗-程序员宅基地

文章浏览阅读899次。https://www.toutiao.com/a6713171323893318151/作者 | 黄小邪/言有三编辑 | 黄小邪/言有三图像预处理算法的好坏直接关系到后续图像处理的效果,如图像分割、目标识别、边缘提取等,为了获取高质量的数字图像,很多时候都需要对图像进行降噪处理,尽可能的保持原始信息完整性(即主要特征)的同时,又能够去除信号中无用的信息。并且,降噪还引出了一..._噪声很大的图片可以降噪吗

Effective Java 【对于所有对象都通用的方法】第13条 谨慎地覆盖clone_为继承设计类有两种选择,但无论选择其中的-程序员宅基地

文章浏览阅读152次。目录谨慎地覆盖cloneCloneable接口并没有包含任何方法,那么它到底有什么作用呢?Object类中的clone()方法如何重写好一个clone()方法1.对于数组类型我可以采用clone()方法的递归2.如果对象是非数组,建议提供拷贝构造器(copy constructor)或者拷贝工厂(copy factory)3.如果为线程安全的类重写clone()方法4.如果为需要被继承的类重写clone()方法总结谨慎地覆盖cloneCloneable接口地目的是作为对象的一个mixin接口(详见第20_为继承设计类有两种选择,但无论选择其中的

毕业设计 基于协同过滤的电影推荐系统-程序员宅基地

文章浏览阅读958次,点赞21次,收藏24次。今天学长向大家分享一个毕业设计项目基于协同过滤的电影推荐系统项目运行效果:项目获取:https://gitee.com/assistant-a/project-sharing21世纪是信息化时代,随着信息技术和网络技术的发展,信息化已经渗透到人们日常生活的各个方面,人们可以随时随地浏览到海量信息,但是这些大量信息千差万别,需要费事费力的筛选、甄别自己喜欢或者感兴趣的数据。对网络电影服务来说,需要用到优秀的协同过滤推荐功能去辅助整个系统。系统基于Python技术,使用UML建模,采用Django框架组合进行设

随便推点

你想要的10G SFP+光模块大全都在这里-程序员宅基地

文章浏览阅读614次。10G SFP+光模块被广泛应用于10G以太网中,在下一代移动网络、固定接入网、城域网、以及数据中心等领域非常常见。下面易天光通信(ETU-LINK)就为大家一一盘点下10G SFP+光模块都有哪些吧。一、10G SFP+双纤光模块10G SFP+双纤光模块是一种常规的光模块,有两个LC光纤接口,传输距离最远可达100公里,常用的10G SFP+双纤光模块有10G SFP+ SR、10G SFP+ LR,其中10G SFP+ SR的传输距离为300米,10G SFP+ LR的传输距离为10公里。_10g sfp+

计算机毕业设计Node.js+Vue基于Web美食网站设计(程序+源码+LW+部署)_基于vue美食网站源码-程序员宅基地

文章浏览阅读239次。该项目含有源码、文档、程序、数据库、配套开发软件、软件安装教程。欢迎交流项目运行环境配置:项目技术:Express框架 + Node.js+ Vue 等等组成,B/S模式 +Vscode管理+前后端分离等等。环境需要1.运行环境:最好是Nodejs最新版,我们在这个版本上开发的。其他版本理论上也可以。2.开发环境:Vscode或HbuilderX都可以。推荐HbuilderX;3.mysql环境:建议是用5.7版本均可4.硬件环境:windows 7/8/10 1G内存以上;_基于vue美食网站源码

oldwain随便写@hexun-程序员宅基地

文章浏览阅读62次。oldwain随便写@hexun链接:http://oldwain.blog.hexun.com/ ...

渗透测试-SQL注入-SQLMap工具_sqlmap拖库-程序员宅基地

文章浏览阅读843次,点赞16次,收藏22次。用这个工具扫描其它网站时,要注意法律问题,同时也比较慢,所以我们以之前写的登录页面为例子扫描。_sqlmap拖库

origin三图合一_神教程:Origin也能玩转图片拼接组合排版-程序员宅基地

文章浏览阅读1.5w次,点赞5次,收藏38次。Origin也能玩转图片的拼接组合排版谭编(华南师范大学学报编辑部,广州 510631)通常,我们利用Origin软件能非常快捷地绘制出一张单独的绘图。但是,我们在论文的撰写过程中,经常需要将多种科学实验图片(电镜图、示意图、曲线图等)组合在一张图片中。大多数人都是采用PPT、Adobe Illustrator、CorelDraw等软件对多种不同类型的图进行拼接的。那么,利用Origin软件能否实..._origin怎么把三个图做到一张图上

51单片机智能电风扇控制系统proteus仿真设计( 仿真+程序+原理图+报告+讲解视频)_电风扇模拟控制系统设计-程序员宅基地

文章浏览阅读4.2k次,点赞4次,收藏51次。51单片机智能电风扇控制系统仿真设计( proteus仿真+程序+原理图+报告+讲解视频)仿真图proteus7.8及以上 程序编译器:keil 4/keil 5 编程语言:C语言 设计编号:S0042。_电风扇模拟控制系统设计