07.JavaWeb基础_Servlet笔记

Servlet基础

是用Java编写的服务器端程序

类关系

Servlet类关系图.bmp

根据生命周期打断点

启动Tomcat调试断点.gif

生命周期执行流程

init()方法

当我们首次访问HelloServlet时,会在tomcat中创建该Servlet,此时会执行init()方法,Java中方法执行是首先从当前类中找是否有此方法,如果没有的话就会往该类的父类里面找.
这就到了HttpServlet方法,发现里面也没有init()方法,继续往上找.
而GenericServlet中有相关的东西

public void init(ServletConfig config) throws ServletException {
    this.config = config;
    this.init();
}

public void init() throws ServletException {
}

第一个是生命周期的方法,而下面的方法是用来被子类覆写的初始化方法,这样的写法既可以实现初始化Servlet,又可以避免子类覆写父类的方法造成父类方法无法执行的问题.

service()方法

service()方法和这个是类似的流程.首先会从HelloServlet中寻找service()方法,没有,进入到HttpServlet中寻找,找到

public void service(ServletRequest req, ServletResponse res) throws ServletException, IOException {
    HttpServletRequest request;
    HttpServletResponse response;
    try {
        request = (HttpServletRequest)req;
        response = (HttpServletResponse)res;
    } catch (ClassCastException var6) {
        throw new ServletException("non-HTTP request or response");
    }

    this.service(request, response);
}

这里会调用一个service()的重载方法,这个重载方法里面有执行其中操作的具体方法,以doGet()为例,虽然HttpServlet中有doGet()的实现,但是我们的HelloServlet中覆写了doGet()方法,具体的实现就看我们的覆写的方法中的操作了.

destory()方法

这个没啥好写的,就是销毁

Servlet总结

  • Servlet是单例,一生只有一个实例。
  • 多个线程同时执行doGet方法,存在线程并发问题(安全问题)
    解决方案:不要使用成员变量
  • 初始化操作:复写init()
  • 业务操作:在doGet()doPost()
  • 所有的servlet都必须在web.xml配置

注意:配置当前使用的servlet(父类HttpServlet和爷类GenericServlet不要配置的)

淘淘商城-Nginx+FTP搭建图片服务器

Nginx

参看[淘淘商城-Nginx笔记]

FTP配置

http://blog.sina.com.cn/s/blog_a97c78020101o8fv.html

参看[淘淘商城-搭建FTP服务器]

整合Nginx和FTP

1.用户登录ftp的根目录

vi /etc/vsftpd/vsftpd.conf

添加如下内容

# 用户登录路径
local_root=/home/
# 锁定用户到各自目录为其根目录
chroot_local_user=YES
# 用户配置目录
user_config_dir=/etc/vsftpd/userconfig

创建/etc/vsftpd/userconfig

mkdir userconfig
cd userconfig/

配置各自用户访问根目录
vim ftpuser
添加

local_root=/home/ftpuser/www/

重启服务

/etc/init.d/vsftpd restart

2.配置nginx root路径为ftp路径

配置Nginx主目录

vi /usr/local/nginx/conf/nginx.conf

修改内容

