引言 < [q{0,
1. 前提 "T~A*a^
2. Java的学习 f#
sDG
3. 目标 (gb
vInZ
4. 联机文档 0>U7]wZKc
5. 章节 @}hdMVi
6. 练习 [wM]w
7. 多媒体CD-ROM 1a'0cSH
8. 源代码 k<k@Tlo
9. 编码样式 Bu7aeBP
10. Java版本 5wa!pR\c
11. 课程和培训 Kk 6i
12. 错误 }!jn%@_y@
13. 封面设计 /N=M9i\;
14. 致谢 0[7tJbN
C |P(,Xp
第1章 对象入门 >"pHk@AW K
1.1 抽象的进步 A (ZtA[G
1.2 对象的接口 M6z$*?<
1.3 实现方案的隐藏 SAokW,
1.4 方案的重复使用 7loIjT7
1.5 继承:重新使用接口 [*d<LAnuWP
1.5.1 改善基础类 m&k l_f7
1.5.2 等价和类似关系 9lc{{)m2)
1.6 多形对象的互换使用 p~h[4hP
1.6.1 动态绑定 AzFS6<_
1.6.2 抽象的基础类和接口 !.[H!-V.
1.7 对象的创建和存在时间 y1kI^B
1.7.1 集合与继承器 lBNB8c0e"{
1.7.2 单根结构 }cERCS\t
1.7.3 集合库与方便使用集合 %KW NY(m
1.7.4 清除时的困境:由谁负责清除? }/M`G]wT#
1.8 违例控制:解决错误 P/T`q:<H
1.9 多线程 ETrL3W<
1.10 永久性 c>L#(D\\
1.11 Java和因特网 }/}eZCaG
1.11.1 什么是Web? @8U8> 'zDE
1.11.2 客户端编程 nD#uOep9
1.11.3 服务器端编程 N.q0D5 :
1.11.4 一个独立的领域:应用程序 =|_k a8{?
1.12 分析和设计 I4MZJAYk
1.12.1 不要迷失 / e]R0NI
1.12.2 阶段0:拟出一个计划 \>c1Z5H>
1.12.3 阶段1:要制作什么? 1|oE3
1.12.4 阶段2:开始构建? -Rj3cx
1.12.5 阶段3:正式创建 +=#@1k~
1.12.6 阶段4:校订 /gq\.+'{
1.12.7 计划的回报 $(&+NJ$U$
1.13 Java还是C++? H<ZXe!q(nx
c[;=7-+
第2章 一切都是对象 YAYwrKt
2.1 用句柄操纵对象 y{J7^o(_~
2.2 必须创建所有对象 &-p!Lg&D
2.2.1 保存在什么地方 QHw{@*
2.2.2 特殊情况:主类型 $fQ'q3
2.2.3 Java中的数组 M
nDaag
2.3 绝对不要清除对象 YL9Tsw
2.3.1 作用域 A4f;ftB
2.3.2 对象的作用域 o5< w2(
2.4 新建数据类型:类 CzG/=#IU
2.4.1 字段和方法 ?/^{sW'
|
2.5 方法、自变量和返回值 {|R +|ow
2.5.1 自变量列表 'Jl3%axR
2.6 构建Java程序 9 N9Q#o$!.
2.6.1 名字的可见性 A5%cgr% 6
2.6.2 使用其他组件 Vl0Y'@{
2.6.3 static关键字 7WEoyd
2.7 我们的第一个Java程序 CAbT9Wz&
2.8 注释和嵌入文档 4|xQQv
2.8.1 注释文档 9DaoMOPEI
2.8.2 具体语法 h#]LXs
2.8.3 嵌入HTML vz`r
!xj)
2.8.4 @see:引用其他类 !-s 6B
2.8.5 类文档标记 !=(M P:
2.8.6 变量文档标记 z-;yDB:~t
2.8.7 方法文档标记 RbJbVFz8C
2.8.8 文档示例 Zie t-@}
2.9 编码样式 MFsW
2.10 总结 a\Dw*h?b~
2.11 练习 {#H'K*j{
Tizjh&*^
第3章 控制程序流程 <k7q9"\4
3.1 使用Java运算符 'T*h0xX
3.1.1 优先级 bXK$H=S Bz
3.1.2 赋值 8|-064i>
3.1.3 算术运算符 Y$N D
3.1.4 自动递增和递减
`d!~)D
3.1.5 关系运算符 k~pbXA*u
3.1.6 逻辑运算符 4Q^i"jT
3.1.7 按位运算符 0j2M< W#
3.1.8 移位运算符 :hUt7/3c
3.1.9 三元if-else运算符 JbW!V Y
3.1.10 逗号运算符 psB9~EU&Q
3.1.11 字串运算符+ sr`)l& t?
3.1.12 运算符常规操作规则 7K &j
3.1.13 造型运算符 -0Q^k\X-
3.1.14 Java没有“sizeof” 6cbIs_g
3.1.15 复习计算顺序 yNN_}9
3.1.16 运算符总结 [C( >e0r
3.2 执行控制 02~GT_)$^
3.2.1 真和假 za[;d4<}k
3.2.2 if-else o]m56
3.2.3 反复 z)&GF$*
3.2.4 do-while i0*6o3h
3.2.5 for F=8gtk|U
3.2.6 中断和继续 <!I^ xo[
3.2.7 切换 ~VaO,8&+L
3.3 总结 h} <Ie <
3.4 练习 {ZD'l5jU
,)P6fa/
第4章 初始化和清除 * AjJf)o
4.1 由构建器保证初始化 (S
k+nD
4.2 方法过载 $Qq5Fx9kU
4.2.1 区分过载方法 F)4;:".zna
4.2.2 主类型的过载 j[\aGS7u
4.2.3 返回值过载 -
u'5xn7
4.2.4 默认构建器 C4
@"@kbr
4.2.5 this关键字 WU<C7
4.3 清除:收尾和垃圾收集
#dm"!I>g
4.3.1 finalize()用途何在 h6C:`0o
4.3.2 必须执行清除 -MT.qhx
4.4 成员初始化 TH#5j.uUs
4.4.1 规定初始化 `=rDB7!$yL
4.4.2 构建器初始化 i 79;;9M
4.5 数组初始化 e6igx
4.5.1 多维数组 g)#.|d+
4.6 总结 8i'EO6
4.7 练习 rH5'+x K
h@"u==0
第5章 隐藏实施过程 'Z ,T,zW
5.1 包:库单元 LwS>jNJx
5.1.1 创建独一无二的包名 {1]/ok2k5
5.1.2 自定义工具库 C4/p5J
5.1.3 利用导入改变行为 %<Te&6NU'
5.1.4 包的停用 0w<qj T^U
5.2 Java访问指示符 l/o
4bkV
5.2.1 “友好的” e-/+e64Q@
5.2.2 public:接口访问 %)@3V8 OI
5.2.3 private:不能接触 Zi~-m]9U
5.2.4 protected:“友好的一种” iH[ .u{h
5.3 接口与实现 b_xGCBC
5.4 类访问 {[Vkht}
5.5 总结 mYiIwm1cb(
5.6 练习 2v9T&xo=
1!`B8y)
第6章 类再生 @8qo(7<~Q
6.1 合成的语法 o
9] 2
6.2 继承的语法 z'd*z[L~
6.2.1 初始化基础类 sQ8_j
6.3 合成与继承的结合 -Rz%<`
6.3.1 确保正确的清除 #Mmr{4m
6.3.2 名字的隐藏 NA9N#;
6.4 到底选择合成还是继承 )+xHv
6.5 protected zQhc
V
6.6 递增开发 S6pvbaMZ
6.7 上溯造型 h|S6LgB
6.7.1 何谓“上溯造型”? FR9*WI
6.8 final关键字 '}eA2Q>BV
6.8.1 final数据 Q( \2(x\
6.8.2 final方法 &H<n76G
6.8.3 final类 ~*,e &I
6.8.4 final的注意事项 ss>p
6.9 初始化和类装载 K{#1O=Gi
6.9.1 继承初始化 #:6gFfk0<
6.10 总结 G'qGsKf\
6.11 练习 6}9`z8
tfb_K4h6,
第7章 多形性 o(_~
st<
7.1 上溯造型 7y2-8eL
7.1.1 为什么要上溯造型 dN)!B!*aI
7.2 深入理解 FY6!)/P0I7
7.2.1 方法调用的绑定 mfpL?N
7.2.2 产生正确的行为 AD/7k3:
7.2.3 扩展性 +rA:/!b)Y
7.3 覆盖与过载 Cvf^3~q
7.4 抽象类和方法 G)'(%rl
7.5 接口 4RXF.kJ3=
7.5.1 Java的“多重继承” 'HdOW[3o
7.5.2 通过继承扩展接口 ek3,ss3
7.5.3 常数分组 A(<"oAe|
7.5.4 初始化接口中的字段 ;x=r.3OQy
7.6 内部类 @rT}V>2I
7.6.1 内部类和上溯造型 Sd0y=!Pj=
7.6.2 方法和作用域中的内部类 lp=8RbQYC
7.6.3 链接到外部类 37M?m$BL
7.6.4 static内部类 ?()$imb*
7.6.5 引用外部类对象 -WX{y Ci
7.6.6 从内部类继承 XdR^,;pWE
7.6.7 内部类可以覆盖吗? #MhieG5
7.6.8 内部类标识符 GE0,d
7.6.9 为什么要用内部类:控制框架 .oR_r1\y
7.7 构建器和多形性 D^%DYp
7.7.1 构建器的调用顺序 HECZZnM
7.7.2 继承和finalize() >
l@o\
7.7.3 构建器内部的多形性方法的行为 D>~S-]
7.8 通过继承进行设计 cA8"Ft{P)
7.8.1 纯继承与扩展 Q|`sYm'.
7.8.2 下溯造型与运行期类型标识 ?{Gf'Y}y&
7.9 总结 Z$'483<
7.10 练习 PN&;3z Z
$_P*Bk)
第8章 对象的容纳 j;+!BKWy4
8.1 数组 "ccP,#Y
8.1.1 数组和第一类对象 (Z72 3)
8.1.2 数组的返回 s{]2~Z^2od
8.2 集合 &oyj8
8.2.1 缺点:类型未知 tbP
;iK'
8.3 枚举器(反复器) MSMgaw?
8.4 集合的类型 ^8#;>+7R
8.4.1 Vector *ydU3LG7
8.4.2 BitSet HAi'0%"
8.4.3 Stack -'9sn/
8.4.4 Hashtable QE8aYPSFf
8.4.5 再论枚举器 =!RlU)w
8.5 排序 :$#";t|
8.6 通用集合库 @9wug!,
8.7 新集合 cG`R\$
8.7.1 使用Collections /4f4H?A -
8.7.2 使用Lists =
+=k(*
8.7.3 使用Sets nd 5w|83
8.7.4 使用Maps G2{.Ew
8.7.5 决定实施方案 ?l3PDorR
8.7.6 未支持的操作 u=5~^ 9
8.7.7 排序和搜索 !VfP#B6.
8.7.8 实用工具 r^$4]@Wn
8.8 总结 Ao:<aX,=
8.9 练习 ?oc#$fcQ~
nDhD"rc
第9章 违例差错控制 C6 XZZ
9.1 基本违例 xfE:r:
9.1.1 违例自变量 pd[?TyVK;
9.2 违例的捕获 9Xu
O\+z
9.2.1 try块 T%\f$jh6
9.2.2 违例控制器 C(^IX"9 #
9.2.3 违例规范 jo-qP4w
9.2.4 捕获所有违例 Ba9le|c5
9.2.5 重新“掷”出违例 v/Z!Wp1LV
9.3 标准Java违例 n;eK2+}]
9.3.1 RuntimeException的特殊情况 tw`{\kWG
9.4 创建自己的违例 1P'R-I
9.5 违例的限制 Wn9b</tf
9.6 用finally清除 BpGK`0H
9.6.1 用finally做什么 %:P&!F\?
9.6.2 缺点:丢失的违例 ,MtN_V-
9.7 构建器 KV!!D{VS`@
9.8 违例匹配 >4zH\T!
9.8.1 违例准则 Ny.s
u?E
9.9 总结 w<3g1n7R
9.10 练习 FE`:1
Z5vpo$l
第10章 Java IO系统 nI-^
10.1 输入和输出 "Zh6j)[o
10.1.1 InputStream的类型 f/r@9\x
10.1.2 OutputStream的类型 \>@'wl
10.2 增添属性和有用的接口 k_2W*2'S
10.2.1 通过FilterInputStream从InputStream里读入数据 d@:4se-q+
10.2.2 通过FilterOutputStream向OutputStream里写入数据 aK%i=6j!
10.3 本身的缺陷:RandomAccessFile %U.aRSf/
10.4 File类 X"KX_)GZD
10.4.1 目录列表器 n
2k&yL+a
10.4.2 检查与创建目录 &wlSOC')j
10.5 IO流的典型应用 em87`Hj^lo
10.5.1 输入流 )<D(Mb2p|
10.5.2 输出流 llf|d'5Nl
10.5.3 快捷文件处理 G|Du/XYh
10.5.4 从标准输入中读取数据 \&&jzU2
10.5.5 管道数据流 O>=D1no*
10.6 StreamTokenizer
W6&s_ (
10.6.1 StringTokenizer H)s$0Xd
10.7 Java 1.1的IO流 m#E%,
rT
10.7.1 数据的发起与接收 *E Z'S+wR
10.7.2 修改数据流的行为 ;t.LLd
10.7.3 未改变的类 =]pcC
10.7.4 一个例子 o*OaYF'8
10.7.5 重定向标准IO >k6RmN
10.8 压缩 ^<0azza/(
10.8.1 用GZIP进行简单压缩 \X=?+|
9
10.8.2 用Zip进行多文件保存 IT3xX=|b
10.8.3 Java归档(jar)实用程序 PD$gW`V
10.9 对象串联 !,mv 7Yj
10.9.1 寻找类 >-s\$8En'
10.9.2 序列化的控制 Ie#LZti
10.9.3 利用“持久性” Y5}<7s\UDO
10.10 总结 7*e7P[LQU
10.11 练习 vwZrvjP2
*}Al0\q0M
第11章 运行期类型鉴定 L6^Qn%:OTd
11.1 对RTTI的需要 :
kVEB<G
11.1.1 Class对象 D&0@k'
11.1.2 造型前的检查 PKoB~wLH
11.2 RTTI语法 !BN7 B
11.3 反射:运行期类信息 !$&3h-l[
11.3.1 一个类方法提取器 s2*^ PG
11.4 总结 NR8YVO)5$
11.5 练习 ,[To)x5o
FK
MuRy|
第12章 传递和返回对象 87P.K Yy
12.1 传递句柄 hw?'aXK{
12.1.1 别名问题 o(,u"c/Or
12.2 制作本地副本 /dU-$}>ZI
12.2.1 按值传递 MtYi8"+<e.
12.2.2 克隆对象 Rc2| o.'y
12.2.3 使类具有克隆能力 |OXufV?I
12.2.4 成功的克隆 RO-ABFEi(
12.2.5 Object.clone()的效果 0jY#,t?>
12.2.6 克隆合成对象 "o
+" Jd
12.2.7 用Vector进行深层复制 ;><9R@0
12.2.8 通过序列化进行深层复制 Zh?1+Sz&
12.2.9 使克隆具有更大的深度 tBNkVh(c
12.2.10 为什么有这个奇怪的设计 :I(-@2?{
12.3 克隆的控制 %KxL{HY
12.3.1 副本构建器 ?@"B:#l
12.4 只读类 u:+wuyu
12.4.1 创建只读类 ^<0u~u)%T
12.4.2 “一成不变”的弊端 BJgg-z{Y
12.4.3 不变字串 ;dt&*]wA
12.4.4 String和StringBuffer类 0*66m:C2
12.4.5 字串的特殊性
WH F>J
12.5 总结 $:I~y|
!1
12.6 练习 .6Swc?
\Uun2.K
第13章 创建窗口和程序片 Ol4)*/oZ
13.1 为何要用AWT? -1ke3
13.2 基本程序片 zi~_[l-
13.2.1 程序片的测试 gn&jNuGg
13.2.2 一个更图形化的例子 (>*<<a22
13.2.3 框架方法的演示 |A_yr/f
13.3 制作按钮 %k8} IBL
13.4 捕获事件 uJ<sa;
13.5 文本字段 =4z:Df
13.6 文本区域 KsM2?aqwf_
13.7 标签 S$q:hXZ#e
13.8 复选框 g:/l5~b
13.9 单选钮 HFD5*Z~M
13.10 下拉列表 ' bio:1
13.11 列表框 }
FcWzi
13.11.1 handleEvent() C<CE!|sfr
13.12 布局的控制 KDP7u
13.12.1 FlowLayout Ic4>kKh
13.12.2 BorderLayout S_cba(0-|\
13.12.3 GridLayout T)Byws
13.12.4 CardLayout
^|DI9G(Bs
13.12.5 GridBagLayout 24_F`" :-=
13.13 action的替用品 {l=!
13.14 程序片的局限 V.PbAN
13.14.1 程序片的优点 "WOY`su>
13.15 视窗化应用 %of#VSk
13.15.1 菜单 LGq}wxq
13.15.2 对话框 BBy"qkTe
13.16 新型AWT pX
]K-
13.16.1 新的事件模型 s$e0;C!D
13.16.2 事件和接收者类型 U@v=q9'W
13.16.3 用Java 1.1 AWT制作窗口和程序片 `INcZr"
13.16.4 再探早期示例 1P&XG@
13.16.5 动态绑定事件 {.2A+JT,
13.16.6 将商业逻辑与UI逻辑区分开 tE/s|v#O
13.16.7 推荐编码方法 w~>V2u_-
13.17 Java 1.1 UI API O#Ab1FQn
13.17.1 桌面颜色 c;l
d
13.17.2 打印 xe[Cuy$P
13.17.3 剪贴板 _@0>yMZ^
13.18 可视编程和Beans T!eb=oy
13.18.1 什么是Bean j;eR9jI$T
13.18.2 用Introspector提取BeanInfo z8+3/jLN0B
13.18.3 一个更复杂的Bean RrSo`q-h+
13.18.4 Bean的封装 \MM(w&
13.18.5 更复杂的Bean支持 +^hFs7je)
13.18.6 Bean更多的知识 X"<t3l(+
13.19 Swing入门 @~Z:W<X
13.19.1 Swing有哪些优点 A<.`HCv2
13.19.2 方便的转换 jvn:W{'Q
13.19.3 显示框架 }~$zdgMT
13.19.4 工具提示 <N^2|*3
13.19.5 边框 CZg$I&x
13.19.6 按钮 Qy |*[
13.19.7 按钮组 tv,iCV
13.19.8 图标 nb0V~W
13.19.9 菜单 v0`E
lkaN
13.19.10 弹出式菜单 wYZFW'5p
13.19.11 列表框和组合框 _O~DJ"
13.19.12 滑杆和进度指示条 D 4^2F(YRX
13.19.13 树 MnUal}MO
13.19.14 表格 Hw[u Sv8
13.19.15 卡片式对话框 x]6OE]]8L
13.19.16 Swing消息框 nE)?P*$3 Z
13.19.17 Swing更多的知识 lCg'K(|"
13.20 总结 ?cf9q@eAH
13.21 练习 rmutw~nHD
W>036
第14章 多线程 C0;c'4(
14.1 反应灵敏的用户界面 %{s<h6{R
14.1.1 从线程继承 k_.%(ZE
14.1.2 针对用户界面的多线程 n\&[^Q#b|
14.1.3 用主类合并线程 .0;Z:x_3
14.1.4 制作多个线程 '"Q;54S**
14.1.5 Daemon线程 &^^zm9{
14.2 共享有限的资源 ;_),?(
14.2.1 资源访问的错误方法 ]Ow
A>fb
14.2.2 Java如何共享资源 )xIk#>)
14.2.3 回顾Java Beans <zH24[
14.3 堵塞
?"[b408-
14.3.1 为何会堵塞 dX-Xzg
14.3.2 死锁 7*]O]6rP
14.4 优先级 _!kL7qJ"
14.4.1 线程组 5Og. :4
14.5 回顾runnable oVyOiWo\Z
14.5.1 过多的线程 .<j\"X(
14.6 总结 {j.5!Nj]B
14.7 练习 !8M]n
E9t[Mb %0
第15章 网络编程 Gb.r!W8
15.1 机器的标识 ~itrM3^"w
15.1.1 服务器和客户机 u{maE ,
15.1.2 端口:机器内独一无二的场所 ]Ec\!,54u
15.2 套接字 6VpT*,2d~
15.2.1 一个简单的服务器和客户机程序 [f,; +Ze
15.3 服务多个客户 8R}CvzI
15.4 数据报 ,cD(s(6+
15.5 一个Web应用 E$dPu
15.5.1 服务器应用 l -us j%\
15.5.2 NameSender程序片 =B_vQJF2
15.5.3 15.5.3 要注意的问题 `)$'1,]u
15.6 Java与CGI的沟通 >AI65g
15.6.1 CGI数据的编码 oF[l<OY4
15.6.2 程序片 ?]SSmZpk
15.6.3 用C++写的CGI程序 7<['4*u
15.6.4 POST的概念
< TJzp
15.7 用JDBC连接数据库 6Pc3 ;X~
15.7.1 获得学习示例 Q[J%
15.7.2 查找程序的GUI版本 oHH-joYnn
15.7.3 JDBC API为何如何复杂 $lmGMljF
15.8 远程方法 ;LM`B^Q]s
15.8.1 远程接口概念 WrBiAh,
15.8.2 远程接口的实施 l1XA9>n
15.8.3 创建根与干 yUcWX bT@
15.8.4 使用远程对象 eSNi6RvE
15.8.5 RMI的替选方案 (;-<
@~2
15.9 总结 &|'k)6Rx
15.10 练习 [A99e`
'B0=
"7
第16章 设计范式 v?Q&06PMRc
16.1 范式的概念 U{Xg#UN
16.1.1 单子 qELy'\
16.1.2 范式分类 nS'hdeoW
16.2 观察器范式 gOaL4tu
16.3 模拟垃圾回收站 fH[Yc>(oj
16.4 改进设计 60?/Z2w5
16.4.1 “制作更多的对象” WBdC}S
}3t
16.4.2 用于原型创建的一个范式 7kJ =C
16.5 抽象的应用 J+tpBPmb
16.6 多重派遣 Ao>] ~r0
16.6.1 实现双重派遣 GqB]^snh
16.7 访问器范式 V4kt&61
16.8 RTTI有害吗 8V$pdz| [
16.9 总结 $2N)m:X0
16.10 练习 @X\-c2=
:(gZ\q">k
第17章 项目 8wJfGY
17.1 文字处理 L}r#KfIb
17.1.1 提取代码列表 .=rS,Tpo
17.1.2 检查大小写样式 /~RY{ c@#L
17.2 方法查找工具 iOk^RDG+
17.3 复杂性理论 b7&5>Q/g
17.4 总结 Hxe!68{aR
17.5 练习 Bg.~#H
{akS K
附录A 使用非Java代码 >S\D+1PV
A.1 Java固有接口 _Ec9g^I10
A.1.1 调用固有方法 V?x&.C2Z
A.1.2 访问JNI函数:JNIEnv自变量 ft$@':F
A.1.3 传递和使用Java对象 }',/~T6
A.1.4 JNI和Java违例 yr=$a3web;
A.1.5 JNI和线程处理 >9F&x>~
A.1.6 使用现成代码 `a*_b9
A.2 微软的解决方案 op!8\rM<e
A.3 J/Direct zF'LbQz0[
A.3.1 @dll.import引导命令 t2V|moG
A.3.2 com.ms.win32包 w"?H4
A.3.3 汇集 `<}Q4p
A.3.4 编写回调函数 X)P;UVR0
A.3.5 其他J/Direct特性 =z_.RE
A.4 本原接口(RNI) /1A3
Sw
A.4.1 RNI总结 @APv?>$)
A.5 Java/COM集成 %#5yC|o9Pn
A.5.1 COM基础 [=f(u
wY>g
A.5.2 MS Java/COM集成 4KH8dau.fF
A.5.3 用Java设计COM服务器 <UI^~Azc#
A.5.4 用Java设计COM客户 V $'~2v{_
A.5.5 ActiveX/Beans集成 k'H+l]=
A.5.6 固有方法与程序片的注意事项 l-?B1gd,l
A.6 CORBA :2+,?#W
A.6.1 CORBA基础 !h\>[ O
A.6.2 一个例子 wrtJ8O(
A.6.3 Java程序片和CORBA la37cG
A.6.4 比较CORBA与RMI Q)`3&b
A.7 总结 ^t X}5i`P
."HDUo2D7
附录B 对比C++和Java dY|~"6d)
.K
I6<k/
附录C Java编程规则 W|C>X=zTi
J3
Y-d7=|
附录D 性能 &A}@@d
D.1 基本方法 z`)i"O]-K_
D.2 寻找瓶颈 Pn&