




已阅读5页,还剩26页未读, 继续免费阅读
版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
*表示重要性,星号越多越重要day1 一、 Web应用基础1.B-S架构(*) 架构的发展 c/s vs. b/s c/s架构 (client客户端-server服务端) (胖客户端:要求客户端运行业务;把业务放到服务器端,则是瘦客户端) 典型的c/s应用:ftp工具、QQ、邮件系统、杀毒软件. 1.建立在tcp/ip协议之上,有自己的通信规则(建立业务) 2.需要相互配合才能完成一个完整业务逻辑 3.允许多个客户端程序同时接入一个server程序(并发) 4.每一个client(机器)都必须安装客户软件 5.修改了server程序,通常client程序都要修改(升级) 优点:利用客户端的计算能力,分担服务器的负荷(大型网络游戏就利用这点) 缺点:用户必须安装客户端程序;客户端需要升级(麻烦) b/s架构 (browser - web server(cluster集群) (极瘦客户端:最低限度地减少客户端程序,只需要browser(浏览器) 1.基于http协议(应用层) 2.几乎所有的业务逻辑处理都在server完成 3.支持并发 4.client要求很少,只需要安装browser(浏览器) 5.修改server之后,client不需要任何变化 6.server端开发技术:html/js,xhtml,. php,asp,jsp,servlet 缺点:所有业务都在服务器端完成,服务器负荷大。 优点:支持高并发访问;不需另外安装软件(只需浏览器),免去更新的麻烦。2.开发一个静态的Web应用(*) 1)下载一个tomcat服务器 2)web服务器-Tomcat的启动和配置(熟练使用和配置) 先进入Tomcat主目录下的bin目录 / %catalina_home%/bin window平台:启动-startup.bat 关闭-shutdown.bat Linux平台:启动-startup.sh 或 catalina.sh run /catalina单次启动;startup启动直至shutdown 关闭-shutdown.sh 或 Ctrl+C 测试: 打开浏览器,访问 Tomcat 首页:http:/localhost:8080 或者 :8080/ 获取进程Id强行杀死tomcat进程 ps ef|grep tomcat -查看tomcat的进程id kill -9 tomcat进程Id号 - 强行杀死tomcat进程 3)介绍Web应用的结构 严格定义为两部分 Web应用的根目录下有一个特定的子目录称为WEB-INF,其中包含不能被客户访问的专用Web应用程序软件, 包括Servlet类文件、部署描述符web.xml、外部库以及其他任何由此应用程序使用的专用文件。 所有位于WEB-INF之外的文件都被看作是公共的,它们可以从客户端被访问。资源包括HTML页面、JSP页面和图像等。 web.xml的模板(一个web.xml中可以配置多个Servlet): servlet的名字1 /servlet的逻辑名 servlet类全名1 /类的完全限定名 servlet的名字2 servlet类全名2 servlet的名字1 /要和servlet标签中的相同 指定servlet相对于应用目录的路径 /servlet的访问路径 servlet的名字2 指定servlet相对于应用目录的路径 index.jsp/指定默认的欢迎页面 4)注意事项3.介绍Web服务器的一些特点 什么是静态内容?什么是动态内容? 静态:返回页面每个客户端都相同。 动态:各个客户端的页面各不相同。 产生动态内容的Web辅助应用:CGI & Servlet CGI(Common Gateway Interface): 可以产生动态内容,跨语言(C/C+, Perl, python.) 1.本身是一个进程(数据共享,跨进程调用) -性能较差 2.可移植性很差(本地语言) 3.安全性差-可以访问本地的操作系统,容易受黑客攻击 web server / servlet container 1.性能好 -多线程(而不是进程) 2.可移植性 -相对好 3.安全性4.Http基础(*) HTTP(Hyper Text Transfer Protocol)是超文本传输协议的缩写,它用于传送 WWW 方式的数据。 HTTP 协议采用了请求/响应模型。基于 TCP/IP 之上的协议,在 Web 上广泛使用。无状态。 如果来自服务器的响应是 html 页面,那么 html 内容会嵌入到 Http 响应中。 Http 会话:是一个简单的请求-响应序列。 1)请求响应模型 请求的关键要素:1.http方法(要完成的动作) 2.要访问的页面(URL请求) 3.表单参数 响应的关键要素:1.状态码(请求是否成功) 2.内容类型(txt,img,html等) 3.返回内容(具体资源:html,图片等) 2)URL的分析 URL(Uniform Resoure Locator)统一资源定位器。Web 上的每一个资源都有唯一的地址,采用的就是 url 格式 3)使用Http Monitor截获http的请求与响应信息 Http 请求方法包括:Get()方法;Post()方法; 其他方法:Head, Trace, Put, Delete, Connect 等 Get()方法: Http 最简单的方法,其最主要的任务:从服务器上获取资源 Post()方法:更强大的请求方法,不仅可以请求资源,还可以向服务器发送表单数据 4)分析http请求的信息 客户端向服务器发送一个请求,请求头包含:1.请求的方法; 2.URI; 3.协议版本; 4.以及包含请求修饰符; 5.客户信息和内容的类似于 MIME 的消息结构 5)分析http响应的信息 截获内容: HTTP/1.1 200 OK / HTTP/1.1 是web服务器使用的版本;200 是状态码;OK 是状态码的文本版本 ETag: W/472-1198101802343 Last-Modified: Wed, 19 Dec 2007 22:03:22 GMT Content-Type: text/html / MIME类型:告诉浏览器所要接收的是哪一类型的数据。以供浏览器显示。 Content-Length: 472 Date: Wed, 19 Dec 2007 22:05:44 GMT Server: Apache-Coyote/1.1 /以上是Http响应的首部 /响应体中包含了Html以及其他要显示的内容 User Login Http 响应状态码分析(具体代码含义参看 http 代码对照表): 1xx:信息响应类,表示接收到请求并且继续处理 2xx:处理成功响应类,表示动作被成功接收、理解和接受 3xx:重定向响应类,为了完成指定的动作,必须接受进一步处理 4xx:客户端错误,客户请求包含语法错误或者是不能正确执行 如:404无法找到,表示无法找到指定位置的资源。 5xx:服务端错误,服务器不能正确执行一个正确的请求二、Servlet基础1.介绍Servlet的基本概念(*) (Servlet、Servlet容器、Servlet vs. CGI)图示 1)它是由 java 编写的、服务端的小程序。相对而言,Applet和javaScript是客户端小程序。 2)基于Http协议的,运行在web服务器内的。Servlet和CGI都是运行在Web服务器上,用来生成Web页面。 3)没有 main 方法。是接受来自网络的请求(form表单,以及其他的请求),并对不同请求作出不同的响应。 4)由容器管理和调用。这个web容器可以控制Servlet对象的生命周期,控制请求由Servlet对象处理。 5)Web 服务器的辅助应用,处理特定的用户请求,并返回响应。 web服务器,这里的服务器不是硬件概念,而是软件,常用的web服务器有 Tomcat,Jboss等 Tomcat是一个用java语言编写的web服务器,所以需要有相应的java运行环境,也就是JVM,还要配置tomcat的具体路径。 6)继承 java 的优点,与协议、平台无关2.开发第一个Servlet应用(*) Web应用的结构 开发一个Servlet应用的步骤 1)建立 Web 应用目录(注意细节:各目录的位置、名称与包含的内容) 2)编写 Java 代码,如:HelloWorld.java 把生成的.class 文件放置到 WebRoot/WEB-INF/classes 目录下 3)编写配置文件:web.xml 放置到 WebRoot/WEB-INF/ 目录下 4)把整个 web 应用放到 %catalina_home%/webapps /放到 tomcat的 webapps目录下3.分析Servlet应用(*) 1)Servlet的继承体系 javax.servlet.Servlet接口 - GenericServlet抽象类 - HttpServlet - 自定义类 所有的servlet都必须实现该接口 处理各种协议(包括http) 专职处理http协议 也可继承GenericServlet 2)Servlet接口;GenericServlet抽象类;HttpServlet类 Servlet接口(位置:javax.servlet.Servlet) 定义了特定的生命周期方法: init(ServletConfig config) service(HttpServletRequest request, HttpServletResponse response) destroy() GenericServlet implements Servlet 实现了 init(ServletConfig config)、destroy()等方法;并提供了 init()方法。 还实现了 ServletConfig 接口。由于这是抽象类,所以必须实现抽象方法:service() HttpServlet extends GenericServlet (位置:javax.servlet.http.HttpServlet) 具有父类 GenericServlet 类似的方法和对象,是使用 Servlet 编程常用的类 支持 HTTP 的 post 和 get 等方法。 3)容器如何找到指定的Servlet?(图示) Servlet有 3 个名字: url 名、 应用中的逻辑名、 实际的名字 配置文件 web.xml 都把这几个名字一一对应起来了 Servlet是受容器管理的 request对象、response对象都是容器根据客户端请求而产生的。 容器根据url请求,通过DD(web.xml)配置(url名,内部逻辑名,完全限定名)定位具体的servlet。 容器根据客户请求创建/分配一个线程,并调用servlet的service()方法,把request对象、respone对象作为参数传过去 service()方法会根据客户端请求的Http方法调用不同的HttpServlet的方法(doGet(),doPost(),.); Servlet使用response对象把处理完的结果响应到客户端。 service()方法结束后,线程要么撤销,要么返回容器的线程池;request对象、response对象都被gc回收。4.使用Servlet来发送用户输入的信息 1)开发用户登录的应用 2)如何使用Form表单来提交数据 (1)action: 表单提交给服务器的哪个资源处理 login 为在 web.xml 中配置的 url-pattern 或者是 jsp、html 文件等。 (2)Get vs. Post方法的比较(使用Http Monitor) GET-通过 URL 提交表单信息,由于受到 URL 长度的限制,只能传递大约 1024(1K)字节。 通过 http monitor 工具截获请求信息(下面仅写出请求行,请求首部略): GET /welcome/login.html?username=zhangsan&password=lisi HTTP/1.1 /GET是请求方式; /welcome/login.html是请求路径; ?号后面是参数列表,以&分隔; 最后是协议版本 POST-通过 Http Body 提交表单信息,传输的数据量大,可以达到 2M。 由请求行、请求首部、消息体 三部分组成。 参数放在消息体中,长度不再受限,更安全;而 GET 必须把参数放在请求行。 通过 http monitor 工具截获请求信息(大略): POST /welcome/login.html HTTP/1.1 /请求行 Accept: . Accept-Language: zh-cn,en-us;q=0.5 . Host: . /从请求行以下到这行,都是请求首部 username=zhangsan&password=zhangsan /消息体(有效负载) enctype: multipart/form-data :在form中设置此属性后,传输的就是二进制数据,常可用于上传文件5.如何在服务器端获得表单提供的数据 1)HttpServletRequest 对象 由 Web Container 封装了用户的 Http 请求数据包而生成,可通过它获得所有跟用户请求数据报相关的信息。 getProtocol():String 返回对应的协议 (如:HTTP/1.1) getMethod():String 返回 http 对应的方法 (Get|Post) getRemoteHost():String 返回远程主机名 (如:) getRemotePort():int 返回远程端口号 (如:55013) getHeader(String config):String 返回http对应的首部信息(参数如Accept-Language) getParameter(String name):String 返回指定参数(name)的值(value) (如:zhangsan) getParameterValues(String name):String 返回指定输入参数(名为 name)的所有值(输入的参数列表) getParameterNames():Enumeration 返回请求对象中全部输入参数的名称 (如:java.util.Hashtable$Enumerator1ff395a) 2)解决 Servlet 的中文乱码问题 (1)响应输出静态页面时,处理中文信息乱码: response.setContentType(text/html; charset=utf-8); (2)获取数据过程中,处理中文输入乱码(3 种方法): 方法一:设置字符编码来解决 post 方法提交表单中文乱码问题。 request.setCharacterEncoding(gbk); response.setContentType(text/html;charset=utf-8); 必须在第一个 request.getParameter(name); 之前执行上面的语句。 方法二:重新生成指定编码的字符串 String name = new String(request.getParamete(name).getBytes(iso-8859-1); 方法三:修改服务器的编码设置解决 get 方法提交表单中文乱码问题 例如:Tomcat,通过修改%TOMCAT%/conf/server.xml 加入 URIEncoding=utf-8 (3)静态 html 页面的中文化问题 6.请求路径 请求路径是把请求导向到一个 servlet 来响应服务。它是由几个重要的部分来组成的。 通过 HttpRequest 对象,暴露了如下信息(对照应用的目录结构): 1)上下文路径(Context Path) 该路径的前缀是和 ServletContext 相关的。 如果 Context 就是 Web 服务器的 URL 命名空间的缺省的根上下文时,那么上下文路径将会是一个空的字符串。 如果上下文并不是服务器的命名空间的根,则上下文路径就以“/”开始,但不能以“/”结束。 2)Servlet 路径(Servlet Path) 该路径就是直接与激活该请求的相应映射,它也是以“/”开头。 但要注意的是,如果指定的请求是与“/”模式相匹配,则此时 Servlet 路径就是一个空字符串。 3)路径信息(PathInfo) 请求路径中除了上下文路径和 Servlet 路径以外的部分。 当没有额外的路径时路径信息就是空的(null),或者它会以“/”开始的字符串。 在 HttpServletRequest 接口中定义如下的方法来访问请求路径的相应信息: getContextPath(); getServletPath(); getPathInfo(); 值得注意的是,请求 URI 和路径的各部分之间的 URL 编码的不同之外,下面的等式恒成立: requestURI = contextPath + servletPath + pathInfoDay21.配置开发环境 如何利用IDE开发(构建)一个web project 部署web project到Tomcat 结果是把WebRoot下的内容打成包,发布到webapps中2.在集成环境开发过程中的注意事项 如果修改了Java文件(修改了方法签名除外),无需重启,也无需重部署 如果修改了html文件,无需重启,但需要重部署 修改了web.xml,系统会自动重部署3.Servlet的生命周期 生命周期有哪几个过程(4个)? 每个过程都对应有特殊的生命周期方法 装载&实例化构造方法 初始化init() 只调用一次,并且在service()之前完成 处理请求service() 处理客户的请求,每个请求都在单独的线程中完成,可多次调用 销毁destroy() 只调用一次,通常在停止WEB应用或者是Web应用重启时 1)实例化(两种时机): A.配置了元素,启动应用时实例化 其中,配置数值建议从1开始计数,值越小,载入的优先级越高。 优先级提倡从1开始,1以下的数字,有些容器不理会。负数则被认为是“默认”。 B.如果没有配置,则在第一次请求时才实例化 2)初始化init()/init(ServletConfig config) 实例化之后,容器马上调用init()方法来初始化。 对 Servlet 将要使用的资源作初始化,如读入配置文件信息等(只执行一次)。 在初始化过程中,容器会创建 ServletConfig 对象并把它作为 init 方法的参数传入。 该配置对象允许 servlet 从访问 Web 应用的配置信息中读取出初始化参数的名-值对。 Servlet 提供2个初始化方法: init() 和 init(ServletConfig config) 。 3)请求处理service() 初始化后,容器会调用 servlet 的 service()方法,向客户端提供服务 service()能够被多客户端多次调用(每个请求都要执行一次) 4)销毁destroy() 在Web Container停止 Web App 或 WebApp 被停止/reload 时调用此方法。4.ServletConfig & ServletContext 1)ServletConfig servlet访问配置数据的一个对象,由容器创建,每个servlet独享 仅对本 servlet 有效,一个 servlet 的 ServletConfig 对象不能被另一个 servlet 访问。 主要用来读取 web.xml 中配置的Servlet初始信息,不能被其它 Servlet 共享。还可以用于访问 ServletContext。 2)ServletContext ServletContext 是多个 Servlet 共享数据的对象。 对同一个 Web 应用中的任何 servlet,在任何时间都有效。 对应整个 Web Application 运行环境,可以用来读取 web.xml 中配置的应用初始信息,写日志,共享数据等 ServletContext 被所有 Servlet 共享。可以理解为真正意义上的全局对象 3)如何获取ServletConfig A.init(ServletConfig config). 容器在Servlet初始化之前已经创建了ServletConfig 但如果Override此方法,需要记得调用:super.init(config);/其实是调用GenericServlet的init(config) /这里的config只是方法内部的变量。如果其它方法中需调用,还得:this.config=config;给成员变量的config赋值 B.其他地方(包括在init()方法中),可以使用Servlet API本身的方法 this.getServletConfig(); /任何方法中都可以调用,包括init(ServletConfig config)方法 注:这也是为什么把这个知识点放在这里的原因 4)如何获取ServletContext A.config.getServletContext();/前提是config在之前已获取 B.Servlet API提供了 this.getServletContext();/没有config也可以用 5)注意:不能在Servlet中使用this.getServletConfig或者this.getServletContext()来初始化成员变量 因为创建成员变量优先于构造方法和init方法;而 config 和 context 只有调用了 init 方法之后才初始化 6)利用ServletContext.log()来写日志 如:this.log(String+(new Date(); 7)例子 使用ServletContext来获取参数 使用SerlvetConfig来获取Servlet的5.产生动态内容的另一个方面:根据业务逻辑进行请求传递(页面跳转) RequestDispatcher(请求分发器) forward(request, response)/跳转到其他资源 include(request, response)/包含其他资源 如何获取RequestDispatcher request.getRequestDispatcher(page) servletcontext.getRequestDispatcher(page); 两者之间的区别?(后面会详细讲述)6.网上书店(打折优惠) SerlvetConfig来获取Servlet的 ServletContext来获取参数 RequestDispatcher进行页面包含Day3访问资源,Servlet如何与数据库构建应用系统1.两种访问DB的方法 1)直接连接DB,通过JDBC API 2)配置Tomcat的连接池 server.xml配置 web.xml:应用引用资源 init():通过JNDI API来获取DB Connection 2种方法都需要在Servlet的init()方法中,把DB Connection注入到Servlet中2、用Tomcat的jndi服务获取数据源 第一步:为 Tomcat 配置连接池: 修改tomcat/conf/server.xml 在节点中加入 Resource 配置 /Web应用的根 第二步:在应用中配置资源引用 (此步骤可以省略) 修改web.xml jdbc/tarena /资源引用名 javax.sql.DataSource /资源的类型 Container /Application Shareable /Unshareable 第三步:在 Servlet 的 init 方法中通过 JNDI 接口来获取 DataSource Context ctx=new InitialContext(); DataSource ds=(DataSource)ctx.lookup(java:comp/env/jdbc/tarena); Connection con=ds.getConnection();3.如何构建一个Web应用系统(Servlet + JDBC + DB) 分层设计的思想:表示层(view) + Dao层 Servlet层的设计: 废弃“为每个请求提供一个Servlet”的做法,引入Action接口与参数控制 Dao模式 工厂模式:DaoFactoryday4 会话管理 Cookie机制 Session机制 HTTP协议与状态保持:Http是一个无状态协议1. 实现状态保持的方案: 1)修改Http协议,使得它支持状态保持(难做到) 2)Cookies:通过客户端来保持状态信息 Cookie是服务器发给客户端的特殊信息 cookie是以文本的方式保存在客户端,每次请求时都带上它 3)Session:通过服务器端来保持状态信息 Session是服务器和客户端之间的一系列的交互动作 服务器为每个客户端开辟内存空间,从而保持状态信息 由于需要客户端也要持有一个标识(id),因此,也要求服务器端和客户端传输该标识, 标识(id)可以借助Cookie机制或者其他的途径来保存2. COOKIE机制 1)Cookie的基本特点 Cookie保存在客户端 只能保存字符串对象,不能保存对象类型 需要客户端浏览器的支持:客户端可以不支持,浏览器用户可能会禁用Cookie 2)采用Cookie需要解决的问题 Cookie的创建 通常是在服务器端创建的(当然也可以通过javascript来创建) 服务器通过在http的响应头加上特殊的指示,那么浏览器在读取这个指示后就会生成相应的cookie了 Cookie存放的内容 业务信息(key,value) 过期时间 域和路径 浏览器是如何通过Cookie和服务器通信? 通过请求与响应,cookie在服务器和客户端之间传递 每次请求和响应都把cookie信息加载到响应头中;依靠cookie的key传递。3. COOKIE编程 1)Cookie类 Servlet API封装了一个类:javax.servlet.http.Cookie,封装了对Cookie的操作,包括: public Cookie(String name, String value) /构造方法,用来创建一个Cookie HttpServletRequest.getCookies() /从Http请求中可以获取Cookies HttpServletResponse.addCookie(Cookie) /往Http响应添加Cookie public int getMaxAge() /获取Cookie的过期时间值 public void setMaxAge(int expiry) /设置Cookie的过期时间值 2)Cookie的创建 Cookie是一个名值对(key=value),而且不管是key还是value都是字符串 如: Cookie visit = new Cookie(visit, 1); 3)Cookie的类型过期时间 会话Cookie Cookie.setMaxAge(-1);/负整数 保存在浏览器的内存中,也就是说关闭了浏览器,cookie就会丢失 普通cookie Cookie.setMaxAge(60);/正整数,单位是秒 表示浏览器在1分钟内不继续访问服务器,Cookie就会被过时失效并销毁(通常保存在文件中) 注意: cookie.setMaxAge(0);/等价于不支持Cookie;4. SESSION机制 每次客户端发送请求,服务断都检查是否含有sessionId。 如果有,则根据sessionId检索出session并处理;如果没有,则创建一个session,并绑定一个不重复的sessionId。 1)基本特点 状态信息保存在服务器端。这意味着安全性更高 通过类似与Hashtable的数据结构来保存 能支持任何类型的对象(session中可含有多个对象) 2)保存会话id的技术(1) Cookie 这是默认的方式,在客户端与服务器端传递JSeesionId 缺点:客户端可能禁用Cookie 表单隐藏字段 在被传递回客户端之前,在 form 里面加入一个hidden域,设置JSeesionId: URL重写 直接在URL后附加上session id的信息 HttpServletResponse对象中,提供了如下的方法: encodeURL(url); /url为相对路径5. SESSION编程 1)HttpSession接口 Servlet API定义了接口:javax.servlet.http.HttpSession, Servlet容器必须实现它,用以跟踪状态。 当浏览器与Servlet容器建立一个http会话时,容器就会通过此接口自动产生一个HttpSession对象 2)获取Session HttpServletRequest对象获取session,返回HttpSession: request.getSession(); /表示如果session对象不存在,就创建一个新的会话 request.getSession(true); /等价于上面这句;如果session对象不存在,就创建一个新的会话 request.getSession(false); /表示如果session对象不存在就返回 null,不会创建新的会话对象 3)Session存取信息 session.setAttribute(String name,Object o) /往session中保存信息 Object session.getAttribute(String name) /从session对象中根据名字获取信息 4)设置Session的有效时间 public void setMaxInactiveInterval(int interval) 设置最大非活动时间间隔,单位秒; 如果参数interval是负值,表示永不过时。零则是不支持session。 通过配置web.xml来设置会话超时,单位是分钟 1 允许两种方式并存,但前者优先级更高 5)其他常用的API HttpSession.invalidate() /手工销毁Session boolean HttpSession.isN
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 2025年名师中国注册市场营销师职业资格认证模拟题
- 2025年偏摆检查仪项目建议书
- 2025年静脉输液耗材项目发展计划
- 2025年超细铜粉项目建议书
- 江西省南昌市南昌县2024-2025学年四年级上学期期末数学试题
- 河北省廊坊市文安县第一中学(5-18 班)2025-2026学年高二上学期开学生物试题(无答案)
- 抗疫课件模板
- 行政知识考试题及答案
- 关于消防的题目及答案
- 批注法阅读实例课件
- 农贸市场消防整改报告
- (高清版)DZT 0337-2020 矿产地质勘查规范 油砂
- 【培训课件】5S培训课程讲义
- 2000-2015年考研英语一真题及详细解析
- 2021年10月自考健康教育与健康促进试题及答案
- 假性软骨发育不全综合征介绍演示培训课件
- 保险行业纳税筹划案例分析
- 私立民办高中学校项目建议书
- 比亚迪汽车发展史
- 他们创造了数学:50位著名数学家的故事
- 财务管理-企业筹资方式
评论
0/150
提交评论