server {
        listen       80;
        server_name  localhost;

        #charset koi8-r;

        #access_log  logs/host.access.log  main;

        location / {
            root   /home/ftpuser/www/;
            index  index.html index.htm;
        }

此时:

Nginx主目录:/home/ftpuser/www/

Vsftpd.config : local_root=/home/
Userconfig/ftpuser : local_root=/home/ftpuser/www/

由于我们使用的是自定义root路径,如果权限不够,在访问图片的时候可能会报403错误。所以我们需要在nginx.conf 里配置user root
QQ截图20160905140758.png

添加如下内容

QQ截图20160905144137.png

此时上传的根目录变成了 : /images
资源的访问url : http://192.168.2.11/images/a.jpg

杂项

客户端连接FTP服务器:

QQ截图20160905142301.png

CentOS使用命令行启动

打开/etc/inittab 文件

vi /etc/inittab

在默认的 run level 设置中,可以看到第一行书写如:id:5:initdefault:(默认的 run level 等级为 5,即图形界面)

将第一行的 5 修改为 3 即可。保存文件后重启系统你就可以看见是启动的文本界面了。

淘淘商城-搭建FTP服务器

1.前期准备

因为穷,所以使用的是VMware虚拟机来搭建

  1. 安装CentOS虚拟机(如果你有服务器更好)
  2. 准备IP地址(虚拟机内网 : 192.168.2.100),并配置好
  3. 准备客户端连接软件(FileZilla/Xftp/EditPlus)

2.开始安装

2.1安装vsftpd组件

[root@localhost ~]# yum -y install vsftpd

安装完后,有/etc/vsftpd/vsftpd.conf 文件,是vsftp的配置文件。

2.2添加一个ftp用户并设置密码

[root@localhost ~]# useradd ftpuser

此用户就是用来登录ftp服务器用的。登录后默认的路径为 /home/ftpuser

[root@localhost ~]# passwd ftpuser

输入两次密码后修改密码。 密码:123

2.3防火墙开启21端口

因为ftp默认的端口为21,而centos默认是没有开启的,所以要修改iptables文件

[root@localhost ~]# vim /etc/sysconfig/iptables

在行上面有22 -j ACCEPT 下面另起一行输入跟那行差不多的,只是把22换成21,然后:wq保存。

修改后的内容如下:

# Firewall configuration written by system-config-firewall
# Manual customization of this file is not recommended.
*filter
:INPUT ACCEPT [0:0]
:FORWARD ACCEPT [0:0]
:OUTPUT ACCEPT [0:0]
-A INPUT -m state --state ESTABLISHED,RELATED -j ACCEPT
-A INPUT -p icmp -j ACCEPT
-A INPUT -i lo -j ACCEPT
-A INPUT -m state --state NEW -m tcp -p tcp --dport 21 -j ACCEPT
-A INPUT -m state --state NEW -m tcp -p tcp --dport 22 -j ACCEPT
-A INPUT -j REJECT --reject-with icmp-host-prohibited
-A FORWARD -j REJECT --reject-with icmp-host-prohibited
COMMIT

还要运行下,重启iptables防火墙

[root@localhost ~]# service iptables restart

2.4修改selinux

外网是可以访问上去了,可是发现没法返回目录(使用ftp的主动模式,被动模式还是无法访问),也上传不了,因为selinux作怪了。

执行以下命令查看状态:

[root@localhost ~]# getsebool -a | grep ftp  
allow_ftpd_anon_write --> off
allow_ftpd_full_access --> off
allow_ftpd_use_cifs --> off
allow_ftpd_use_nfs --> off
ftp_home_dir --> off
ftpd_connect_db --> off
ftpd_use_passive_mode --> off
httpd_enable_ftp_server --> off
tftp_anon_write --> off

执行上面命令,再返回的结果看到allow_ftpd_full_access和ftp_home_dir两行都是off,代表,没有开启外网的访问

[root@localhost ~]# setsebool -P allow_ftpd_full_access on
[root@localhost ~]# setsebool -P ftp_home_dir on

这样应该没问题了(如果,还是不行,看看是不是用了ftp客户端工具用了passive模式访问了,如提示Entering Passive mode,就代表是passive模式,默认是不行的,因为ftp passive模式被iptables挡住了,下面会讲怎么开启,如果懒得开的话,就看看你客户端ftp是否有port模式的选项,或者把passive模式的选项去掉。如果客户端还是不行,看看客户端上的主机的电脑是否开了防火墙,关吧)

2.5关闭匿名访问

[root@localhost ~]# vim /etc/vsftpd/vsftpd.conf

anonymous_enable=YES改为anonymous_enable=NO

3.关闭匿名访问.png

重启ftp服务

[root@localhost ~]# service vsftpd restart

2.6开启被动模式

默认是开启的,但是要指定一个端口范围,打开vsftpd.conf文件,在后面加上

pasv_min_port=30000
pasv_max_port=30999

表示端口范围为30000~30999,这个可以随意改。改完重启一下vsftpd
由于指定这段端口范围,iptables也要相应的开启这个范围,所以像上面那样打开iptables文件。
也是在21上下面另起一行,更那行差不多,只是把21 改为30000:30999,然后:wq保存

修改后的内容如下:

# Generated by iptables-save v1.4.7 on Sat Sep  3 13:04:11 2016
*filter
:INPUT ACCEPT [0:0]
:FORWARD ACCEPT [0:0]
:OUTPUT ACCEPT [9:1116]
-A INPUT -p tcp -m tcp --dport 22 -j ACCEPT
-A INPUT -p tcp -m tcp --dport 21 -j ACCEPT
-A INPUT -p tcp -m tcp --dport 30000:30999 -j ACCEPT
-A INPUT -p tcp -m tcp --dport 80 -j ACCEPT
-A INPUT -m state --state RELATED,ESTABLISHED -j ACCEPT
-A INPUT -p icmp -j ACCEPT
-A INPUT -i lo -j ACCEPT
-A INPUT -p tcp -m state --state NEW -m tcp --dport 22 -j ACCEPT
-A INPUT -j REJECT --reject-with icmp-host-prohibited
-A FORWARD -j REJECT --reject-with icmp-host-prohibited
COMMIT
# Completed on Sat Sep  3 13:04:11 2016

重启iptables防火墙

[root@localhost ~]# service iptables restart

2.7设置开机启动vsftpd ftp服务

[root@localhost ~]# chkconfig vsftpd on

3.客户端连接工具介绍

3.1FileZilla

哈哈 这个没用过 虽然视频里面讲的是这个工具

3.2Xftp

一图胜千言
QQ截图20161102120402.png

3.3EditPlus

五图胜千言

1.png

2.png

3.png

4.png

5.png

总结

我竟然不知道EditPlus居然能连接FTP服务器,这样的话用这个东西来修改服务器配置就方便多了,不用再一点一点vim了(原谅我只会vim简单操作).

说明

这篇文章整理自黑马JavaEE的课程笔记中,外加了点自己的实践的东西
加油学,闪光之前总是要经历黑暗.^_^

淘淘商城-Nginx笔记

Nginx

1.介绍

Nginx(“engine x”)是一款是由俄罗斯的程序设计师Igor Sysoev所开发高性能的 Web和 反向代理 服务器,也是一个 IMAP/POP3/SMTP 代理服务器。
在高连接并发的情况下,Nginx是Apache服务器不错的替代品。

2.安装Nginx

2.1 安装环境

配置Nginx的安装环境:

yum install gcc-c++
yum install -y pcre pcre-devel
yum install -y zlib zlib-devel
yum install -y openssl openssl-devel

2.2 安装Nginx

下载nginx

wget http://nginx.org/download/nginx-1.8.0.tar.gz

解压

tar -zxvf nginx-1.8.0.tar.gz

进入目录

cd nginx-1.8.0

配置configure

./configure \
--prefix=/usr/local/nginx \
--pid-path=/var/run/nginx/nginx.pid \
--lock-path=/var/lock/nginx.lock \
--error-log-path=/var/log/nginx/error.log \
--http-log-path=/var/log/nginx/access.log \
--with-http_gzip_static_module \
--http-client-body-temp-path=/var/temp/nginx/client \
--http-proxy-temp-path=/var/temp/nginx/proxy \
--http-fastcgi-temp-path=/var/temp/nginx/fastcgi \
--http-uwsgi-temp-path=/var/temp/nginx/uwsgi \
--http-scgi-temp-path=/var/temp/nginx/scgi

注意:上边将临时文件目录指定为/var/temp/nginx,需要在/var下创建temp及nginx目录

编译生成Makefile

make

安装

make install

nginx安装成功截图

nginx安装成功.png

3.Nginx常用命令

3.1 启动Nginx

cd /usr/local/nginx/sbin/
./nginx 

查询nginx进程:
QQ截图20160905111916.png

4780是nginx主进程的进程id,4781是nginx工作进程的进程id

注意:执行./nginx启动nginx,这里可以-c指定加载的nginx配置文件,如下:
./nginx -c /usr/local/nginx/conf/nginx.conf
如果不指定-c,nginx在启动时默认加载conf/nginx.conf文件,此文件的地址也可以在编译安装nginx时指定./configure的参数(–conf-path= 指向配置文件(nginx.conf))

3.2 停止nginx

方式1 : 快速停止

cd /usr/local/nginx/sbin
./nginx -s stop

此方式相当于先查出nginx进程id再使用kill命令强制杀掉进程。
方式2 : 完整停止(建议使用):

cd /usr/local/nginx/sbin
./nginx -s quit

此方式停止步骤是待nginx进程处理任务完毕进行停止。

3.3 重启nginx

方式1 : 先停止再启动(建议使用)
对nginx进行重启相当于先停止nginx再启动nginx,即先执行停止命令再执行启动命令。

./nginx -s quit
./nginx

方式2 : 重新加载配置文件
当nginx的配置文件nginx.conf修改后,要想让配置生效需要重启nginx,使用-s reload不用先停止nginx再启动nginx即可将配置信息在nginx中生效

./nginx -s reload

Nginx无法站外访问?

QQ截图20160905112152.png

这是一个很尴尬的场面,传智的教程里面没有提到这一部分内容.此部分内容来自互联网和我自己的实践.

刚安装好nginx一个常见的问题是无法站外访问,本机wget、telnet都正常。而服务器之外,不管是局域网的其它主机还是互联网的主机都无法访问站点。如果用telnet的话,提示:
正在连接到192.168.0.xxx…不能打开到主机的连接, 在端口 80: 连接失败
CentOS防火墙在虚拟机的CENTOS装好APACHE不能用,郁闷,解决方法如下

/sbin/iptables -I INPUT -p tcp --dport 80 -j ACCEPT 
/sbin/iptables -I INPUT -p tcp --dport 22 -j ACCEPT 

如果还无法访问,则需配置一下Linux防火墙:

iptables -I INPUT 5 -i eth0 -p tcp --dport 80 -m state --state NEW,ESTABLISHED -j ACCEPT

然后保存:

/etc/rc.d/init.d/iptables save 

centos 5.3,5.4以上的版本需要用

service iptables save 
来实现保存到配置文件。 

这样重启计算机后,CentOS防火墙默认已经开放了80和22端口。
这里应该也可以不重启计算机:

/etc/init.d/iptables restart 

CentOS防火墙的关闭,关闭其服务即可:

查看CentOS防火墙信息:/etc/init.d/iptables status 
关闭CentOS防火墙服务:/etc/init.d/iptables stop 

永久关闭?不知道怎么个永久法:

chkconfig –level 35 iptables off。

QQ截图20160905112441.png

再次访问即可使用
QQ截图20160905112534.png

4.开机自启动nginx

4.1 编写shell脚本

这里使用的是编写shell脚本的方式来处理

vi /etc/init.d/nginx

输入下面的代码

#!/bin/bash
# nginx Startup script for the Nginx HTTP Server
# it is v.0.0.2 version.
# chkconfig: - 85 15
# description: Nginx is a high-performance web and proxy server.
#              It has a lot of features, but it's not for everyone.
# processname: nginx
# pidfile: /var/run/nginx.pid
# config: /usr/local/nginx/conf/nginx.conf
nginxd=/usr/local/nginx/sbin/nginx
nginx_config=/usr/local/nginx/conf/nginx.conf
nginx_pid=/var/run/nginx.pid
RETVAL=0
prog="nginx"
# Source function library.
. /etc/rc.d/init.d/functions
# Source networking configuration.
. /etc/sysconfig/network
# Check that networking is up.
[ ${NETWORKING} = "no" ] && exit 0
[ -x $nginxd ] || exit 0
# Start nginx daemons functions.
start() {
if [ -e $nginx_pid ];then
   echo "nginx already running...."
   exit 1
fi
   echo -n $"Starting $prog: "
   daemon $nginxd -c ${nginx_config}
   RETVAL=$?
   echo
   [ $RETVAL = 0 ] && touch /var/lock/subsys/nginx
   return $RETVAL
}
# Stop nginx daemons functions.
stop() {
        echo -n $"Stopping $prog: "
        killproc $nginxd
        RETVAL=$?
        echo
        [ $RETVAL = 0 ] && rm -f /var/lock/subsys/nginx /var/run/nginx.pid
}
# reload nginx service functions.
reload() {
    echo -n $"Reloading $prog: "
    #kill -HUP `cat ${nginx_pid}`
    killproc $nginxd -HUP
    RETVAL=$?
    echo
}
# See how we were called.
case "$1" in
start)
        start
        ;;
stop)
        stop
        ;;
reload)
        reload
        ;;
restart)
        stop
        start
        ;;
status)
        status $prog
        RETVAL=$?
        ;;
*)
        echo $"Usage: $prog {start|stop|restart|reload|status|help}"
        exit 1
esac
exit $RETVAL

然后 :wq 保存并退出(别告诉我你不会这个)

4.2 设置文件的访问权限

chmod a+x /etc/init.d/nginx   (a+x ==> all user can execute  所有用户可执行)

这样在控制台就很容易的操作nginx了:查看Nginx当前状态、启动Nginx、停止Nginx、重启Nginx…

如果修改了nginx的配置文件nginx.conf,也可以使用上面的命令重新加载新的配置文件并运行,可以将此命令加入到rc.local文件中,这样开机的时候nginx就默认启动了

4.3 加入到rc.local文件中

vi /etc/rc.local

加入一行 /etc/init.d/nginx start 保存并退出,下次重启会生效。

#!/bin/sh
#
# This script will be executed *after* all the other init scripts.
# You can put your own initialization stuff in here if you don't
# want to do the full Sys V style init stuff.

touch /var/lock/subsys/local

# nginx auto start
/etc/init.d/nginx start

Nginx应用-反向代理(Reverse Proxy)

Hibernate笔记一

Hibernate介绍

Hibernate是轻量级JavaEE应用的持久层解决方案,是一个关系数据库ORM框架.Hibernate的持久化方案,将用户从原始的JDBC底层SQL访问中解放出来,用户无须关注底层数据库操作,只要通过操作映射到数据表的Java对象,就可以对数据库进行增删改查

ORM (Object Relational Mapping对象关系映射)
ORM 就是通过将Java对象映射到数据库表,通过操作Java对象,就可以完成对数据表的操作

流行数据库框架 
1. JPA(Java Persistence API) : 通过注解描述对象与数据表映射关系 (只有接口规范)
2. Hibernate : 最流行ORM框架,通过对象-关系映射配置,可以完全脱离底层SQL , Hibernate实现JPA规范 
3. MyBatis : 本是apache的一个开源项目 iBatis,支持普通 SQL查询,存储过程和高级映射的优秀持久层框架 (企业主流)
    * MyBaits 并不是完全ORM , 需要在xml中配置SQL语句
4. Apache DBUtils(超链接到DBUtils那节笔记里面去)
5. Spring JDBCTemplate

SQL语句封装程度 Hibernate > MyBatis > Apache DBUtils 、Spring JDBCTemplate

Hibernate中的ORM实现

数据库中的表与java中的类对应
表中的记录是与类中的对象对应
表中的字段是与类中的属性对应

ORM对应关系.png

Hibernate体系结构

Hibernate架构.png

Hibernate目录结构

Hibernate目录结构.png

Hibernate使用

1.导入jar包

导包的内容参见上一步,目录结构中导入即可

特别介绍: log4j(链接上log4j的单独的笔记内容)

2.编写类和关系映射配置xx.hbm.xml

public class Customer {
    private int id;
    private String name;
    private int age;
    private String city;
    ...
}

hibernate 完全ORM,只需要操作Customer类对象, 自动生成SQL 操作customer 表
但是需要为实体类和数据表进行关系映射配置
1.在实体类所在的包下创建名为:类名.hbm.xml文件,eg:Customer.hbm.xml
2.配置规则参见 hibernate3.jar org/hibernate/hibernate-mapping-3.0.dtd





















配置属性到列映射时,指定类型,类型有三种写法
第一种 java类型 java.lang.String
第二种 hibernate类型 string
第三种 SQL类型 varchar(20)

3.配置核心配置文件hibernate.cfg.xml

Hibernate框架支持两种 Hibernate属性配置方式
1.hibernate.properties
采用properties方式,必须手动编程加载 hbm文件或者 持久化类
2.hibernate.cfg.xml
采用XML配置方式,可以通过配置添加hbm文件
规则参见 hibernate3.jar /org/hibernate/hibernate-configuration-3.0.dtd

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-configuration PUBLIC
    "-//Hibernate/Hibernate Configuration DTD 3.0//EN"
    "http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
    <!-- 会话连接工厂,建立数据库连接需要SessionFactory -->
    <session-factory>
        <!-- JDBC连接基本参数 -->
        <property name="hibernate.connection.driver_class">com.mysql.jdbc.Driver</property>
        <property name="hibernate.connection.url">jdbc:mysql:///hibernatetest</property>
        <property name="hibernate.connection.username">root</property>
        <property name="hibernate.connection.password">1234</property>
        <!-- 配置数据库方言,便于生成一些与数据库相关SQL方言 -->
        <property name="hibernate.dialect">org.hibernate.dialect.MySQL5Dialect</property>
        <!-- DDL策略   可以根据需要自动创建数据表 -->
        <property name="hibernate.hbm2ddl.auto">update</property>
        <!-- 将SQL语句 输出到控制台 -->
        <property name="hibernate.show_sql">true</property>
        <property name="hibernate.format_sql">true</property>

        <!-- 导入映射的po类的xx.hbm.xml映射文件-->            
        <mapping resource="cn/itcast/domain/Customer.hbm.xml"></mapping>
    </session-factory>
</hibernate-configuration>

注意:
配置文件必须放在根目录下
通常采用方式2来配置

编程操作使用Hibernate框架

web层的javabean叫vo viewobject

service层 bo business object

dao层 po 持久化对象
po类需要有个无参构造,get/set方法,私有化属性,属性使用包装类型,不要用基本类型,类不能为final(因为要为他生成代理),
还需要有一个与表中主键映射的字段


hbm映射介绍

pojo是个啥? 视频两分钟的时候介绍了下

基础加强--框架加强

新特性

Static Import : 静态导入,当前类中可以直接使用静态资源(字段|方法)

格式:import static java.lang.Math.*;

Varargs :可变参数

public void demo(String... args){ 当成数组使用 }

Autoboxing/Unboxing 自动装箱、自动拆箱(基本类型 <–> 包装类)

装箱:基本类型 --> 包装类。例如:Integer i = 10;  --> Integer m = new Integer(10);
拆箱:包装类 --> 基本类型。例如:int j = i;  --> int n = m.intValue();

Enhanced for Loop 增强for循环

for(类型 变量 : 容器){}   --> 容器:数组、Iterable接口(iterator方法)

Typesafe Enums 枚举

public class Hello {}
public enum Color {}

泛型

特点

1.将运行时的问题提升到编译时
2.方法级别的泛型定义主要是用在工具类中的
    public class TestGeneric2 {
        public void demo(){
            int m = 10;
            String s = init("abc");
            Integer i = init(123);
        }
        //方法级别的定义  MyBeanUtils
        public <T> T init(T obj){
            //算法
            System.out.println(obj);
            return obj;
        }
    }

泛型擦除

以下两个方法不能同时存在
public void init(List<String> list){}
public void init(List<Integer> list){}
类型擦除之后,两个方法是完全相同

反射中Class对象的三种获取方式:

Class.forName("全限定类名");
String.class
obj.getClass

反射和泛型混合使用的例子:

public class Dao<T> {
    private Session session;
    private Class beanClass;
    public Dao() {
        //1 在运行时,获得当前运行类 父类中泛型的实际参数
        //this.getClass().getSuperclass(); //jdk1.4之前,获得父类,不含泛型信息
        // * 获得被参数化的类型  ArrayList<Integer> ,含有泛型信息
        ParameterizedType type = (ParameterizedType) this.getClass().getGenericSuperclass();
        //2 获得所有的实际参数类型。类<A,B,C,D> ,返回数组,获得第一个实际参数
        Type firstType = type.getActualTypeArguments()[0];

        this.beanClass = (Class) firstType;

        /*
            //回顾 反射 api
            beanClass.getName(); //类名
            beanClass.getMethod(name, parameterTypes)  //方法
            beanClass.getConstructor(parameterTypes) //构造
            beanClass.getField(name)  //字段
        */
        System.out.println(beanClass);
    }

    /* 泛型 ,将运行时问题 提升 编译时
     * 泛型实参只有在运行时才可以确定,编译时 T 不能确定类型(User | Book) 只有在运行时才可以确定
     * 思想:如何在运行时获得实际类型参数?
     */
    public T findById(Integer id){
        return (T) session.get(beanClass, id);
    }
}

注解

介绍

注解就是类,用于修饰对象(类/构造/字段/方法等),常用语取代xml配置.
但是,开发中常用注解+xml混合使用

JDK提供的常用注解

@Override
    在jdk1.5表示子类覆写父类的方法
    在jdk1.6表示子类实现父接口的方法
@Deprecated
    表示被修饰对象已经过期了,过期的方法依旧可以使用.
    以下情况被标记过期:1.安全问题 2.有新的API
@SuppressWarnings
    抑制警告,如果有警告通过编辑器不进行警告
        deprecation:如果过期不警告
        rawtypes:没有泛型
        unused:未使用
        null:空指针
        serial:序列化
        all:所有
    建议:尽量不使用抑制

自定义注解

注解架构

定义注解
使用注解
解析注解

基本语法

定义注解使用关键字 : `@interface` ,和定义类class相似
    public @interface xxx{}
成员格式: 修饰符 返回值 属性名() [default 默认值]
    eg: public String username();
    注意点:
        1.修饰符:默认值 public abstract,且只能是这两个
        2.返回值:只能是基本类型/字符串String/Class/注解/枚举,以及以上类型的一维数组.

使用注解—重点!!!

注解出现的目的就是用来代替xml的

AJAX笔记

Ajax介绍&编写流程

使用异步的方式从浏览器端发送请求,请求服务器端资源,并获得内容一种技术。
ajax 不是新技术,是多个技术整合:javascript、html、css、xml,XMLHttpRequest
a.png

使用步骤:

1.获得核心类(引擎)
2.编写回调函数,获得响应内容
3.建立连接
4发送请求

同步操作 : 访问servlet地址栏改变
异步操作 : 访问servlet地址栏不改变

Jsonlib

需要导入的jar包:

commons-beanutils-1.8.3.jar
commons-collections-3.2.1.jar
commons-lang-2.6.jar
commons-logging-1.1.1.jar
ezmorph-1.0.6.jar
json-lib-2.4-jdk15.jar

Java对象转json:

JSONArray.fromObject(javaObject).toString()

Json转Java对象:

JSONObject obj = JSONObject.fromObject(jsonStr);
Bean bean = (Bean) JSONObject.toBean(obj, Bean.class);

注意:

JsonConfig配置信息,设置排除的字段

User user = new User("u001", "jack", "1234");
// 配置信息 --设置排除
JsonConfig jsonConfig = new JsonConfig();
jsonConfig.setExcludes(new String[]{"username","password"});

// java 对象  转换 json  --> {'k':'v',.....}
String str = JSONObject.fromObject(user,jsonConfig).toString();
System.out.println(str);// {"id":"u001"}

XStream

需要导入的jar包:

xmlpull-1.1.3.1.jar
xpp3_min-1.1.4c.jar
xstream-1.4.7.jar

Java对象转xml:

public void demo02(){
    User user = new User("u007", "路家豪", "123");
    //1 核心类
    XStream xStream = new XStream();
    // * 设置别名,如果不设置的话会使用User类的全限定类名来作为根元素
    xStream.alias("user", User.class);
    //2 转换成xml
    String xmlData = xStream.toXML(user);
    System.out.println(xmlData);
    /*
     * <user>
          <id>u007</id>
          <username>路家豪</username>
          <password>123</password>
        </user>
     */
}

设置属性的值:
public void demo03(){
    User user = new User("u007", "路家豪", "123");
    //1 核心类
    XStream xStream = new XStream();
    // * 设置元素别名
    xStream.alias("user", User.class);
    // * 设置属性别名 , 三个参数分别标售:指定javabean/property属性名/xml中显示的属性名
    xStream.aliasAttribute(User.class, "id", "id");
    //2 转换成xml
    String xmlData = xStream.toXML(user);
    System.out.println(xmlData);

    /*
     * <user id="u007">
          <username>路家豪</username>
          <password>123</password>
        </user>
     */
}

忽略字段:
public void demo06(){
    User user = new User("u007", "路家豪", "123");
    //1 核心类
    XStream xStream = new XStream();
    // * 设置元素别名
    xStream.alias("user", User.class);
    // * 设置属性别名 , 指定javabean property 使用其他别名
    xStream.aliasAttribute(User.class, "id", "id");
    // * 忽略指定的字段
    xStream.omitField(User.class, "username");

    //2 转换成xml
    String xmlData = xStream.toXML(user);
    System.out.println(xmlData);
    /*
     * <user id="u007">
          <password>123</password>
        </user>
     */
}

xml转Java对象:

public void demo04(){
    String str = "<user id='u007'><username>路家豪</username><password>123</password></user>";
    //1 核心类
    XStream xStream = new XStream();
    // * 设置元素别名
    xStream.alias("user", User.class);
    // * 设置属性别名 , 指定javabean property 使用其他别名
    xStream.aliasAttribute(User.class, "id", "id");

    //2 xml转换成 javabean
    User user = (User)xStream.fromXML(str);
    System.out.println(user);
}

案例一:判断用户名是否存在

html代码:

<input id="name" type="text" name="name" onblur="sendData(this)" onfocus="document.getElementById('spanId').innerHTML=''"/>
<span id="spanId"></span>

js代码:

<script type="text/javascript">
    function sendData(obj) {
        // 0.获得输入的数据
        var inputValue = obj.value;
        /* 1 创建核心类
             XMLHttpRequest ajax引擎,不同的浏览器对此对象的创建方式不同。存在浏览器兼容问题
             * 在所有现代浏览器中(包括 IE 7):
                xmlhttp=new XMLHttpRequest()
             * 在 Internet Explorer 5 和 6 中:
                xmlhttp=new ActiveXObject("Microsoft.XMLHTTP")
         */
        var xmlhttp = null;
        if (window.XMLHttpRequest) {// code for all new browsers
            xmlhttp = new XMLHttpRequest();
        } else if (window.ActiveXObject) {// code for IE5 and IE6
            xmlhttp = new ActiveXObject("Microsoft.XMLHTTP");
        }
        /* 2 设置回调函数,通过属性设置 onreadystatechange
             * 目的:发送请求之后可以获得服务器响应内容。
             * 一般情况使用匿名函数实现
             * ajax引擎在不同的阶段都会调用回调函数。
             * 属性:readyState 和 stauts
             2.1 readyState 返回整形数据,表示当前执行某一个阶段。
                 * 0  初始化状态。核心对象创建时默认值
                 * 1 open() 方法已调用,连接创建完成之后,ajax引擎将此状态修改1
                 * 2 send() 方法已调用,发送请求
                 * 3  接受中, 所有响应头部都已经接收到。响应体开始接收但未完成。
                 * 4  响应已经完全接收。【掌握】--服务器发送的所有数据,已经到ajax引擎内部。
             2.2 status 表示  响应的http状态码
                 * 200 正常【掌握】
                 * 302 重定向
                 * 304 缓存
                 * 404 不存在
                 * 500 服务器异常
         */
        xmlhttp.onreadystatechange = function () {
            if (xmlhttp.readyState == 4 && xmlhttp.status == 200){
                // 3.1 接收服务器响应的数据,获得json数据,注意json也是文本
                var data = xmlhttp.responseText;
                // 3.2 将字符串转化成json对象  使用格式:eval("('abc')");
                var jsonData = eval("("+data+")");

                // 3.3 判断  控制按钮是否可用
                var buttonObj = document.getElementById("buttonId");
                var spanObj = document.getElementById("spanId");
                if (jsonData.flag) {// 可用
                    buttonObj.removeAttribute("disabled");
                    spanObj.style.color = "#3D882D";
                } else {// 不可用
                    buttonObj.setAttribute("disabled","disabled");
                    spanObj.style.color = "#CC0000";
                }
                // 3.4 将信息显示到指定位置
                spanObj.innerHTML = jsonData.msg;
            }
        };
        //3 建立连接
        xmlhttp.open("POST","${pageContext.request.contextPath}/CustomerServlet?method=checkName");

        // 4 POST请求需要设置编码
        xmlhttp.setRequestHeader("content-type", "application/x-www-form-urlencoded");

        //5 发送请求  格式: name=lujiahao&password=1234
        xmlhttp.send("name=" + inputValue);
    }
</script>

Servlet端代码:

/**
 * Ajax查询用户名是否存在
 */
private void checkName(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException{
    // 0响应乱码
    response.setContentType("application/json;charset=UTF-8");
    try {
        // 1.获得数据并封装
        String name = request.getParameter("name");
        // 2.通知service查询所有
        //CustomerServeice customerServeice = new CustomerServiceImpl();
        // customerServeice.checkName(name);

        PrintWriter out = response.getWriter();
        if (name == null || "".equals(name)) {
            out.print("{'flag':false,'msg':'用户名不能为空'}");
            return;
        } else if ("lujiahao".equals(name)) {
            out.print("{'flag':false,'msg':'用户名已经存在'}");
        } else {
            out.print("{'flag':true,'msg':'用户名可用'}");
        }
    } catch (Exception e){
        // 1.打印日志
        e.printStackTrace();
    }
}

案例二:省市县三级联动

Listener-监听器

TODO:这一部分不太明白,需要后面在好好看看吧

分页

实现思路

QQ截图20160725151218.png

分页语法

select ..... limit 参数1,参数2;
参数1 ,表示开始索引,从0开始。startIndex  ,算法 (pageNum - 1) * pageSize
参数2 ,表示每页显示个数。pageSize

Servlet的写法

/**
 * 查询所有--分页查询
 */
private void findAllWithPage(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException{
    try {
        // 1.获得数据并封装
        // 获得当前页码
        String pageNumStr = request.getParameter("pageNum");
        int pageNum = 1;
        try {
            // 这种处理方式可以防止在地址栏里面输入非数字导致转化异常
            pageNum = Integer.parseInt(pageNumStr);
        } catch (Exception e){
        }

        // * 每页显示个数(固定值)
        int pageSize = 5;
        // 2.通知service查询所有
        CustomerServeice customerServeice = new CustomerServiceImpl();
        PageBean<Customer> pageBean = customerServeice.findAllCustomerWithPage(pageNum, pageSize);

        // 3.显示
        // 存放到request作用域中--每一次查询都是新的数据
        request.setAttribute("pageBean",pageBean);
        // servlet到jsp中显示,一次请求需要使用请求转发
        request.getRequestDispatcher("/pages/show_all_page.jsp").forward(request,response);
    } catch (Exception e){
        // 1.打印日志
        e.printStackTrace();
        // 2.请求转发到消息界面
        request.setAttribute("msg","查询失败,请稍后重试.");
        request.getRequestDispatcher("/pages/message.jsp").forward(request,response);
    }
}

Service写法

@Override
public PageBean<Customer> findAllCustomerWithPage(int pageNum, int pageSize) {
    // 一般分页的逻辑是要在service里面来写
    // 创建PageBean封装分页所有的数据,并将封装后的数据返回

    // 1.查询总记录数
    int totalRecord = customerDao.getTotalRecord();
    // 2.创建PageBean
    PageBean<Customer> pageBean = new PageBean<>(pageNum,pageSize,totalRecord);
    // 3.查询结果
    List<Customer> data = customerDao.findAll(pageBean.getStartIndex(), pageBean.getPageSize());
    pageBean.setData(data);
    return pageBean;
}

Dao写法

@Override
public List<Customer> findAll(int startIndex, int pageSize) {
    try {
        String sql = "select * from t_customer limit ?,?";
        Object[] params = {startIndex,pageSize};
        return runner.query(sql,new BeanListHandler<Customer>(Customer.class),params);
    } catch (Exception e){
        throw new DaoException(e);
    }
}

JSP界面

当前${pageBean.pageNum}页,共${pageBean.totalPage}页,总条数${pageBean.totalRecord}条<br/>

<c:url value="${pageContext.request.contextPath}/CustomerServlet" var="baseUrl">
    <c:param name="method" value="findAllWithPage"></c:param>
    <c:param name="pageNum" value=""></c:param>
</c:url>
<c:if test="${pageBean.pageNum > 1}">
    <a href="${baseUrl}1">首页</a>
    <a href="${baseUrl}${pageBean.pageNum-1}">上一页</a>
</c:if>
<c:forEach begin="${pageBean.start}" end="${pageBean.end}" var="num">
    <a href="${baseUrl}${num}">${num}</a>
</c:forEach>
<c:if test="${pageBean.pageNum < pageBean.totalPage}">
    <a href="${baseUrl}${pageBean.pageNum+1}">下一页</a>
    <a href="${baseUrl}${pageBean.totalPage}">尾页</a>
</c:if>

PageBean的封装

/**
 * 分页的封装类
 *      封装的两种思路:
 *          -- 集合Map(Android中常见的请求参数的封装)
 *          -- 自定义JavaBean对象
 * Created by lujiahao on 2016/7/25.
 */
public class PageBean<T> {

    // 分页必备选项,为下面计算提供数据
    private int pageNum;            // 当前页(第几页)
    private int pageSize;           // 每页显示个数
    private int totalRecord;        // 总记录数(总条数)---这个数据需要通过查询来获得

    // 通过计算获得的数据
    private int startIndex;         // 分页开始的索引
    private int totalPage;          // 总分页数

    // 分页查询的结果
    private List<T> data;           // 查询分页的数据--使用泛型的目的是为了方便复用

    // 导航条动态显示  首页 上一页 1 2 3 4 下一页 尾页
    private int start;              // 循环开始
    private int end;                // 循环结束

    public PageBean(int pageNum, int pageSize, int totalRecord) {
        // 构造方法中初始化三个必备选项
        this.pageNum = pageNum;
        this.pageSize = pageSize;
        this.totalRecord = totalRecord;

        // 处理地址栏输入负页数
        if (this.pageNum < 1) {
            this.pageNum = 1;
        }

        // 计算分页开始的索引:(当前页 - 1) * 每页显示个数
        this.startIndex = (this.pageNum - 1) * this.pageSize;

        // 计算总分页数
        if (this.totalRecord % this.pageSize == 0){
            // 能整除,总分页数 = 总记录数 / 每页显示个数
            this.totalPage = this.totalRecord / this.pageSize;
        } else {
            // 不能整除,需要加一页用来存不够一页的数据
            this.totalPage = this.totalRecord / this.pageSize + 1;
        }
        // 上面的快捷算法---暂时理解不了啊
        //this.totalPage = (this.totalRecord + (this.pageSize - 1)) / this.pageSize;

        // 导航条动态显示  默认显示10页
        this.start = 1;
        this.end = 10;
        // 总页数不够10页
        if (this.totalPage <= 10){
            this.end = this.totalPage;
        } else {
            // 总页数大于10页
            // 页数要求 前五后四
            this.start = this.pageNum -5;
            this.end = this.pageNum +4;

            // 当pageNum=1时,其实页数至少是1
            if (this.start < 1){
                this.start = 1;
                this.end = 10;
            }

            // 当pageNum到最后一页事
            if (this.end > this.totalPage){
                this.end = this.totalPage;
                this.start = this.totalPage -9; // 9 = 5 + 4
            }
        }
    }
    ...get/set
}

注意点

容错的处理非常巧妙:
这是在Servlet中的处理方式

String pageNumStr = request.getParameter("pageNum");
int pageNum = 1;
try {
    // 这种处理方式可以防止在地址栏里面输入非数字导致转化异常
    pageNum = Integer.parseInt(pageNumStr);
} catch (Exception e){
}

这是在PageBean中的处理方式

// 处理地址栏输入负页数
if (this.pageNum < 1) {
    this.pageNum = 1;
}

JDBC笔记二

1. JDBC批处理

Statement 和 PreparedStatement 都提供批处理。
批处理:批量处理sql语句。
Statement的批处理,可以一次性执行不同的sql语句。应用场景:系统初始化(创建数据库,创建不同表)
PreparedStatement 的批处理,只能执行一条sql语句,实际参数可以批量。应用场景:数据的初始化

1.1 Statement

addBatch(sql)  ,给批处理缓存中添加sql语句。
clearBatch();  清空批处理缓存。
executeBatch() , 执行批处理缓存所有的sql语句。注意:执行完成之后缓存中的内容仍然存在。

1.2 PreparedStatement

addBatch()  , 将实际参数批量设置缓存中。注意:获得预处理之前必须提供sql语句。

mysql 默认没开启批处理。
通过URL之后参数开启, ?rewriteBatchedStatements = true

2. 事务

2.1 什么是事务

  • 事务:一组业务逻辑操作,要么全部成功,要么全部不成功。
  • 事务特性:ACID
    • 原子性:一个事务是不可划分的一个整体。要么成功,要么不成功。
    • 一致性:事务操作前后数据一致。(数据完整)。转账前后,两个人和一样的。
    • 隔离性:多个事务对同一个内容并发操作。
    • 持久性:事务提交,操作成功了。不能改变了。保存到数据库中了。
  • 隔离问题
    • 脏读:一个事务 读到 另一个 事务 没有提交的数据。
    • 不可重复读:一个事务 读到 另一个事务 已经提交的数据。(update更新)
    • 虚读(幻读):一个事务 读到 另一个事务 已经提交的数据。(insert插入) –理论时
  • 事务隔离级别:用于解决隔离问题
    1. 读未提交:read uncommitted,一个事务读到另一个事务没有提交的数据。存放问题:3个
    2. 读已提交:read committed,一个事务读到另一个事务提交的数据。解决:脏读,存在2个问题
    3. 可重复读:repeatable read ,一个事务中读到数据重复的。及时另一个事务已经提交。解决:脏读、不可重复读,存放1个问题。
    4. 串行化,serializable,单事务,同时只能一个事务在操作。另一个事务挂起(暂停)。解决:3个问题。
  • 默认隔离级别
    mysql默认隔离级别:repeatable read
    oracle默认隔离级别:read committed
    对比:
    性能:read uncommitted > read committed > repeatable read > serializable
    安全:read uncommitted < read committed < repeatable read < serializable

2.2 事务操作

  • mysql 命令操作
    开启事务:mysql> start transaction; –开启相当于关闭自动提交
    提交事务:mysql> commit; –全部成功
    回滚事务:mysql> rollback; –保证全部不成功
  • jdbc java代码操作【掌握】– JDBC中必须使用Connection连接进行事务操作。
    开启事务:conn.setAutoCommit(false);
    提交事务:conn.commit();
    回滚事务:conn.rollback();

mysql 默认事务提交的,及每执行一条sql语句就是一个事务。
扩展:oracle事务默认不提交,需要手动提交。

2.3 保存点

Savepoint 保存点,用于记录程序执行位置,方便可以随意回滚指定的位置。spring 事务的传播行为
AB整体(必须),CD整体(可选)

Savepoint savepoint = null;
try{
   // 1 开启事务
   conn.setAutoCommit(false);
   A
   B
   // 记录保存点
   savepoint = conn.setSavepoint();
   C
   D

   //2 提交ABCD
   conn.commit();
} catch(){
  if(savepoint != null){
      //CD 有异常,回滚到CD之前
      conn.rollback(savepoint);
      // 提交AB
      conn.commit();
  } else {
     //AB有异常 ,回滚到最开始处
     conn.rollback();
  }
}

2.4 丢失更新

案例

A 查询数据,username = ‘jack’ ,password = ‘1234’
B 查询数据,username=”jack”, password=”1234”
A 更新用户名 username=”rose”,password=’1234’ –> username=”rose”,password=”1234”
B 更新密码 password=”9999” ,username=”jack” –> username=”jack”,password=’9999’
丢失更新:最后更新数据,将前面更新的数据覆盖了。

解决方案:采用锁机制。

乐观锁:丢失更新肯定不会发生。

给表中添加一个字段(标识),用于记录操作版本。
username="jack",password="1234",version="1" ,比较版本号,如果一样,修改版本自动+1.。如果不一样,必须先查询,再更新。

悲观锁:丢失更新肯定会发生。采用数据库锁机制。

读锁:共享锁,大家可以一起读。
    select .... from .... lock in share mode;
写锁:排他锁,只能一个进行写,不能有其他锁(写、读),所有更新update操作丢将自动获得写锁。
    select ... from ... for update;

注意:数据库的锁,必须在事务中使用。
只要事务操作完成(commit|rollback|超时)自动释放锁

3. 事务实例

4. 连接池

1.为什么使用连接池:连接Connection 创建与销毁 比较耗时的。为了提供性能,开发连接池。
2.什么是连接池

javaee规范规定:连接池必须实现接口,javax.sql.DataSource (数据源)
为了获得连接 getConnection()
连接池给调用者提供连接,当调用者使用,此链接只能供调动者是使用,其他人不能使用。
当调用者使用完成之后,必须归还给连接池。连接必须重复使用。

3.自定义连接池
4.第三方连接池

DBCP,apache
C3P0 ,hibernate 整合
tomcat 内置(JNDI)

4.1 DBCP

1.Apache提供的
2.导入jar包

commons-dbcp-1.4.jar 核心包
commons-pool-1.6.jar 依赖包

3.核心类

public class BasicDataSource implements DataSource

4.手动调用

//1 创建核心类
BasicDataSource dataSource = new BasicDataSource();
//2 配置4个基本参数
dataSource.setDriverClassName("com.mysql.jdbc.Driver");
dataSource.setUrl("jdbc:mysql:///day16_db");
dataSource.setUsername("root");
dataSource.setPassword("1234");
//3 管理连接配置
dataSource.setMaxActive(30);    //最大活动数
dataSource.setMaxIdle(20);        //最大空闲数
dataSource.setMinIdle(10);        //最小空闲数
dataSource.setInitialSize(15);    //初始化个数

5.配置调用

DBCP采用properties文件,key=value ,key为 BasicDataSource属性(及setter获得)
//0 读取配置文件
InputStream is = TestDBCP.class.getClassLoader().getResourceAsStream("dbcpconfig.properties");
Properties properties = new Properties();
properties.load(is);
//1 加载配置文件,获得配置信息
DataSource dataSource = BasicDataSourceFactory.createDataSource(properties);

6.配置文件 dbcpconfig.properties

#连接设置
driverClassName=com.mysql.jdbc.Driver
url=jdbc:mysql://localhost:3306/day16_db
username=root
password=1234

#<!-- 初始化连接 -->
initialSize=10

#最大连接数量
maxActive=50

#<!-- 最大空闲连接 -->
maxIdle=20

#<!-- 最小空闲连接 -->
minIdle=5

#<!-- 超时等待时间以毫秒为单位 6000毫秒/1000等于60秒 -->
maxWait=60000


#JDBC驱动建立连接时附带的连接属性属性的格式必须为这样:[属性名=property;] 
#注意:"user" 与 "password" 两个属性会被明确地传递,因此这里不需要包含他们。
connectionProperties=useUnicode=true;characterEncoding=gbk

#指定由连接池所创建的连接的自动提交(auto-commit)状态。
defaultAutoCommit=true

#driver default 指定由连接池所创建的连接的只读(read-only)状态。
#如果没有设置该值,则“setReadOnly”方法将不被调用。(某些驱动并不支持只读模式,如:Informix)
defaultReadOnly=

#driver default 指定由连接池所创建的连接的事务级别(TransactionIsolation)。
#可用值为下列之一:(详情可见javadoc。)NONE,READ_UNCOMMITTED, READ_COMMITTED, REPEATABLE_READ, SERIALIZABLE
defaultTransactionIsolation=READ_UNCOMMITTED

4.2 C3P0

1.第三方提供,非常优秀
2.导入jar包

c3p0-0.9.2-pre5.jar 核心包
mchange-commons-java-0.2.3.jar 依赖包
c3p0-oracle-thin-extras-0.9.2-pre5.jar 使用oracle的依赖

3.核心类

ComboPooledDataSource

4.手动调用

//1 核心类 (日志级别:debug info warn  error)
ComboPooledDataSource dataSource = new ComboPooledDataSource();
//2 基本4项
dataSource.setDriverClass("com.mysql.jdbc.Driver");
dataSource.setJdbcUrl("jdbc:mysql:///day16_db");
dataSource.setUser("root");
dataSource.setPassword("1234");
//3 优化
dataSource.setMaxPoolSize(30);        //最大连接池数
dataSource.setMinPoolSize(10);        //最小连接池数
dataSource.setInitialPoolSize(15);    //初始化连接池数
dataSource.setAcquireIncrement(5);    //每一次增强个数

5.配置调用

加载位置:WEB-INF/classes  (classpath , src)
配置文件名称:c3p0-config.xml
//1 c3p0...jar 将自动加载配置文件。打包后从WEB-INF/classes (src)目录加载名称为c3p0-config.xml的文件
//ComboPooledDataSource dataSource = new ComboPooledDataSource(); //自动从配置文件 <default-config>
ComboPooledDataSource dataSource = new ComboPooledDataSource("itheima"); //手动指定配置文件 <named-config name="itheima"> 

6.配置文件 c3p0-config.xml

<c3p0-config>
    <!-- 默认配置,如果没有指定则使用这个配置 -->
    <default-config>
        <property name="driverClass">com.mysql.jdbc.Driver</property>
        <property name="jdbcUrl">jdbc:mysql:///day16_db</property>
        <property name="user">root</property>
        <property name="password">1234</property>

        <property name="checkoutTimeout">30000</property>
        <property name="idleConnectionTestPeriod">30</property>
        <property name="initialPoolSize">10</property>
        <property name="maxIdleTime">30</property>
        <property name="maxPoolSize">100</property>
        <property name="minPoolSize">10</property>
        <property name="maxStatements">200</property>
        <user-overrides user="test-user">
            <property name="maxPoolSize">10</property>
            <property name="minPoolSize">1</property>
            <property name="maxStatements">0</property>
        </user-overrides>
    </default-config> 
    <!-- 命名的配置 -->
    <named-config name="itheima">
        <property name="driverClass">com.mysql.jdbc.Driver</property>
        <property name="jdbcUrl">jdbc:mysql:///day16_db</property>
        <property name="user">root</property>
        <property name="password">1234</property>
    <!-- 如果池中数据连接不够时一次增长多少个 -->
        <property name="acquireIncrement">5</property>
        <property name="initialPoolSize">20</property>
        <property name="minPoolSize">10</property>
        <property name="maxPoolSize">40</property>
        <property name="maxStatements">0</property>
        <property name="maxStatementsPerConnection">5</property>
    </named-config>
</c3p0-config> 

4.3 JNDI

  • tomcat管理连接池
  • JNDI(Java Naming and Directory Interface,Java命名和目录接口)是SUN公司提供的一种标准的Java命名系统接口,用于存放java任意对象,给对象进行命名(使用目录结构)。例如:/a/b/c
  • 作用:在多个项目之间共享数据,只需要传递名称,就可以获得对象。
  • 理解:JNDI 就是一个容器,用于存放任意对象。
  • tomcat 将连接池存放 JNDI容器,可以获得使用。使用前提必须通知tomcat进行存放,默认情况下tomcat没有存放。

配置:
1.给tomcat配置数据源(连接池),使用 标签

方式1:%tomcat%/conf/server.xml -->  在<Host>标签下添加<Context>
方式2:%tomcat%/conf/Catalina/localhost/day16.xml  ---> <Context>
    day16/META-INF/Context.xml  在项目的META-INF下创建Context.xml,会自动发布到“方法2”指定位置
Context.xml中的内容:
    <Context>
        <!-- 
            name 存放进去名称
            auth 存放位置
            type 确定存放内容,tomcat将通过指定接口创建实例,底层使用DBCP
            其他都是DBCP属性设置
         -->
      <Resource name="jdbc/itheima" auth="Container" type="javax.sql.DataSource"
                   maxActive="100" maxIdle="30" maxWait="10000"
                   username="root" password="1234" driverClassName="com.mysql.jdbc.Driver"
                   url="jdbc:mysql://localhost:3306/day16_db"/>
    </Context>

2 从JNDI容器获取,在当前项目web.xml配置

<!-- 给当前项目配置,从JNDI容器获得指定名称内容 -->
<resource-ref>
  <res-ref-name>jdbc/itheima</res-ref-name>
  <res-type>javax.sql.DataSource</res-type>
  <res-auth>Container</res-auth>
</resource-ref>

3.使用

//0 包名:javax.naming
//1 初始化JNDI容器
Context context = new InitialContext();
//2 获得数据源   固定 “java:/comp/env”
DataSource dataSource = (DataSource)context.lookup("java:/comp/env/jdbc/itheima");